208 lines
6.4 KiB
Python
208 lines
6.4 KiB
Python
"""Sensors for the Elexa Guardian integration."""
|
|
from typing import Callable, Dict, Optional
|
|
|
|
from homeassistant.config_entries import ConfigEntry
|
|
from homeassistant.const import (
|
|
DEVICE_CLASS_BATTERY,
|
|
DEVICE_CLASS_TEMPERATURE,
|
|
PERCENTAGE,
|
|
TEMP_FAHRENHEIT,
|
|
TIME_MINUTES,
|
|
)
|
|
from homeassistant.core import HomeAssistant, callback
|
|
from homeassistant.helpers.dispatcher import async_dispatcher_connect
|
|
from homeassistant.helpers.update_coordinator import DataUpdateCoordinator
|
|
|
|
from . import PairedSensorEntity, ValveControllerEntity
|
|
from .const import (
|
|
API_SENSOR_PAIRED_SENSOR_STATUS,
|
|
API_SYSTEM_DIAGNOSTICS,
|
|
API_SYSTEM_ONBOARD_SENSOR_STATUS,
|
|
CONF_UID,
|
|
DATA_COORDINATOR,
|
|
DATA_UNSUB_DISPATCHER_CONNECT,
|
|
DOMAIN,
|
|
SIGNAL_PAIRED_SENSOR_COORDINATOR_ADDED,
|
|
)
|
|
|
|
SENSOR_KIND_BATTERY = "battery"
|
|
SENSOR_KIND_TEMPERATURE = "temperature"
|
|
SENSOR_KIND_UPTIME = "uptime"
|
|
|
|
SENSOR_ATTRS_MAP = {
|
|
SENSOR_KIND_BATTERY: ("Battery", DEVICE_CLASS_BATTERY, None, PERCENTAGE),
|
|
SENSOR_KIND_TEMPERATURE: (
|
|
"Temperature",
|
|
DEVICE_CLASS_TEMPERATURE,
|
|
None,
|
|
TEMP_FAHRENHEIT,
|
|
),
|
|
SENSOR_KIND_UPTIME: ("Uptime", None, "mdi:timer", TIME_MINUTES),
|
|
}
|
|
|
|
PAIRED_SENSOR_SENSORS = [SENSOR_KIND_BATTERY, SENSOR_KIND_TEMPERATURE]
|
|
VALVE_CONTROLLER_SENSORS = [SENSOR_KIND_TEMPERATURE, SENSOR_KIND_UPTIME]
|
|
|
|
|
|
async def async_setup_entry(
|
|
hass: HomeAssistant, entry: ConfigEntry, async_add_entities: Callable
|
|
) -> None:
|
|
"""Set up Guardian switches based on a config entry."""
|
|
|
|
@callback
|
|
def add_new_paired_sensor(uid: str) -> None:
|
|
"""Add a new paired sensor."""
|
|
coordinator = hass.data[DOMAIN][DATA_COORDINATOR][entry.entry_id][
|
|
API_SENSOR_PAIRED_SENSOR_STATUS
|
|
][uid]
|
|
|
|
entities = []
|
|
for kind in PAIRED_SENSOR_SENSORS:
|
|
name, device_class, icon, unit = SENSOR_ATTRS_MAP[kind]
|
|
entities.append(
|
|
PairedSensorSensor(
|
|
entry, coordinator, kind, name, device_class, icon, unit
|
|
)
|
|
)
|
|
|
|
async_add_entities(entities, True)
|
|
|
|
# Handle adding paired sensors after HASS startup:
|
|
hass.data[DOMAIN][DATA_UNSUB_DISPATCHER_CONNECT][entry.entry_id].append(
|
|
async_dispatcher_connect(
|
|
hass,
|
|
SIGNAL_PAIRED_SENSOR_COORDINATOR_ADDED.format(entry.data[CONF_UID]),
|
|
add_new_paired_sensor,
|
|
)
|
|
)
|
|
|
|
sensors = []
|
|
|
|
# Add all valve controller-specific binary sensors:
|
|
for kind in VALVE_CONTROLLER_SENSORS:
|
|
name, device_class, icon, unit = SENSOR_ATTRS_MAP[kind]
|
|
sensors.append(
|
|
ValveControllerSensor(
|
|
entry,
|
|
hass.data[DOMAIN][DATA_COORDINATOR][entry.entry_id],
|
|
kind,
|
|
name,
|
|
device_class,
|
|
icon,
|
|
unit,
|
|
)
|
|
)
|
|
|
|
# Add all paired sensor-specific binary sensors:
|
|
for coordinator in hass.data[DOMAIN][DATA_COORDINATOR][entry.entry_id][
|
|
API_SENSOR_PAIRED_SENSOR_STATUS
|
|
].values():
|
|
for kind in PAIRED_SENSOR_SENSORS:
|
|
name, device_class, icon, unit = SENSOR_ATTRS_MAP[kind]
|
|
sensors.append(
|
|
PairedSensorSensor(
|
|
entry, coordinator, kind, name, device_class, icon, unit
|
|
)
|
|
)
|
|
|
|
async_add_entities(sensors)
|
|
|
|
|
|
class PairedSensorSensor(PairedSensorEntity):
|
|
"""Define a binary sensor related to a Guardian valve controller."""
|
|
|
|
def __init__(
|
|
self,
|
|
entry: ConfigEntry,
|
|
coordinator: DataUpdateCoordinator,
|
|
kind: str,
|
|
name: str,
|
|
device_class: Optional[str],
|
|
icon: Optional[str],
|
|
unit: Optional[str],
|
|
) -> None:
|
|
"""Initialize."""
|
|
super().__init__(entry, coordinator, kind, name, device_class, icon)
|
|
|
|
self._state = None
|
|
self._unit = unit
|
|
|
|
@property
|
|
def available(self) -> bool:
|
|
"""Return whether the entity is available."""
|
|
return self.coordinator.last_update_success
|
|
|
|
@property
|
|
def state(self) -> str:
|
|
"""Return the sensor state."""
|
|
return self._state
|
|
|
|
@property
|
|
def unit_of_measurement(self) -> str:
|
|
"""Return the unit of measurement of this entity, if any."""
|
|
return self._unit
|
|
|
|
@callback
|
|
def _async_update_from_latest_data(self) -> None:
|
|
"""Update the entity."""
|
|
if self._kind == SENSOR_KIND_BATTERY:
|
|
self._state = self.coordinator.data["battery"]
|
|
elif self._kind == SENSOR_KIND_TEMPERATURE:
|
|
self._state = self.coordinator.data["temperature"]
|
|
|
|
|
|
class ValveControllerSensor(ValveControllerEntity):
|
|
"""Define a generic Guardian sensor."""
|
|
|
|
def __init__(
|
|
self,
|
|
entry: ConfigEntry,
|
|
coordinators: Dict[str, DataUpdateCoordinator],
|
|
kind: str,
|
|
name: str,
|
|
device_class: Optional[str],
|
|
icon: Optional[str],
|
|
unit: Optional[str],
|
|
) -> None:
|
|
"""Initialize."""
|
|
super().__init__(entry, coordinators, kind, name, device_class, icon)
|
|
|
|
self._state = None
|
|
self._unit = unit
|
|
|
|
@property
|
|
def available(self) -> bool:
|
|
"""Return whether the entity is available."""
|
|
if self._kind == SENSOR_KIND_TEMPERATURE:
|
|
return self.coordinators[
|
|
API_SYSTEM_ONBOARD_SENSOR_STATUS
|
|
].last_update_success
|
|
if self._kind == SENSOR_KIND_UPTIME:
|
|
return self.coordinators[API_SYSTEM_DIAGNOSTICS].last_update_success
|
|
return False
|
|
|
|
@property
|
|
def state(self) -> str:
|
|
"""Return the sensor state."""
|
|
return self._state
|
|
|
|
@property
|
|
def unit_of_measurement(self) -> str:
|
|
"""Return the unit of measurement of this entity, if any."""
|
|
return self._unit
|
|
|
|
async def _async_continue_entity_setup(self) -> None:
|
|
"""Register API interest (and related tasks) when the entity is added."""
|
|
if self._kind == SENSOR_KIND_TEMPERATURE:
|
|
self.async_add_coordinator_update_listener(API_SYSTEM_ONBOARD_SENSOR_STATUS)
|
|
|
|
@callback
|
|
def _async_update_from_latest_data(self) -> None:
|
|
"""Update the entity."""
|
|
if self._kind == SENSOR_KIND_TEMPERATURE:
|
|
self._state = self.coordinators[API_SYSTEM_ONBOARD_SENSOR_STATUS].data[
|
|
"temperature"
|
|
]
|
|
elif self._kind == SENSOR_KIND_UPTIME:
|
|
self._state = self.coordinators[API_SYSTEM_DIAGNOSTICS].data["uptime"]
|