mirror of
https://github.com/python-kasa/python-kasa.git
synced 2025-01-08 22:07:06 +00:00
f8a46f74cd
Clean up and consolidate the processing of raw component query responses and simplify the code paths for creating smartcam child devices when supported.
259 lines
9.4 KiB
Python
259 lines
9.4 KiB
Python
"""Module for SmartCamDevice."""
|
|
|
|
from __future__ import annotations
|
|
|
|
import logging
|
|
from typing import Any, cast
|
|
|
|
from ..device import _DeviceInfo
|
|
from ..device_type import DeviceType
|
|
from ..module import Module
|
|
from ..protocols.smartcamprotocol import _ChildCameraProtocolWrapper
|
|
from ..smart import SmartChildDevice, SmartDevice
|
|
from ..smart.smartdevice import ComponentsRaw
|
|
from .modules import ChildDevice, DeviceModule
|
|
from .smartcammodule import SmartCamModule
|
|
|
|
_LOGGER = logging.getLogger(__name__)
|
|
|
|
|
|
class SmartCamDevice(SmartDevice):
|
|
"""Class for smart cameras."""
|
|
|
|
# Modules that are called as part of the init procedure on first update
|
|
FIRST_UPDATE_MODULES = {DeviceModule, ChildDevice}
|
|
|
|
@staticmethod
|
|
def _get_device_type_from_sysinfo(sysinfo: dict[str, Any]) -> DeviceType:
|
|
"""Find type to be displayed as a supported device category."""
|
|
if (
|
|
sysinfo
|
|
and (device_type := sysinfo.get("device_type"))
|
|
and device_type.endswith("HUB")
|
|
):
|
|
return DeviceType.Hub
|
|
return DeviceType.Camera
|
|
|
|
@staticmethod
|
|
def _get_device_info(
|
|
info: dict[str, Any], discovery_info: dict[str, Any] | None
|
|
) -> _DeviceInfo:
|
|
"""Get model information for a device."""
|
|
basic_info = info["getDeviceInfo"]["device_info"]["basic_info"]
|
|
short_name = basic_info["device_model"]
|
|
long_name = discovery_info["device_model"] if discovery_info else short_name
|
|
device_type = SmartCamDevice._get_device_type_from_sysinfo(basic_info)
|
|
fw_version_full = basic_info["sw_version"]
|
|
firmware_version, firmware_build = fw_version_full.split(" ", maxsplit=1)
|
|
return _DeviceInfo(
|
|
short_name=basic_info["device_model"],
|
|
long_name=long_name,
|
|
brand="tapo",
|
|
device_family=basic_info["device_type"],
|
|
device_type=device_type,
|
|
hardware_version=basic_info["hw_version"],
|
|
firmware_version=firmware_version,
|
|
firmware_build=firmware_build,
|
|
requires_auth=True,
|
|
region=basic_info.get("region"),
|
|
)
|
|
|
|
def _update_internal_info(self, info_resp: dict) -> None:
|
|
"""Update the internal device info."""
|
|
info = self._try_get_response(info_resp, "getDeviceInfo")
|
|
self._info = self._map_info(info["device_info"])
|
|
|
|
def _update_children_info(self) -> None:
|
|
"""Update the internal child device info from the parent info."""
|
|
if child_info := self._try_get_response(
|
|
self._last_update, "getChildDeviceList", {}
|
|
):
|
|
for info in child_info["child_device_list"]:
|
|
child_id = info["device_id"]
|
|
if child_id not in self._children:
|
|
_LOGGER.debug(
|
|
"Skipping child update for %s, probably unsupported device",
|
|
child_id,
|
|
)
|
|
continue
|
|
self._children[child_id]._update_internal_state(info)
|
|
|
|
async def _initialize_smart_child(
|
|
self, info: dict, child_components_raw: ComponentsRaw
|
|
) -> SmartDevice:
|
|
"""Initialize a smart child device attached to a smartcam device."""
|
|
child_id = info["device_id"]
|
|
child_protocol = _ChildCameraProtocolWrapper(child_id, self.protocol)
|
|
try:
|
|
initial_response = await child_protocol.query(
|
|
{"get_connect_cloud_state": None}
|
|
)
|
|
except Exception as ex:
|
|
_LOGGER.exception("Error initialising child %s: %s", child_id, ex)
|
|
|
|
return await SmartChildDevice.create(
|
|
parent=self,
|
|
child_info=info,
|
|
child_components_raw=child_components_raw,
|
|
protocol=child_protocol,
|
|
last_update=initial_response,
|
|
)
|
|
|
|
async def _initialize_children(self) -> None:
|
|
"""Initialize children for hubs."""
|
|
child_info_query = {
|
|
"getChildDeviceList": {"childControl": {"start_index": 0}},
|
|
"getChildDeviceComponentList": {"childControl": {"start_index": 0}},
|
|
}
|
|
resp = await self.protocol.query(child_info_query)
|
|
self.internal_state.update(resp)
|
|
|
|
smart_children_components = {
|
|
child["device_id"]: child
|
|
for child in resp["getChildDeviceComponentList"]["child_component_list"]
|
|
}
|
|
children = {}
|
|
for info in resp["getChildDeviceList"]["child_device_list"]:
|
|
if (
|
|
(category := info.get("category"))
|
|
and category in SmartChildDevice.CHILD_DEVICE_TYPE_MAP
|
|
and (child_id := info.get("device_id"))
|
|
and (child_components := smart_children_components.get(child_id))
|
|
):
|
|
children[child_id] = await self._initialize_smart_child(
|
|
info, child_components
|
|
)
|
|
else:
|
|
_LOGGER.debug("Child device type not supported: %s", info)
|
|
|
|
self._children = children
|
|
|
|
async def _initialize_modules(self) -> None:
|
|
"""Initialize modules based on component negotiation response."""
|
|
for mod in SmartCamModule.REGISTERED_MODULES.values():
|
|
if (
|
|
mod.REQUIRED_COMPONENT
|
|
and mod.REQUIRED_COMPONENT not in self._components
|
|
):
|
|
continue
|
|
module = mod(self, mod._module_name())
|
|
if await module._check_supported():
|
|
self._modules[module.name] = module
|
|
|
|
async def _initialize_features(self) -> None:
|
|
"""Initialize device features."""
|
|
for module in self.modules.values():
|
|
module._initialize_features()
|
|
for feat in module._module_features.values():
|
|
self._add_feature(feat)
|
|
|
|
for child in self._children.values():
|
|
await child._initialize_features()
|
|
|
|
async def _query_setter_helper(
|
|
self, method: str, module: str, section: str, params: dict | None = None
|
|
) -> dict:
|
|
res = await self.protocol.query({method: {module: {section: params}}})
|
|
|
|
return res
|
|
|
|
async def _query_getter_helper(
|
|
self, method: str, module: str, sections: str | list[str]
|
|
) -> Any:
|
|
res = await self.protocol.query({method: {module: {"name": sections}}})
|
|
|
|
return res
|
|
|
|
@staticmethod
|
|
def _parse_components(components_raw: ComponentsRaw) -> dict[str, int]:
|
|
return {
|
|
str(comp["name"]): int(comp["version"])
|
|
for comp in components_raw["app_component_list"]
|
|
}
|
|
|
|
async def _negotiate(self) -> None:
|
|
"""Perform initialization.
|
|
|
|
We fetch the device info and the available components as early as possible.
|
|
If the device reports supporting child devices, they are also initialized.
|
|
"""
|
|
initial_query = {
|
|
"getDeviceInfo": {"device_info": {"name": ["basic_info", "info"]}},
|
|
"getAppComponentList": {"app_component": {"name": "app_component_list"}},
|
|
}
|
|
resp = await self.protocol.query(initial_query)
|
|
self._last_update.update(resp)
|
|
self._update_internal_info(resp)
|
|
|
|
self._components_raw = cast(
|
|
ComponentsRaw, resp["getAppComponentList"]["app_component"]
|
|
)
|
|
self._components = self._parse_components(self._components_raw)
|
|
|
|
if "childControl" in self._components and not self.children:
|
|
await self._initialize_children()
|
|
|
|
def _map_info(self, device_info: dict) -> dict:
|
|
basic_info = device_info["basic_info"]
|
|
return {
|
|
"model": basic_info["device_model"],
|
|
"device_type": basic_info["device_type"],
|
|
"alias": basic_info["device_alias"],
|
|
"fw_ver": basic_info["sw_version"],
|
|
"hw_ver": basic_info["hw_version"],
|
|
"mac": basic_info["mac"],
|
|
"hwId": basic_info.get("hw_id"),
|
|
"oem_id": basic_info["oem_id"],
|
|
"device_id": basic_info["dev_id"],
|
|
}
|
|
|
|
@property
|
|
def is_on(self) -> bool:
|
|
"""Return true if the device is on."""
|
|
if (camera := self.modules.get(Module.Camera)) and not camera.disabled:
|
|
return camera.is_on
|
|
|
|
return True
|
|
|
|
async def set_state(self, on: bool) -> dict:
|
|
"""Set the device state."""
|
|
if (camera := self.modules.get(Module.Camera)) and not camera.disabled:
|
|
return await camera.set_state(on)
|
|
|
|
return {}
|
|
|
|
@property
|
|
def device_type(self) -> DeviceType:
|
|
"""Return the device type."""
|
|
if self._device_type == DeviceType.Unknown:
|
|
self._device_type = self._get_device_type_from_sysinfo(self._info)
|
|
return self._device_type
|
|
|
|
@property
|
|
def alias(self) -> str | None:
|
|
"""Returns the device alias or nickname."""
|
|
if self._info:
|
|
return self._info.get("alias")
|
|
return None
|
|
|
|
async def set_alias(self, alias: str) -> dict:
|
|
"""Set the device name (alias)."""
|
|
return await self.protocol.query(
|
|
{
|
|
"setDeviceAlias": {"system": {"sys": {"dev_alias": alias}}},
|
|
}
|
|
)
|
|
|
|
@property
|
|
def hw_info(self) -> dict:
|
|
"""Return hardware info for the device."""
|
|
return {
|
|
"sw_ver": self._info.get("hw_ver"),
|
|
"hw_ver": self._info.get("fw_ver"),
|
|
"mac": self._info.get("mac"),
|
|
"type": self._info.get("type"),
|
|
"hwId": self._info.get("hwId"),
|
|
"dev_name": self.alias,
|
|
"oemId": self._info.get("oem_id"),
|
|
}
|