Home Assistant Unofficial Reference 2024.12.1
__init__.py
Go to the documentation of this file.
1 """The rest component."""
2 
3 from __future__ import annotations
4 
5 import asyncio
6 from collections.abc import Coroutine
7 import contextlib
8 from datetime import timedelta
9 import logging
10 from typing import Any
11 
12 import httpx
13 import voluptuous as vol
14 
15 from homeassistant.components.binary_sensor import DOMAIN as BINARY_SENSOR_DOMAIN
16 from homeassistant.components.sensor import DOMAIN as SENSOR_DOMAIN
17 from homeassistant.const import (
18  CONF_AUTHENTICATION,
19  CONF_HEADERS,
20  CONF_METHOD,
21  CONF_PARAMS,
22  CONF_PASSWORD,
23  CONF_PAYLOAD,
24  CONF_RESOURCE,
25  CONF_RESOURCE_TEMPLATE,
26  CONF_SCAN_INTERVAL,
27  CONF_TIMEOUT,
28  CONF_USERNAME,
29  CONF_VERIFY_SSL,
30  HTTP_DIGEST_AUTHENTICATION,
31  SERVICE_RELOAD,
32  Platform,
33 )
34 from homeassistant.core import HomeAssistant, ServiceCall, callback
35 from homeassistant.exceptions import HomeAssistantError
36 from homeassistant.helpers import discovery, template
37 from homeassistant.helpers.entity_component import DEFAULT_SCAN_INTERVAL
38 from homeassistant.helpers.reload import (
39  async_integration_yaml_config,
40  async_reload_integration_platforms,
41 )
42 from homeassistant.helpers.typing import ConfigType, DiscoveryInfoType
43 from homeassistant.helpers.update_coordinator import DataUpdateCoordinator
44 from homeassistant.util.async_ import create_eager_task
45 
46 from .const import (
47  CONF_ENCODING,
48  CONF_PAYLOAD_TEMPLATE,
49  CONF_SSL_CIPHER_LIST,
50  COORDINATOR,
51  DEFAULT_SSL_CIPHER_LIST,
52  DOMAIN,
53  PLATFORM_IDX,
54  REST,
55  REST_DATA,
56  REST_IDX,
57 )
58 from .data import RestData
59 from .schema import CONFIG_SCHEMA, RESOURCE_SCHEMA # noqa: F401
60 
61 _LOGGER = logging.getLogger(__name__)
62 
63 PLATFORMS = [
64  Platform.BINARY_SENSOR,
65  Platform.NOTIFY,
66  Platform.SENSOR,
67  Platform.SWITCH,
68 ]
69 
70 COORDINATOR_AWARE_PLATFORMS = [SENSOR_DOMAIN, BINARY_SENSOR_DOMAIN]
71 
72 
73 async def async_setup(hass: HomeAssistant, config: ConfigType) -> bool:
74  """Set up the rest platforms."""
76 
77  async def reload_service_handler(service: ServiceCall) -> None:
78  """Remove all user-defined groups and load new ones from config."""
79  conf = None
80  with contextlib.suppress(HomeAssistantError):
81  conf = await async_integration_yaml_config(hass, DOMAIN)
82  if conf is None:
83  return
84  await async_reload_integration_platforms(hass, DOMAIN, PLATFORMS)
86  await _async_process_config(hass, conf)
87 
88  hass.services.async_register(
89  DOMAIN, SERVICE_RELOAD, reload_service_handler, schema=vol.Schema({})
90  )
91 
92  return await _async_process_config(hass, config)
93 
94 
95 @callback
96 def _async_setup_shared_data(hass: HomeAssistant) -> None:
97  """Create shared data for platform config and rest coordinators."""
98  hass.data[DOMAIN] = {key: [] for key in (REST_DATA, *COORDINATOR_AWARE_PLATFORMS)}
99 
100 
101 async def _async_process_config(hass: HomeAssistant, config: ConfigType) -> bool:
102  """Process rest configuration."""
103  if DOMAIN not in config:
104  return True
105 
106  refresh_coroutines: list[Coroutine[Any, Any, None]] = []
107  load_coroutines: list[Coroutine[Any, Any, None]] = []
108  rest_config: list[ConfigType] = config[DOMAIN]
109  for rest_idx, conf in enumerate(rest_config):
110  scan_interval: timedelta = conf.get(CONF_SCAN_INTERVAL, DEFAULT_SCAN_INTERVAL)
111  resource_template: template.Template | None = conf.get(CONF_RESOURCE_TEMPLATE)
112  payload_template: template.Template | None = conf.get(CONF_PAYLOAD_TEMPLATE)
113  rest = create_rest_data_from_config(hass, conf)
114  coordinator = _rest_coordinator(
115  hass, rest, resource_template, payload_template, scan_interval
116  )
117  refresh_coroutines.append(coordinator.async_refresh())
118  hass.data[DOMAIN][REST_DATA].append({REST: rest, COORDINATOR: coordinator})
119 
120  for platform_domain in COORDINATOR_AWARE_PLATFORMS:
121  if platform_domain not in conf:
122  continue
123 
124  for platform_conf in conf[platform_domain]:
125  hass.data[DOMAIN][platform_domain].append(platform_conf)
126  platform_idx = len(hass.data[DOMAIN][platform_domain]) - 1
127 
128  load_coroutine = discovery.async_load_platform(
129  hass,
130  platform_domain,
131  DOMAIN,
132  {REST_IDX: rest_idx, PLATFORM_IDX: platform_idx},
133  config,
134  )
135  load_coroutines.append(load_coroutine)
136 
137  if refresh_coroutines:
138  await asyncio.gather(*(create_eager_task(coro) for coro in refresh_coroutines))
139 
140  if load_coroutines:
141  await asyncio.gather(*(create_eager_task(coro) for coro in load_coroutines))
142 
143  return True
144 
145 
147  hass: HomeAssistant, platform_domain: str, discovery_info: DiscoveryInfoType
148 ) -> tuple[ConfigType, DataUpdateCoordinator[None], RestData]:
149  """Get the config and coordinator for the platform from discovery."""
150  shared_data = hass.data[DOMAIN][REST_DATA][discovery_info[REST_IDX]]
151  conf: ConfigType = hass.data[DOMAIN][platform_domain][discovery_info[PLATFORM_IDX]]
152  coordinator: DataUpdateCoordinator[None] = shared_data[COORDINATOR]
153  rest: RestData = shared_data[REST]
154  if rest.data is None:
155  await coordinator.async_request_refresh()
156  return conf, coordinator, rest
157 
158 
160  hass: HomeAssistant,
161  rest: RestData,
162  resource_template: template.Template | None,
163  payload_template: template.Template | None,
164  update_interval: timedelta,
165 ) -> DataUpdateCoordinator[None]:
166  """Wrap a DataUpdateCoordinator around the rest object."""
167  if resource_template or payload_template:
168 
169  async def _async_refresh_with_templates() -> None:
170  if resource_template:
171  rest.set_url(resource_template.async_render(parse_result=False))
172  if payload_template:
173  rest.set_payload(payload_template.async_render(parse_result=False))
174  await rest.async_update()
175 
176  update_method = _async_refresh_with_templates
177  else:
178  update_method = rest.async_update
179 
180  return DataUpdateCoordinator(
181  hass,
182  _LOGGER,
183  config_entry=None,
184  name="rest data",
185  update_method=update_method,
186  update_interval=update_interval,
187  )
188 
189 
190 def create_rest_data_from_config(hass: HomeAssistant, config: ConfigType) -> RestData:
191  """Create RestData from config."""
192  resource: str | None = config.get(CONF_RESOURCE)
193  resource_template: template.Template | None = config.get(CONF_RESOURCE_TEMPLATE)
194  method: str = config[CONF_METHOD]
195  payload: str | None = config.get(CONF_PAYLOAD)
196  payload_template: template.Template | None = config.get(CONF_PAYLOAD_TEMPLATE)
197  verify_ssl: bool = config[CONF_VERIFY_SSL]
198  ssl_cipher_list: str = config.get(CONF_SSL_CIPHER_LIST, DEFAULT_SSL_CIPHER_LIST)
199  username: str | None = config.get(CONF_USERNAME)
200  password: str | None = config.get(CONF_PASSWORD)
201  headers: dict[str, str] | None = config.get(CONF_HEADERS)
202  params: dict[str, str] | None = config.get(CONF_PARAMS)
203  timeout: int = config[CONF_TIMEOUT]
204  encoding: str = config[CONF_ENCODING]
205  if resource_template is not None:
206  resource = resource_template.async_render(parse_result=False)
207 
208  if payload_template is not None:
209  payload = payload_template.async_render(parse_result=False)
210 
211  if not resource:
212  raise HomeAssistantError("Resource not set for RestData")
213 
214  auth: httpx.DigestAuth | tuple[str, str] | None = None
215  if username and password:
216  if config.get(CONF_AUTHENTICATION) == HTTP_DIGEST_AUTHENTICATION:
217  auth = httpx.DigestAuth(username, password)
218  else:
219  auth = (username, password)
220 
221  return RestData(
222  hass,
223  method,
224  resource,
225  encoding,
226  auth,
227  headers,
228  params,
229  payload,
230  verify_ssl,
231  ssl_cipher_list,
232  timeout,
233  )
bool _async_process_config(HomeAssistant hass, ConfigType config)
Definition: __init__.py:101
RestData create_rest_data_from_config(HomeAssistant hass, ConfigType config)
Definition: __init__.py:190
None _async_setup_shared_data(HomeAssistant hass)
Definition: __init__.py:96
bool async_setup(HomeAssistant hass, ConfigType config)
Definition: __init__.py:73
tuple[ConfigType, DataUpdateCoordinator[None], RestData] async_get_config_and_coordinator(HomeAssistant hass, str platform_domain, DiscoveryInfoType discovery_info)
Definition: __init__.py:148
DataUpdateCoordinator[None] _rest_coordinator(HomeAssistant hass, RestData rest, template.Template|None resource_template, template.Template|None payload_template, timedelta update_interval)
Definition: __init__.py:165
None async_reload_integration_platforms(HomeAssistant hass, str integration_domain, Iterable[str] platform_domains)
Definition: reload.py:30
ConfigType|None async_integration_yaml_config(HomeAssistant hass, str integration_name)
Definition: reload.py:142