Skip to content

Commit 1c0e7e1

Browse files
Delay marking a node as unavailable (#568)
Co-authored-by: Martin Hjelmare <marhje52@gmail.com>
1 parent 2415d73 commit 1c0e7e1

File tree

1 file changed

+22
-9
lines changed

1 file changed

+22
-9
lines changed

matter_server/server/device_controller.py

+22-9
Original file line numberDiff line numberDiff line change
@@ -63,6 +63,8 @@
6363
LOGGER = logging.getLogger(__name__)
6464
MAX_POLL_INTERVAL = 600
6565
MAX_COMMISSION_RETRIES = 3
66+
NODE_RESUBSCRIBE_ATTEMPTS_UNAVAILABLE = 3
67+
NODE_RESUBSCRIBE_TIMEOUT_OFFLINE = 30 * 60 * 1000
6668

6769
MDNS_TYPE_OPERATIONAL_NODE = "_matter._tcp.local."
6870
MDNS_TYPE_COMMISSIONABLE_NODE = "_matterc._udp.local."
@@ -851,7 +853,7 @@ async def _subscribe_node(self, node_id: int) -> None:
851853
node_logger.debug("Re-using existing subscription.")
852854
return
853855
async with node_lock:
854-
node_logger.debug("Unsubscribing from existing subscription.")
856+
node_logger.info("Unsubscribing from existing subscription.")
855857
await self._call_sdk(prev_sub.Shutdown)
856858
del self._subscriptions[node_id]
857859

@@ -976,14 +978,25 @@ def resubscription_attempted(
976978
terminationError,
977979
nextResubscribeIntervalMsec,
978980
)
979-
# mark node as unavailable and signal consumers
980-
# we debounce it a bit so we only mark the node unavailable
981-
# at the second resubscription attempt
982-
if node.available and self._last_subscription_attempt[node_id] >= 1:
983-
# NOTE: if the node is (re)discovered by mdns, that callback will
984-
# take care of resubscribing to the node
981+
resubscription_attempt = self._last_subscription_attempt[node_id] + 1
982+
self._last_subscription_attempt[node_id] = resubscription_attempt
983+
# Mark node as unavailable and signal consumers.
984+
# We debounce it a bit so we only mark the node unavailable
985+
# after some resubscription attempts and we shutdown the subscription
986+
# if the resubscription interval exceeds 30 minutes (TTL of mdns).
987+
# The node will be auto picked up by mdns if it's alive again.
988+
if (
989+
node.available
990+
and resubscription_attempt >= NODE_RESUBSCRIBE_ATTEMPTS_UNAVAILABLE
991+
):
992+
node.available = False
993+
self.server.signal_event(EventType.NODE_UPDATED, node)
994+
LOGGER.info("Marked node %s as unavailable", node_id)
995+
if (
996+
not node.available
997+
and nextResubscribeIntervalMsec > NODE_RESUBSCRIBE_TIMEOUT_OFFLINE
998+
):
985999
asyncio.create_task(self._node_offline(node_id))
986-
self._last_subscription_attempt[node_id] += 1
9871000

9881001
def resubscription_succeeded(
9891002
transaction: Attribute.SubscriptionTransaction,
@@ -1249,7 +1262,7 @@ async def _node_offline(self, node_id: int) -> None:
12491262
# shutdown existing subscriptions
12501263
if sub := self._subscriptions.pop(node_id, None):
12511264
await self._call_sdk(sub.Shutdown)
1252-
# mark node as unavailable
1265+
# mark node as unavailable (if it wasn't already)
12531266
node = self._nodes[node_id]
12541267
if not node.available:
12551268
return # nothing to do to

0 commit comments

Comments
 (0)