core/homeassistant/components/zoneminder/sensor.py

181 lines
4.9 KiB
Python

"""Support for ZoneMinder sensors."""
from __future__ import annotations
import logging
import voluptuous as vol
from zoneminder.monitor import TimePeriod
from homeassistant.components.sensor import (
PLATFORM_SCHEMA,
SensorEntity,
SensorEntityDescription,
)
from homeassistant.const import CONF_MONITORED_CONDITIONS
from homeassistant.core import HomeAssistant
import homeassistant.helpers.config_validation as cv
from homeassistant.helpers.entity_platform import AddEntitiesCallback
from homeassistant.helpers.typing import ConfigType, DiscoveryInfoType
from . import DOMAIN as ZONEMINDER_DOMAIN
_LOGGER = logging.getLogger(__name__)
CONF_INCLUDE_ARCHIVED = "include_archived"
DEFAULT_INCLUDE_ARCHIVED = False
SENSOR_TYPES: tuple[SensorEntityDescription, ...] = (
SensorEntityDescription(
key="all",
name="Events",
),
SensorEntityDescription(
key="hour",
name="Events Last Hour",
),
SensorEntityDescription(
key="day",
name="Events Last Day",
),
SensorEntityDescription(
key="week",
name="Events Last Week",
),
SensorEntityDescription(
key="month",
name="Events Last Month",
),
)
SENSOR_KEYS: list[str] = [desc.key for desc in SENSOR_TYPES]
PLATFORM_SCHEMA = PLATFORM_SCHEMA.extend(
{
vol.Optional(
CONF_INCLUDE_ARCHIVED, default=DEFAULT_INCLUDE_ARCHIVED
): cv.boolean,
vol.Optional(CONF_MONITORED_CONDITIONS, default=["all"]): vol.All(
cv.ensure_list, [vol.In(SENSOR_KEYS)]
),
}
)
def setup_platform(
hass: HomeAssistant,
config: ConfigType,
add_entities: AddEntitiesCallback,
discovery_info: DiscoveryInfoType | None = None,
) -> None:
"""Set up the ZoneMinder sensor platform."""
include_archived = config[CONF_INCLUDE_ARCHIVED]
monitored_conditions = config[CONF_MONITORED_CONDITIONS]
sensors: list[SensorEntity] = []
for zm_client in hass.data[ZONEMINDER_DOMAIN].values():
if not (monitors := zm_client.get_monitors()):
_LOGGER.warning("Could not fetch any monitors from ZoneMinder")
for monitor in monitors:
sensors.append(ZMSensorMonitors(monitor))
sensors.extend(
[
ZMSensorEvents(monitor, include_archived, description)
for description in SENSOR_TYPES
if description.key in monitored_conditions
]
)
sensors.append(ZMSensorRunState(zm_client))
add_entities(sensors)
class ZMSensorMonitors(SensorEntity):
"""Get the status of each ZoneMinder monitor."""
def __init__(self, monitor):
"""Initialize monitor sensor."""
self._monitor = monitor
self._state = None
self._is_available = None
@property
def name(self):
"""Return the name of the sensor."""
return f"{self._monitor.name} Status"
@property
def native_value(self):
"""Return the state of the sensor."""
return self._state
@property
def available(self):
"""Return True if Monitor is available."""
return self._is_available
def update(self):
"""Update the sensor."""
if not (state := self._monitor.function):
self._state = None
else:
self._state = state.value
self._is_available = self._monitor.is_available
class ZMSensorEvents(SensorEntity):
"""Get the number of events for each monitor."""
_attr_native_unit_of_measurement = "Events"
def __init__(self, monitor, include_archived, description: SensorEntityDescription):
"""Initialize event sensor."""
self.entity_description = description
self._monitor = monitor
self._include_archived = include_archived
self.time_period = TimePeriod.get_time_period(description.key)
@property
def name(self):
"""Return the name of the sensor."""
return f"{self._monitor.name} {self.time_period.title}"
def update(self):
"""Update the sensor."""
self._attr_native_value = self._monitor.get_events(
self.time_period, self._include_archived
)
class ZMSensorRunState(SensorEntity):
"""Get the ZoneMinder run state."""
def __init__(self, client):
"""Initialize run state sensor."""
self._state = None
self._is_available = None
self._client = client
@property
def name(self):
"""Return the name of the sensor."""
return "Run State"
@property
def native_value(self):
"""Return the state of the sensor."""
return self._state
@property
def available(self):
"""Return True if ZoneMinder is available."""
return self._is_available
def update(self):
"""Update the sensor."""
self._state = self._client.get_active_state()
self._is_available = self._client.is_available