mirror of
https://github.com/python-kasa/python-kasa.git
synced 2025-01-09 06:17:08 +00:00
acd0202cab
Introduces the child camera protocol wrapper, required to get the child device info with the new protocol.
182 lines
6.1 KiB
Python
182 lines
6.1 KiB
Python
"""Module for SmartCamera Protocol."""
|
|
|
|
from __future__ import annotations
|
|
|
|
import logging
|
|
from pprint import pformat as pf
|
|
from typing import Any
|
|
|
|
from ..exceptions import (
|
|
AuthenticationError,
|
|
DeviceError,
|
|
KasaException,
|
|
_RetryableError,
|
|
)
|
|
from ..json import dumps as json_dumps
|
|
from ..smartprotocol import SmartProtocol
|
|
from .sslaestransport import (
|
|
SMART_AUTHENTICATION_ERRORS,
|
|
SMART_RETRYABLE_ERRORS,
|
|
SmartErrorCode,
|
|
)
|
|
|
|
_LOGGER = logging.getLogger(__name__)
|
|
|
|
|
|
class SmartCameraProtocol(SmartProtocol):
|
|
"""Class for SmartCamera Protocol."""
|
|
|
|
async def _handle_response_lists(
|
|
self, response_result: dict[str, Any], method, retry_count
|
|
):
|
|
pass
|
|
|
|
def _handle_response_error_code(self, resp_dict: dict, method, raise_on_error=True):
|
|
error_code_raw = resp_dict.get("error_code")
|
|
try:
|
|
error_code = SmartErrorCode.from_int(error_code_raw)
|
|
except ValueError:
|
|
_LOGGER.warning(
|
|
"Device %s received unknown error code: %s", self._host, error_code_raw
|
|
)
|
|
error_code = SmartErrorCode.INTERNAL_UNKNOWN_ERROR
|
|
|
|
if error_code is SmartErrorCode.SUCCESS:
|
|
return
|
|
|
|
if not raise_on_error:
|
|
resp_dict["result"] = error_code
|
|
return
|
|
|
|
msg = (
|
|
f"Error querying device: {self._host}: "
|
|
+ f"{error_code.name}({error_code.value})"
|
|
+ f" for method: {method}"
|
|
)
|
|
if error_code in SMART_RETRYABLE_ERRORS:
|
|
raise _RetryableError(msg, error_code=error_code)
|
|
if error_code in SMART_AUTHENTICATION_ERRORS:
|
|
raise AuthenticationError(msg, error_code=error_code)
|
|
raise DeviceError(msg, error_code=error_code)
|
|
|
|
async def close(self) -> None:
|
|
"""Close the underlying transport."""
|
|
await self._transport.close()
|
|
|
|
async def _execute_query(
|
|
self, request: str | dict, *, retry_count: int, iterate_list_pages: bool = True
|
|
) -> dict:
|
|
debug_enabled = _LOGGER.isEnabledFor(logging.DEBUG)
|
|
|
|
if isinstance(request, dict):
|
|
if len(request) == 1:
|
|
method = next(iter(request))
|
|
if method == "multipleRequest":
|
|
params = request["multipleRequest"]
|
|
req = {"method": "multipleRequest", "params": params}
|
|
elif method[:3] == "set":
|
|
params = next(iter(request[method]))
|
|
req = {
|
|
"method": method[:3],
|
|
params: request[method][params],
|
|
}
|
|
else:
|
|
return await self._execute_multiple_query(request, retry_count)
|
|
else:
|
|
return await self._execute_multiple_query(request, retry_count)
|
|
else:
|
|
# If method like getSomeThing then module will be some_thing
|
|
method = request
|
|
snake_name = "".join(
|
|
["_" + i.lower() if i.isupper() else i for i in method]
|
|
).lstrip("_")
|
|
params = snake_name[4:]
|
|
req = {"method": snake_name[:3], params: {}}
|
|
|
|
smart_request = json_dumps(req)
|
|
if debug_enabled:
|
|
_LOGGER.debug(
|
|
"%s >> %s",
|
|
self._host,
|
|
pf(smart_request),
|
|
)
|
|
response_data = await self._transport.send(smart_request)
|
|
|
|
if debug_enabled:
|
|
_LOGGER.debug(
|
|
"%s << %s",
|
|
self._host,
|
|
pf(response_data),
|
|
)
|
|
|
|
if "error_code" in response_data:
|
|
# H200 does not return an error code
|
|
self._handle_response_error_code(response_data, method)
|
|
|
|
# TODO need to update handle response lists
|
|
|
|
if method[:3] == "set":
|
|
return {}
|
|
if method == "multipleRequest":
|
|
return {method: response_data["result"]}
|
|
return {method: {params: response_data[params]}}
|
|
|
|
|
|
class _ChildCameraProtocolWrapper(SmartProtocol):
|
|
"""Protocol wrapper for controlling child devices.
|
|
|
|
This is an internal class used to communicate with child devices,
|
|
and should not be used directly.
|
|
|
|
This class overrides query() method of the protocol to modify all
|
|
outgoing queries to use ``controlChild`` command, and unwraps the
|
|
device responses before returning to the caller.
|
|
"""
|
|
|
|
def __init__(self, device_id: str, base_protocol: SmartProtocol):
|
|
self._device_id = device_id
|
|
self._protocol = base_protocol
|
|
self._transport = base_protocol._transport
|
|
|
|
async def query(self, request: str | dict, retry_count: int = 3) -> dict:
|
|
"""Wrap request inside controlChild envelope."""
|
|
return await self._query(request, retry_count)
|
|
|
|
async def _query(self, request: str | dict, retry_count: int = 3) -> dict:
|
|
"""Wrap request inside controlChild envelope."""
|
|
if not isinstance(request, dict):
|
|
raise KasaException("Child requests must be dictionaries.")
|
|
requests = []
|
|
methods = []
|
|
for key, val in request.items():
|
|
request = {
|
|
"method": "controlChild",
|
|
"params": {
|
|
"childControl": {
|
|
"device_id": self._device_id,
|
|
"request_data": {"method": key, "params": val},
|
|
}
|
|
},
|
|
}
|
|
methods.append(key)
|
|
requests.append(request)
|
|
|
|
multipleRequest = {"multipleRequest": {"requests": requests}}
|
|
|
|
response = await self._protocol.query(multipleRequest, retry_count)
|
|
|
|
responses = response["multipleRequest"]["responses"]
|
|
response_dict = {}
|
|
for index_id, response in enumerate(responses):
|
|
response_data = response["result"]["response_data"]
|
|
method = methods[index_id]
|
|
self._handle_response_error_code(
|
|
response_data, method, raise_on_error=False
|
|
)
|
|
response_dict[method] = response_data.get("result")
|
|
|
|
return response_dict
|
|
|
|
async def close(self) -> None:
|
|
"""Do nothing as the parent owns the protocol."""
|