Skip to content

Commit b63395a

Browse files
committed
Add fallback scanner for Matter nodes
1 parent b05c9b4 commit b63395a

File tree

1 file changed

+38
-5
lines changed

1 file changed

+38
-5
lines changed

matter_server/server/device_controller.py

+38-5
Original file line numberDiff line numberDiff line change
@@ -72,6 +72,7 @@
7272
NODE_PING_TIMEOUT = 10
7373
NODE_PING_TIMEOUT_BATTERY_POWERED = 60
7474
NODE_MDNS_BACKOFF = 60
75+
FALLBACK_NODE_SCANNER_INTERVAL = 1800
7576

7677
MDNS_TYPE_OPERATIONAL_NODE = "_matter._tcp.local."
7778
MDNS_TYPE_COMMISSIONABLE_NODE = "_matterc._udp.local."
@@ -113,6 +114,7 @@ def __init__(
113114
self._aiobrowser: AsyncServiceBrowser | None = None
114115
self._aiozc: AsyncZeroconf | None = None
115116
self._sdk_executor = ThreadPoolExecutor(max_workers=1)
117+
self._fallback_node_scanner_timer: asyncio.TimerHandle | None = None
116118

117119
async def initialize(self) -> None:
118120
"""Async initialize of controller."""
@@ -174,20 +176,24 @@ async def start(self) -> None:
174176
services,
175177
handlers=[self._on_mdns_service_state_change],
176178
)
179+
# set-up fallback node scanner
180+
asyncio.create_task(self._fallback_node_scanner())
177181

178182
async def stop(self) -> None:
179183
"""Handle logic on server stop."""
180184
if self.chip_controller is None:
181185
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
187187
if self._aiobrowser:
188188
await self._aiobrowser.async_cancel()
189+
if self._fallback_node_scanner_timer:
190+
self._fallback_node_scanner_timer.cancel()
189191
if self._aiozc:
190192
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 = {}
191197
# shutdown the sdk device controller
192198
await self._call_sdk(self.chip_controller.Shutdown)
193199
LOGGER.debug("Stopped.")
@@ -1250,3 +1256,30 @@ async def _node_offline(self, node_id: int) -> None:
12501256
node.available = False
12511257
self.server.signal_event(EventType.NODE_UPDATED, node)
12521258
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

Comments
 (0)