|
1 | 1 | """Tuya devices."""
|
2 | 2 |
|
| 3 | +from __future__ import annotations |
| 4 | + |
3 | 5 | from collections.abc import Callable
|
4 | 6 | import dataclasses
|
5 | 7 | import datetime
|
6 | 8 | import enum
|
7 | 9 | import logging
|
8 |
| -from typing import Any, Optional, Union |
| 10 | +from typing import Any |
9 | 11 |
|
10 | 12 | from zigpy.quirks import BaseCustomDevice, CustomCluster, CustomDevice
|
11 | 13 | import zigpy.types as t
|
| 14 | +from zigpy.typing import AddressingMode |
12 | 15 | from zigpy.zcl import BaseAttributeDefs, foundation
|
13 | 16 | from zigpy.zcl.clusters.closures import WindowCovering
|
14 | 17 | from zigpy.zcl.clusters.general import Basic, LevelControl, OnOff, PowerConfiguration
|
@@ -155,16 +158,16 @@ class TuyaData(t.Struct):
|
155 | 158 | @property
|
156 | 159 | def payload(
|
157 | 160 | self,
|
158 |
| - ) -> Union[ |
159 |
| - t.int32s_be, |
160 |
| - t.Bool, |
161 |
| - t.CharacterString, |
162 |
| - t.enum8, |
163 |
| - t.bitmap8, |
164 |
| - t.bitmap16, |
165 |
| - t.bitmap32, |
166 |
| - t.LVBytes, |
167 |
| - ]: |
| 161 | + ) -> ( |
| 162 | + t.int32s_be |
| 163 | + | t.Bool |
| 164 | + | t.CharacterString |
| 165 | + | t.enum8 |
| 166 | + | t.bitmap8 |
| 167 | + | t.bitmap16 |
| 168 | + | t.bitmap32 |
| 169 | + | t.LVBytes |
| 170 | + ): |
168 | 171 | """Payload accordingly to data point type."""
|
169 | 172 | if self.dp_type == TuyaDPType.VALUE:
|
170 | 173 | return t.int32s_be.deserialize(self.raw)[0]
|
@@ -294,11 +297,11 @@ class NoManufacturerCluster(CustomCluster):
|
294 | 297 |
|
295 | 298 | async def command(
|
296 | 299 | self,
|
297 |
| - command_id: Union[foundation.GeneralCommand, int, t.uint8_t], |
| 300 | + command_id: foundation.GeneralCommand | int | t.uint8_t, |
298 | 301 | *args,
|
299 |
| - manufacturer: Optional[Union[int, t.uint16_t]] = None, |
| 302 | + manufacturer: int | t.uint16_t | None = None, |
300 | 303 | expect_reply: bool = True,
|
301 |
| - tsn: Optional[Union[int, t.uint8_t]] = None, |
| 304 | + tsn: int | t.uint8_t | None = None, |
302 | 305 | **kwargs: Any,
|
303 | 306 | ):
|
304 | 307 | """Override the default Cluster command."""
|
@@ -410,11 +413,9 @@ def tuya_mcu_command(self, command: Command):
|
410 | 413 | def handle_cluster_request(
|
411 | 414 | self,
|
412 | 415 | hdr: foundation.ZCLHeader,
|
413 |
| - args: tuple, |
| 416 | + args: list[Any], |
414 | 417 | *,
|
415 |
| - dst_addressing: Optional[ |
416 |
| - Union[t.Addressing.Group, t.Addressing.IEEE, t.Addressing.NWK] |
417 |
| - ] = None, |
| 418 | + dst_addressing: AddressingMode | None = None, |
418 | 419 | ) -> None:
|
419 | 420 | """Handle time request."""
|
420 | 421 |
|
@@ -463,11 +464,9 @@ class TuyaManufClusterAttributes(TuyaManufCluster):
|
463 | 464 | def handle_cluster_request(
|
464 | 465 | self,
|
465 | 466 | hdr: foundation.ZCLHeader,
|
466 |
| - args: tuple, |
| 467 | + args: list[Any], |
467 | 468 | *,
|
468 |
| - dst_addressing: Optional[ |
469 |
| - Union[t.Addressing.Group, t.Addressing.IEEE, t.Addressing.NWK] |
470 |
| - ] = None, |
| 469 | + dst_addressing: AddressingMode | None = None, |
471 | 470 | ) -> None:
|
472 | 471 | """Handle cluster request."""
|
473 | 472 | if hdr.command_id not in (0x0001, 0x0002):
|
@@ -600,11 +599,12 @@ def switch_event(self, channel, state):
|
600 | 599 |
|
601 | 600 | async def command(
|
602 | 601 | self,
|
603 |
| - command_id: Union[foundation.GeneralCommand, int, t.uint8_t], |
| 602 | + command_id: foundation.GeneralCommand | int | t.uint8_t, |
604 | 603 | *args,
|
605 |
| - manufacturer: Optional[Union[int, t.uint16_t]] = None, |
| 604 | + manufacturer: int | t.uint16_t | None = None, |
606 | 605 | expect_reply: bool = True,
|
607 |
| - tsn: Optional[Union[int, t.uint8_t]] = None, |
| 606 | + tsn: int | t.uint8_t | None = None, |
| 607 | + **kwargs: Any, |
608 | 608 | ):
|
609 | 609 | """Override the default Cluster command."""
|
610 | 610 |
|
@@ -636,11 +636,9 @@ class TuyaManufacturerClusterOnOff(TuyaManufCluster):
|
636 | 636 | def handle_cluster_request(
|
637 | 637 | self,
|
638 | 638 | hdr: foundation.ZCLHeader,
|
639 |
| - args: tuple[TuyaManufCluster.Command], |
| 639 | + args: list[Any], |
640 | 640 | *,
|
641 |
| - dst_addressing: Optional[ |
642 |
| - Union[t.Addressing.Group, t.Addressing.IEEE, t.Addressing.NWK] |
643 |
| - ] = None, |
| 641 | + dst_addressing: AddressingMode | None = None, |
644 | 642 | ) -> None:
|
645 | 643 | """Handle cluster request."""
|
646 | 644 |
|
@@ -757,11 +755,12 @@ async def write_attributes(self, attributes, manufacturer=None):
|
757 | 755 | # pylint: disable=W0236
|
758 | 756 | async def command(
|
759 | 757 | self,
|
760 |
| - command_id: Union[foundation.GeneralCommand, int, t.uint8_t], |
| 758 | + command_id: foundation.GeneralCommand | int | t.uint8_t, |
761 | 759 | *args,
|
762 |
| - manufacturer: Optional[Union[int, t.uint16_t]] = None, |
| 760 | + manufacturer: int | t.uint16_t | None = None, |
763 | 761 | expect_reply: bool = True,
|
764 |
| - tsn: Optional[Union[int, t.uint8_t]] = None, |
| 762 | + tsn: int | t.uint8_t | None = None, |
| 763 | + **kwargs: Any, |
765 | 764 | ):
|
766 | 765 | """Implement thermostat commands."""
|
767 | 766 |
|
@@ -1052,10 +1051,8 @@ def handle_cluster_request(
|
1052 | 1051 | hdr: foundation.ZCLHeader,
|
1053 | 1052 | args: list[Any],
|
1054 | 1053 | *,
|
1055 |
| - dst_addressing: Optional[ |
1056 |
| - Union[t.Addressing.Group, t.Addressing.IEEE, t.Addressing.NWK] |
1057 |
| - ] = None, |
1058 |
| - ): |
| 1054 | + dst_addressing: AddressingMode | None = None, |
| 1055 | + ) -> None: |
1059 | 1056 | """Handle press_types command."""
|
1060 | 1057 | # normally if default response sent, TS004x wouldn't send such repeated zclframe (with same sequence number),
|
1061 | 1058 | # but for stability reasons (e. g. the case the response doesn't arrive the device), we can simply ignore it
|
@@ -1152,11 +1149,9 @@ class TuyaManufacturerWindowCover(TuyaManufCluster):
|
1152 | 1149 | def handle_cluster_request(
|
1153 | 1150 | self,
|
1154 | 1151 | hdr: foundation.ZCLHeader,
|
1155 |
| - args: tuple[TuyaManufCluster.Command], |
| 1152 | + args: list[Any], |
1156 | 1153 | *,
|
1157 |
| - dst_addressing: Optional[ |
1158 |
| - Union[t.Addressing.Group, t.Addressing.IEEE, t.Addressing.NWK] |
1159 |
| - ] = None, |
| 1154 | + dst_addressing: AddressingMode | None = None, |
1160 | 1155 | ) -> None:
|
1161 | 1156 | """Handle cluster request."""
|
1162 | 1157 | # Tuya Specific Cluster Commands
|
@@ -1246,11 +1241,12 @@ def cover_event(self, attribute, value):
|
1246 | 1241 |
|
1247 | 1242 | def command(
|
1248 | 1243 | self,
|
1249 |
| - command_id: Union[foundation.GeneralCommand, int, t.uint8_t], |
| 1244 | + command_id: foundation.GeneralCommand | int | t.uint8_t, |
1250 | 1245 | *args,
|
1251 |
| - manufacturer: Optional[Union[int, t.uint16_t]] = None, |
| 1246 | + manufacturer: int | t.uint16_t | None = None, |
1252 | 1247 | expect_reply: bool = True,
|
1253 |
| - tsn: Optional[Union[int, t.uint8_t]] = None, |
| 1248 | + tsn: int | t.uint8_t | None = None, |
| 1249 | + **kwargs: Any, |
1254 | 1250 | ):
|
1255 | 1251 | """Override the default Cluster command."""
|
1256 | 1252 | if manufacturer is None:
|
@@ -1353,11 +1349,9 @@ class TuyaManufacturerLevelControl(TuyaManufCluster):
|
1353 | 1349 | def handle_cluster_request(
|
1354 | 1350 | self,
|
1355 | 1351 | hdr: foundation.ZCLHeader,
|
1356 |
| - args: tuple[TuyaManufCluster.Command], |
| 1352 | + args: list[Any], |
1357 | 1353 | *,
|
1358 |
| - dst_addressing: Optional[ |
1359 |
| - Union[t.Addressing.Group, t.Addressing.IEEE, t.Addressing.NWK] |
1360 |
| - ] = None, |
| 1354 | + dst_addressing: AddressingMode | None = None, |
1361 | 1355 | ) -> None:
|
1362 | 1356 | """Handle cluster request."""
|
1363 | 1357 | tuya_payload = args[0]
|
@@ -1411,11 +1405,11 @@ def level_event(self, channel, state):
|
1411 | 1405 |
|
1412 | 1406 | def command(
|
1413 | 1407 | self,
|
1414 |
| - command_id: Union[foundation.GeneralCommand, int, t.uint8_t], |
| 1408 | + command_id: foundation.GeneralCommand | int | t.uint8_t, |
1415 | 1409 | *args,
|
1416 |
| - manufacturer: Optional[Union[int, t.uint16_t]] = None, |
| 1410 | + manufacturer: int | t.uint16_t | None = None, |
1417 | 1411 | expect_reply: bool = True,
|
1418 |
| - tsn: Optional[Union[int, t.uint8_t]] = None, |
| 1412 | + tsn: int | t.uint8_t | None = None, |
1419 | 1413 | **kwargs: Any,
|
1420 | 1414 | ):
|
1421 | 1415 | """Override the default Cluster command."""
|
@@ -1458,16 +1452,9 @@ class DPToAttributeMapping:
|
1458 | 1452 | """Container for datapoint to cluster attribute update mapping."""
|
1459 | 1453 |
|
1460 | 1454 | ep_attribute: str
|
1461 |
| - attribute_name: Union[str, tuple] |
1462 |
| - converter: Optional[ |
1463 |
| - Callable[ |
1464 |
| - [ |
1465 |
| - Any, |
1466 |
| - ], |
1467 |
| - Any, |
1468 |
| - ] |
1469 |
| - ] = None |
1470 |
| - endpoint_id: Optional[int] = None |
| 1455 | + attribute_name: str | tuple[str, ...] |
| 1456 | + converter: Callable[[Any], Any] | None = None |
| 1457 | + endpoint_id: int | None = None |
1471 | 1458 |
|
1472 | 1459 |
|
1473 | 1460 | @dataclasses.dataclass
|
@@ -1574,11 +1561,9 @@ def __init__(self, *args, **kwargs):
|
1574 | 1561 | def handle_cluster_request(
|
1575 | 1562 | self,
|
1576 | 1563 | hdr: foundation.ZCLHeader,
|
1577 |
| - args: tuple, |
| 1564 | + args: list[Any], |
1578 | 1565 | *,
|
1579 |
| - dst_addressing: Optional[ |
1580 |
| - Union[t.Addressing.Group, t.Addressing.IEEE, t.Addressing.NWK] |
1581 |
| - ] = None, |
| 1566 | + dst_addressing: AddressingMode | None = None, |
1582 | 1567 | ) -> None:
|
1583 | 1568 | """Handle cluster specific request."""
|
1584 | 1569 |
|
|
0 commit comments