refactor(topology): introduce TopologyRepository protocol with DTO return types
Replace repo: BaseRepository with a structural TopologyRepository protocol in persistence.py and allocator.py. All read methods now return typed DTOs (TopologySummary, LANRow, DeckyRow, EdgeRow) instead of raw dicts, eliminating silent field-shape regressions across the topology subsystem. TopologySummary gains email_personas and language_default so api_personas.py can continue reading those fields via attribute access. hydrate() converts DTOs to dicts before passing to _backfill_decky_configs, keeping the mutable working-state function dict-based at its boundary. All production callers (router handlers, mutator, CLI, heartbeat) migrated from dict/get access to attribute access. 134 tests pass.
This commit is contained in:
@@ -284,13 +284,13 @@ async def reconcile_agent_resyncs(repo: BaseRepository) -> int:
|
||||
return 0
|
||||
drained = 0
|
||||
for topo in pending:
|
||||
tid = topo["id"]
|
||||
tid = topo.id
|
||||
try:
|
||||
await _deployer.resync_agent_topology(repo, tid)
|
||||
await repo.set_topology_resync(tid, False)
|
||||
drained += 1
|
||||
log.info("topology %s resynced to agent %s",
|
||||
tid, topo.get("target_host_uuid"))
|
||||
tid, topo.target_host_uuid)
|
||||
except Exception as exc: # noqa: BLE001
|
||||
log.warning(
|
||||
"topology %s resync failed (will retry): %s", tid, exc,
|
||||
|
||||
@@ -121,10 +121,10 @@ async def _materialise_lan_change(
|
||||
topology = await repo.get_topology(topology_id)
|
||||
if topology is None:
|
||||
return
|
||||
status = topology.get("status")
|
||||
status = topology.status
|
||||
if status not in ("active", "degraded"):
|
||||
return
|
||||
if topology.get("target_host_uuid"):
|
||||
if topology.target_host_uuid:
|
||||
_log.info(
|
||||
"live LAN op skipped (agent-pinned topology=%s); next agent push will reconcile",
|
||||
topology_id,
|
||||
@@ -291,9 +291,9 @@ async def _live_topology_or_none(
|
||||
topology = await repo.get_topology(topology_id)
|
||||
if topology is None:
|
||||
return None
|
||||
if topology.get("status") not in ("active", "degraded"):
|
||||
if topology.status not in ("active", "degraded"):
|
||||
return None
|
||||
if topology.get("target_host_uuid"):
|
||||
if topology.target_host_uuid:
|
||||
_log.info(
|
||||
"live decky op skipped (agent-pinned topology=%s); "
|
||||
"next agent push will reconcile",
|
||||
@@ -1019,7 +1019,7 @@ async def apply_update_lan(
|
||||
return
|
||||
|
||||
topology = await repo.get_topology(topology_id)
|
||||
is_live = bool(topology) and topology.get("status") in ("active", "degraded")
|
||||
is_live = bool(topology) and topology.status in ("active", "degraded")
|
||||
if is_live:
|
||||
hostile = {"subnet", "is_dmz"} & fields.keys()
|
||||
if hostile:
|
||||
|
||||
Reference in New Issue
Block a user