Home Assistant Unofficial Reference 2024.12.1
device_action.py
Go to the documentation of this file.
1 """Provides device actions for Select."""
2 
3 from __future__ import annotations
4 
5 from contextlib import suppress
6 
7 import voluptuous as vol
8 
10  async_get_entity_registry_entry_or_raise,
11  async_validate_entity_schema,
12 )
13 from homeassistant.const import (
14  ATTR_ENTITY_ID,
15  CONF_DEVICE_ID,
16  CONF_DOMAIN,
17  CONF_ENTITY_ID,
18  CONF_TYPE,
19 )
20 from homeassistant.core import Context, HomeAssistant
21 from homeassistant.exceptions import HomeAssistantError
22 from homeassistant.helpers import entity_registry as er
24 from homeassistant.helpers.entity import get_capability
25 from homeassistant.helpers.typing import ConfigType, TemplateVarsType
26 
27 from .const import (
28  ATTR_CYCLE,
29  ATTR_OPTION,
30  ATTR_OPTIONS,
31  CONF_CYCLE,
32  CONF_OPTION,
33  DOMAIN,
34  SERVICE_SELECT_FIRST,
35  SERVICE_SELECT_LAST,
36  SERVICE_SELECT_NEXT,
37  SERVICE_SELECT_OPTION,
38  SERVICE_SELECT_PREVIOUS,
39 )
40 
41 _ACTION_SCHEMA = vol.Any(
42  cv.DEVICE_ACTION_BASE_SCHEMA.extend(
43  {
44  vol.Required(CONF_TYPE): SERVICE_SELECT_FIRST,
45  vol.Required(CONF_ENTITY_ID): cv.entity_id_or_uuid,
46  }
47  ),
48  cv.DEVICE_ACTION_BASE_SCHEMA.extend(
49  {
50  vol.Required(CONF_TYPE): SERVICE_SELECT_LAST,
51  vol.Required(CONF_ENTITY_ID): cv.entity_id_or_uuid,
52  }
53  ),
54  cv.DEVICE_ACTION_BASE_SCHEMA.extend(
55  {
56  vol.Required(CONF_TYPE): SERVICE_SELECT_NEXT,
57  vol.Required(CONF_ENTITY_ID): cv.entity_id_or_uuid,
58  vol.Optional(CONF_CYCLE, default=True): cv.boolean,
59  }
60  ),
61  cv.DEVICE_ACTION_BASE_SCHEMA.extend(
62  {
63  vol.Required(CONF_TYPE): SERVICE_SELECT_PREVIOUS,
64  vol.Required(CONF_ENTITY_ID): cv.entity_id_or_uuid,
65  vol.Optional(CONF_CYCLE, default=True): cv.boolean,
66  }
67  ),
68  cv.DEVICE_ACTION_BASE_SCHEMA.extend(
69  {
70  vol.Required(CONF_TYPE): SERVICE_SELECT_OPTION,
71  vol.Required(CONF_ENTITY_ID): cv.entity_id_or_uuid,
72  vol.Required(CONF_OPTION): cv.string,
73  }
74  ),
75 )
76 
77 
79  hass: HomeAssistant, config: ConfigType
80 ) -> ConfigType:
81  """Validate config."""
82  return async_validate_entity_schema(hass, config, _ACTION_SCHEMA)
83 
84 
86  hass: HomeAssistant, device_id: str
87 ) -> list[dict[str, str]]:
88  """List device actions for Select devices."""
89  registry = er.async_get(hass)
90  return [
91  {
92  CONF_DEVICE_ID: device_id,
93  CONF_DOMAIN: DOMAIN,
94  CONF_ENTITY_ID: entry.id,
95  CONF_TYPE: service_conf_type,
96  }
97  for service_conf_type in (
98  SERVICE_SELECT_FIRST,
99  SERVICE_SELECT_LAST,
100  SERVICE_SELECT_NEXT,
101  SERVICE_SELECT_OPTION,
102  SERVICE_SELECT_PREVIOUS,
103  )
104  for entry in er.async_entries_for_device(registry, device_id)
105  if entry.domain == DOMAIN
106  ]
107 
108 
110  hass: HomeAssistant,
111  config: ConfigType,
112  variables: TemplateVarsType,
113  context: Context | None,
114 ) -> None:
115  """Execute a device action."""
116  service_data = {ATTR_ENTITY_ID: config[CONF_ENTITY_ID]}
117  if config[CONF_TYPE] == SERVICE_SELECT_OPTION:
118  service_data[ATTR_OPTION] = config[CONF_OPTION]
119  if config[CONF_TYPE] in {SERVICE_SELECT_NEXT, SERVICE_SELECT_PREVIOUS}:
120  service_data[ATTR_CYCLE] = config[CONF_CYCLE]
121 
122  await hass.services.async_call(
123  DOMAIN,
124  config[CONF_TYPE],
125  service_data,
126  blocking=True,
127  context=context,
128  )
129 
130 
132  hass: HomeAssistant, config: ConfigType
133 ) -> dict[str, vol.Schema]:
134  """List action capabilities."""
135  if config[CONF_TYPE] in {SERVICE_SELECT_NEXT, SERVICE_SELECT_PREVIOUS}:
136  return {
137  "extra_fields": vol.Schema(
138  {vol.Optional(CONF_CYCLE, default=True): cv.boolean}
139  )
140  }
141 
142  if config[CONF_TYPE] == SERVICE_SELECT_OPTION:
143  options: list[str] = []
144  with suppress(HomeAssistantError):
146  hass, config[CONF_ENTITY_ID]
147  )
148  options = get_capability(hass, entry.entity_id, ATTR_OPTIONS) or []
149  return {
150  "extra_fields": vol.Schema({vol.Required(CONF_OPTION): vol.In(options)})
151  }
152 
153  return {}
er.RegistryEntry async_get_entity_registry_entry_or_raise(HomeAssistant hass, str entity_registry_id)
Definition: __init__.py:332
ConfigType async_validate_entity_schema(HomeAssistant hass, ConfigType config, VolSchemaType schema)
Definition: __init__.py:344
list[dict[str, str]] async_get_actions(HomeAssistant hass, str device_id)
ConfigType async_validate_action_config(HomeAssistant hass, ConfigType config)
dict[str, vol.Schema] async_get_action_capabilities(HomeAssistant hass, ConfigType config)
None async_call_action_from_config(HomeAssistant hass, ConfigType config, TemplateVarsType variables, Context|None context)
Any|None get_capability(HomeAssistant hass, str entity_id, str capability)
Definition: entity.py:139