mirror of https://github.com/home-assistant/core
212 lines
6.6 KiB
Python
212 lines
6.6 KiB
Python
"""Entity classes for the Airzone integration."""
|
|
|
|
from __future__ import annotations
|
|
|
|
import logging
|
|
from typing import Any
|
|
|
|
from aioairzone.const import (
|
|
API_SYSTEM_ID,
|
|
API_ZONE_ID,
|
|
AZD_AVAILABLE,
|
|
AZD_FIRMWARE,
|
|
AZD_FULL_NAME,
|
|
AZD_HOT_WATER,
|
|
AZD_ID,
|
|
AZD_MAC,
|
|
AZD_MODEL,
|
|
AZD_NAME,
|
|
AZD_SYSTEM,
|
|
AZD_SYSTEMS,
|
|
AZD_THERMOSTAT_FW,
|
|
AZD_THERMOSTAT_MODEL,
|
|
AZD_WEBSERVER,
|
|
AZD_ZONES,
|
|
)
|
|
from aioairzone.exceptions import AirzoneError
|
|
|
|
from homeassistant.config_entries import ConfigEntry
|
|
from homeassistant.exceptions import HomeAssistantError
|
|
from homeassistant.helpers import device_registry as dr
|
|
from homeassistant.helpers.device_registry import DeviceInfo
|
|
from homeassistant.helpers.update_coordinator import CoordinatorEntity
|
|
|
|
from . import AirzoneConfigEntry
|
|
from .const import DOMAIN, MANUFACTURER
|
|
from .coordinator import AirzoneUpdateCoordinator
|
|
|
|
_LOGGER = logging.getLogger(__name__)
|
|
|
|
|
|
class AirzoneEntity(CoordinatorEntity[AirzoneUpdateCoordinator]):
|
|
"""Define an Airzone entity."""
|
|
|
|
_attr_has_entity_name = True
|
|
|
|
def get_airzone_value(self, key: str) -> Any:
|
|
"""Return Airzone entity value by key."""
|
|
raise NotImplementedError
|
|
|
|
|
|
class AirzoneSystemEntity(AirzoneEntity):
|
|
"""Define an Airzone System entity."""
|
|
|
|
def __init__(
|
|
self,
|
|
coordinator: AirzoneUpdateCoordinator,
|
|
entry: AirzoneConfigEntry,
|
|
system_data: dict[str, Any],
|
|
) -> None:
|
|
"""Initialize."""
|
|
super().__init__(coordinator)
|
|
|
|
self.system_id = system_data[AZD_ID]
|
|
|
|
self._attr_device_info = DeviceInfo(
|
|
identifiers={(DOMAIN, f"{entry.entry_id}_{self.system_id}")},
|
|
manufacturer=MANUFACTURER,
|
|
model=self.get_airzone_value(AZD_MODEL),
|
|
name=f"System {self.system_id}",
|
|
sw_version=self.get_airzone_value(AZD_FIRMWARE),
|
|
via_device=(DOMAIN, f"{entry.entry_id}_ws"),
|
|
)
|
|
self._attr_unique_id = entry.unique_id or entry.entry_id
|
|
|
|
@property
|
|
def available(self) -> bool:
|
|
"""Return system availability."""
|
|
return super().available and self.get_airzone_value(AZD_AVAILABLE)
|
|
|
|
def get_airzone_value(self, key: str) -> Any:
|
|
"""Return system value by key."""
|
|
value = None
|
|
if system := self.coordinator.data[AZD_SYSTEMS].get(self.system_id):
|
|
if key in system:
|
|
value = system[key]
|
|
return value
|
|
|
|
|
|
class AirzoneHotWaterEntity(AirzoneEntity):
|
|
"""Define an Airzone Hot Water entity."""
|
|
|
|
def __init__(
|
|
self,
|
|
coordinator: AirzoneUpdateCoordinator,
|
|
entry: ConfigEntry,
|
|
) -> None:
|
|
"""Initialize."""
|
|
super().__init__(coordinator)
|
|
|
|
self._attr_device_info = DeviceInfo(
|
|
identifiers={(DOMAIN, f"{entry.entry_id}_dhw")},
|
|
manufacturer=MANUFACTURER,
|
|
model="DHW",
|
|
name=self.get_airzone_value(AZD_NAME),
|
|
via_device=(DOMAIN, f"{entry.entry_id}_ws"),
|
|
)
|
|
self._attr_unique_id = entry.unique_id or entry.entry_id
|
|
|
|
def get_airzone_value(self, key: str) -> Any:
|
|
"""Return DHW value by key."""
|
|
return self.coordinator.data[AZD_HOT_WATER].get(key)
|
|
|
|
async def _async_update_dhw_params(self, params: dict[str, Any]) -> None:
|
|
"""Send DHW parameters to API."""
|
|
_params = {
|
|
API_SYSTEM_ID: 0,
|
|
**params,
|
|
}
|
|
_LOGGER.debug("update_dhw_params=%s", _params)
|
|
try:
|
|
await self.coordinator.airzone.set_dhw_parameters(_params)
|
|
except AirzoneError as error:
|
|
raise HomeAssistantError(f"Failed to set DHW: {error}") from error
|
|
|
|
self.coordinator.async_set_updated_data(self.coordinator.airzone.data())
|
|
|
|
|
|
class AirzoneWebServerEntity(AirzoneEntity):
|
|
"""Define an Airzone WebServer entity."""
|
|
|
|
def __init__(
|
|
self,
|
|
coordinator: AirzoneUpdateCoordinator,
|
|
entry: ConfigEntry,
|
|
) -> None:
|
|
"""Initialize."""
|
|
super().__init__(coordinator)
|
|
|
|
mac = self.get_airzone_value(AZD_MAC)
|
|
|
|
self._attr_device_info = DeviceInfo(
|
|
connections={(dr.CONNECTION_NETWORK_MAC, mac)},
|
|
identifiers={(DOMAIN, f"{entry.entry_id}_ws")},
|
|
manufacturer=MANUFACTURER,
|
|
model=self.get_airzone_value(AZD_MODEL),
|
|
name=self.get_airzone_value(AZD_FULL_NAME),
|
|
sw_version=self.get_airzone_value(AZD_FIRMWARE),
|
|
)
|
|
self._attr_unique_id = entry.unique_id or entry.entry_id
|
|
|
|
def get_airzone_value(self, key: str) -> Any:
|
|
"""Return system value by key."""
|
|
return self.coordinator.data[AZD_WEBSERVER].get(key)
|
|
|
|
|
|
class AirzoneZoneEntity(AirzoneEntity):
|
|
"""Define an Airzone Zone entity."""
|
|
|
|
def __init__(
|
|
self,
|
|
coordinator: AirzoneUpdateCoordinator,
|
|
entry: ConfigEntry,
|
|
system_zone_id: str,
|
|
zone_data: dict[str, Any],
|
|
) -> None:
|
|
"""Initialize."""
|
|
super().__init__(coordinator)
|
|
|
|
self.system_id = zone_data[AZD_SYSTEM]
|
|
self.system_zone_id = system_zone_id
|
|
self.zone_id = zone_data[AZD_ID]
|
|
|
|
self._attr_device_info = DeviceInfo(
|
|
identifiers={(DOMAIN, f"{entry.entry_id}_{system_zone_id}")},
|
|
manufacturer=MANUFACTURER,
|
|
model=self.get_airzone_value(AZD_THERMOSTAT_MODEL),
|
|
name=zone_data[AZD_NAME],
|
|
sw_version=self.get_airzone_value(AZD_THERMOSTAT_FW),
|
|
via_device=(DOMAIN, f"{entry.entry_id}_{self.system_id}"),
|
|
)
|
|
self._attr_unique_id = entry.unique_id or entry.entry_id
|
|
|
|
@property
|
|
def available(self) -> bool:
|
|
"""Return zone availability."""
|
|
return super().available and self.get_airzone_value(AZD_AVAILABLE)
|
|
|
|
def get_airzone_value(self, key: str) -> Any:
|
|
"""Return zone value by key."""
|
|
value = None
|
|
if zone := self.coordinator.data[AZD_ZONES].get(self.system_zone_id):
|
|
if key in zone:
|
|
value = zone[key]
|
|
return value
|
|
|
|
async def _async_update_hvac_params(self, params: dict[str, Any]) -> None:
|
|
"""Send HVAC parameters to API."""
|
|
_params = {
|
|
API_SYSTEM_ID: self.system_id,
|
|
API_ZONE_ID: self.zone_id,
|
|
**params,
|
|
}
|
|
_LOGGER.debug("update_hvac_params=%s", _params)
|
|
try:
|
|
await self.coordinator.airzone.set_hvac_parameters(_params)
|
|
except AirzoneError as error:
|
|
raise HomeAssistantError(
|
|
f"Failed to set zone {self.entity_id}: {error}"
|
|
) from error
|
|
|
|
self.coordinator.async_set_updated_data(self.coordinator.airzone.data())
|