-
-
Notifications
You must be signed in to change notification settings - Fork 568
/
device.py
356 lines (291 loc) · 11.4 KB
/
device.py
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
import logging
from enum import Enum
from typing import Any, Dict, List, Optional, Union, cast, final # noqa: F401
import click
from .click_common import DeviceGroupMeta, LiteralParamType, command
from .descriptorcollection import DescriptorCollection
from .descriptors import AccessFlags, ActionDescriptor, Descriptor, PropertyDescriptor
from .deviceinfo import DeviceInfo
from .devicestatus import DeviceStatus
from .exceptions import (
DeviceError,
DeviceInfoUnavailableException,
PayloadDecodeException,
)
from .miioprotocol import MiIOProtocol
_LOGGER = logging.getLogger(__name__)
class UpdateState(Enum):
Downloading = "downloading"
Installing = "installing"
Failed = "failed"
Idle = "idle"
class Device(metaclass=DeviceGroupMeta):
"""Base class for all device implementations.
This is the main class providing the basic protocol handling for devices using the
``miIO`` protocol. This class should not be initialized directly but a device-
specific class inheriting it should be used instead of it.
"""
retry_count = 3
timeout = 5
_mappings: dict[str, Any] = {}
_supported_models: list[str] = []
def __init_subclass__(cls, **kwargs):
"""Overridden to register all integrations to the factory."""
super().__init_subclass__(**kwargs)
from .devicefactory import DeviceFactory
DeviceFactory.register(cls)
def __init__(
self,
ip: Optional[str] = None,
token: Optional[str] = None,
start_id: int = 0,
debug: int = 0,
lazy_discover: bool = True,
timeout: Optional[int] = None,
*,
model: Optional[str] = None,
) -> None:
self.ip = ip
self.token: Optional[str] = token
self._model: Optional[str] = model
self._info: Optional[DeviceInfo] = None
# TODO: use _info's noneness instead?
self._initialized: bool = False
self._descriptors: DescriptorCollection = DescriptorCollection(device=self)
timeout = timeout if timeout is not None else self.timeout
self._debug = debug
self._protocol = MiIOProtocol(
ip, token, start_id, debug, lazy_discover, timeout
)
def send(
self,
command: str,
parameters: Optional[Any] = None,
retry_count: Optional[int] = None,
*,
extra_parameters=None,
) -> Any:
"""Send a command to the device.
Basic format of the request:
{"id": 1234, "method": command, "parameters": parameters}
`extra_parameters` allows passing elements to the top-level of the request.
This is necessary for some devices, such as gateway devices, which expect
the sub-device identifier to be on the top-level.
:param str command: Command to send
:param dict parameters: Parameters to send
:param int retry_count: How many times to retry on error
:param dict extra_parameters: Extra top-level parameters
:param str model: Force model to avoid autodetection
"""
retry_count = retry_count if retry_count is not None else self.retry_count
return self._protocol.send(
command, parameters, retry_count, extra_parameters=extra_parameters
)
def send_handshake(self):
"""Send initial handshake to the device."""
return self._protocol.send_handshake()
@command(
click.argument("command", type=str, required=True),
click.argument("parameters", type=LiteralParamType(), required=False),
)
def raw_command(self, command, parameters):
"""Send a raw command to the device. This is mostly useful when trying out
commands which are not implemented by a given device instance.
:param str command: Command to send
:param dict parameters: Parameters to send
"""
return self.send(command, parameters)
@command(
skip_autodetect=True,
)
def info(self, *, skip_cache=False) -> DeviceInfo:
"""Get (and cache) miIO protocol information from the device.
This includes information about connected wlan network, and hardware and
software versions.
:param skip_cache bool: Skip the cache
"""
if self._info is not None and not skip_cache:
return self._info
return self._fetch_info()
def _fetch_info(self) -> DeviceInfo:
"""Perform miIO.info query on the device and cache the result."""
try:
devinfo = DeviceInfo(self.send("miIO.info"))
self._info = devinfo
_LOGGER.debug("Detected model %s", devinfo.model)
return devinfo
except PayloadDecodeException as ex:
raise DeviceInfoUnavailableException(
"Unable to request miIO.info from the device"
) from ex
def _initialize_descriptors(self) -> None:
"""Initialize the device descriptors.
This will add descriptors defined in the implementation class and the status class.
This can be overridden to add additional descriptors to the device.
If you do so, do not forget to call this method.
"""
if self._initialized:
return
self._descriptors.descriptors_from_object(self)
# Read descriptors from the status class
self._descriptors.descriptors_from_object(self.status.__annotations__["return"])
if not self._descriptors:
_LOGGER.warning(
"'%s' does not specify any descriptors, please considering creating a PR.",
self.__class__.__name__,
)
self._initialized = True
@property
def device_id(self) -> int:
"""Return the device id (did)."""
if not self._protocol._device_id:
self.send_handshake()
return int.from_bytes(self._protocol._device_id, byteorder="big")
@property
def raw_id(self) -> int:
"""Return the last used protocol sequence id."""
return self._protocol.raw_id
@property
def supported_models(self) -> list[str]:
"""Return a list of supported models."""
return list(self._mappings.keys()) or self._supported_models
@property
def model(self) -> str:
"""Return device model."""
if self._model is not None:
return self._model
return self.info().model
def update(self, url: str, md5: str):
"""Start an OTA update."""
payload = {
"mode": "normal",
"install": "1",
"app_url": url,
"file_md5": md5,
"proc": "dnld install",
}
return self.send("miIO.ota", payload)[0] == "ok"
def update_progress(self) -> int:
"""Return current update progress [0-100]."""
return self.send("miIO.get_ota_progress")[0]
def update_state(self):
"""Return current update state."""
return UpdateState(self.send("miIO.get_ota_state")[0])
def configure_wifi(self, ssid, password, uid=0, extra_params=None):
"""Configure the wifi settings."""
if extra_params is None:
extra_params = {}
params = {"ssid": ssid, "passwd": password, "uid": uid, **extra_params}
return self.send("miIO.config_router", params)[0]
def get_properties(
self, properties, *, property_getter="get_prop", max_properties=None
):
"""Request properties in slices based on given max_properties.
This is necessary as some devices have limitation on how many
properties can be queried at once.
If `max_properties` is None, all properties are requested at once.
:param list properties: List of properties to query from the device.
:param int max_properties: Number of properties that can be requested at once.
:return: List of property values.
"""
_props = properties.copy()
values = []
while _props:
values.extend(self.send(property_getter, _props[:max_properties]))
if max_properties is None:
break
_props[:] = _props[max_properties:]
properties_count = len(properties)
values_count = len(values)
if properties_count != values_count:
_LOGGER.debug(
"Count (%s) of requested properties does not match the "
"count (%s) of received values.",
properties_count,
values_count,
)
return values
@command()
def status(self) -> DeviceStatus:
"""Return device status."""
raise NotImplementedError()
@command()
def descriptors(self) -> DescriptorCollection[Descriptor]:
"""Return a collection containing all descriptors for the device."""
if not self._initialized:
self._initialize_descriptors()
return self._descriptors
@command()
def actions(self) -> DescriptorCollection[ActionDescriptor]:
"""Return device actions."""
return DescriptorCollection(
{
k: v
for k, v in self.descriptors().items()
if isinstance(v, ActionDescriptor)
},
device=self,
)
@final
@command()
def settings(self) -> DescriptorCollection[PropertyDescriptor]:
"""Return settable properties."""
return DescriptorCollection(
{
k: v
for k, v in self.descriptors().items()
if isinstance(v, PropertyDescriptor) and v.access & AccessFlags.Write
},
device=self,
)
@final
@command()
def sensors(self) -> DescriptorCollection[PropertyDescriptor]:
"""Return read-only properties."""
return DescriptorCollection(
{
k: v
for k, v in self.descriptors().items()
if v.access == AccessFlags.Read
},
device=self,
)
def supports_miot(self) -> bool:
"""Return True if the device supports miot commands.
This requests a single property (siid=1, piid=1) and returns True on success.
"""
try:
self.send("get_properties", [{"did": "dummy", "siid": 1, "piid": 1}])
except DeviceError as ex:
_LOGGER.debug("miot query failed, likely non-miot device: %s", repr(ex))
return False
return True
@command(
click.argument("name"),
click.argument("params", type=LiteralParamType(), required=False),
name="call",
)
def call_action(self, name: str, params=None):
"""Call action by name."""
try:
act = self.actions()[name]
except KeyError:
raise ValueError("Unable to find action '%s'" % name)
if params is None:
return act.method()
return act.method(params)
@command(
click.argument("name"),
click.argument("params", type=LiteralParamType(), required=True),
name="set",
)
def change_setting(self, name: str, params=None):
"""Change setting value."""
try:
setting = self.settings()[name]
except KeyError:
raise ValueError("Unable to find setting '%s'" % name)
params = params if params is not None else []
return setting.setter(params)
def __repr__(self):
return f"<{self.__class__.__name__}: {self.ip} (token: {self.token})>"