73
73
NODE_SUBSCRIPTION_CEILING_BATTERY_POWERED = 600
74
74
NODE_RESUBSCRIBE_ATTEMPTS_UNAVAILABLE = 3
75
75
NODE_RESUBSCRIBE_TIMEOUT_OFFLINE = 30 * 60 * 1000
76
+ NODE_RESUBSCRIBE_FORCE_TIMEOUT = 5
76
77
NODE_PING_TIMEOUT = 10
77
78
NODE_PING_TIMEOUT_BATTERY_POWERED = 60
78
79
NODE_MDNS_BACKOFF = 610 # must be higher than (highest) sub ceiling
@@ -978,7 +979,10 @@ def event_callback(
978
979
self .event_history .append (node_event )
979
980
980
981
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 )
982
986
983
987
self .server .signal_event (EventType .NODE_EVENT , node_event )
984
988
@@ -1359,15 +1363,41 @@ def _write_node_state(self, node_id: int, force: bool = False) -> None:
1359
1363
force = force ,
1360
1364
)
1361
1365
1362
- def _node_unavailable (self , node_id : int ) -> None :
1366
+ def _node_unavailable (
1367
+ self , node_id : int , force_resubscription : bool = False
1368
+ ) -> None :
1363
1369
"""Mark node as unavailable."""
1364
1370
# mark node as unavailable (if it wasn't already)
1365
1371
node = self ._nodes [node_id ]
1366
1372
if not node .available :
1367
1373
return
1368
1374
node .available = False
1369
1375
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
+ )
1371
1401
1372
1402
async def _node_offline (self , node_id : int ) -> None :
1373
1403
"""Mark node as offline."""
@@ -1379,7 +1409,8 @@ async def _node_offline(self, node_id: int) -> None:
1379
1409
return # nothing to do to
1380
1410
node .available = False
1381
1411
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" )
1383
1414
1384
1415
async def _fallback_node_scanner (self ) -> None :
1385
1416
"""Scan for operational nodes in the background that are missed by mdns."""
0 commit comments