Home Assistant Unofficial Reference 2024.12.1
device_action.py
Go to the documentation of this file.
1 """Provides device automations for Alarm control panel."""
2 
3 from __future__ import annotations
4 
5 from typing import Final
6 
7 import voluptuous as vol
8 
9 from homeassistant.components.device_automation import async_validate_entity_schema
10 from homeassistant.const import (
11  ATTR_CODE,
12  ATTR_ENTITY_ID,
13  CONF_CODE,
14  CONF_DEVICE_ID,
15  CONF_DOMAIN,
16  CONF_ENTITY_ID,
17  CONF_TYPE,
18  SERVICE_ALARM_ARM_AWAY,
19  SERVICE_ALARM_ARM_HOME,
20  SERVICE_ALARM_ARM_NIGHT,
21  SERVICE_ALARM_ARM_VACATION,
22  SERVICE_ALARM_DISARM,
23  SERVICE_ALARM_TRIGGER,
24 )
25 from homeassistant.core import Context, HomeAssistant
26 from homeassistant.helpers import entity_registry as er
28 from homeassistant.helpers.entity import get_supported_features
29 from homeassistant.helpers.typing import ConfigType, TemplateVarsType
30 
31 from . import ATTR_CODE_ARM_REQUIRED, DOMAIN
32 from .const import AlarmControlPanelEntityFeature
33 
34 ACTION_TYPES: Final[set[str]] = {
35  "arm_away",
36  "arm_home",
37  "arm_night",
38  "arm_vacation",
39  "disarm",
40  "trigger",
41 }
42 
43 _ACTION_SCHEMA: Final = cv.DEVICE_ACTION_BASE_SCHEMA.extend(
44  {
45  vol.Required(CONF_TYPE): vol.In(ACTION_TYPES),
46  vol.Required(CONF_ENTITY_ID): cv.entity_id_or_uuid,
47  vol.Optional(CONF_CODE): cv.string,
48  }
49 )
50 
51 
53  hass: HomeAssistant, config: ConfigType
54 ) -> ConfigType:
55  """Validate config."""
56  return async_validate_entity_schema(hass, config, _ACTION_SCHEMA)
57 
58 
60  hass: HomeAssistant, device_id: str
61 ) -> list[dict[str, str]]:
62  """List device actions for Alarm control panel devices."""
63  registry = er.async_get(hass)
64  actions = []
65 
66  # Get all the integrations entities for this device
67  for entry in er.async_entries_for_device(registry, device_id):
68  if entry.domain != DOMAIN:
69  continue
70 
71  supported_features = get_supported_features(hass, entry.entity_id)
72 
73  base_action: dict = {
74  CONF_DEVICE_ID: device_id,
75  CONF_DOMAIN: DOMAIN,
76  CONF_ENTITY_ID: entry.id,
77  }
78 
79  # Add actions for each entity that belongs to this integration
80  if supported_features & AlarmControlPanelEntityFeature.ARM_AWAY:
81  actions.append({**base_action, CONF_TYPE: "arm_away"})
82  if supported_features & AlarmControlPanelEntityFeature.ARM_HOME:
83  actions.append({**base_action, CONF_TYPE: "arm_home"})
84  if supported_features & AlarmControlPanelEntityFeature.ARM_NIGHT:
85  actions.append({**base_action, CONF_TYPE: "arm_night"})
86  if supported_features & AlarmControlPanelEntityFeature.ARM_VACATION:
87  actions.append({**base_action, CONF_TYPE: "arm_vacation"})
88  actions.append({**base_action, CONF_TYPE: "disarm"})
89  if supported_features & AlarmControlPanelEntityFeature.TRIGGER:
90  actions.append({**base_action, CONF_TYPE: "trigger"})
91 
92  return actions
93 
94 
96  hass: HomeAssistant,
97  config: ConfigType,
98  variables: TemplateVarsType,
99  context: Context | None,
100 ) -> None:
101  """Execute a device action."""
102  service_data = {ATTR_ENTITY_ID: config[CONF_ENTITY_ID]}
103  if CONF_CODE in config:
104  service_data[ATTR_CODE] = config[CONF_CODE]
105 
106  if config[CONF_TYPE] == "arm_away":
107  service = SERVICE_ALARM_ARM_AWAY
108  elif config[CONF_TYPE] == "arm_home":
109  service = SERVICE_ALARM_ARM_HOME
110  elif config[CONF_TYPE] == "arm_night":
111  service = SERVICE_ALARM_ARM_NIGHT
112  elif config[CONF_TYPE] == "arm_vacation":
113  service = SERVICE_ALARM_ARM_VACATION
114  elif config[CONF_TYPE] == "disarm":
115  service = SERVICE_ALARM_DISARM
116  elif config[CONF_TYPE] == "trigger":
117  service = SERVICE_ALARM_TRIGGER
118 
119  await hass.services.async_call(
120  DOMAIN, service, service_data, blocking=True, context=context
121  )
122 
123 
125  hass: HomeAssistant, config: ConfigType
126 ) -> dict[str, vol.Schema]:
127  """List action capabilities."""
128  # We need to refer to the state directly because ATTR_CODE_ARM_REQUIRED is not a
129  # capability attribute
130  registry = er.async_get(hass)
131  entity_id = er.async_resolve_entity_id(registry, config[CONF_ENTITY_ID])
132  state = hass.states.get(entity_id) if entity_id else None
133  code_required = state.attributes.get(ATTR_CODE_ARM_REQUIRED) if state else False
134 
135  if config[CONF_TYPE] == "trigger" or (
136  config[CONF_TYPE] != "disarm" and not code_required
137  ):
138  return {}
139 
140  return {"extra_fields": vol.Schema({vol.Optional(CONF_CODE): str})}
dict[str, vol.Schema] async_get_action_capabilities(HomeAssistant hass, ConfigType config)
ConfigType async_validate_action_config(HomeAssistant hass, ConfigType config)
list[dict[str, str]] async_get_actions(HomeAssistant hass, str device_id)
None async_call_action_from_config(HomeAssistant hass, ConfigType config, TemplateVarsType variables, Context|None context)
ConfigType async_validate_entity_schema(HomeAssistant hass, ConfigType config, VolSchemaType schema)
Definition: __init__.py:344
int get_supported_features(HomeAssistant hass, str entity_id)
Definition: entity.py:169