core/homeassistant/components/deconz/number.py

130 lines
4.0 KiB
Python

"""Support for configuring different deCONZ numbers."""
from __future__ import annotations
from collections.abc import Callable, Coroutine
from dataclasses import dataclass
from typing import Any
from pydeconz.gateway import DeconzSession
from pydeconz.interfaces.sensors import SensorResources
from pydeconz.models.event import EventType
from pydeconz.models.sensor import SensorBase as PydeconzSensorBase
from pydeconz.models.sensor.presence import Presence
from homeassistant.components.number import (
DOMAIN as NUMBER_DOMAIN,
NumberEntity,
NumberEntityDescription,
)
from homeassistant.config_entries import ConfigEntry
from homeassistant.const import EntityCategory
from homeassistant.core import HomeAssistant, callback
from homeassistant.helpers.entity_platform import AddEntitiesCallback
from .entity import DeconzDevice
from .hub import DeconzHub
@dataclass(frozen=True, kw_only=True)
class DeconzNumberDescription[_T: (Presence, PydeconzSensorBase)](
NumberEntityDescription
):
"""Class describing deCONZ number entities."""
instance_check: type[_T]
name_suffix: str
set_fn: Callable[[DeconzSession, str, int], Coroutine[Any, Any, dict[str, Any]]]
update_key: str
value_fn: Callable[[_T], float | None]
ENTITY_DESCRIPTIONS: tuple[DeconzNumberDescription, ...] = (
DeconzNumberDescription[Presence](
key="delay",
instance_check=Presence,
name_suffix="Delay",
set_fn=lambda api, id, v: api.sensors.presence.set_config(id=id, delay=v),
update_key="delay",
value_fn=lambda device: device.delay,
native_max_value=65535,
native_min_value=0,
native_step=1,
entity_category=EntityCategory.CONFIG,
),
DeconzNumberDescription[Presence](
key="duration",
instance_check=Presence,
name_suffix="Duration",
set_fn=lambda api, id, v: api.sensors.presence.set_config(id=id, duration=v),
update_key="duration",
value_fn=lambda device: device.duration,
native_max_value=65535,
native_min_value=0,
native_step=1,
entity_category=EntityCategory.CONFIG,
),
)
async def async_setup_entry(
hass: HomeAssistant,
config_entry: ConfigEntry,
async_add_entities: AddEntitiesCallback,
) -> None:
"""Set up the deCONZ number entity."""
hub = DeconzHub.get_hub(hass, config_entry)
hub.entities[NUMBER_DOMAIN] = set()
@callback
def async_add_sensor(_: EventType, sensor_id: str) -> None:
"""Add sensor from deCONZ."""
sensor = hub.api.sensors.presence[sensor_id]
for description in ENTITY_DESCRIPTIONS:
if (
not isinstance(sensor, description.instance_check)
or description.value_fn(sensor) is None
):
continue
async_add_entities([DeconzNumber(sensor, hub, description)])
hub.register_platform_add_device_callback(
async_add_sensor,
hub.api.sensors.presence,
always_ignore_clip_sensors=True,
)
class DeconzNumber(DeconzDevice[SensorResources], NumberEntity):
"""Representation of a deCONZ number entity."""
TYPE = NUMBER_DOMAIN
entity_description: DeconzNumberDescription
def __init__(
self,
device: SensorResources,
hub: DeconzHub,
description: DeconzNumberDescription,
) -> None:
"""Initialize deCONZ number entity."""
self.entity_description = description
self.unique_id_suffix = description.key
self._name_suffix = description.name_suffix
self._update_key = description.update_key
super().__init__(device, hub)
@property
def native_value(self) -> float | None:
"""Return the value of the sensor property."""
return self.entity_description.value_fn(self._device)
async def async_set_native_value(self, value: float) -> None:
"""Set sensor config."""
await self.entity_description.set_fn(
self.hub.api,
self._device.resource_id,
int(value),
)