|
| 1 | +import base64 |
| 2 | + |
| 3 | +from datetime import timezone |
| 4 | +import os |
| 5 | +from uid2_client import encryption_block_size |
| 6 | +from uid2_client.advertising_token_version import AdvertisingTokenVersion |
| 7 | +from uid2_client.encryption import _encrypt_data_v1, _encrypt_gcm, _PayloadType |
| 8 | +from uid2_client.identity_scope import IdentityScope |
| 9 | +from uid2_client.identity_type import IdentityType |
| 10 | +from uid2_client.keys import * |
| 11 | +from uid2_client.uid2_base64_url_coder import Uid2Base64UrlCoder |
| 12 | + |
| 13 | + |
| 14 | +class Params: |
| 15 | + def __init__(self, expiry=dt.datetime.now(tz=timezone.utc) + dt.timedelta(hours=1), |
| 16 | + identity_scope=IdentityScope.UID2.value, identity_type=IdentityType.Email.value): |
| 17 | + self.identity_scope = identity_scope |
| 18 | + self.identity_type = identity_type |
| 19 | + self.token_expiry = expiry |
| 20 | + if not isinstance(expiry, dt.datetime): |
| 21 | + self.token_expiry = dt.datetime.now(tz=timezone.utc) + expiry |
| 22 | + |
| 23 | + |
| 24 | +def default_params(): |
| 25 | + return Params() |
| 26 | + |
| 27 | + |
| 28 | +class UID2TokenGenerator: |
| 29 | + @staticmethod |
| 30 | + def generate_uid2_token_v2(id_str, master_key, site_id, site_key, params = default_params(), version=2): |
| 31 | + id = bytes(id_str, 'utf-8') |
| 32 | + identity = int.to_bytes(site_id, 4, 'big') |
| 33 | + identity += int.to_bytes(len(id), 4, 'big') |
| 34 | + identity += id |
| 35 | + # old privacy_bits |
| 36 | + identity += int.to_bytes(0, 4, 'big') |
| 37 | + identity += int.to_bytes(int((dt.datetime.now(tz=timezone.utc) - dt.timedelta(hours=1)).timestamp()) * 1000, 8, |
| 38 | + 'big') |
| 39 | + identity_iv = bytes([10, 11, 12, 13, 14, 15, 16, 1, 2, 3, 4, 5, 6, 7, 8, 9]) |
| 40 | + expiry = params.token_expiry |
| 41 | + master_payload = int.to_bytes(int(expiry.timestamp()) * 1000, 8, 'big') |
| 42 | + master_payload += _encrypt_data_v1(identity, key=site_key, iv=identity_iv) |
| 43 | + master_iv = bytes([21, 22, 23, 24, 25, 26, 27, 28, 29, 30, 31, 32, 33, 34, 35, 36]) |
| 44 | + |
| 45 | + token = int.to_bytes(version, 1, 'big') |
| 46 | + token += _encrypt_data_v1(master_payload, key=master_key, iv=master_iv) |
| 47 | + |
| 48 | + return base64.b64encode(token).decode('ascii') |
| 49 | + |
| 50 | + @staticmethod |
| 51 | + def generate_uid2_token_v3(id_str, master_key, site_id, site_key, params = default_params()): |
| 52 | + return UID2TokenGenerator.generate_uid2_token_with_debug_info(id_str, master_key, site_id, site_key, params, |
| 53 | + AdvertisingTokenVersion.ADVERTISING_TOKEN_V3.value) |
| 54 | + |
| 55 | + @staticmethod |
| 56 | + def generate_uid2_token_v4(id_str, master_key, site_id, site_key, params = default_params()): |
| 57 | + return UID2TokenGenerator.generate_uid2_token_with_debug_info(id_str, master_key, site_id, site_key, params, |
| 58 | + AdvertisingTokenVersion.ADVERTISING_TOKEN_V4.value) |
| 59 | + |
| 60 | + @staticmethod |
| 61 | + def generate_uid2_token_with_debug_info(id_str, master_key, site_id, site_key, params, version): |
| 62 | + id = base64.b64decode(id_str) |
| 63 | + |
| 64 | + site_payload = int.to_bytes(site_id, 4, 'big') |
| 65 | + site_payload += int.to_bytes(0, 8, 'big') # publisher id |
| 66 | + site_payload += int.to_bytes(0, 4, 'big') # client key id |
| 67 | + |
| 68 | + site_payload += int.to_bytes(0, 4, 'big') # privacy bits |
| 69 | + site_payload += int.to_bytes(int((dt.datetime.now(tz=timezone.utc) - dt.timedelta(hours=1)).timestamp()) * 1000, |
| 70 | + 8, 'big') # established |
| 71 | + site_payload += int.to_bytes(int((dt.datetime.now(tz=timezone.utc) - dt.timedelta(hours=1)).timestamp()) * 1000, |
| 72 | + 8, 'big') # refreshed |
| 73 | + site_payload += id |
| 74 | + |
| 75 | + expiry = params.token_expiry |
| 76 | + master_payload = int.to_bytes(int(expiry.timestamp()) * 1000, 8, 'big') |
| 77 | + master_payload += int.to_bytes(int((dt.datetime.now(tz=timezone.utc)).timestamp()) * 1000, 8, 'big') # created |
| 78 | + |
| 79 | + master_payload += int.to_bytes(0, 4, 'big') # operator site id |
| 80 | + master_payload += int.to_bytes(0, 1, 'big') # operator type |
| 81 | + master_payload += int.to_bytes(0, 4, 'big') # operator version |
| 82 | + master_payload += int.to_bytes(0, 4, 'big') # operator key id |
| 83 | + |
| 84 | + master_payload += int.to_bytes(site_key.key_id, 4, 'big') |
| 85 | + master_payload += _encrypt_gcm(site_payload, None, site_key.secret) |
| 86 | + |
| 87 | + token = int.to_bytes((params.identity_scope << 4 | params.identity_type << 2), 1, 'big') |
| 88 | + token += int.to_bytes(version, 1, 'big') |
| 89 | + token += int.to_bytes(master_key.key_id, 4, 'big') |
| 90 | + token += _encrypt_gcm(master_payload, None, master_key.secret) |
| 91 | + |
| 92 | + if version == AdvertisingTokenVersion.ADVERTISING_TOKEN_V4.value: |
| 93 | + return Uid2Base64UrlCoder.encode(token) |
| 94 | + else: |
| 95 | + return base64.b64encode(token).decode('ascii') |
| 96 | + |
| 97 | + @staticmethod |
| 98 | + def encrypt_data_v2(data, key, site_id, now): |
| 99 | + iv = os.urandom(encryption_block_size) |
| 100 | + result = int.to_bytes(_PayloadType.ENCRYPTED_DATA.value, 1, 'big') |
| 101 | + result += int.to_bytes(1, 1, 'big') # version |
| 102 | + result += int.to_bytes(int(now.timestamp() * 1000), 8, 'big') |
| 103 | + result += int.to_bytes(site_id, 4, 'big') |
| 104 | + result += _encrypt_data_v1(data, key, iv) |
| 105 | + return base64.b64encode(result).decode('ascii') |
0 commit comments