From 4bd1c0eb28475f484870797fe7ae8b2ea1f4d4e7 Mon Sep 17 00:00:00 2001 From: Alberto Geniola Date: Tue, 10 Aug 2021 15:18:28 +0200 Subject: [PATCH] Increased default timeout to 10 seconds (also for subdevices) Improved command timeout error logging --- meross_iot/controller/device.py | 4 ++-- meross_iot/controller/known/subdevice.py | 4 ++-- meross_iot/manager.py | 18 ++++++++++++++---- meross_iot/model/exception.py | 5 ++++- 4 files changed, 22 insertions(+), 9 deletions(-) diff --git a/meross_iot/controller/device.py b/meross_iot/controller/device.py index 8222e287..3683a26d 100644 --- a/meross_iot/controller/device.py +++ b/meross_iot/controller/device.py @@ -252,7 +252,7 @@ async def _execute_command(self, method: str, namespace: Namespace, payload: dict, - timeout: float = 5, + timeout: float = 10, skip_rate_limits: bool = False, drop_on_overquota: bool = True ) -> dict: @@ -344,7 +344,7 @@ async def _execute_command(self, method: str, namespace: Namespace, payload: dict, - timeout: float = 5, + timeout: float = 10, skip_rate_limits: bool = False, drop_on_overquota: bool = True ) -> dict: diff --git a/meross_iot/controller/known/subdevice.py b/meross_iot/controller/known/subdevice.py index 3b7a2530..c133529a 100644 --- a/meross_iot/controller/known/subdevice.py +++ b/meross_iot/controller/known/subdevice.py @@ -26,7 +26,7 @@ async def _execute_command(self, method: str, namespace: Namespace, payload: dict, - timeout: float = 5, + timeout: float = 10, skip_rate_limits: bool = False, drop_on_overquota: bool = True) -> dict: raise NotImplementedError("This method should never be called directly for subdevices.") @@ -151,7 +151,7 @@ async def _execute_command(self, method: str, namespace: Namespace, payload: dict, - timeout: float = 5, + timeout: float = 10, skip_rate_limits: bool = False, drop_on_overquota: bool = True) -> dict: raise NotImplementedError("This method should never be called directly for subdevices.") diff --git a/meross_iot/manager.py b/meross_iot/manager.py index 8d349d20..c9617535 100644 --- a/meross_iot/manager.py +++ b/meross_iot/manager.py @@ -129,6 +129,13 @@ def __init__( self._user_topic = build_client_user_topic(user_id=self._cloud_creds.user_id) self._limiter = rate_limiter + def _get_client_from_domain_port(self, client: mqtt.Client) -> Tuple[Optional[str], Optional[int]]: + for k, v in self._mqtt_clients.items(): + if v == client: + domain, port = k.split(":") + return domain, int(port) + return None, None + async def _async_get_create_mqtt_client(self, domain: str, port: int) -> mqtt.Client: """ Retrieves the mqtt_client for the given domain/port combination. @@ -819,6 +826,8 @@ async def async_execute_cmd_client(self, namespace=namespace, payload=payload, timeout=timeout, + skip_rate_limiting_check=skip_rate_limiting_check, + drop_on_overquota=drop_on_overquota ) elif rate_limiting_action == RateLimitResultStrategy.DropCall: raise RateLimitExceeded() @@ -838,12 +847,12 @@ async def async_execute_cmd_client(self, future=fut, target_device_uuid=destination_device_uuid, message=message, - timeout=timeout, + timeout=timeout ) return response.get("payload") async def _async_send_and_wait_ack( - self, client: mqtt.Client, future: Future, target_device_uuid: str, message: bytes, timeout: float + self, client: mqtt.Client, future: Future, target_device_uuid: str, message: bytes, timeout: float, ): message_info = client.publish( topic=build_device_request_topic(target_device_uuid), payload=message @@ -851,11 +860,12 @@ async def _async_send_and_wait_ack( try: return await asyncio.wait_for(future, timeout, loop=self._loop) except TimeoutError as e: + domain, port = self._get_client_from_domain_port(client=client) _LOGGER.error( f"Timeout occurred while waiting a response for message {message} sent to device uuid " - f"{target_device_uuid}. Timeout was: {timeout} seconds." + f"{target_device_uuid}. Timeout was: {timeout} seconds. Mqtt Host: {domain}:{port}." ) - raise CommandTimeoutError() + raise CommandTimeoutError(message=str(message), target_device_uuid=target_device_uuid, timeout=timeout) async def _notify_connection_drop(self): for d in self._device_registry.find_all_by(): diff --git a/meross_iot/model/exception.py b/meross_iot/model/exception.py index 80a2c22b..a2c5f1ac 100644 --- a/meross_iot/model/exception.py +++ b/meross_iot/model/exception.py @@ -3,7 +3,10 @@ class UnconnectedError(Exception): class CommandTimeoutError(Exception): - pass + def __init__(self, message: str, target_device_uuid: str, timeout: float): + self.message = message + self.target_device_uuid = target_device_uuid + self.timeout=timeout class MqttError(Exception):