2021-02-15 19:11:27 +00:00
|
|
|
"""Code to handle a Xiaomi Device."""
|
2021-10-07 16:30:17 +00:00
|
|
|
import datetime
|
|
|
|
from enum import Enum
|
2021-07-28 08:52:43 +00:00
|
|
|
from functools import partial
|
2021-02-15 19:11:27 +00:00
|
|
|
import logging
|
2022-03-21 14:29:11 +00:00
|
|
|
from typing import Any, TypeVar
|
2021-02-15 19:11:27 +00:00
|
|
|
|
2021-06-25 19:25:51 +00:00
|
|
|
from construct.core import ChecksumError
|
2021-02-15 19:11:27 +00:00
|
|
|
from miio import Device, DeviceException
|
|
|
|
|
2022-03-29 09:12:43 +00:00
|
|
|
from homeassistant.const import ATTR_CONNECTIONS, CONF_MODEL
|
2021-02-15 19:11:27 +00:00
|
|
|
from homeassistant.helpers import device_registry as dr
|
2021-10-28 22:37:55 +00:00
|
|
|
from homeassistant.helpers.entity import DeviceInfo, Entity
|
2022-03-21 14:29:11 +00:00
|
|
|
from homeassistant.helpers.update_coordinator import (
|
|
|
|
CoordinatorEntity,
|
|
|
|
DataUpdateCoordinator,
|
|
|
|
)
|
2021-02-15 19:11:27 +00:00
|
|
|
|
2022-03-29 09:12:43 +00:00
|
|
|
from .const import CONF_MAC, DOMAIN, AuthException, SetupException
|
2021-02-15 19:11:27 +00:00
|
|
|
|
|
|
|
_LOGGER = logging.getLogger(__name__)
|
|
|
|
|
2022-03-21 14:29:11 +00:00
|
|
|
_T = TypeVar("_T", bound=DataUpdateCoordinator[Any])
|
|
|
|
|
2021-02-15 19:11:27 +00:00
|
|
|
|
|
|
|
class ConnectXiaomiDevice:
|
|
|
|
"""Class to async connect to a Xiaomi Device."""
|
|
|
|
|
|
|
|
def __init__(self, hass):
|
|
|
|
"""Initialize the entity."""
|
|
|
|
self._hass = hass
|
|
|
|
self._device = None
|
|
|
|
self._device_info = None
|
|
|
|
|
|
|
|
@property
|
|
|
|
def device(self):
|
|
|
|
"""Return the class containing all connections to the device."""
|
|
|
|
return self._device
|
|
|
|
|
|
|
|
@property
|
|
|
|
def device_info(self):
|
|
|
|
"""Return the class containing device info."""
|
|
|
|
return self._device_info
|
|
|
|
|
|
|
|
async def async_connect_device(self, host, token):
|
|
|
|
"""Connect to the Xiaomi Device."""
|
|
|
|
_LOGGER.debug("Initializing with host %s (token %s...)", host, token[:5])
|
2021-06-25 19:25:51 +00:00
|
|
|
|
2021-02-15 19:11:27 +00:00
|
|
|
try:
|
|
|
|
self._device = Device(host, token)
|
|
|
|
# get the device info
|
|
|
|
self._device_info = await self._hass.async_add_executor_job(
|
|
|
|
self._device.info
|
|
|
|
)
|
2021-06-25 19:25:51 +00:00
|
|
|
except DeviceException as error:
|
|
|
|
if isinstance(error.__cause__, ChecksumError):
|
2021-10-14 23:25:44 +00:00
|
|
|
raise AuthException(error) from error
|
2021-06-25 19:25:51 +00:00
|
|
|
|
2021-10-14 23:25:44 +00:00
|
|
|
raise SetupException(
|
|
|
|
f"DeviceException during setup of xiaomi device with host {host}"
|
|
|
|
) from error
|
2021-06-25 19:25:51 +00:00
|
|
|
|
2021-02-15 19:11:27 +00:00
|
|
|
_LOGGER.debug(
|
|
|
|
"%s %s %s detected",
|
|
|
|
self._device_info.model,
|
|
|
|
self._device_info.firmware_version,
|
|
|
|
self._device_info.hardware_version,
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
|
|
class XiaomiMiioEntity(Entity):
|
|
|
|
"""Representation of a base Xiaomi Miio Entity."""
|
|
|
|
|
|
|
|
def __init__(self, name, device, entry, unique_id):
|
|
|
|
"""Initialize the Xiaomi Miio Device."""
|
|
|
|
self._device = device
|
|
|
|
self._model = entry.data[CONF_MODEL]
|
|
|
|
self._mac = entry.data[CONF_MAC]
|
|
|
|
self._device_id = entry.unique_id
|
|
|
|
self._unique_id = unique_id
|
|
|
|
self._name = name
|
2021-07-28 08:52:43 +00:00
|
|
|
self._available = None
|
2021-02-15 19:11:27 +00:00
|
|
|
|
|
|
|
@property
|
|
|
|
def unique_id(self):
|
|
|
|
"""Return an unique ID."""
|
|
|
|
return self._unique_id
|
|
|
|
|
|
|
|
@property
|
|
|
|
def name(self):
|
|
|
|
"""Return the name of this entity, if any."""
|
|
|
|
return self._name
|
|
|
|
|
|
|
|
@property
|
2021-10-28 22:37:55 +00:00
|
|
|
def device_info(self) -> DeviceInfo:
|
2021-02-15 19:11:27 +00:00
|
|
|
"""Return the device info."""
|
2021-10-28 22:37:55 +00:00
|
|
|
device_info = DeviceInfo(
|
|
|
|
identifiers={(DOMAIN, self._device_id)},
|
|
|
|
manufacturer="Xiaomi",
|
|
|
|
model=self._model,
|
|
|
|
name=self._name,
|
|
|
|
)
|
2021-02-22 12:01:02 +00:00
|
|
|
|
|
|
|
if self._mac is not None:
|
2021-10-28 22:37:55 +00:00
|
|
|
device_info[ATTR_CONNECTIONS] = {(dr.CONNECTION_NETWORK_MAC, self._mac)}
|
2021-02-22 12:01:02 +00:00
|
|
|
|
|
|
|
return device_info
|
2021-07-28 08:52:43 +00:00
|
|
|
|
|
|
|
|
2022-03-21 14:29:11 +00:00
|
|
|
class XiaomiCoordinatedMiioEntity(CoordinatorEntity[_T]):
|
2021-07-28 08:52:43 +00:00
|
|
|
"""Representation of a base a coordinated Xiaomi Miio Entity."""
|
|
|
|
|
2022-07-17 13:15:24 +00:00
|
|
|
_attr_has_entity_name = True
|
|
|
|
|
|
|
|
def __init__(self, device, entry, unique_id, coordinator):
|
2021-07-28 08:52:43 +00:00
|
|
|
"""Initialize the coordinated Xiaomi Miio Device."""
|
|
|
|
super().__init__(coordinator)
|
|
|
|
self._device = device
|
|
|
|
self._model = entry.data[CONF_MODEL]
|
|
|
|
self._mac = entry.data[CONF_MAC]
|
|
|
|
self._device_id = entry.unique_id
|
|
|
|
self._device_name = entry.title
|
|
|
|
self._unique_id = unique_id
|
|
|
|
|
|
|
|
@property
|
|
|
|
def unique_id(self):
|
|
|
|
"""Return an unique ID."""
|
|
|
|
return self._unique_id
|
|
|
|
|
|
|
|
@property
|
2021-10-28 22:37:55 +00:00
|
|
|
def device_info(self) -> DeviceInfo:
|
2021-07-28 08:52:43 +00:00
|
|
|
"""Return the device info."""
|
2021-10-28 22:37:55 +00:00
|
|
|
device_info = DeviceInfo(
|
|
|
|
identifiers={(DOMAIN, self._device_id)},
|
|
|
|
manufacturer="Xiaomi",
|
|
|
|
model=self._model,
|
|
|
|
name=self._device_name,
|
|
|
|
)
|
2021-07-28 08:52:43 +00:00
|
|
|
|
|
|
|
if self._mac is not None:
|
2021-10-28 22:37:55 +00:00
|
|
|
device_info[ATTR_CONNECTIONS] = {(dr.CONNECTION_NETWORK_MAC, self._mac)}
|
2021-07-28 08:52:43 +00:00
|
|
|
|
|
|
|
return device_info
|
|
|
|
|
|
|
|
async def _try_command(self, mask_error, func, *args, **kwargs):
|
|
|
|
"""Call a miio device command handling error messages."""
|
|
|
|
try:
|
|
|
|
result = await self.hass.async_add_executor_job(
|
|
|
|
partial(func, *args, **kwargs)
|
|
|
|
)
|
|
|
|
|
|
|
|
_LOGGER.debug("Response received from miio device: %s", result)
|
|
|
|
|
|
|
|
return True
|
|
|
|
except DeviceException as exc:
|
|
|
|
if self.available:
|
|
|
|
_LOGGER.error(mask_error, exc)
|
|
|
|
|
|
|
|
return False
|
2021-10-07 16:30:17 +00:00
|
|
|
|
|
|
|
@classmethod
|
|
|
|
def _extract_value_from_attribute(cls, state, attribute):
|
|
|
|
value = getattr(state, attribute)
|
|
|
|
if isinstance(value, Enum):
|
|
|
|
return value.value
|
|
|
|
if isinstance(value, datetime.timedelta):
|
|
|
|
return cls._parse_time_delta(value)
|
|
|
|
if isinstance(value, datetime.time):
|
|
|
|
return cls._parse_datetime_time(value)
|
|
|
|
if isinstance(value, datetime.datetime):
|
|
|
|
return cls._parse_datetime_datetime(value)
|
2021-11-03 16:28:11 +00:00
|
|
|
|
2021-11-01 16:40:15 +00:00
|
|
|
if value is None:
|
|
|
|
_LOGGER.debug("Attribute %s is None, this is unexpected", attribute)
|
2021-10-07 16:30:17 +00:00
|
|
|
|
|
|
|
return value
|
|
|
|
|
|
|
|
@staticmethod
|
|
|
|
def _parse_time_delta(timedelta: datetime.timedelta) -> int:
|
2021-11-03 16:28:11 +00:00
|
|
|
return int(timedelta.total_seconds())
|
2021-10-07 16:30:17 +00:00
|
|
|
|
|
|
|
@staticmethod
|
2022-07-09 17:59:11 +00:00
|
|
|
def _parse_datetime_time(initial_time: datetime.time) -> str:
|
2021-10-07 16:30:17 +00:00
|
|
|
time = datetime.datetime.now().replace(
|
2022-07-09 17:59:11 +00:00
|
|
|
hour=initial_time.hour, minute=initial_time.minute, second=0, microsecond=0
|
2021-10-07 16:30:17 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
if time < datetime.datetime.now():
|
|
|
|
time += datetime.timedelta(days=1)
|
|
|
|
|
|
|
|
return time.isoformat()
|
|
|
|
|
|
|
|
@staticmethod
|
|
|
|
def _parse_datetime_datetime(time: datetime.datetime) -> str:
|
|
|
|
return time.isoformat()
|