"""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 Camera, ChildDevice, DeviceModule, Time 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 def _try_add_listen_module(self) -> None: try: import onvif # type: ignore[import-untyped] # noqa: F401 except ImportError: return from .modules.listen import Listen self._modules[Listen._module_name()] = Listen(self, Listen._module_name()) async def _initialize_modules(self) -> None: """Initialize modules based on component negotiation response.""" for mod in SmartCamModule.REGISTERED_MODULES.values(): required_component = cast(str, mod.REQUIRED_COMPONENT) if ( required_component in self._components or any( self.sys_info.get(key) is not None for key in mod.SYSINFO_LOOKUP_KEYS ) or mod in self.FIRST_UPDATE_MODULES or mod is Time ): module = mod(self, mod._module_name()) if await module._check_supported(): self._modules[module.name] = module if self._device_type is DeviceType.Camera: self._modules[Camera._module_name()] = Camera(self, Camera._module_name()) if Module.Motion in self._modules: self._try_add_listen_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("fw_ver"), "hw_ver": self._info.get("hw_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"), }