Refactor Totalconnect binary sensor (#115629)

This commit is contained in:
Joost Lekkerkerker 2024-04-21 17:36:19 +02:00 committed by GitHub
parent ec066472ae
commit 95b858648e
No known key found for this signature in database
GPG Key ID: B5690EEEBB952194

View File

@ -1,7 +1,12 @@
"""Interfaces with TotalConnect sensors.""" """Interfaces with TotalConnect sensors."""
from collections.abc import Callable
from dataclasses import dataclass
import logging import logging
from total_connect_client.location import TotalConnectLocation
from total_connect_client.zone import TotalConnectZone
from homeassistant.components.binary_sensor import ( from homeassistant.components.binary_sensor import (
BinarySensorDeviceClass, BinarySensorDeviceClass,
BinarySensorEntity, BinarySensorEntity,
@ -12,7 +17,9 @@ from homeassistant.const import EntityCategory
from homeassistant.core import HomeAssistant from homeassistant.core import HomeAssistant
from homeassistant.helpers.device_registry import DeviceInfo from homeassistant.helpers.device_registry import DeviceInfo
from homeassistant.helpers.entity_platform import AddEntitiesCallback from homeassistant.helpers.entity_platform import AddEntitiesCallback
from homeassistant.helpers.update_coordinator import CoordinatorEntity
from . import TotalConnectDataUpdateCoordinator
from .const import DOMAIN from .const import DOMAIN
LOW_BATTERY = "low_battery" LOW_BATTERY = "low_battery"
@ -23,172 +30,194 @@ ZONE = "zone"
_LOGGER = logging.getLogger(__name__) _LOGGER = logging.getLogger(__name__)
@dataclass(frozen=True, kw_only=True)
class TotalConnectZoneBinarySensorEntityDescription(BinarySensorEntityDescription):
"""Describes TotalConnect binary sensor entity."""
device_class_fn: Callable[[TotalConnectZone], BinarySensorDeviceClass] | None = None
is_on_fn: Callable[[TotalConnectZone], bool]
def get_security_zone_device_class(zone: TotalConnectZone) -> BinarySensorDeviceClass:
"""Return the device class of a TotalConnect security zone."""
if zone.is_type_fire():
return BinarySensorDeviceClass.SMOKE
if zone.is_type_carbon_monoxide():
return BinarySensorDeviceClass.GAS
if zone.is_type_motion():
return BinarySensorDeviceClass.MOTION
if zone.is_type_medical():
return BinarySensorDeviceClass.SAFETY
if zone.is_type_temperature():
return BinarySensorDeviceClass.PROBLEM
return BinarySensorDeviceClass.DOOR
SECURITY_BINARY_SENSOR = TotalConnectZoneBinarySensorEntityDescription(
key=ZONE,
name="",
device_class_fn=get_security_zone_device_class,
is_on_fn=lambda zone: zone.is_faulted() or zone.is_triggered(),
)
NO_BUTTON_BINARY_SENSORS: tuple[TotalConnectZoneBinarySensorEntityDescription, ...] = (
TotalConnectZoneBinarySensorEntityDescription(
key=LOW_BATTERY,
device_class=BinarySensorDeviceClass.BATTERY,
entity_category=EntityCategory.DIAGNOSTIC,
name=" low battery",
is_on_fn=lambda zone: zone.is_low_battery(),
),
TotalConnectZoneBinarySensorEntityDescription(
key=TAMPER,
device_class=BinarySensorDeviceClass.TAMPER,
entity_category=EntityCategory.DIAGNOSTIC,
name=f" {TAMPER}",
is_on_fn=lambda zone: zone.is_tampered(),
),
)
@dataclass(frozen=True, kw_only=True)
class TotalConnectAlarmBinarySensorEntityDescription(BinarySensorEntityDescription):
"""Describes TotalConnect binary sensor entity."""
is_on_fn: Callable[[TotalConnectLocation], bool]
LOCATION_BINARY_SENSORS: tuple[TotalConnectAlarmBinarySensorEntityDescription, ...] = (
TotalConnectAlarmBinarySensorEntityDescription(
key=LOW_BATTERY,
device_class=BinarySensorDeviceClass.BATTERY,
entity_category=EntityCategory.DIAGNOSTIC,
name=" low battery",
is_on_fn=lambda location: location.is_low_battery(),
),
TotalConnectAlarmBinarySensorEntityDescription(
key=TAMPER,
device_class=BinarySensorDeviceClass.TAMPER,
entity_category=EntityCategory.DIAGNOSTIC,
name=f" {TAMPER}",
is_on_fn=lambda location: location.is_cover_tampered(),
),
TotalConnectAlarmBinarySensorEntityDescription(
key=POWER,
device_class=BinarySensorDeviceClass.POWER,
entity_category=EntityCategory.DIAGNOSTIC,
name=f" {POWER}",
is_on_fn=lambda location: location.is_ac_loss(),
),
)
async def async_setup_entry( async def async_setup_entry(
hass: HomeAssistant, entry: ConfigEntry, async_add_entities: AddEntitiesCallback hass: HomeAssistant, entry: ConfigEntry, async_add_entities: AddEntitiesCallback
) -> None: ) -> None:
"""Set up TotalConnect device sensors based on a config entry.""" """Set up TotalConnect device sensors based on a config entry."""
sensors: list = [] sensors: list = []
client_locations = hass.data[DOMAIN][entry.entry_id].client.locations coordinator: TotalConnectDataUpdateCoordinator = hass.data[DOMAIN][entry.entry_id]
client_locations = coordinator.client.locations
for location_id, location in client_locations.items(): for location_id, location in client_locations.items():
sensors.append(TotalConnectAlarmLowBatteryBinarySensor(location)) sensors.extend(
sensors.append(TotalConnectAlarmTamperBinarySensor(location)) TotalConnectAlarmBinarySensor(coordinator, description, location)
sensors.append(TotalConnectAlarmPowerBinarySensor(location)) for description in LOCATION_BINARY_SENSORS
)
for zone in location.zones.values(): for zone in location.zones.values():
sensors.append(TotalConnectZoneSecurityBinarySensor(location_id, zone)) sensors.append(
TotalConnectZoneBinarySensor(
coordinator, SECURITY_BINARY_SENSOR, location_id, zone
)
)
if not zone.is_type_button(): if not zone.is_type_button():
sensors.append(TotalConnectLowBatteryBinarySensor(location_id, zone)) sensors.extend(
sensors.append(TotalConnectTamperBinarySensor(location_id, zone)) TotalConnectZoneBinarySensor(
coordinator,
description,
location_id,
zone,
)
for description in NO_BUTTON_BINARY_SENSORS
)
async_add_entities(sensors, True) async_add_entities(sensors)
class TotalConnectZoneBinarySensor(BinarySensorEntity): class TotalConnectZoneBinarySensor(
CoordinatorEntity[TotalConnectDataUpdateCoordinator], BinarySensorEntity
):
"""Represent an TotalConnect zone.""" """Represent an TotalConnect zone."""
def __init__(self, location_id, zone): entity_description: TotalConnectZoneBinarySensorEntityDescription
def __init__(
self,
coordinator: TotalConnectDataUpdateCoordinator,
entity_description: TotalConnectZoneBinarySensorEntityDescription,
location_id: str,
zone: TotalConnectZone,
) -> None:
"""Initialize the TotalConnect status.""" """Initialize the TotalConnect status."""
super().__init__(coordinator)
self.entity_description = entity_description
self._location_id = location_id self._location_id = location_id
self._zone = zone self._zone = zone
self._attr_name = f"{zone.description}{self.entity_description.name}" self._attr_name = f"{zone.description}{entity_description.name}"
self._attr_unique_id = ( self._attr_unique_id = f"{location_id}_{zone.zoneid}_{entity_description.key}"
f"{location_id}_{zone.zoneid}_{self.entity_description.key}"
)
self._attr_is_on = None self._attr_is_on = None
self._attr_extra_state_attributes = { self._attr_extra_state_attributes = {
"zone_id": self._zone.zoneid, "zone_id": zone.zoneid,
"location_id": self._location_id, "location_id": self._location_id,
"partition": self._zone.partition, "partition": zone.partition,
} }
identifier = zone.sensor_serial_number or f"zone_{zone.zoneid}"
@property self._attr_device_info = DeviceInfo(
def device_info(self) -> DeviceInfo: name=zone.description,
"""Return device info."""
identifier = self._zone.sensor_serial_number or f"zone_{self._zone.zoneid}"
return DeviceInfo(
name=self._zone.description,
identifiers={(DOMAIN, identifier)}, identifiers={(DOMAIN, identifier)},
serial_number=self._zone.sensor_serial_number, serial_number=zone.sensor_serial_number,
)
class TotalConnectZoneSecurityBinarySensor(TotalConnectZoneBinarySensor):
"""Represent an TotalConnect security zone."""
entity_description: BinarySensorEntityDescription = BinarySensorEntityDescription(
key=ZONE, name=""
) )
@property @property
def device_class(self): def is_on(self) -> bool:
"""Return the state of the entity."""
return self.entity_description.is_on_fn(self._zone)
@property
def device_class(self) -> BinarySensorDeviceClass | None:
"""Return the class of this zone.""" """Return the class of this zone."""
if self._zone.is_type_fire(): if self.entity_description.device_class_fn:
return BinarySensorDeviceClass.SMOKE return self.entity_description.device_class_fn(self._zone)
if self._zone.is_type_carbon_monoxide(): return super().device_class
return BinarySensorDeviceClass.GAS
if self._zone.is_type_motion():
return BinarySensorDeviceClass.MOTION
if self._zone.is_type_medical():
return BinarySensorDeviceClass.SAFETY
if self._zone.is_type_temperature():
return BinarySensorDeviceClass.PROBLEM
return BinarySensorDeviceClass.DOOR
def update(self):
"""Return the state of the device."""
if self._zone.is_faulted() or self._zone.is_triggered():
self._attr_is_on = True
else:
self._attr_is_on = False
class TotalConnectLowBatteryBinarySensor(TotalConnectZoneBinarySensor): class TotalConnectAlarmBinarySensor(
"""Represent an TotalConnect zone low battery status.""" CoordinatorEntity[TotalConnectDataUpdateCoordinator], BinarySensorEntity
):
"""Represent a TotalConnect alarm device binary sensors."""
entity_description: BinarySensorEntityDescription = BinarySensorEntityDescription( entity_description: TotalConnectAlarmBinarySensorEntityDescription
key=LOW_BATTERY,
device_class=BinarySensorDeviceClass.BATTERY,
entity_category=EntityCategory.DIAGNOSTIC,
name=" low battery",
)
def update(self): def __init__(
"""Return the state of the device.""" self,
self._attr_is_on = self._zone.is_low_battery() coordinator: TotalConnectDataUpdateCoordinator,
entity_description: TotalConnectAlarmBinarySensorEntityDescription,
location: TotalConnectLocation,
class TotalConnectTamperBinarySensor(TotalConnectZoneBinarySensor): ) -> None:
"""Represent an TotalConnect zone tamper status."""
entity_description: BinarySensorEntityDescription = BinarySensorEntityDescription(
key=TAMPER,
device_class=BinarySensorDeviceClass.TAMPER,
entity_category=EntityCategory.DIAGNOSTIC,
name=f" {TAMPER}",
)
def update(self):
"""Return the state of the device."""
self._attr_is_on = self._zone.is_tampered()
class TotalConnectAlarmBinarySensor(BinarySensorEntity):
"""Represent an TotalConnect alarm device binary sensors."""
def __init__(self, location):
"""Initialize the TotalConnect alarm device binary sensor.""" """Initialize the TotalConnect alarm device binary sensor."""
super().__init__(coordinator)
self.entity_description = entity_description
self._location = location self._location = location
self._attr_name = f"{location.location_name}{self.entity_description.name}" self._attr_name = f"{location.location_name}{entity_description.name}"
self._attr_unique_id = f"{location.location_id}_{self.entity_description.key}" self._attr_unique_id = f"{location.location_id}_{entity_description.key}"
self._attr_is_on = None
self._attr_extra_state_attributes = { self._attr_extra_state_attributes = {
"location_id": self._location.location_id, "location_id": location.location_id,
} }
@property
class TotalConnectAlarmLowBatteryBinarySensor(TotalConnectAlarmBinarySensor): def is_on(self) -> bool:
"""Represent an TotalConnect Alarm low battery status.""" """Return the state of the entity."""
return self.entity_description.is_on_fn(self._location)
entity_description: BinarySensorEntityDescription = BinarySensorEntityDescription(
key=LOW_BATTERY,
device_class=BinarySensorDeviceClass.BATTERY,
entity_category=EntityCategory.DIAGNOSTIC,
name=" low battery",
)
def update(self):
"""Return the state of the device."""
self._attr_is_on = self._location.is_low_battery()
class TotalConnectAlarmTamperBinarySensor(TotalConnectAlarmBinarySensor):
"""Represent an TotalConnect alarm tamper status."""
entity_description: BinarySensorEntityDescription = BinarySensorEntityDescription(
key=TAMPER,
device_class=BinarySensorDeviceClass.TAMPER,
entity_category=EntityCategory.DIAGNOSTIC,
name=f" {TAMPER}",
)
def update(self):
"""Return the state of the device."""
self._attr_is_on = self._location.is_cover_tampered()
class TotalConnectAlarmPowerBinarySensor(TotalConnectAlarmBinarySensor):
"""Represent an TotalConnect alarm power status."""
entity_description: BinarySensorEntityDescription = BinarySensorEntityDescription(
key=POWER,
device_class=BinarySensorDeviceClass.POWER,
entity_category=EntityCategory.DIAGNOSTIC,
name=f" {POWER}",
)
def update(self):
"""Return the state of the device."""
self._attr_is_on = not self._location.is_ac_loss()