131 lines
3.6 KiB
Python
131 lines
3.6 KiB
Python
"""Melnor integration models."""
|
|
|
|
from collections.abc import Callable
|
|
from datetime import timedelta
|
|
import logging
|
|
from typing import TypeVar
|
|
|
|
from melnor_bluetooth.device import Device, Valve
|
|
|
|
from homeassistant.components.number import EntityDescription
|
|
from homeassistant.core import HomeAssistant, callback
|
|
from homeassistant.helpers.entity import DeviceInfo
|
|
from homeassistant.helpers.update_coordinator import (
|
|
CoordinatorEntity,
|
|
DataUpdateCoordinator,
|
|
)
|
|
|
|
from .const import DOMAIN
|
|
|
|
_LOGGER = logging.getLogger(__name__)
|
|
|
|
|
|
class MelnorDataUpdateCoordinator(DataUpdateCoordinator[Device]):
|
|
"""Melnor data update coordinator."""
|
|
|
|
_device: Device
|
|
|
|
def __init__(self, hass: HomeAssistant, device: Device) -> None:
|
|
"""Initialize my coordinator."""
|
|
super().__init__(
|
|
hass,
|
|
_LOGGER,
|
|
name="Melnor Bluetooth",
|
|
update_interval=timedelta(seconds=5),
|
|
)
|
|
self._device = device
|
|
|
|
async def _async_update_data(self):
|
|
"""Update the device state."""
|
|
|
|
await self._device.fetch_state()
|
|
return self._device
|
|
|
|
|
|
class MelnorBluetoothEntity(CoordinatorEntity[MelnorDataUpdateCoordinator]):
|
|
"""Base class for melnor entities."""
|
|
|
|
_device: Device
|
|
_attr_has_entity_name = True
|
|
|
|
def __init__(
|
|
self,
|
|
coordinator: MelnorDataUpdateCoordinator,
|
|
) -> None:
|
|
"""Initialize a melnor base entity."""
|
|
super().__init__(coordinator)
|
|
|
|
self._device = coordinator.data
|
|
|
|
self._attr_device_info = DeviceInfo(
|
|
identifiers={(DOMAIN, self._device.mac)},
|
|
manufacturer="Melnor",
|
|
model=self._device.model,
|
|
name=self._device.name,
|
|
)
|
|
|
|
@callback
|
|
def _handle_coordinator_update(self) -> None:
|
|
"""Handle updated data from the coordinator."""
|
|
self._device = self.coordinator.data
|
|
self.async_write_ha_state()
|
|
|
|
@property
|
|
def available(self) -> bool:
|
|
"""Return True if entity is available."""
|
|
return self._device.is_connected
|
|
|
|
|
|
class MelnorZoneEntity(MelnorBluetoothEntity):
|
|
"""Base class for valves that define themselves as child devices."""
|
|
|
|
_valve: Valve
|
|
|
|
def __init__(
|
|
self,
|
|
coordinator: MelnorDataUpdateCoordinator,
|
|
entity_description: EntityDescription,
|
|
valve: Valve,
|
|
) -> None:
|
|
"""Initialize a valve entity."""
|
|
super().__init__(coordinator)
|
|
|
|
self._attr_unique_id = (
|
|
f"{self._device.mac}-zone{valve.id}-{entity_description.key}"
|
|
)
|
|
self.entity_description = entity_description
|
|
|
|
self._valve = valve
|
|
|
|
self._attr_device_info = DeviceInfo(
|
|
identifiers={(DOMAIN, f"{self._device.mac}-zone{self._valve.id}")},
|
|
manufacturer="Melnor",
|
|
name=f"Zone {valve.id + 1}",
|
|
via_device=(DOMAIN, self._device.mac),
|
|
)
|
|
|
|
|
|
T = TypeVar("T", bound=EntityDescription)
|
|
|
|
|
|
def get_entities_for_valves(
|
|
coordinator: MelnorDataUpdateCoordinator,
|
|
descriptions: list[T],
|
|
function: Callable[
|
|
[Valve, T],
|
|
CoordinatorEntity[MelnorDataUpdateCoordinator],
|
|
],
|
|
) -> list[CoordinatorEntity[MelnorDataUpdateCoordinator]]:
|
|
"""Get descriptions for valves."""
|
|
entities = []
|
|
|
|
# This device may not have 4 valves total, but the library will only expose the right number of valves
|
|
for i in range(1, 5):
|
|
valve = coordinator.data[f"zone{i}"]
|
|
|
|
if valve is not None:
|
|
for description in descriptions:
|
|
entities.append(function(valve, description))
|
|
|
|
return entities
|