import requests import random import apns import trio import gateway_fetch from base64 import b64decode, b64encode import urllib3 urllib3.disable_warnings() PHONE_IP = "192.168.5.120" API_PORT = 8080 GATEWAY = "22223333" def register(push_token: bytes, no_parse = False, gateway = None) -> tuple[str, bytes]: """Forwards a registration request to the phone and returns the phone number, signature for the provided push token""" if gateway is None: print("Requesting device MCC+MNC for gateway detection...") mccmnc = requests.get(f"http://{PHONE_IP}:{API_PORT}/info").text print("MCC+MNC received! " + mccmnc) print("Determining gateway...") gateway = gateway_fetch.getGatewayMCCMNC(mccmnc) if gateway is not None: print("Gateway found! " + str(gateway)) else: print("No gateway was provided, and automatic gateway detection failed. Please run again with the --gateway flag.") # gateway = GATEWAY raise token = push_token.hex().upper() req_id = random.randint(0, 2**32) sms = f"REG-REQ?v=3;t={token};r={req_id};" print("Sending message and waiting for response...") r = requests.get(f"http://{PHONE_IP}:{API_PORT}/register", params={"sms": sms, "gateway": gateway}, timeout=16) print("Received response from device!") if no_parse: print("Now do the next part and rerun with --pdu") exit() return parse_pdu(r.text, req_id) # if r.text.split("?")[0] != "REG-RESP": # raise Exception(f"Failed to register: {r.text}") # #assert r.text.split("?")[0] == "REG-RESP" # resp_params = r.text.split("?")[1] # resp_params = resp_params.split(";") # resp_params = {param.split("=")[0]: param.split("=")[1] for param in resp_params} # assert resp_params["v"] == "3" # assert resp_params["r"] == str(req_id) # signature = bytes.fromhex(resp_params["s"]) # return resp_params["n"], signature def parse_pdu(r: str, req_id: int | None = None): if r.split("?")[0] != "REG-RESP": raise Exception(f"Failed to register: {r}") #assert r.text.split("?")[0] == "REG-RESP" resp_params = r.split("?")[1] resp_params = resp_params.split(";") resp_params = {param.split("=")[0]: param.split("=")[1] for param in resp_params} assert resp_params["v"] == "3" if req_id is not None: assert resp_params["r"] == str(req_id) signature = bytes.fromhex(resp_params["s"]) return resp_params["n"], signature # async def main(): # # Open test.json # try: # with open("test.json", "r") as f: # test_json = f.read() # import json # test_json = json.loads(test_json) # except FileNotFoundError: # test_json = {} # creds = apns.PushCredentials( # test_json.get("push_key", ""), # test_json.get("push_cert", ""), # b64decode(test_json["push_token"]) if "push_token" in test_json else b"", # ) # async with apns.APNSConnection.start(creds) as connection: # connection.credentials # #number, sig = register(connection.credentials.token) # if "register_sig" not in test_json: # try: # number, sig = register(connection.credentials.token) # test_json["register_sig"] = sig.hex() # test_json["number"] = number # except Exception as e: # print(e) # sig = None # number = None # else: # sig = bytes.fromhex(test_json["register_sig"]) # number = test_json["number"] # if sig is not None and number is not None: # from ids import profile # phone_auth_keypair = profile.get_phone_cert(number, connection.credentials.token, [sig]) # test_json["auth_key"] = phone_auth_keypair.key # test_json["auth_cert"] = phone_auth_keypair.cert # out_json = { # "push_key": creds.private_key, # "push_cert": creds.cert, # "push_token": b64encode(creds.token).decode("utf-8"), # } # test_json.update(out_json) # with open("test.json", "w") as f: # import json # f.write(json.dumps(test_json, indent=4)) # if __name__ == "__main__": # trio.run(main)