2022-05-27 16:41:34 +02:00
|
|
|
#!/usr/bin/env python3
|
2022-10-04 00:01:06 +02:00
|
|
|
from abc import ABC, abstractmethod, abstractproperty
|
2022-05-27 16:41:34 +02:00
|
|
|
from datetime import datetime, timedelta, timezone
|
|
|
|
from email.utils import parsedate_to_datetime
|
|
|
|
import os, requests, re, time
|
|
|
|
import tempfile
|
|
|
|
import shutil
|
|
|
|
|
|
|
|
class Downloader(ABC):
|
|
|
|
|
|
|
|
extensions = ('.ota', '.ota.signed', '.zigbee', '.fw2', '.sbl-ota')
|
2022-10-04 00:01:06 +02:00
|
|
|
archive_extensions = [ ext for sublist in [ x[1] for x in shutil.get_unpack_formats() ] for ext in sublist ]
|
|
|
|
otau_path = os.path.join(os.path.expanduser('~/otau/'))
|
|
|
|
log_path = os.path.join(otau_path, 'log.log')
|
2022-05-27 16:41:34 +02:00
|
|
|
|
2022-12-25 23:52:39 +01:00
|
|
|
def __init__(self, verbose):
|
|
|
|
self.verbose = verbose
|
|
|
|
|
2022-05-27 16:41:34 +02:00
|
|
|
@abstractmethod
|
2022-10-04 00:01:06 +02:00
|
|
|
def get_url_list(self):
|
2022-05-27 16:41:34 +02:00
|
|
|
pass
|
|
|
|
|
2022-10-04 00:01:06 +02:00
|
|
|
def perform_downloads(self):
|
2022-05-27 16:41:34 +02:00
|
|
|
print("")
|
2022-10-04 00:01:06 +02:00
|
|
|
print(f"Putting {self.__class__.__name__} updates in {self.otau_path}")
|
|
|
|
if not os.path.exists(self.otau_path):
|
|
|
|
os.makedirs(self.otau_path)
|
2022-05-27 16:41:34 +02:00
|
|
|
|
|
|
|
cnt = 0
|
2022-10-04 00:01:06 +02:00
|
|
|
retries = self.get_url_list()
|
2022-05-27 16:41:34 +02:00
|
|
|
delay = None
|
|
|
|
while cnt == 0 or (cnt < 50 and delay):
|
2022-10-04 00:01:06 +02:00
|
|
|
retries, delay = self.handle_downloads(retries, delay)
|
2022-05-27 16:41:34 +02:00
|
|
|
cnt += 1
|
|
|
|
|
2022-10-04 00:01:06 +02:00
|
|
|
def handle_downloads(self, lst, delay):
|
2022-05-27 16:41:34 +02:00
|
|
|
retries = []
|
|
|
|
|
|
|
|
if delay:
|
|
|
|
nowish = datetime.now(timezone.utc) + timedelta(seconds=-2)
|
|
|
|
if delay > nowish:
|
|
|
|
wait = (delay - nowish).seconds + 1
|
2022-12-25 23:52:39 +01:00
|
|
|
if self.verbose:
|
|
|
|
print(f"Some downloads were deferred by the server. Waiting {wait} seconds until retry", end='', flush=True)
|
2022-05-27 16:41:34 +02:00
|
|
|
ix = 0
|
|
|
|
while ix < wait:
|
|
|
|
print('.', end='', flush=True)
|
|
|
|
ix += 1
|
|
|
|
time.sleep(1)
|
|
|
|
print("", flush=True)
|
|
|
|
|
|
|
|
newDelay = None
|
|
|
|
|
|
|
|
for (url, filename) in lst:
|
2022-10-04 00:01:06 +02:00
|
|
|
ret = self.download_file(url, filename, retries)
|
2022-05-27 16:41:34 +02:00
|
|
|
if ret is None or isinstance(ret, datetime):
|
|
|
|
if ret is not None or newDelay is None or ret > newDelay:
|
|
|
|
newDelay = ret
|
|
|
|
continue
|
|
|
|
|
|
|
|
fname, firmwarecontent = ret
|
2022-10-04 00:01:06 +02:00
|
|
|
self.handle_content(fname, firmwarecontent, url)
|
2022-05-27 16:41:34 +02:00
|
|
|
return retries, newDelay
|
|
|
|
|
2022-10-04 00:01:06 +02:00
|
|
|
def download_file(self, url, filename, retries):
|
|
|
|
if filename and os.path.isfile(os.path.join(self.otau_path, filename)):
|
2022-12-25 23:52:39 +01:00
|
|
|
if self.verbose:
|
|
|
|
print(f"{filename} skipped. A file with that name already exists")
|
2022-05-27 16:41:34 +02:00
|
|
|
return None
|
|
|
|
|
|
|
|
response = requests.get(url)
|
|
|
|
if 'Retry-After' in response.headers:
|
|
|
|
timestamp = parsedate_to_datetime(response.headers['Date'])
|
|
|
|
delay = timedelta(seconds=int(response.headers['Retry-After']) + 1)
|
|
|
|
retries.append((url, filename))
|
|
|
|
return timestamp + delay
|
|
|
|
|
|
|
|
fname: str = filename
|
|
|
|
|
|
|
|
if 'Content-Disposition' in response.headers:
|
|
|
|
contentDisposition = response.headers['Content-Disposition']
|
|
|
|
contentDisposition = re.findall("filename=(.+)", contentDisposition)[0]
|
|
|
|
contentDisposition = contentDisposition.split(";")
|
|
|
|
fname = contentDisposition[0]
|
|
|
|
|
2022-10-04 00:01:06 +02:00
|
|
|
|
2022-05-27 16:41:34 +02:00
|
|
|
return fname, response.content
|
|
|
|
|
2022-10-04 00:01:06 +02:00
|
|
|
def handle_content(self, fname: str, firmwarecontent, src: str):
|
|
|
|
if fname.lower().endswith(self.extensions):
|
|
|
|
fullname = os.path.join(self.otau_path, fname)
|
2022-05-27 16:41:34 +02:00
|
|
|
|
|
|
|
if not os.path.isfile(fullname):
|
|
|
|
file = open(fullname, "wb")
|
|
|
|
file.write(firmwarecontent)
|
|
|
|
file.close()
|
|
|
|
print(f"{fname} downloaded")
|
2022-10-04 00:01:06 +02:00
|
|
|
self.write_log(src, fname, len(firmwarecontent))
|
2022-05-27 16:41:34 +02:00
|
|
|
else:
|
2022-12-25 23:52:39 +01:00
|
|
|
if self.verbose:
|
|
|
|
print(f"{fname} skipped. A file with that name already exists")
|
2022-05-27 16:41:34 +02:00
|
|
|
else:
|
|
|
|
with tempfile.TemporaryDirectory() as tmpdirname:
|
|
|
|
fullname = os.path.join(tmpdirname, fname)
|
|
|
|
|
|
|
|
if not os.path.isfile(fullname):
|
|
|
|
file = open(fullname, 'wb')
|
|
|
|
file.write(firmwarecontent)
|
|
|
|
file.close()
|
2022-10-04 00:01:06 +02:00
|
|
|
if list(filter(lambda ext: fname.endswith(ext), self.archive_extensions)):
|
|
|
|
shutil.unpack_archive(fullname, tmpdirname)
|
2022-12-25 23:52:39 +01:00
|
|
|
if self.verbose:
|
|
|
|
print(f"Downloaded and unpacked {fname}")
|
2022-10-04 00:01:06 +02:00
|
|
|
for f in self.filtered_filelist(tmpdirname):
|
|
|
|
target = os.path.join(self.otau_path, os.path.basename(f))
|
|
|
|
if not os.path.isfile(target):
|
|
|
|
shutil.copyfile(f, target)
|
2022-12-25 23:52:39 +01:00
|
|
|
if self.verbose:
|
|
|
|
print(f"Extracted {os.path.basename(f)}")
|
2022-10-04 00:01:06 +02:00
|
|
|
self.write_log(fname, os.path.basename(f), os.path.getsize(f))
|
|
|
|
else:
|
2022-12-25 23:52:39 +01:00
|
|
|
if self.verbose:
|
|
|
|
print('%s skipped. A file with that name already exists' % os.path.basename(f))
|
2022-10-04 00:01:06 +02:00
|
|
|
else:
|
|
|
|
print(f"{fname} is not a supported file type")
|
|
|
|
|
|
|
|
def filtered_filelist(self, rootDir):
|
2022-05-27 16:41:34 +02:00
|
|
|
return [os.path.join(r, fn)
|
|
|
|
for r, ds, fs in os.walk(rootDir)
|
|
|
|
for fn in fs if fn.endswith(self.extensions)]
|
2022-10-04 00:01:06 +02:00
|
|
|
|
|
|
|
def write_log(self, src, fname, size):
|
2023-08-01 12:57:54 +02:00
|
|
|
with open(self.log_path, "at") as o:
|
|
|
|
o.write(src.ljust(100) + fname.ljust(50) + str(size).ljust(16) + '\n')
|