Skip to content

Commit

Permalink
fix: fix type error, wrong use of any and Any (XiaoMi#338)
Browse files Browse the repository at this point in the history
* fix: fix type error, wrong use of any and Any

* fix: wrong use of session close

* fix: fix test_lan type error

* fix: remove __del__

* feat: oauth, http add deinit_async
  • Loading branch information
topsworld authored Dec 22, 2024
1 parent afef709 commit c1867e2
Show file tree
Hide file tree
Showing 18 changed files with 237 additions and 173 deletions.
4 changes: 2 additions & 2 deletions custom_components/xiaomi_home/climate.py
Original file line number Diff line number Diff line change
Expand Up @@ -47,7 +47,7 @@
"""
from __future__ import annotations
import logging
from typing import Optional
from typing import Any, Optional

from homeassistant.config_entries import ConfigEntry
from homeassistant.core import HomeAssistant
Expand Down Expand Up @@ -415,7 +415,7 @@ def swing_mode(self) -> Optional[str]:
return SWING_OFF
return None

def __ac_state_changed(self, prop: MIoTSpecProperty, value: any) -> None:
def __ac_state_changed(self, prop: MIoTSpecProperty, value: Any) -> None:
del prop
if not isinstance(value, str):
_LOGGER.error(
Expand Down
20 changes: 16 additions & 4 deletions custom_components/xiaomi_home/config_flow.py
Original file line number Diff line number Diff line change
Expand Up @@ -315,6 +315,9 @@ async def async_step_oauth(self, user_input=None):
_LOGGER.error('task_oauth exception, %s', error)
self._config_error_reason = str(error)
return self.async_show_progress_done(next_step_id='oauth_error')
if self._miot_oauth:
await self._miot_oauth.deinit_async()
self._miot_oauth = None
return self.async_show_progress_done(next_step_id='homes_select')
return self.async_show_progress(
step_id='oauth',
Expand All @@ -336,10 +339,16 @@ async def __check_oauth_async(self) -> None:
try:
auth_info = await self._miot_oauth.get_access_token_async(
code=oauth_code)
self._miot_http = MIoTHttpClient(
cloud_server=self._cloud_server,
client_id=OAUTH2_CLIENT_ID,
access_token=auth_info['access_token'])
if not self._miot_http:
self._miot_http = MIoTHttpClient(
cloud_server=self._cloud_server,
client_id=OAUTH2_CLIENT_ID,
access_token=auth_info['access_token'])
else:
self._miot_http.update_http_header(
cloud_server=self._cloud_server,
client_id=OAUTH2_CLIENT_ID,
access_token=auth_info['access_token'])
self._auth_info = auth_info
# Gen uuid
self._uuid = hashlib.sha256(
Expand Down Expand Up @@ -449,6 +458,9 @@ async def __check_oauth_async(self) -> None:

# Auth success, unregister oauth webhook
webhook_async_unregister(self.hass, webhook_id=self._virtual_did)
if self._miot_http:
await self._miot_http.deinit_async()
self._miot_http = None
_LOGGER.info(
'__check_oauth_async, webhook.async_unregister: %s',
self._virtual_did)
Expand Down
3 changes: 2 additions & 1 deletion custom_components/xiaomi_home/event.py
Original file line number Diff line number Diff line change
Expand Up @@ -46,6 +46,7 @@
Event entities for Xiaomi Home.
"""
from __future__ import annotations
from typing import Any

from homeassistant.config_entries import ConfigEntry
from homeassistant.core import HomeAssistant
Expand Down Expand Up @@ -84,6 +85,6 @@ def __init__(self, miot_device: MIoTDevice, spec: MIoTSpecEvent) -> None:
# Set device_class
self._attr_device_class = spec.device_class

def on_event_occurred(self, name: str, arguments: list[dict[int, any]]):
def on_event_occurred(self, name: str, arguments: list[dict[int, Any]]):
"""An event is occurred."""
self._trigger_event(event_type=name, event_attributes=arguments)
2 changes: 1 addition & 1 deletion custom_components/xiaomi_home/fan.py
Original file line number Diff line number Diff line change
Expand Up @@ -93,7 +93,7 @@ class Fan(MIoTServiceEntity, FanEntity):
_speed_min: Optional[int]
_speed_max: Optional[int]
_speed_step: Optional[int]
_mode_list: Optional[dict[any, any]]
_mode_list: Optional[dict[Any, Any]]

def __init__(
self, miot_device: MIoTDevice, entity_data: MIoTEntityData
Expand Down
4 changes: 2 additions & 2 deletions custom_components/xiaomi_home/humidifier.py
Original file line number Diff line number Diff line change
Expand Up @@ -47,7 +47,7 @@
"""
from __future__ import annotations
import logging
from typing import Optional
from typing import Any, Optional

from homeassistant.config_entries import ConfigEntry
from homeassistant.core import HomeAssistant
Expand Down Expand Up @@ -97,7 +97,7 @@ class Humidifier(MIoTServiceEntity, HumidifierEntity):
_prop_target_humidity: Optional[MIoTSpecProperty]
_prop_humidity: Optional[MIoTSpecProperty]

_mode_list: dict[any, any]
_mode_list: dict[Any, Any]

def __init__(
self, miot_device: MIoTDevice, entity_data: MIoTEntityData
Expand Down
4 changes: 2 additions & 2 deletions custom_components/xiaomi_home/light.py
Original file line number Diff line number Diff line change
Expand Up @@ -47,7 +47,7 @@
"""
from __future__ import annotations
import logging
from typing import Optional
from typing import Any, Optional

from homeassistant.config_entries import ConfigEntry
from homeassistant.core import HomeAssistant
Expand Down Expand Up @@ -102,7 +102,7 @@ class Light(MIoTServiceEntity, LightEntity):
_prop_mode: Optional[MIoTSpecProperty]

_brightness_scale: Optional[tuple[int, int]]
_mode_list: Optional[dict[any, any]]
_mode_list: Optional[dict[Any, Any]]

def __init__(
self, miot_device: MIoTDevice, entity_data: MIoTEntityData
Expand Down
8 changes: 4 additions & 4 deletions custom_components/xiaomi_home/miot/common.py
Original file line number Diff line number Diff line change
Expand Up @@ -48,9 +48,9 @@
import json
from os import path
import random
from typing import Optional
from typing import Any, Optional
import hashlib
from paho.mqtt.client import MQTTMatcher
from paho.mqtt.matcher import MQTTMatcher
import yaml

MIOT_ROOT_PATH: str = path.dirname(path.abspath(__file__))
Expand Down Expand Up @@ -87,7 +87,7 @@ def randomize_int(value: int, ratio: float) -> int:
class MIoTMatcher(MQTTMatcher):
"""MIoT Pub/Sub topic matcher."""

def iter_all_nodes(self) -> any:
def iter_all_nodes(self) -> Any:
"""Return an iterator on all nodes with their paths and contents."""
def rec(node, path_):
# pylint: disable=protected-access
Expand All @@ -97,7 +97,7 @@ def rec(node, path_):
yield from rec(child, path_ + [part])
return rec(self._root, [])

def get(self, topic: str) -> Optional[any]:
def get(self, topic: str) -> Optional[Any]:
try:
return self[topic]
except KeyError:
Expand Down
81 changes: 64 additions & 17 deletions custom_components/xiaomi_home/miot/miot_client.py
Original file line number Diff line number Diff line change
@@ -1,7 +1,52 @@
# -*- coding: utf-8 -*-
"""MIoT client instance."""
"""
Copyright (C) 2024 Xiaomi Corporation.
The ownership and intellectual property rights of Xiaomi Home Assistant
Integration and related Xiaomi cloud service API interface provided under this
license, including source code and object code (collectively, "Licensed Work"),
are owned by Xiaomi. Subject to the terms and conditions of this License, Xiaomi
hereby grants you a personal, limited, non-exclusive, non-transferable,
non-sublicensable, and royalty-free license to reproduce, use, modify, and
distribute the Licensed Work only for your use of Home Assistant for
non-commercial purposes. For the avoidance of doubt, Xiaomi does not authorize
you to use the Licensed Work for any other purpose, including but not limited
to use Licensed Work to develop applications (APP), Web services, and other
forms of software.
You may reproduce and distribute copies of the Licensed Work, with or without
modifications, whether in source or object form, provided that you must give
any other recipients of the Licensed Work a copy of this License and retain all
copyright and disclaimers.
Xiaomi provides the Licensed Work on an "AS IS" BASIS, WITHOUT WARRANTIES OR
CONDITIONS OF ANY KIND, either express or implied, including, without
limitation, any warranties, undertakes, or conditions of TITLE, NO ERROR OR
OMISSION, CONTINUITY, RELIABILITY, NON-INFRINGEMENT, MERCHANTABILITY, or
FITNESS FOR A PARTICULAR PURPOSE. In any event, you are solely responsible
for any direct, indirect, special, incidental, or consequential damages or
losses arising from the use or inability to use the Licensed Work.
Xiaomi reserves all rights not expressly granted to you in this License.
Except for the rights expressly granted by Xiaomi under this License, Xiaomi
does not authorize you in any form to use the trademarks, copyrights, or other
forms of intellectual property rights of Xiaomi and its affiliates, including,
without limitation, without obtaining other written permission from Xiaomi, you
shall not use "Xiaomi", "Mijia" and other words related to Xiaomi or words that
may make the public associate with Xiaomi in any form to publicize or promote
the software or hardware devices that use the Licensed Work.
Xiaomi has the right to immediately terminate all your authorization under this
License in the event:
1. You assert patent invalidation, litigation, or other claims against patents
or other intellectual property rights of Xiaomi or its affiliates; or,
2. You make, have made, manufacture, sell, or offer to sell products that knock
off Xiaomi or its affiliates' products.
MIoT client instance.
"""
from copy import deepcopy
from typing import Callable, Optional, final
from typing import Any, Callable, Optional, final
import asyncio
import json
import logging
Expand Down Expand Up @@ -36,9 +81,9 @@
@dataclass
class MIoTClientSub:
"""MIoT client subscription."""
topic: str = None
handler: Callable[[dict, any], None] = None
handler_ctx: any = None
topic: Optional[str]
handler: Callable[[dict, Any], None]
handler_ctx: Any = None

def __str__(self) -> str:
return f'{self.topic}, {id(self.handler)}, {id(self.handler_ctx)}'
Expand Down Expand Up @@ -345,6 +390,8 @@ async def deinit_async(self) -> None:
if self._show_devices_changed_notify_timer:
self._show_devices_changed_notify_timer.cancel()
self._show_devices_changed_notify_timer = None
await self._oauth.deinit_async()
await self._http.deinit_async()
# Remove notify
self._persistence_notify(
self.__gen_notify_key('dev_list_changed'), None, None)
Expand Down Expand Up @@ -526,7 +573,7 @@ async def refresh_user_cert_async(self) -> bool:
return False

async def set_prop_async(
self, did: str, siid: int, piid: int, value: any
self, did: str, siid: int, piid: int, value: Any
) -> bool:
if did not in self._device_list_cache:
raise MIoTClientError(f'did not exist, {did}')
Expand Down Expand Up @@ -612,7 +659,7 @@ def request_refresh_prop(
0.2, lambda: self._main_loop.create_task(
self.__refresh_props_handler()))

async def get_prop_async(self, did: str, siid: int, piid: int) -> any:
async def get_prop_async(self, did: str, siid: int, piid: int) -> Any:
if did not in self._device_list_cache:
raise MIoTClientError(f'did not exist, {did}')

Expand Down Expand Up @@ -717,8 +764,8 @@ async def action_async(
return None

def sub_prop(
self, did: str, handler: Callable[[dict, any], None],
siid: int = None, piid: int = None, handler_ctx: any = None
self, did: str, handler: Callable[[dict, Any], None],
siid: int = None, piid: int = None, handler_ctx: Any = None
) -> bool:
if did not in self._device_list_cache:
raise MIoTClientError(f'did not exist, {did}')
Expand All @@ -741,8 +788,8 @@ def unsub_prop(self, did: str, siid: int = None, piid: int = None) -> bool:
return True

def sub_event(
self, did: str, handler: Callable[[dict, any], None],
siid: int = None, eiid: int = None, handler_ctx: any = None
self, did: str, handler: Callable[[dict, Any], None],
siid: int = None, eiid: int = None, handler_ctx: Any = None
) -> bool:
if did not in self._device_list_cache:
raise MIoTClientError(f'did not exist, {did}')
Expand All @@ -764,8 +811,8 @@ def unsub_event(self, did: str, siid: int = None, eiid: int = None) -> bool:
return True

def sub_device_state(
self, did: str, handler: Callable[[str, MIoTDeviceState, any], None],
handler_ctx: any = None
self, did: str, handler: Callable[[str, MIoTDeviceState, Any], None],
handler_ctx: Any = None
) -> bool:
"""Call callback handler in main loop"""
if did not in self._device_list_cache:
Expand Down Expand Up @@ -1060,7 +1107,7 @@ async def __on_miot_lan_state_change(self, state: bool) -> None:

@final
def __on_cloud_device_state_changed(
self, did: str, state: MIoTDeviceState, ctx: any
self, did: str, state: MIoTDeviceState, ctx: Any
) -> None:
_LOGGER.info('cloud device state changed, %s, %s', did, state)
cloud_device = self._device_list_cloud.get(did, None)
Expand Down Expand Up @@ -1110,7 +1157,7 @@ async def __on_gw_device_list_changed(

@final
async def __on_lan_device_state_changed(
self, did: str, state: dict, ctx: any
self, did: str, state: dict, ctx: Any
) -> None:
_LOGGER.info('lan device state changed, %s, %s', did, state)
lan_state_new: bool = state.get('online', False)
Expand Down Expand Up @@ -1146,7 +1193,7 @@ async def __on_lan_device_state_changed(
self.__request_show_devices_changed_notify()

@final
def __on_prop_msg(self, params: dict, ctx: any) -> None:
def __on_prop_msg(self, params: dict, ctx: Any) -> None:
"""params MUST contain did, siid, piid, value"""
# BLE device has no online/offline msg
try:
Expand All @@ -1158,7 +1205,7 @@ def __on_prop_msg(self, params: dict, ctx: any) -> None:
_LOGGER.error('on prop msg error, %s, %s', params, err)

@final
def __on_event_msg(self, params: dict, ctx: any) -> None:
def __on_event_msg(self, params: dict, ctx: Any) -> None:
try:
subs: list[MIoTClientSub] = list(self._sub_tree.iter_match(
f'{params["did"]}/e/{params["siid"]}/{params["eiid"]}'))
Expand Down
22 changes: 12 additions & 10 deletions custom_components/xiaomi_home/miot/miot_cloud.py
Original file line number Diff line number Diff line change
Expand Up @@ -51,7 +51,7 @@
import logging
import re
import time
from typing import Optional
from typing import Any, Optional
from urllib.parse import urlencode
import aiohttp

Expand Down Expand Up @@ -94,10 +94,11 @@ def __init__(
self._oauth_host = DEFAULT_OAUTH2_API_HOST
else:
self._oauth_host = f'{cloud_server}.{DEFAULT_OAUTH2_API_HOST}'
self._session = aiohttp.ClientSession()
self._session = aiohttp.ClientSession(loop=self._main_loop)

def __del__(self):
self._session.close()
async def deinit_async(self) -> None:
if self._session and not self._session.closed:
await self._session.close()

def set_redirect_url(self, redirect_url: str) -> None:
if not isinstance(redirect_url, str) or redirect_url.strip() == '':
Expand Down Expand Up @@ -250,10 +251,11 @@ def __init__(
cloud_server=cloud_server, client_id=client_id,
access_token=access_token)

self._session = aiohttp.ClientSession()
self._session = aiohttp.ClientSession(loop=self._main_loop)

def __del__(self):
self._session.close()
async def deinit_async(self) -> None:
if self._session and not self._session.closed:
await self._session.close()

def update_http_header(
self, cloud_server: Optional[str] = None,
Expand Down Expand Up @@ -581,7 +583,7 @@ async def get_devices_async(
self, home_ids: list[str] = None
) -> dict[str, dict]:
homeinfos = await self.get_homeinfos_async()
homes: dict[str, dict[str, any]] = {}
homes: dict[str, dict[str, Any]] = {}
devices: dict[str, dict] = {}
for device_type in ['home_list', 'share_home_list']:
homes.setdefault(device_type, {})
Expand Down Expand Up @@ -661,7 +663,7 @@ async def get_props_async(self, params: list) -> list:
raise MIoTHttpError('invalid response result')
return res_obj['result']

async def __get_prop_async(self, did: str, siid: int, piid: int) -> any:
async def __get_prop_async(self, did: str, siid: int, piid: int) -> Any:
results = await self.get_props_async(
params=[{'did': did, 'siid': siid, 'piid': piid}])
if not results:
Expand Down Expand Up @@ -722,7 +724,7 @@ async def __get_prop_handler(self) -> bool:

async def get_prop_async(
self, did: str, siid: int, piid: int, immediately: bool = False
) -> any:
) -> Any:
if immediately:
return await self.__get_prop_async(did, siid, piid)
key: str = f'{did}.{siid}.{piid}'
Expand Down
Loading

0 comments on commit c1867e2

Please sign in to comment.