2023-04-05 23:52:14 +00:00
|
|
|
from __future__ import annotations
|
|
|
|
|
2023-04-06 17:19:13 +00:00
|
|
|
import courier, albert
|
2023-04-06 14:38:29 +00:00
|
|
|
from hashlib import sha1
|
2023-04-06 01:01:07 +00:00
|
|
|
|
|
|
|
|
2023-04-06 17:19:13 +00:00
|
|
|
def _serialize_field(id: int, value: bytes) -> bytes:
|
|
|
|
return id.to_bytes() + len(value).to_bytes(2, "big") + value
|
|
|
|
|
|
|
|
|
|
|
|
def _serialize_payload(id: int, fields: list[(int, bytes)]) -> bytes:
|
|
|
|
payload = b""
|
|
|
|
|
|
|
|
for fid, value in fields:
|
2023-04-07 14:33:03 +00:00
|
|
|
if fid is not None:
|
|
|
|
payload += _serialize_field(fid, value)
|
2023-04-06 17:19:13 +00:00
|
|
|
|
|
|
|
return id.to_bytes() + len(payload).to_bytes(4, "big") + payload
|
|
|
|
|
|
|
|
|
|
|
|
def _deserialize_field(stream: bytes) -> tuple[int, bytes]:
|
|
|
|
id = int.from_bytes(stream[:1], "big")
|
|
|
|
length = int.from_bytes(stream[1:3], "big")
|
|
|
|
value = stream[3 : 3 + length]
|
|
|
|
return id, value
|
|
|
|
|
|
|
|
# Note: Takes a stream, not a buffer, as we do not know the length of the payload
|
2023-04-07 05:48:07 +00:00
|
|
|
# WILL BLOCK IF THE STREAM IS EMPTY
|
2023-04-06 17:19:13 +00:00
|
|
|
def _deserialize_payload(stream) -> tuple[int, list[tuple[int, bytes]]] | None:
|
|
|
|
id = int.from_bytes(stream.read(1), "big")
|
|
|
|
|
|
|
|
if id == 0x0:
|
|
|
|
return None
|
|
|
|
|
|
|
|
length = int.from_bytes(stream.read(4), "big")
|
|
|
|
|
|
|
|
buffer = stream.read(length)
|
|
|
|
|
|
|
|
fields = []
|
|
|
|
|
|
|
|
while len(buffer) > 0:
|
|
|
|
fid, value = _deserialize_field(buffer)
|
|
|
|
fields.append((fid, value))
|
|
|
|
buffer = buffer[3 + len(value) :]
|
|
|
|
|
|
|
|
return id, fields
|
|
|
|
|
2023-04-07 13:28:32 +00:00
|
|
|
def _deserialize_payload_from_buffer(buffer: bytes) -> tuple[int, list[tuple[int, bytes]]] | None:
|
|
|
|
id = int.from_bytes(buffer[:1], "big")
|
|
|
|
|
|
|
|
if id == 0x0:
|
|
|
|
return None
|
|
|
|
|
|
|
|
length = int.from_bytes(buffer[1:5], "big")
|
|
|
|
|
|
|
|
buffer = buffer[5:]
|
|
|
|
|
|
|
|
if len(buffer) < length:
|
|
|
|
raise Exception("Buffer is too short")
|
|
|
|
|
|
|
|
fields = []
|
|
|
|
|
|
|
|
while len(buffer) > 0:
|
|
|
|
fid, value = _deserialize_field(buffer)
|
|
|
|
fields.append((fid, value))
|
|
|
|
buffer = buffer[3 + len(value) :]
|
|
|
|
|
|
|
|
return id, fields
|
|
|
|
|
2023-04-06 17:19:13 +00:00
|
|
|
|
|
|
|
# Returns the value of the first field with the given id
|
|
|
|
def _get_field(fields: list[tuple[int, bytes]], id: int) -> bytes:
|
|
|
|
for field_id, value in fields:
|
|
|
|
if field_id == id:
|
|
|
|
return value
|
|
|
|
return None
|
|
|
|
|
|
|
|
|
|
|
|
class APNSConnection:
|
|
|
|
def __init__(self, private_key=None, cert=None):
|
|
|
|
# Generate the private key and certificate if they're not provided
|
|
|
|
if private_key is None or cert is None:
|
|
|
|
self.private_key, self.cert = albert.generate_push_cert()
|
2023-04-06 01:01:07 +00:00
|
|
|
else:
|
2023-04-06 17:19:13 +00:00
|
|
|
self.private_key, self.cert = private_key, cert
|
2023-04-06 01:01:07 +00:00
|
|
|
|
2023-04-06 17:19:13 +00:00
|
|
|
self.sock = courier.connect(self.private_key, self.cert)
|
2023-04-06 01:01:07 +00:00
|
|
|
|
2023-04-06 17:19:13 +00:00
|
|
|
def connect(self, token: bytes = None):
|
|
|
|
if token is None:
|
|
|
|
payload = _serialize_payload(7, [(2, 0x01.to_bytes())])
|
|
|
|
else:
|
|
|
|
payload = _serialize_payload(7, [(1, token), (2, 0x01.to_bytes())])
|
|
|
|
|
|
|
|
self.sock.write(payload)
|
|
|
|
|
|
|
|
payload = _deserialize_payload(self.sock)
|
|
|
|
|
|
|
|
if payload == None or payload[0] != 8 or _get_field(payload[1], 1) != 0x00.to_bytes():
|
2023-04-06 01:01:07 +00:00
|
|
|
raise Exception("Failed to connect")
|
|
|
|
|
2023-04-06 17:19:13 +00:00
|
|
|
self.token = _get_field(payload[1], 3)
|
2023-04-06 01:01:07 +00:00
|
|
|
|
2023-04-06 14:38:29 +00:00
|
|
|
def filter(self, topics: list[str]):
|
2023-04-06 17:19:13 +00:00
|
|
|
fields = [(1, self.token)]
|
|
|
|
|
|
|
|
for topic in topics:
|
|
|
|
fields.append((2, sha1(topic.encode()).digest()))
|
|
|
|
|
|
|
|
payload = _serialize_payload(9, fields)
|
|
|
|
|
|
|
|
self.sock.write(payload)
|
2023-04-07 05:48:07 +00:00
|
|
|
|
|
|
|
def send_message(self, token: bytes, topic: str, payload: str):
|
|
|
|
# Current time in UNIX nanoseconds
|
|
|
|
import time
|
|
|
|
# Expire in 5 minutes
|
|
|
|
expiry = int(time.time()) + 500
|
|
|
|
|
|
|
|
payload = _serialize_payload(0x0a,
|
|
|
|
[(1, sha1(topic.encode()).digest()),
|
|
|
|
(2, token),
|
|
|
|
(3, payload.encode("utf-8")),
|
|
|
|
(4, (3864024149).to_bytes(4, "big")),
|
|
|
|
(5, expiry.to_bytes(4, "big")),
|
|
|
|
(6, time.time_ns().to_bytes(8, "big")),
|
|
|
|
(7, 0x00.to_bytes()),
|
|
|
|
(0xf, self.token)])
|
|
|
|
|
|
|
|
print(payload)
|
|
|
|
|
|
|
|
self.sock.write(payload)
|
|
|
|
|
|
|
|
payload = _deserialize_payload(self.sock)
|
|
|
|
|
|
|
|
print(payload)
|
|
|
|
|
|
|
|
|
|
|
|
# TODO: Find a way to make this non-blocking
|
|
|
|
def expect_message(self) -> tuple[int, list[tuple[int, bytes]]] | None:
|
|
|
|
return _deserialize_payload(self.sock)
|