core/homeassistant/components/onvif/sensor.py

103 lines
3.8 KiB
Python

"""Support for ONVIF binary sensors."""
from __future__ import annotations
from datetime import date, datetime
from decimal import Decimal
from homeassistant.components.sensor import RestoreSensor, SensorDeviceClass
from homeassistant.config_entries import ConfigEntry
from homeassistant.core import HomeAssistant, callback
from homeassistant.helpers import entity_registry as er
from homeassistant.helpers.entity_platform import AddEntitiesCallback
from homeassistant.helpers.typing import StateType
from homeassistant.util.enum import try_parse_enum
from .base import ONVIFBaseEntity
from .const import DOMAIN
from .device import ONVIFDevice
async def async_setup_entry(
hass: HomeAssistant,
config_entry: ConfigEntry,
async_add_entities: AddEntitiesCallback,
) -> None:
"""Set up a ONVIF binary sensor."""
device: ONVIFDevice = hass.data[DOMAIN][config_entry.unique_id]
entities = {
event.uid: ONVIFSensor(event.uid, device)
for event in device.events.get_platform("sensor")
}
ent_reg = er.async_get(hass)
for entry in er.async_entries_for_config_entry(ent_reg, config_entry.entry_id):
if entry.domain == "sensor" and entry.unique_id not in entities:
entities[entry.unique_id] = ONVIFSensor(entry.unique_id, device, entry)
async_add_entities(entities.values())
uids_by_platform = device.events.get_uids_by_platform("sensor")
@callback
def async_check_entities() -> None:
"""Check if we have added an entity for the event."""
nonlocal uids_by_platform
if not (missing := uids_by_platform.difference(entities)):
return
new_entities: dict[str, ONVIFSensor] = {
uid: ONVIFSensor(uid, device) for uid in missing
}
if new_entities:
entities.update(new_entities)
async_add_entities(new_entities.values())
device.events.async_add_listener(async_check_entities)
class ONVIFSensor(ONVIFBaseEntity, RestoreSensor):
"""Representation of a ONVIF sensor event."""
_attr_should_poll = False
def __init__(
self, uid, device: ONVIFDevice, entry: er.RegistryEntry | None = None
) -> None:
"""Initialize the ONVIF binary sensor."""
self._attr_unique_id = uid
if entry is not None:
self._attr_device_class = try_parse_enum(
SensorDeviceClass, entry.original_device_class
)
self._attr_entity_category = entry.entity_category
self._attr_name = entry.name
self._attr_native_unit_of_measurement = entry.unit_of_measurement
else:
event = device.events.get_uid(uid)
assert event
self._attr_device_class = try_parse_enum(
SensorDeviceClass, event.device_class
)
self._attr_entity_category = event.entity_category
self._attr_entity_registry_enabled_default = event.entity_enabled
self._attr_name = f"{device.name} {event.name}"
self._attr_native_unit_of_measurement = event.unit_of_measurement
self._attr_native_value = event.value
super().__init__(device)
@property
def native_value(self) -> StateType | date | datetime | Decimal:
"""Return the value reported by the sensor."""
assert self._attr_unique_id is not None
if (event := self.device.events.get_uid(self._attr_unique_id)) is not None:
return event.value
return self._attr_native_value
async def async_added_to_hass(self) -> None:
"""Connect to dispatcher listening for entity data notifications."""
self.async_on_remove(
self.device.events.async_add_listener(self.async_write_ha_state)
)
if (last_sensor_data := await self.async_get_last_sensor_data()) is not None:
self._attr_native_value = last_sensor_data.native_value