|
72 | 72 | NODE_PING_TIMEOUT = 10
|
73 | 73 | NODE_PING_TIMEOUT_BATTERY_POWERED = 60
|
74 | 74 | NODE_MDNS_BACKOFF = 60
|
| 75 | +FALLBACK_NODE_SCANNER_INTERVAL = 1800 |
75 | 76 |
|
76 | 77 | MDNS_TYPE_OPERATIONAL_NODE = "_matter._tcp.local."
|
77 | 78 | MDNS_TYPE_COMMISSIONABLE_NODE = "_matterc._udp.local."
|
@@ -113,6 +114,7 @@ def __init__(
|
113 | 114 | self._aiobrowser: AsyncServiceBrowser | None = None
|
114 | 115 | self._aiozc: AsyncZeroconf | None = None
|
115 | 116 | self._sdk_executor = ThreadPoolExecutor(max_workers=1)
|
| 117 | + self._fallback_node_scanner_timer: asyncio.TimerHandle | None = None |
116 | 118 |
|
117 | 119 | async def initialize(self) -> None:
|
118 | 120 | """Async initialize of controller."""
|
@@ -174,20 +176,24 @@ async def start(self) -> None:
|
174 | 176 | services,
|
175 | 177 | handlers=[self._on_mdns_service_state_change],
|
176 | 178 | )
|
| 179 | + # set-up fallback node scanner |
| 180 | + asyncio.create_task(self._fallback_node_scanner()) |
177 | 181 |
|
178 | 182 | async def stop(self) -> None:
|
179 | 183 | """Handle logic on server stop."""
|
180 | 184 | if self.chip_controller is None:
|
181 | 185 | raise RuntimeError("Device Controller not initialized.")
|
182 |
| - # unsubscribe all node subscriptions |
183 |
| - for sub in self._subscriptions.values(): |
184 |
| - await self._call_sdk(sub.Shutdown) |
185 |
| - self._subscriptions = {} |
186 |
| - # shutdown (and cleanup) mdns browser |
| 186 | + # shutdown (and cleanup) mdns browser and fallback node scanner |
187 | 187 | if self._aiobrowser:
|
188 | 188 | await self._aiobrowser.async_cancel()
|
| 189 | + if self._fallback_node_scanner_timer: |
| 190 | + self._fallback_node_scanner_timer.cancel() |
189 | 191 | if self._aiozc:
|
190 | 192 | await self._aiozc.async_close()
|
| 193 | + # unsubscribe all node subscriptions |
| 194 | + for sub in self._subscriptions.values(): |
| 195 | + await self._call_sdk(sub.Shutdown) |
| 196 | + self._subscriptions = {} |
191 | 197 | # shutdown the sdk device controller
|
192 | 198 | await self._call_sdk(self.chip_controller.Shutdown)
|
193 | 199 | LOGGER.debug("Stopped.")
|
@@ -1250,3 +1256,30 @@ async def _node_offline(self, node_id: int) -> None:
|
1250 | 1256 | node.available = False
|
1251 | 1257 | self.server.signal_event(EventType.NODE_UPDATED, node)
|
1252 | 1258 | LOGGER.info("Marked node %s as offline", node_id)
|
| 1259 | + |
| 1260 | + async def _fallback_node_scanner(self) -> None: |
| 1261 | + """Scan for operational nodes in the background that are missed by mdns.""" |
| 1262 | + # This code could/should be removed in the future and is added to have a fallback |
| 1263 | + # to discover operational nodes that got somehow missed by zeroconf. |
| 1264 | + # the issue in zeroconf is being investigated and in the meanwhile we have this fallback. |
| 1265 | + for node_id, node in self._nodes.items(): |
| 1266 | + if node.available: |
| 1267 | + continue |
| 1268 | + now = time.time() |
| 1269 | + last_seen = self._node_last_seen.get(node_id, 0) |
| 1270 | + if now - last_seen < FALLBACK_NODE_SCANNER_INTERVAL: |
| 1271 | + continue |
| 1272 | + if await self.ping_node(node_id, attempts=3): |
| 1273 | + LOGGER.info("Node %s discovered using fallback ping", node_id) |
| 1274 | + await self._setup_node(node_id) |
| 1275 | + |
| 1276 | + def reschedule_self() -> None: |
| 1277 | + self._fallback_node_scanner_timer = None |
| 1278 | + asyncio.create_task(self._fallback_node_scanner()) |
| 1279 | + |
| 1280 | + # reschedule task to run at next interval |
| 1281 | + if TYPE_CHECKING: |
| 1282 | + assert self.server.loop |
| 1283 | + self._fallback_node_scanner_timer = self.server.loop.call_later( |
| 1284 | + FALLBACK_NODE_SCANNER_INTERVAL, reschedule_self |
| 1285 | + ) |
0 commit comments