"""Selectors for Home Assistant.""" from __future__ import annotations from collections.abc import Callable from typing import Any, cast import voluptuous as vol from homeassistant.const import CONF_MODE, CONF_UNIT_OF_MEASUREMENT from homeassistant.util import decorator SELECTORS = decorator.Registry() def validate_selector(config: Any) -> dict: """Validate a selector.""" if not isinstance(config, dict): raise vol.Invalid("Expected a dictionary") if len(config) != 1: raise vol.Invalid(f"Only one type can be specified. Found {', '.join(config)}") selector_type = list(config)[0] if (selector_class := SELECTORS.get(selector_type)) is None: raise vol.Invalid(f"Unknown selector type {selector_type} found") # Selectors can be empty if config[selector_type] is None: return {selector_type: {}} return { selector_type: cast(dict, selector_class.CONFIG_SCHEMA(config[selector_type])) } class Selector: """Base class for selectors.""" CONFIG_SCHEMA: Callable @SELECTORS.register("entity") class EntitySelector(Selector): """Selector of a single entity.""" CONFIG_SCHEMA = vol.Schema( { # Integration that provided the entity vol.Optional("integration"): str, # Domain the entity belongs to vol.Optional("domain"): str, # Device class of the entity vol.Optional("device_class"): str, } ) @SELECTORS.register("device") class DeviceSelector(Selector): """Selector of a single device.""" CONFIG_SCHEMA = vol.Schema( { # Integration linked to it with a config entry vol.Optional("integration"): str, # Manufacturer of device vol.Optional("manufacturer"): str, # Model of device vol.Optional("model"): str, # Device has to contain entities matching this selector vol.Optional("entity"): EntitySelector.CONFIG_SCHEMA, } ) @SELECTORS.register("area") class AreaSelector(Selector): """Selector of a single area.""" CONFIG_SCHEMA = vol.Schema( { vol.Optional("entity"): vol.Schema( { vol.Optional("domain"): str, vol.Optional("device_class"): str, vol.Optional("integration"): str, } ), vol.Optional("device"): vol.Schema( { vol.Optional("integration"): str, vol.Optional("manufacturer"): str, vol.Optional("model"): str, } ), } ) @SELECTORS.register("number") class NumberSelector(Selector): """Selector of a numeric value.""" CONFIG_SCHEMA = vol.Schema( { vol.Required("min"): vol.Coerce(float), vol.Required("max"): vol.Coerce(float), vol.Optional("step", default=1): vol.All( vol.Coerce(float), vol.Range(min=1e-3) ), vol.Optional(CONF_UNIT_OF_MEASUREMENT): str, vol.Optional(CONF_MODE, default="slider"): vol.In(["box", "slider"]), } ) @SELECTORS.register("addon") class AddonSelector(Selector): """Selector of a add-on.""" CONFIG_SCHEMA = vol.Schema({}) @SELECTORS.register("boolean") class BooleanSelector(Selector): """Selector of a boolean value.""" CONFIG_SCHEMA = vol.Schema({}) @SELECTORS.register("time") class TimeSelector(Selector): """Selector of a time value.""" CONFIG_SCHEMA = vol.Schema({}) @SELECTORS.register("target") class TargetSelector(Selector): """Selector of a target value (area ID, device ID, entity ID etc). Value should follow cv.ENTITY_SERVICE_FIELDS format. """ CONFIG_SCHEMA = vol.Schema( { vol.Optional("entity"): vol.Schema( { vol.Optional("domain"): str, vol.Optional("device_class"): str, vol.Optional("integration"): str, } ), vol.Optional("device"): vol.Schema( { vol.Optional("integration"): str, vol.Optional("manufacturer"): str, vol.Optional("model"): str, } ), } ) @SELECTORS.register("action") class ActionSelector(Selector): """Selector of an action sequence (script syntax).""" CONFIG_SCHEMA = vol.Schema({}) @SELECTORS.register("object") class ObjectSelector(Selector): """Selector for an arbitrary object.""" CONFIG_SCHEMA = vol.Schema({}) @SELECTORS.register("text") class StringSelector(Selector): """Selector for a multi-line text string.""" CONFIG_SCHEMA = vol.Schema({vol.Optional("multiline", default=False): bool}) @SELECTORS.register("select") class SelectSelector(Selector): """Selector for an single-choice input select.""" CONFIG_SCHEMA = vol.Schema( {vol.Required("options"): vol.All([str], vol.Length(min=1))} )