gibMacOS/Scripts/downloader.py

163 lines
6.4 KiB
Python
Raw Normal View History

import sys, os, time, ssl, gzip
from io import BytesIO
2018-12-28 01:11:18 +01:00
# Python-aware urllib stuff
if sys.version_info >= (3, 0):
2019-03-07 05:02:22 +01:00
from urllib.request import urlopen, Request
2018-12-28 01:11:18 +01:00
else:
# Import urllib2 to catch errors
import urllib2
2019-03-07 05:02:22 +01:00
from urllib2 import urlopen, Request
2018-12-28 01:11:18 +01:00
class Downloader:
2019-03-07 05:02:22 +01:00
def __init__(self,**kwargs):
self.ua = kwargs.get("useragent",{"User-Agent":"Mozilla"})
# Provide reasonable default logic to workaround macOS CA file handling
cafile = ssl.get_default_verify_paths().openssl_cafile
try:
# If default OpenSSL CA file does not exist, use that from certifi
if not os.path.exists(cafile):
import certifi
cafile = certifi.where()
self.ssl_context = ssl.create_default_context(cafile=cafile)
except:
# None of the above worked, disable certificate verification for now
self.ssl_context = ssl._create_unverified_context()
2018-12-28 01:11:18 +01:00
return
2019-03-07 05:02:22 +01:00
def _decode(self, value, encoding="utf-8", errors="ignore"):
# Helper method to only decode if bytes type
if sys.version_info >= (3,0) and isinstance(value, bytes):
return value.decode(encoding,errors)
return value
def open_url(self, url, headers = None):
# Fall back on the default ua if none provided
headers = self.ua if headers == None else headers
2018-12-28 01:11:18 +01:00
# Wrap up the try/except block so we don't have to do this for each function
try:
response = urlopen(Request(url, headers=headers), context=self.ssl_context)
2018-12-28 01:11:18 +01:00
except Exception as e:
# No fixing this - bail
return None
2018-12-28 01:11:18 +01:00
return response
2019-09-07 22:16:41 +02:00
def get_size(self, size, suffix=None, use_1024=False, round_to=2, strip_zeroes=False):
# size is the number of bytes
# suffix is the target suffix to locate (B, KB, MB, etc) - if found
# use_2014 denotes whether or not we display in MiB vs MB
# round_to is the number of dedimal points to round our result to (0-15)
# strip_zeroes denotes whether we strip out zeroes
# Failsafe in case our size is unknown
2018-12-28 01:11:18 +01:00
if size == -1:
return "Unknown"
2019-09-07 22:16:41 +02:00
# Get our suffixes based on use_1024
ext = ["B","KiB","MiB","GiB","TiB","PiB"] if use_1024 else ["B","KB","MB","GB","TB","PB"]
div = 1024 if use_1024 else 1000
2018-12-28 01:11:18 +01:00
s = float(size)
2019-09-07 22:16:41 +02:00
s_dict = {} # Initialize our dict
# Iterate the ext list, and divide by 1000 or 1024 each time to setup the dict {ext:val}
2018-12-28 01:11:18 +01:00
for e in ext:
s_dict[e] = s
2019-09-07 22:16:41 +02:00
s /= div
# Get our suffix if provided - will be set to None if not found, or if started as None
suffix = next((x for x in ext if x.lower() == suffix.lower()),None) if suffix else suffix
# Get the largest value that's still over 1
biggest = suffix if suffix else next((x for x in ext[::-1] if s_dict[x] >= 1), "B")
# Determine our rounding approach - first make sure it's an int; default to 2 on error
try:round_to=int(round_to)
except:round_to=2
round_to = 0 if round_to < 0 else 15 if round_to > 15 else round_to # Ensure it's between 0 and 15
bval = round(s_dict[biggest], round_to)
# Split our number based on decimal points
a,b = str(bval).split(".")
# Check if we need to strip or pad zeroes
b = b.rstrip("0") if strip_zeroes else b.ljust(round_to,"0") if round_to > 0 else ""
return "{:,}{} {}".format(int(a),"" if not b else "."+b,biggest)
2018-12-28 01:11:18 +01:00
def _progress_hook(self, response, bytes_so_far, total_size):
if total_size > 0:
percent = float(bytes_so_far) / total_size
percent = round(percent*100, 2)
t_s = self.get_size(total_size)
try:
b_s = self.get_size(bytes_so_far, t_s.split(" ")[1])
except:
b_s = self.get_size(bytes_so_far)
sys.stdout.write("Downloaded {} of {} ({:.2f}%)\r".format(b_s, t_s, percent))
else:
2019-03-26 23:05:32 +01:00
b_s = self.get_size(bytes_so_far)
2019-03-07 05:02:22 +01:00
sys.stdout.write("Downloaded {}\r".format(b_s))
2018-12-28 01:11:18 +01:00
2019-03-07 05:02:22 +01:00
def get_string(self, url, progress = True, headers = None):
response = self.open_url(url, headers)
2018-12-28 01:11:18 +01:00
if not response:
return None
2020-05-13 13:15:26 +02:00
CHUNK = 1024 * 1024
2018-12-28 01:11:18 +01:00
bytes_so_far = 0
try:
total_size = int(response.headers['Content-Length'])
except:
total_size = -1
chunk_so_far = "".encode("utf-8")
while True:
chunk = response.read(CHUNK)
bytes_so_far += len(chunk)
if progress:
self._progress_hook(response, bytes_so_far, total_size)
if not chunk:
break
chunk_so_far += chunk
2019-03-07 05:02:22 +01:00
return self._decode(chunk_so_far)
2018-12-28 01:11:18 +01:00
2019-03-07 05:02:22 +01:00
def get_bytes(self, url, progress = True, headers = None):
response = self.open_url(url, headers)
2018-12-28 01:11:18 +01:00
if not response:
return None
2020-05-13 13:15:26 +02:00
CHUNK = 1024 * 1024
2018-12-28 01:11:18 +01:00
bytes_so_far = 0
try:
total_size = int(response.headers['Content-Length'])
except:
total_size = -1
chunk_so_far = "".encode("utf-8")
while True:
chunk = response.read(CHUNK)
bytes_so_far += len(chunk)
if progress:
self._progress_hook(response, bytes_so_far, total_size)
if not chunk:
break
chunk_so_far += chunk
if response.headers.get('Content-Encoding') == "gzip":
fileobj = BytesIO(chunk_so_far)
file = gzip.GzipFile(fileobj=fileobj)
return file.read()
2018-12-28 01:11:18 +01:00
return chunk_so_far
2019-03-07 05:02:22 +01:00
def stream_to_file(self, url, file, progress = True, headers = None):
response = self.open_url(url, headers)
2018-12-28 01:11:18 +01:00
if not response:
return None
2020-05-13 13:15:26 +02:00
CHUNK = 1024 * 1024
2018-12-28 01:11:18 +01:00
bytes_so_far = 0
try:
total_size = int(response.headers['Content-Length'])
except:
total_size = -1
with open(file, 'wb') as f:
while True:
chunk = response.read(CHUNK)
bytes_so_far += len(chunk)
if progress:
self._progress_hook(response, bytes_so_far, total_size)
if not chunk:
break
f.write(chunk)
if os.path.exists(file):
return file
else:
return None