mirror of
https://github.com/malmeloo/FindMy.py.git
synced 2026-04-17 19:53:53 +02:00
Merge pull request #219 from lanrat/airtag_example
update airtag example
This commit is contained in:
@@ -8,14 +8,19 @@ import argparse
|
||||
import logging
|
||||
import sys
|
||||
from pathlib import Path
|
||||
from typing import TYPE_CHECKING
|
||||
|
||||
from _login import get_account_sync
|
||||
|
||||
from findmy import FindMyAccessory
|
||||
|
||||
# Path where login session will be stored.
|
||||
if TYPE_CHECKING:
|
||||
from findmy.accessory import RollingKeyPairSource
|
||||
from findmy.keys import HasHashedPublicKey
|
||||
|
||||
# Default 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"
|
||||
DEFAULT_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!
|
||||
@@ -30,32 +35,67 @@ ANISETTE_LIBS_PATH = "ani_libs.bin"
|
||||
|
||||
logging.basicConfig(level=logging.INFO)
|
||||
|
||||
BATTERY_LEVEL = {0b00: "Full", 0b01: "Medium", 0b10: "Low", 0b11: "Very Low"}
|
||||
|
||||
def main(airtag_path: Path) -> int:
|
||||
# Step 0: create an accessory key generator
|
||||
airtag = FindMyAccessory.from_json(airtag_path)
|
||||
|
||||
def get_battery_level(status: int) -> str:
|
||||
"""Extract battery level from status byte."""
|
||||
battery_id = (status >> 6) & 0b11
|
||||
return BATTERY_LEVEL.get(battery_id, "Unknown")
|
||||
|
||||
|
||||
def get_airtag_name(airtag: HasHashedPublicKey | RollingKeyPairSource, path: Path) -> str:
|
||||
"""Get a human-readable name for an airtag, with fallbacks."""
|
||||
if isinstance(airtag, FindMyAccessory):
|
||||
if airtag.name:
|
||||
return airtag.name
|
||||
if airtag.identifier:
|
||||
return airtag.identifier
|
||||
return path.stem # filename without extension
|
||||
|
||||
|
||||
def main(airtag_paths: list[Path], store_path: str) -> int:
|
||||
# Step 0: create accessory key generators for all paths
|
||||
airtags = [FindMyAccessory.from_json(path) for path in airtag_paths]
|
||||
airtag_to_path: dict[HasHashedPublicKey | RollingKeyPairSource, Path] = dict(
|
||||
zip(airtags, airtag_paths, strict=False)
|
||||
)
|
||||
|
||||
# Step 1: log into an Apple account
|
||||
acc = get_account_sync(STORE_PATH, ANISETTE_SERVER, ANISETTE_LIBS_PATH)
|
||||
acc = get_account_sync(store_path, ANISETTE_SERVER, ANISETTE_LIBS_PATH)
|
||||
print(f"Logged in as: {acc.account_name} ({acc.first_name} {acc.last_name})")
|
||||
|
||||
# step 2: fetch reports!
|
||||
location = acc.fetch_location(airtag)
|
||||
locations = acc.fetch_location(airtags)
|
||||
|
||||
# step 3: print 'em
|
||||
print("Last known location:")
|
||||
print(f" - {location}")
|
||||
print("Last known locations:")
|
||||
for airtag, path in airtag_to_path.items():
|
||||
location = locations.get(airtag) # type: ignore[union-attr]
|
||||
name = get_airtag_name(airtag, path)
|
||||
if location:
|
||||
battery = get_battery_level(location.status)
|
||||
print(f" - {name}: {location} (Battery: {battery})")
|
||||
else:
|
||||
print(f" - {name}: No location found")
|
||||
|
||||
# step 4: save current account state to disk
|
||||
acc.to_json(STORE_PATH)
|
||||
airtag.to_json(airtag_path)
|
||||
acc.to_json(store_path)
|
||||
for airtag, path in zip(airtags, airtag_paths, strict=False):
|
||||
airtag.to_json(path)
|
||||
|
||||
return 0
|
||||
|
||||
|
||||
if __name__ == "__main__":
|
||||
parser = argparse.ArgumentParser()
|
||||
parser.add_argument("airtag_path", type=Path)
|
||||
parser.add_argument("airtag_paths", type=Path, nargs="+")
|
||||
parser.add_argument(
|
||||
"--store-path",
|
||||
type=str,
|
||||
default=DEFAULT_STORE_PATH,
|
||||
help=f"Path to account session file (default: {DEFAULT_STORE_PATH})",
|
||||
)
|
||||
args = parser.parse_args()
|
||||
|
||||
sys.exit(main(args.airtag_path))
|
||||
sys.exit(main(args.airtag_paths, args.store_path))
|
||||
|
||||
Reference in New Issue
Block a user