core/homeassistant/components/siren/__init__.py

226 lines
7.4 KiB
Python

"""Component to interface with various sirens/chimes."""
from __future__ import annotations
from datetime import timedelta
from functools import partial
import logging
from typing import TYPE_CHECKING, Any, TypedDict, cast, final
import voluptuous as vol
from homeassistant.config_entries import ConfigEntry
from homeassistant.const import SERVICE_TOGGLE, SERVICE_TURN_OFF, SERVICE_TURN_ON
from homeassistant.core import HomeAssistant, ServiceCall
import homeassistant.helpers.config_validation as cv
from homeassistant.helpers.config_validation import ( # noqa: F401
PLATFORM_SCHEMA,
PLATFORM_SCHEMA_BASE,
)
from homeassistant.helpers.deprecation import (
all_with_deprecated_constants,
check_if_deprecated_constant,
dir_with_deprecated_constants,
)
from homeassistant.helpers.entity import ToggleEntity, ToggleEntityDescription
from homeassistant.helpers.entity_component import EntityComponent
from homeassistant.helpers.typing import ConfigType
from .const import ( # noqa: F401
_DEPRECATED_SUPPORT_DURATION,
_DEPRECATED_SUPPORT_TONES,
_DEPRECATED_SUPPORT_TURN_OFF,
_DEPRECATED_SUPPORT_TURN_ON,
_DEPRECATED_SUPPORT_VOLUME_SET,
ATTR_AVAILABLE_TONES,
ATTR_DURATION,
ATTR_TONE,
ATTR_VOLUME_LEVEL,
DOMAIN,
SirenEntityFeature,
)
if TYPE_CHECKING:
from functools import cached_property
else:
from homeassistant.backports.functools import cached_property
_LOGGER = logging.getLogger(__name__)
SCAN_INTERVAL = timedelta(seconds=60)
TURN_ON_SCHEMA = {
vol.Optional(ATTR_TONE): vol.Any(vol.Coerce(int), cv.string),
vol.Optional(ATTR_DURATION): cv.positive_int,
vol.Optional(ATTR_VOLUME_LEVEL): cv.small_float,
}
class SirenTurnOnServiceParameters(TypedDict, total=False):
"""Represent possible parameters to siren.turn_on service data dict type."""
tone: int | str
duration: int
volume_level: float
# mypy: disallow-any-generics
def process_turn_on_params(
siren: SirenEntity, params: SirenTurnOnServiceParameters
) -> SirenTurnOnServiceParameters:
"""Process turn_on service params.
Filters out unsupported params and validates the rest.
"""
if not siren.supported_features & SirenEntityFeature.TONES:
params.pop(ATTR_TONE, None)
elif (tone := params.get(ATTR_TONE)) is not None:
# Raise an exception if the specified tone isn't available
is_tone_dict_value = bool(
isinstance(siren.available_tones, dict)
and tone in siren.available_tones.values()
)
if (
not siren.available_tones
or tone not in siren.available_tones
and not is_tone_dict_value
):
raise ValueError(
f"Invalid tone specified for entity {siren.entity_id}: {tone}, "
"check the available_tones attribute for valid tones to pass in"
)
# If available tones is a dict, and the tone provided is a dict value, we need
# to transform it to the corresponding dict key before returning
if is_tone_dict_value:
assert isinstance(siren.available_tones, dict)
params[ATTR_TONE] = next(
key for key, value in siren.available_tones.items() if value == tone
)
if not siren.supported_features & SirenEntityFeature.DURATION:
params.pop(ATTR_DURATION, None)
if not siren.supported_features & SirenEntityFeature.VOLUME_SET:
params.pop(ATTR_VOLUME_LEVEL, None)
return params
async def async_setup(hass: HomeAssistant, config: ConfigType) -> bool:
"""Set up siren devices."""
component = hass.data[DOMAIN] = EntityComponent[SirenEntity](
_LOGGER, DOMAIN, hass, SCAN_INTERVAL
)
await component.async_setup(config)
async def async_handle_turn_on_service(
siren: SirenEntity, call: ServiceCall
) -> None:
"""Handle turning a siren on."""
data = {
k: v
for k, v in call.data.items()
if k in (ATTR_TONE, ATTR_DURATION, ATTR_VOLUME_LEVEL)
}
await siren.async_turn_on(
**process_turn_on_params(siren, cast(SirenTurnOnServiceParameters, data))
)
component.async_register_entity_service(
SERVICE_TURN_ON,
TURN_ON_SCHEMA,
async_handle_turn_on_service,
[SirenEntityFeature.TURN_ON],
)
component.async_register_entity_service(
SERVICE_TURN_OFF, {}, "async_turn_off", [SirenEntityFeature.TURN_OFF]
)
component.async_register_entity_service(
SERVICE_TOGGLE,
{},
"async_toggle",
[SirenEntityFeature.TURN_ON | SirenEntityFeature.TURN_OFF],
)
return True
async def async_setup_entry(hass: HomeAssistant, entry: ConfigEntry) -> bool:
"""Set up a config entry."""
component: EntityComponent[SirenEntity] = hass.data[DOMAIN]
return await component.async_setup_entry(entry)
async def async_unload_entry(hass: HomeAssistant, entry: ConfigEntry) -> bool:
"""Unload a config entry."""
component: EntityComponent[SirenEntity] = hass.data[DOMAIN]
return await component.async_unload_entry(entry)
class SirenEntityDescription(ToggleEntityDescription, frozen_or_thawed=True):
"""A class that describes siren entities."""
available_tones: list[int | str] | dict[int, str] | None = None
CACHED_PROPERTIES_WITH_ATTR_ = {
"available_tones",
"supported_features",
}
class SirenEntity(ToggleEntity, cached_properties=CACHED_PROPERTIES_WITH_ATTR_):
"""Representation of a siren device."""
_entity_component_unrecorded_attributes = frozenset({ATTR_AVAILABLE_TONES})
entity_description: SirenEntityDescription
_attr_available_tones: list[int | str] | dict[int, str] | None
_attr_supported_features: SirenEntityFeature = SirenEntityFeature(0)
@final
@property
def capability_attributes(self) -> dict[str, Any] | None:
"""Return capability attributes."""
if (
self.supported_features & SirenEntityFeature.TONES
and self.available_tones is not None
):
return {ATTR_AVAILABLE_TONES: self.available_tones}
return None
@cached_property
def available_tones(self) -> list[int | str] | dict[int, str] | None:
"""Return a list of available tones.
Requires SirenEntityFeature.TONES.
"""
if hasattr(self, "_attr_available_tones"):
return self._attr_available_tones
if hasattr(self, "entity_description"):
return self.entity_description.available_tones
return None
@cached_property
def supported_features(self) -> SirenEntityFeature:
"""Return the list of supported features."""
features = self._attr_supported_features
if type(features) is int: # noqa: E721
new_features = SirenEntityFeature(features)
self._report_deprecated_supported_features_values(new_features)
return new_features
return features
# As we import deprecated constants from the const module, we need to add these two functions
# otherwise this module will be logged for using deprecated constants and not the custom component
# These can be removed if no deprecated constant are in this module anymore
__getattr__ = partial(check_if_deprecated_constant, module_globals=globals())
__dir__ = partial(
dir_with_deprecated_constants, module_globals_keys=[*globals().keys()]
)
__all__ = all_with_deprecated_constants(globals())