Skip to content

Commit 0ca4099

Browse files
committed
Switch (back) to a single wildcard subscription per node
1 parent 59e1cbd commit 0ca4099

File tree

1 file changed

+4
-96
lines changed

1 file changed

+4
-96
lines changed

matter_server/server/device_controller.py

+4-96
Original file line numberDiff line numberDiff line change
@@ -74,28 +74,6 @@
7474
)
7575

7676

77-
BASE_SUBSCRIBE_ATTRIBUTES: tuple[Attribute.AttributePath, ...] = (
78-
# all endpoints, BasicInformation cluster
79-
Attribute.AttributePath(
80-
EndpointId=None, ClusterId=Clusters.BasicInformation.id, Attribute=None
81-
),
82-
# all endpoints, BridgedDeviceBasicInformation (bridges only)
83-
Attribute.AttributePath(
84-
EndpointId=None,
85-
ClusterId=Clusters.BridgedDeviceBasicInformation.id,
86-
Attribute=None,
87-
),
88-
# networkinterfaces attribute on general diagnostics cluster,
89-
# so we have the most accurate IP addresses for ping/diagnostics
90-
Attribute.AttributePath(
91-
EndpointId=0, Attribute=Clusters.GeneralDiagnostics.Attributes.NetworkInterfaces
92-
),
93-
# active fabrics attribute - to speedup node diagnostics
94-
Attribute.AttributePath(
95-
EndpointId=0, Attribute=Clusters.OperationalCredentials.Attributes.Fabrics
96-
),
97-
)
98-
9977
# pylint: disable=too-many-lines,too-many-locals,too-many-statements,too-many-branches,too-many-instance-attributes
10078

10179

@@ -114,8 +92,6 @@ def __init__(
11492
# we keep the last events in memory so we can include them in the diagnostics dump
11593
self.event_history: deque[Attribute.EventReadResult] = deque(maxlen=25)
11694
self._subscriptions: dict[int, Attribute.SubscriptionTransaction] = {}
117-
self._attr_subscriptions: dict[int, list[Attribute.AttributePath]] = {}
118-
self._resub_debounce_timer: dict[int, asyncio.TimerHandle] = {}
11995
self._nodes: dict[int, MatterNodeData] = {}
12096
self._last_subscription_attempt: dict[int, int] = {}
12197
self.wifi_credentials_set: bool = False
@@ -697,40 +673,9 @@ async def subscribe_attribute(
697673
The given attribute path(s) will be added to the list of attributes that
698674
are watched for the given node. This is persistent over restarts.
699675
"""
700-
if self.chip_controller is None:
701-
raise RuntimeError("Device Controller not initialized.")
702-
703-
if node_id not in self._nodes:
704-
raise NodeNotExists(
705-
f"Node {node_id} does not exist or has not been interviewed."
706-
)
707-
708-
node = self._nodes[node_id]
709-
assert node is not None
710-
711-
# work out added subscriptions
712-
if not isinstance(attribute_path, list):
713-
attribute_path = [attribute_path]
714-
attribute_paths = {parse_attribute_path(x) for x in attribute_path}
715-
prev_subs = set(node.attribute_subscriptions)
716-
node.attribute_subscriptions.update(attribute_paths)
717-
if prev_subs == node.attribute_subscriptions:
718-
return # nothing to do
719-
# save updated node data
720-
self._write_node_state(node_id)
721-
722-
# (re)setup node subscription
723-
# this could potentially be called multiple times within a short timeframe
724-
# so debounce it a bit
725-
def resubscribe() -> None:
726-
self._resub_debounce_timer.pop(node_id, None)
727-
asyncio.create_task(self._subscribe_node(node_id))
728-
729-
if existing_timer := self._resub_debounce_timer.pop(node_id, None):
730-
existing_timer.cancel()
731-
assert self.server.loop is not None
732-
self._resub_debounce_timer[node_id] = self.server.loop.call_later(
733-
5, resubscribe
676+
LOGGER.warning(
677+
"The subscribe_attribute command has been deprecated and will be removed from"
678+
" a future version. You no longer need to call this to subscribe to attribute changes."
734679
)
735680

736681
@api_command(APICommand.PING_NODE)
@@ -809,52 +754,15 @@ async def _subscribe_node(self, node_id: int) -> None:
809754
node_lock = self._get_node_lock(node_id)
810755
node = self._nodes[node_id]
811756

812-
# work out all (current) attribute subscriptions
813-
attr_subscriptions: list[Attribute.AttributePath] = list(
814-
BASE_SUBSCRIBE_ATTRIBUTES
815-
)
816-
for (
817-
endpoint_id,
818-
cluster_id,
819-
attribute_id,
820-
) in node.attribute_subscriptions:
821-
attr_path = Attribute.AttributePath(
822-
EndpointId=endpoint_id,
823-
ClusterId=cluster_id,
824-
AttributeId=attribute_id,
825-
)
826-
if attr_path in attr_subscriptions:
827-
continue
828-
attr_subscriptions.append(attr_path)
829-
830-
if node.is_bridge or len(attr_subscriptions) > 9:
831-
# A matter device can officially only handle 3 attribute paths per subscription
832-
# and a maximum of 3 concurrent subscriptions per fabric.
833-
# We cheat a bit here and use one single subscription for up to 9 paths,
834-
# because in our experience that is more stable than multiple subscriptions
835-
# to the same device. If we have more than 9 paths to watch for a node,
836-
# we switch to a wildcard subscription.
837-
attr_subscriptions = [Attribute.AttributePath()] # wildcard
838-
839757
# check if we already have setup subscriptions for this node,
840758
# if so, we need to unsubscribe first unless nothing changed
841759
# in the attribute paths we want to subscribe.
842760
if prev_sub := self._subscriptions.get(node_id, None):
843-
if (
844-
node.available
845-
and self._attr_subscriptions.get(node_id) == attr_subscriptions
846-
):
847-
# the current subscription already matches, no need to re-setup
848-
node_logger.debug("Re-using existing subscription.")
849-
return
850761
async with node_lock:
851762
node_logger.info("Unsubscribing from existing subscription.")
852763
await self._call_sdk(prev_sub.Shutdown)
853764
del self._subscriptions[node_id]
854765

855-
# store our list of subscriptions for this node
856-
self._attr_subscriptions[node_id] = attr_subscriptions
857-
858766
# determine if node is battery powered sleeping device
859767
# Endpoint 0, ThreadNetworkDiagnostics Cluster, routingRole attribute
860768
battery_powered = (
@@ -1016,7 +924,7 @@ def resubscription_succeeded(
1016924
eventLoop=loop,
1017925
device=device.deviceProxy,
1018926
devCtrl=self.chip_controller,
1019-
attributes=attr_subscriptions,
927+
attributes=[Attribute.AttributePath()], # wildcard
1020928
events=[
1021929
Attribute.EventPath(
1022930
EndpointId=None, Cluster=None, Event=None, Urgent=1

0 commit comments

Comments
 (0)