2019-05-30 16:48:58 +00:00
|
|
|
"""Runtime entry data for ESPHome stored in hass.data."""
|
2021-03-17 22:49:01 +00:00
|
|
|
from __future__ import annotations
|
|
|
|
|
2019-05-30 16:48:58 +00:00
|
|
|
import asyncio
|
2021-03-17 22:49:01 +00:00
|
|
|
from typing import TYPE_CHECKING, Any, Callable
|
2019-05-30 16:48:58 +00:00
|
|
|
|
|
|
|
from aioesphomeapi import (
|
2019-07-31 19:25:30 +00:00
|
|
|
COMPONENT_TYPE_TO_INFO,
|
2019-06-18 15:41:45 +00:00
|
|
|
BinarySensorInfo,
|
2019-07-31 19:25:30 +00:00
|
|
|
CameraInfo,
|
|
|
|
ClimateInfo,
|
|
|
|
CoverInfo,
|
2019-11-26 02:00:58 +00:00
|
|
|
DeviceInfo,
|
|
|
|
EntityInfo,
|
|
|
|
EntityState,
|
2019-07-31 19:25:30 +00:00
|
|
|
FanInfo,
|
|
|
|
LightInfo,
|
|
|
|
SensorInfo,
|
|
|
|
SwitchInfo,
|
|
|
|
TextSensorInfo,
|
2019-11-26 02:00:58 +00:00
|
|
|
UserService,
|
2019-07-31 19:25:30 +00:00
|
|
|
)
|
2019-05-30 16:48:58 +00:00
|
|
|
import attr
|
|
|
|
|
2019-06-18 15:41:45 +00:00
|
|
|
from homeassistant.config_entries import ConfigEntry
|
2020-01-29 21:59:45 +00:00
|
|
|
from homeassistant.core import callback
|
2019-05-30 16:48:58 +00:00
|
|
|
from homeassistant.helpers.dispatcher import async_dispatcher_send
|
|
|
|
from homeassistant.helpers.storage import Store
|
|
|
|
from homeassistant.helpers.typing import HomeAssistantType
|
|
|
|
|
2020-07-14 17:30:30 +00:00
|
|
|
if TYPE_CHECKING:
|
|
|
|
from . import APIClient
|
|
|
|
|
2020-10-30 08:02:00 +00:00
|
|
|
SAVE_DELAY = 120
|
|
|
|
|
2019-06-18 15:41:45 +00:00
|
|
|
# Mapping from ESPHome info type to HA platform
|
|
|
|
INFO_TYPE_TO_PLATFORM = {
|
2019-07-31 19:25:30 +00:00
|
|
|
BinarySensorInfo: "binary_sensor",
|
|
|
|
CameraInfo: "camera",
|
|
|
|
ClimateInfo: "climate",
|
|
|
|
CoverInfo: "cover",
|
|
|
|
FanInfo: "fan",
|
|
|
|
LightInfo: "light",
|
|
|
|
SensorInfo: "sensor",
|
|
|
|
SwitchInfo: "switch",
|
|
|
|
TextSensorInfo: "sensor",
|
2019-06-18 15:41:45 +00:00
|
|
|
}
|
|
|
|
|
2019-05-30 16:48:58 +00:00
|
|
|
|
|
|
|
@attr.s
|
|
|
|
class RuntimeEntryData:
|
|
|
|
"""Store runtime data for esphome config entries."""
|
|
|
|
|
2020-07-14 17:30:30 +00:00
|
|
|
entry_id: str = attr.ib()
|
2021-03-18 21:58:19 +00:00
|
|
|
client: APIClient = attr.ib()
|
2020-07-14 17:30:30 +00:00
|
|
|
store: Store = attr.ib()
|
2021-03-17 22:49:01 +00:00
|
|
|
state: dict[str, dict[str, Any]] = attr.ib(factory=dict)
|
|
|
|
info: dict[str, dict[str, Any]] = attr.ib(factory=dict)
|
2019-10-24 20:36:47 +00:00
|
|
|
|
|
|
|
# A second list of EntityInfo objects
|
|
|
|
# This is necessary for when an entity is being removed. HA requires
|
|
|
|
# some static info to be accessible during removal (unique_id, maybe others)
|
|
|
|
# If an entity can't find anything in the info array, it will look for info here.
|
2021-03-17 22:49:01 +00:00
|
|
|
old_info: dict[str, dict[str, Any]] = attr.ib(factory=dict)
|
2019-10-24 20:36:47 +00:00
|
|
|
|
2021-03-22 12:29:39 +00:00
|
|
|
services: dict[int, UserService] = attr.ib(factory=dict)
|
2020-07-14 17:30:30 +00:00
|
|
|
available: bool = attr.ib(default=False)
|
2021-03-17 22:49:01 +00:00
|
|
|
device_info: DeviceInfo | None = attr.ib(default=None)
|
|
|
|
cleanup_callbacks: list[Callable[[], None]] = attr.ib(factory=list)
|
|
|
|
disconnect_callbacks: list[Callable[[], None]] = attr.ib(factory=list)
|
|
|
|
loaded_platforms: set[str] = attr.ib(factory=set)
|
2020-07-14 17:30:30 +00:00
|
|
|
platform_load_lock: asyncio.Lock = attr.ib(factory=asyncio.Lock)
|
2019-05-30 16:48:58 +00:00
|
|
|
|
2020-01-29 21:59:45 +00:00
|
|
|
@callback
|
2019-07-31 19:25:30 +00:00
|
|
|
def async_update_entity(
|
|
|
|
self, hass: HomeAssistantType, component_key: str, key: int
|
|
|
|
) -> None:
|
2019-05-30 16:48:58 +00:00
|
|
|
"""Schedule the update of an entity."""
|
2020-02-24 16:47:52 +00:00
|
|
|
signal = f"esphome_{self.entry_id}_update_{component_key}_{key}"
|
2019-05-30 16:48:58 +00:00
|
|
|
async_dispatcher_send(hass, signal)
|
|
|
|
|
2020-01-29 21:59:45 +00:00
|
|
|
@callback
|
2019-07-31 19:25:30 +00:00
|
|
|
def async_remove_entity(
|
|
|
|
self, hass: HomeAssistantType, component_key: str, key: int
|
|
|
|
) -> None:
|
2019-05-30 16:48:58 +00:00
|
|
|
"""Schedule the removal of an entity."""
|
2020-02-24 16:47:52 +00:00
|
|
|
signal = f"esphome_{self.entry_id}_remove_{component_key}_{key}"
|
2019-05-30 16:48:58 +00:00
|
|
|
async_dispatcher_send(hass, signal)
|
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
async def _ensure_platforms_loaded(
|
2021-03-17 22:49:01 +00:00
|
|
|
self, hass: HomeAssistantType, entry: ConfigEntry, platforms: set[str]
|
2019-07-31 19:25:30 +00:00
|
|
|
):
|
2019-06-18 15:41:45 +00:00
|
|
|
async with self.platform_load_lock:
|
|
|
|
needed = platforms - self.loaded_platforms
|
|
|
|
tasks = []
|
|
|
|
for platform in needed:
|
2019-07-31 19:25:30 +00:00
|
|
|
tasks.append(
|
|
|
|
hass.config_entries.async_forward_entry_setup(entry, platform)
|
|
|
|
)
|
2019-06-18 15:41:45 +00:00
|
|
|
if tasks:
|
|
|
|
await asyncio.wait(tasks)
|
|
|
|
self.loaded_platforms |= needed
|
|
|
|
|
|
|
|
async def async_update_static_infos(
|
2021-03-17 22:49:01 +00:00
|
|
|
self, hass: HomeAssistantType, entry: ConfigEntry, infos: list[EntityInfo]
|
2019-07-31 19:25:30 +00:00
|
|
|
) -> None:
|
2019-05-30 16:48:58 +00:00
|
|
|
"""Distribute an update of static infos to all platforms."""
|
2019-06-18 15:41:45 +00:00
|
|
|
# First, load all platforms
|
|
|
|
needed_platforms = set()
|
|
|
|
for info in infos:
|
|
|
|
for info_type, platform in INFO_TYPE_TO_PLATFORM.items():
|
|
|
|
if isinstance(info, info_type):
|
|
|
|
needed_platforms.add(platform)
|
|
|
|
break
|
|
|
|
await self._ensure_platforms_loaded(hass, entry, needed_platforms)
|
|
|
|
|
|
|
|
# Then send dispatcher event
|
2020-02-24 16:47:52 +00:00
|
|
|
signal = f"esphome_{self.entry_id}_on_list"
|
2019-05-30 16:48:58 +00:00
|
|
|
async_dispatcher_send(hass, signal, infos)
|
|
|
|
|
2020-01-29 21:59:45 +00:00
|
|
|
@callback
|
2019-07-31 19:25:30 +00:00
|
|
|
def async_update_state(self, hass: HomeAssistantType, state: EntityState) -> None:
|
2019-05-30 16:48:58 +00:00
|
|
|
"""Distribute an update of state information to all platforms."""
|
2020-02-24 16:47:52 +00:00
|
|
|
signal = f"esphome_{self.entry_id}_on_state"
|
2019-05-30 16:48:58 +00:00
|
|
|
async_dispatcher_send(hass, signal, state)
|
|
|
|
|
2020-01-29 21:59:45 +00:00
|
|
|
@callback
|
2019-05-30 16:48:58 +00:00
|
|
|
def async_update_device_state(self, hass: HomeAssistantType) -> None:
|
|
|
|
"""Distribute an update of a core device state like availability."""
|
2020-02-24 16:47:52 +00:00
|
|
|
signal = f"esphome_{self.entry_id}_on_device_update"
|
2019-05-30 16:48:58 +00:00
|
|
|
async_dispatcher_send(hass, signal)
|
|
|
|
|
2021-03-17 22:49:01 +00:00
|
|
|
async def async_load_from_store(self) -> tuple[list[EntityInfo], list[UserService]]:
|
2019-05-30 16:48:58 +00:00
|
|
|
"""Load the retained data from store and return de-serialized data."""
|
|
|
|
restored = await self.store.async_load()
|
|
|
|
if restored is None:
|
|
|
|
return [], []
|
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
self.device_info = _attr_obj_from_dict(
|
|
|
|
DeviceInfo, **restored.pop("device_info")
|
|
|
|
)
|
2019-05-30 16:48:58 +00:00
|
|
|
infos = []
|
|
|
|
for comp_type, restored_infos in restored.items():
|
|
|
|
if comp_type not in COMPONENT_TYPE_TO_INFO:
|
|
|
|
continue
|
|
|
|
for info in restored_infos:
|
|
|
|
cls = COMPONENT_TYPE_TO_INFO[comp_type]
|
|
|
|
infos.append(_attr_obj_from_dict(cls, **info))
|
|
|
|
services = []
|
2019-07-31 19:25:30 +00:00
|
|
|
for service in restored.get("services", []):
|
2019-05-30 16:48:58 +00:00
|
|
|
services.append(UserService.from_dict(service))
|
|
|
|
return infos, services
|
|
|
|
|
|
|
|
async def async_save_to_store(self) -> None:
|
|
|
|
"""Generate dynamic data to store and save it to the filesystem."""
|
2019-07-31 19:25:30 +00:00
|
|
|
store_data = {"device_info": attr.asdict(self.device_info), "services": []}
|
2019-05-30 16:48:58 +00:00
|
|
|
|
|
|
|
for comp_type, infos in self.info.items():
|
2019-07-31 19:25:30 +00:00
|
|
|
store_data[comp_type] = [attr.asdict(info) for info in infos.values()]
|
2019-05-30 16:48:58 +00:00
|
|
|
for service in self.services.values():
|
2019-07-31 19:25:30 +00:00
|
|
|
store_data["services"].append(service.to_dict())
|
2019-05-30 16:48:58 +00:00
|
|
|
|
2020-10-30 08:02:00 +00:00
|
|
|
self.store.async_delay_save(lambda: store_data, SAVE_DELAY)
|
2019-05-30 16:48:58 +00:00
|
|
|
|
|
|
|
|
|
|
|
def _attr_obj_from_dict(cls, **kwargs):
|
2019-07-31 19:25:30 +00:00
|
|
|
return cls(**{key: kwargs[key] for key in attr.fields_dict(cls) if key in kwargs})
|