""" Example showing how to fetch locations of an AirTag, or any other FindMy accessory. """ from __future__ import annotations import logging import sys from pathlib import Path from _login import get_account_sync from findmy import FindMyAccessory # Path where login session will be stored. # This is necessary to avoid generating a new session every time we log in. STORE_PATH = "account.json" # URL to LOCAL anisette server. Set to None to use built-in Anisette generator instead (recommended) # IF YOU USE A PUBLIC SERVER, DO NOT COMPLAIN THAT YOU KEEP RUNNING INTO AUTHENTICATION ERRORS! # If you change this value, make sure to remove the account store file. ANISETTE_SERVER = None # Path where Anisette libraries will be stored. # This is only relevant when using the built-in Anisette server. # It can be omitted (set to None) to avoid saving to disk, # but specifying a path is highly recommended to avoid downloading the bundle on every run. ANISETTE_LIBS_PATH = "ani_libs.bin" logging.basicConfig(level=logging.INFO) def main(plist_path: str) -> int: # Step 0: create an accessory key generator with Path(plist_path).open("rb") as f: airtag = FindMyAccessory.from_plist(f) # Step 1: log into an Apple account print("Logging into account") acc = get_account_sync(STORE_PATH, ANISETTE_SERVER, ANISETTE_LIBS_PATH) # step 2: fetch reports! print("Fetching reports") reports = acc.fetch_last_reports(airtag) # step 3: print 'em print() print("Location reports:") for report in sorted(reports): print(f" - {report}") # step 4: save current account state to disk acc.to_json(STORE_PATH) return 0 if __name__ == "__main__": if len(sys.argv) < 2: print(f"Usage: {sys.argv[0]} ", file=sys.stderr) print(file=sys.stderr) print("The plist file should be dumped from MacOS's FindMy app.", file=sys.stderr) sys.exit(1) sys.exit(main(sys.argv[1]))