Skip to content

Commit 08b1335

Browse files
Override Liveness timeout on shutdown (#771)
Co-authored-by: Martin Hjelmare <marhje52@gmail.com>
1 parent b21a6dd commit 08b1335

File tree

2 files changed

+42
-4
lines changed

2 files changed

+42
-4
lines changed

matter_server/server/device_controller.py

+35-4
Original file line numberDiff line numberDiff line change
@@ -73,6 +73,7 @@
7373
NODE_SUBSCRIPTION_CEILING_BATTERY_POWERED = 600
7474
NODE_RESUBSCRIBE_ATTEMPTS_UNAVAILABLE = 3
7575
NODE_RESUBSCRIBE_TIMEOUT_OFFLINE = 30 * 60 * 1000
76+
NODE_RESUBSCRIBE_FORCE_TIMEOUT = 5
7677
NODE_PING_TIMEOUT = 10
7778
NODE_PING_TIMEOUT_BATTERY_POWERED = 60
7879
NODE_MDNS_BACKOFF = 610 # must be higher than (highest) sub ceiling
@@ -978,7 +979,10 @@ def event_callback(
978979
self.event_history.append(node_event)
979980

980981
if isinstance(data.Data, Clusters.BasicInformation.Events.ShutDown):
981-
self._node_unavailable(node_id)
982+
# Force resubscription after a shutdown event. Otherwise we'd have to
983+
# wait for up to NODE_SUBSCRIPTION_CEILING_BATTERY_POWERED minutes for
984+
# the SDK to notice the device is gone.
985+
self._node_unavailable(node_id, True)
982986

983987
self.server.signal_event(EventType.NODE_EVENT, node_event)
984988

@@ -1359,15 +1363,41 @@ def _write_node_state(self, node_id: int, force: bool = False) -> None:
13591363
force=force,
13601364
)
13611365

1362-
def _node_unavailable(self, node_id: int) -> None:
1366+
def _node_unavailable(
1367+
self, node_id: int, force_resubscription: bool = False
1368+
) -> None:
13631369
"""Mark node as unavailable."""
13641370
# mark node as unavailable (if it wasn't already)
13651371
node = self._nodes[node_id]
13661372
if not node.available:
13671373
return
13681374
node.available = False
13691375
self.server.signal_event(EventType.NODE_UPDATED, node)
1370-
LOGGER.info("Marked node %s as unavailable", node_id)
1376+
node_logger = LOGGER.getChild(f"node_{node_id}")
1377+
node_logger.info("Marked node as unavailable")
1378+
if force_resubscription:
1379+
# Make sure the subscriptions are expiring very soon to trigger subscription
1380+
# resumption logic quickly. This is especially important for battery operated
1381+
# devices so subscription resumption logic kicks in quickly.
1382+
node_logger.info(
1383+
"Forcing subscription timeout in %ds", NODE_RESUBSCRIBE_FORCE_TIMEOUT
1384+
)
1385+
asyncio.create_task(
1386+
self._chip_device_controller.subscription_override_liveness_timeout(
1387+
node_id, NODE_RESUBSCRIBE_FORCE_TIMEOUT * 1000
1388+
)
1389+
)
1390+
# Clear the timeout soon after the scheduled timeout above. This causes the
1391+
# SDK to use the default liveness timeout again, which is what we want for
1392+
# the once resumed subscription.
1393+
self._loop.call_later(
1394+
NODE_RESUBSCRIBE_FORCE_TIMEOUT + 1,
1395+
lambda: asyncio.create_task(
1396+
self._chip_device_controller.subscription_override_liveness_timeout(
1397+
node_id, 0
1398+
)
1399+
),
1400+
)
13711401

13721402
async def _node_offline(self, node_id: int) -> None:
13731403
"""Mark node as offline."""
@@ -1379,7 +1409,8 @@ async def _node_offline(self, node_id: int) -> None:
13791409
return # nothing to do to
13801410
node.available = False
13811411
self.server.signal_event(EventType.NODE_UPDATED, node)
1382-
LOGGER.info("Marked node %s as offline", node_id)
1412+
node_logger = LOGGER.getChild(f"node_{node_id}")
1413+
node_logger.info("Marked node as offline")
13831414

13841415
async def _fallback_node_scanner(self) -> None:
13851416
"""Scan for operational nodes in the background that are missed by mdns."""

matter_server/server/sdk.py

+7
Original file line numberDiff line numberDiff line change
@@ -395,6 +395,13 @@ async def shutdown_subscription(self, node_id: int) -> None:
395395
if sub := self._subscriptions.pop(node_id, None):
396396
await self._call_sdk(sub.Shutdown)
397397

398+
async def subscription_override_liveness_timeout(
399+
self, node_id: int, liveness_timeout_ms: int
400+
) -> None:
401+
"""Override the liveness timeout for the subscription of the node."""
402+
if sub := self._subscriptions.get(node_id):
403+
await self._call_sdk(sub.OverrideLivenessTimeoutMs, liveness_timeout_ms)
404+
398405
def node_has_subscription(self, node_id: int) -> bool:
399406
"""Check if a node has an active subscription."""
400407
return node_id in self._subscriptions

0 commit comments

Comments
 (0)