Skip to content

Commit a259a7a

Browse files
committed
introduce allow_sdk_types in parse value helper
1 parent edb1e1b commit a259a7a

File tree

3 files changed

+31
-32
lines changed

3 files changed

+31
-32
lines changed

matter_server/common/helpers/util.py

+20-11
Original file line numberDiff line numberDiff line change
@@ -3,16 +3,17 @@
33
from __future__ import annotations
44

55
import base64
6-
from base64 import b64decode
76
import binascii
7+
import logging
8+
import platform
9+
import socket
10+
from base64 import b64decode
811
from dataclasses import MISSING, asdict, fields, is_dataclass
912
from datetime import datetime
1013
from enum import Enum
1114
from functools import cache
12-
from importlib.metadata import PackageNotFoundError, version as pkg_version
13-
import logging
14-
import platform
15-
import socket
15+
from importlib.metadata import PackageNotFoundError
16+
from importlib.metadata import version as pkg_version
1617
from types import NoneType, UnionType
1718
from typing import (
1819
TYPE_CHECKING,
@@ -108,8 +109,13 @@ def parse_value(
108109
value_type: Any,
109110
default: Any = MISSING,
110111
allow_none: bool = True,
112+
allow_sdk_types: bool = False,
111113
) -> Any:
112-
"""Try to parse a value from raw (json) data and type annotations."""
114+
"""
115+
Try to parse a value from raw (json) data and type annotations.
116+
117+
If allow_sdk_types is False, any SDK specific custom data types will be converted.
118+
"""
113119
# pylint: disable=too-many-return-statements,too-many-branches
114120

115121
if isinstance(value_type, str):
@@ -131,9 +137,9 @@ def parse_value(
131137
return default
132138
if value is None and value_type is NoneType:
133139
return None
134-
if value is None and allow_none:
135-
return None
136140
if value is None and value_type is Nullable:
141+
return Nullable() if allow_sdk_types else None
142+
if value is None and allow_none:
137143
return None
138144
if is_dataclass(value_type) and isinstance(value, dict):
139145
return dataclass_from_dict(value_type, value)
@@ -220,12 +226,12 @@ def parse_value(
220226
if value_type is uint and (
221227
isinstance(value, int) or (isinstance(value, str) and value.isnumeric())
222228
):
223-
return uint(value)
229+
return uint(value) if allow_sdk_types else int(value)
224230
if value_type is float32 and (
225231
isinstance(value, (float, int))
226232
or (isinstance(value, str) and value.isnumeric())
227233
):
228-
return float32(value)
234+
return float32(value) if allow_sdk_types else float(value)
229235

230236
# If we reach this point, we could not match the value with the type and we raise
231237
if not isinstance(value, value_type):
@@ -236,7 +242,9 @@ def parse_value(
236242
return value
237243

238244

239-
def dataclass_from_dict(cls: type[_T], dict_obj: dict, strict: bool = False) -> _T:
245+
def dataclass_from_dict(
246+
cls: type[_T], dict_obj: dict, strict: bool = False, allow_sdk_types: bool = False
247+
) -> _T:
240248
"""
241249
Create (instance of) a dataclass by providing a dict with values.
242250
@@ -259,6 +267,7 @@ def dataclass_from_dict(cls: type[_T], dict_obj: dict, strict: bool = False) ->
259267
type_hints[field.name],
260268
field.default,
261269
allow_none=not strict,
270+
allow_sdk_types=allow_sdk_types,
262271
)
263272
for field in dc_fields
264273
if field.init

matter_server/server/device_controller.py

+11-4
Original file line numberDiff line numberDiff line change
@@ -22,7 +22,7 @@
2222

2323
from matter_server.common.helpers.util import convert_ip_address
2424
from matter_server.common.models import CommissionableNodeData, CommissioningParameters
25-
from matter_server.server.helpers.attributes import parse_attributes_from_read_result, parse_nullable_for_write
25+
from matter_server.server.helpers.attributes import parse_attributes_from_read_result
2626
from matter_server.server.helpers.utils import ping_ip
2727

2828
from ..common.errors import (
@@ -619,8 +619,16 @@ async def write_attribute(
619619
if (node := self._nodes.get(node_id)) is None or not node.available:
620620
raise NodeNotReady(f"Node {node_id} is not (yet) available.")
621621
endpoint_id, cluster_id, attribute_id = parse_attribute_path(attribute_path)
622-
attribute = ALL_ATTRIBUTES[cluster_id][attribute_id]()
623-
attribute.value = parse_nullable_for_write(value)
622+
attribute = cast(
623+
Clusters.ClusterAttributeDescriptor,
624+
ALL_ATTRIBUTES[cluster_id][attribute_id](),
625+
)
626+
attribute.value = parse_value(
627+
name=attribute.__qualname__,
628+
value=value,
629+
value_type=attribute.attribute_type.Type,
630+
allow_sdk_types=True,
631+
)
624632
return await self.chip_controller.WriteAttribute(
625633
nodeid=node_id,
626634
attributes=[(endpoint_id, attribute)],
@@ -1256,4 +1264,3 @@ async def _node_offline(self, node_id: int) -> None:
12561264
node.available = False
12571265
self.server.signal_event(EventType.NODE_UPDATED, node)
12581266
LOGGER.info("Marked node %s as offline", node_id)
1259-
-17
Original file line numberDiff line numberDiff line change
@@ -1,8 +1,6 @@
11
"""Helpers to manage Cluster attributes."""
22

33
from typing import Any
4-
from copy import deepcopy
5-
from chip.clusters import Objects as Clusters
64

75
from ...common.helpers.util import create_attribute_path
86

@@ -25,18 +23,3 @@ def parse_attributes_from_read_result(
2523
)
2624
result[attribute_path] = attr_value
2725
return result
28-
29-
def parse_nullable_for_write(payload):
30-
"""Parses the Python 'None' values to CHIP 'Nullable' type for writing attribute."""
31-
outload = deepcopy(payload)
32-
33-
if isinstance(outload, dict):
34-
for k, v in outload.items():
35-
outload[k] = parse_nullable_for_write(outload[k])
36-
elif isinstance(outload, list):
37-
for i in range(0, len(outload)):
38-
outload[i] = parse_nullable_for_write(outload[i])
39-
else:
40-
outload = Clusters.NullValue if outload is None else outload
41-
42-
return outload

0 commit comments

Comments
 (0)