core/homeassistant/components/livisi/coordinator.py

136 lines
5.0 KiB
Python

"""Code to manage fetching LIVISI data API."""
from __future__ import annotations
from datetime import timedelta
from typing import Any
from aiohttp import ClientConnectorError
from aiolivisi import AioLivisi, LivisiEvent, Websocket
from aiolivisi.errors import TokenExpiredException
from homeassistant.config_entries import ConfigEntry
from homeassistant.core import HomeAssistant
from homeassistant.helpers.dispatcher import async_dispatcher_send
from homeassistant.helpers.update_coordinator import DataUpdateCoordinator, UpdateFailed
from .const import (
AVATAR,
AVATAR_PORT,
CLASSIC_PORT,
CONF_HOST,
CONF_PASSWORD,
DEVICE_POLLING_DELAY,
LIVISI_REACHABILITY_CHANGE,
LIVISI_STATE_CHANGE,
LOGGER,
)
class LivisiDataUpdateCoordinator(DataUpdateCoordinator[list[dict[str, Any]]]):
"""Class to manage fetching LIVISI data API."""
config_entry: ConfigEntry
def __init__(
self, hass: HomeAssistant, config_entry: ConfigEntry, aiolivisi: AioLivisi
) -> None:
"""Initialize my coordinator."""
super().__init__(
hass,
LOGGER,
name="Livisi devices",
update_interval=timedelta(seconds=DEVICE_POLLING_DELAY),
)
self.config_entry = config_entry
self.hass = hass
self.aiolivisi = aiolivisi
self.websocket = Websocket(aiolivisi)
self.devices: set[str] = set()
self.rooms: dict[str, Any] = {}
self.serial_number: str = ""
self.controller_type: str = ""
self.is_avatar: bool = False
self.port: int = 0
async def _async_update_data(self) -> list[dict[str, Any]]:
"""Get device configuration from LIVISI."""
try:
return await self.async_get_devices()
except TokenExpiredException:
await self.aiolivisi.async_set_token(self.aiolivisi.livisi_connection_data)
return await self.async_get_devices()
except ClientConnectorError as exc:
raise UpdateFailed("Failed to get livisi devices from controller") from exc
def _async_dispatcher_send(self, event: str, source: str, data: Any) -> None:
if data is not None:
async_dispatcher_send(self.hass, f"{event}_{source}", data)
async def async_setup(self) -> None:
"""Set up the Livisi Smart Home Controller."""
if not self.aiolivisi.livisi_connection_data:
livisi_connection_data = {
"ip_address": self.config_entry.data[CONF_HOST],
"password": self.config_entry.data[CONF_PASSWORD],
}
await self.aiolivisi.async_set_token(
livisi_connection_data=livisi_connection_data
)
controller_data = await self.aiolivisi.async_get_controller()
if (controller_type := controller_data["controllerType"]) == AVATAR:
self.port = AVATAR_PORT
self.is_avatar = True
else:
self.port = CLASSIC_PORT
self.is_avatar = False
self.controller_type = controller_type
self.serial_number = controller_data["serialNumber"]
async def async_get_devices(self) -> list[dict[str, Any]]:
"""Set the discovered devices list."""
return await self.aiolivisi.async_get_devices()
async def async_get_device_state(self, capability: str, key: str) -> Any | None:
"""Get state from livisi devices."""
response: dict[str, Any] = await self.aiolivisi.async_get_device_state(
capability[1:]
)
if response is None:
return None
return response.get(key, {}).get("value")
async def async_set_all_rooms(self) -> None:
"""Set the room list."""
response: list[dict[str, Any]] = await self.aiolivisi.async_get_all_rooms()
for available_room in response:
available_room_config: dict[str, Any] = available_room["config"]
self.rooms[available_room["id"]] = available_room_config["name"]
def on_data(self, event_data: LivisiEvent) -> None:
"""Define a handler to fire when the data is received."""
self._async_dispatcher_send(
LIVISI_STATE_CHANGE, event_data.source, event_data.onState
)
self._async_dispatcher_send(
LIVISI_STATE_CHANGE, event_data.source, event_data.vrccData
)
self._async_dispatcher_send(
LIVISI_REACHABILITY_CHANGE, event_data.source, event_data.isReachable
)
self._async_dispatcher_send(
LIVISI_STATE_CHANGE, event_data.source, event_data.isOpen
)
async def on_close(self) -> None:
"""Define a handler to fire when the websocket is closed."""
for device_id in self.devices:
self._async_dispatcher_send(LIVISI_REACHABILITY_CHANGE, device_id, False)
await self.websocket.connect(self.on_data, self.on_close, self.port)
async def ws_connect(self) -> None:
"""Connect the websocket."""
await self.websocket.connect(self.on_data, self.on_close, self.port)