|
1
|
|
|
# -*- coding: utf-8 -*- |
|
|
|
|
|
|
2
|
|
|
|
|
3
|
|
|
import binascii |
|
4
|
|
|
import hashlib |
|
5
|
|
|
import random |
|
6
|
|
|
import time |
|
7
|
|
|
import zlib |
|
8
|
|
|
from collections import OrderedDict |
|
9
|
|
|
from math import floor |
|
10
|
|
|
from .. import devices |
|
11
|
|
|
from .tclrequest import TclRequest |
|
12
|
|
|
from .tclresult import DownloadResult |
|
13
|
|
|
|
|
14
|
|
|
|
|
15
|
|
|
VDKEY_B64Z = b"eJwdjwEOwDAIAr8kKFr//7HhmqXp8AIIDrYAgg8byiUXrwRJRXja+d6iNxu0AhUooDCN9rd6rDLxmGIakUVWo3IGCTRWqCAt6X4jGEIUAxgN0eYWnp+LkpHQAg/PsO90ELsy0Npm/n2HbtPndFgGEV31R9OmT4O4nrddjc3Qt6nWscx7e+WRHq5UnOudtjw5skuV09pFhvmqnOEIs4ljPeel1wfLYUF4\n" |
|
16
|
|
|
|
|
17
|
|
|
|
|
18
|
|
|
def get_salt(): |
|
19
|
|
|
"""Generate cryptographic salt.""" |
|
20
|
|
|
millis = floor(time.time() * 1000) |
|
21
|
|
|
tail = "{:06d}".format(random.randint(0, 999999)) |
|
22
|
|
|
return "{}{}".format(str(millis), tail) |
|
23
|
|
|
|
|
24
|
|
|
def get_vk2(params_dict, cltp): |
|
25
|
|
|
"""Generate salted hash of API parameters.""" |
|
26
|
|
|
params_dict["cltp"] = cltp |
|
27
|
|
|
query = "" |
|
28
|
|
|
for key, val in params_dict.items(): |
|
29
|
|
|
if query: |
|
30
|
|
|
query += "&" |
|
31
|
|
|
query += key + "=" + str(val) |
|
32
|
|
|
vdk = zlib.decompress(binascii.a2b_base64(VDKEY_B64Z)) |
|
33
|
|
|
query += vdk.decode("utf-8") |
|
34
|
|
|
engine = hashlib.sha1() |
|
35
|
|
|
engine.update(bytes(query, "utf-8")) |
|
36
|
|
|
hexhash = engine.hexdigest() |
|
37
|
|
|
return hexhash |
|
38
|
|
|
|
|
39
|
|
|
class DownloadRequest(TclRequest): |
|
|
|
|
|
|
40
|
|
|
def __init__(self, device: devices.Device, tvver: str, fw_id: str): |
|
|
|
|
|
|
41
|
|
|
super().__init__() |
|
42
|
|
|
self.uri = "/download_request.php" |
|
43
|
|
|
self.method = "POST" |
|
44
|
|
|
self.device = device |
|
45
|
|
|
self.tvver = tvver |
|
46
|
|
|
self.fw_id = fw_id |
|
47
|
|
|
|
|
48
|
|
|
def get_headers(self): |
|
49
|
|
|
return {"User-Agent": self.device.ua} |
|
50
|
|
|
|
|
51
|
|
|
def get_params(self): |
|
52
|
|
|
params = OrderedDict() |
|
53
|
|
|
params["id"] = self.device.imei |
|
54
|
|
|
params["salt"] = get_salt() |
|
55
|
|
|
params["curef"] = self.device.curef |
|
56
|
|
|
params["fv"] = self.device.fwver |
|
57
|
|
|
params["tv"] = self.tvver |
|
58
|
|
|
params["type"] = self.device.type |
|
59
|
|
|
params["fw_id"] = self.fw_id |
|
60
|
|
|
params["mode"] = self.device.mode |
|
61
|
|
|
params["vk"] = get_vk2(params, self.device.cltp) |
|
62
|
|
|
params["cltp"] = self.device.cltp |
|
63
|
|
|
params["cktp"] = self.device.cktp |
|
64
|
|
|
params["rtd"] = self.device.rtd |
|
65
|
|
|
if self.device.mode == self.device.MODE_STATES["FULL"]: |
|
66
|
|
|
params["foot"] = 1 |
|
67
|
|
|
params["chnl"] = self.device.chnl |
|
68
|
|
|
return params |
|
69
|
|
|
|
|
70
|
|
|
def is_done(self, http_status: int, contents: str) -> bool: |
|
71
|
|
|
if http_status == 200: |
|
72
|
|
|
self.response = contents |
|
73
|
|
|
self.result = DownloadResult(contents) |
|
74
|
|
|
self.success = True |
|
75
|
|
|
return True |
|
76
|
|
|
elif http_status not in [500, 502, 503]: |
|
77
|
|
|
# Errors OTHER than 500, 502 or 503 are probably |
|
78
|
|
|
# errors where we don't need to retry |
|
79
|
|
|
self.error = "HTTP {}".format(http_status) |
|
80
|
|
|
self.success = False |
|
81
|
|
|
return True |
|
82
|
|
|
return False |
|
83
|
|
|
|
The coding style of this project requires that you add a docstring to this code element. Below, you find an example for methods:
If you would like to know more about docstrings, we recommend to read PEP-257: Docstring Conventions.