223 lines
7.3 KiB
Python
223 lines
7.3 KiB
Python
"""The rest component."""
|
|
from __future__ import annotations
|
|
|
|
import asyncio
|
|
from collections.abc import Coroutine
|
|
import contextlib
|
|
from datetime import timedelta
|
|
import logging
|
|
from typing import Any
|
|
|
|
import httpx
|
|
import voluptuous as vol
|
|
|
|
from homeassistant.components.binary_sensor import DOMAIN as BINARY_SENSOR_DOMAIN
|
|
from homeassistant.components.sensor import DOMAIN as SENSOR_DOMAIN
|
|
from homeassistant.const import (
|
|
CONF_AUTHENTICATION,
|
|
CONF_HEADERS,
|
|
CONF_METHOD,
|
|
CONF_PARAMS,
|
|
CONF_PASSWORD,
|
|
CONF_PAYLOAD,
|
|
CONF_RESOURCE,
|
|
CONF_RESOURCE_TEMPLATE,
|
|
CONF_SCAN_INTERVAL,
|
|
CONF_TIMEOUT,
|
|
CONF_USERNAME,
|
|
CONF_VERIFY_SSL,
|
|
HTTP_DIGEST_AUTHENTICATION,
|
|
SERVICE_RELOAD,
|
|
Platform,
|
|
)
|
|
from homeassistant.core import HomeAssistant, ServiceCall, callback
|
|
from homeassistant.exceptions import HomeAssistantError
|
|
from homeassistant.helpers import discovery, template
|
|
from homeassistant.helpers.entity_component import DEFAULT_SCAN_INTERVAL
|
|
from homeassistant.helpers.reload import (
|
|
async_integration_yaml_config,
|
|
async_reload_integration_platforms,
|
|
)
|
|
from homeassistant.helpers.typing import ConfigType, DiscoveryInfoType
|
|
from homeassistant.helpers.update_coordinator import DataUpdateCoordinator
|
|
|
|
from .const import (
|
|
CONF_ENCODING,
|
|
CONF_SSL_CIPHER_LIST,
|
|
COORDINATOR,
|
|
DEFAULT_SSL_CIPHER_LIST,
|
|
DOMAIN,
|
|
PLATFORM_IDX,
|
|
REST,
|
|
REST_DATA,
|
|
REST_IDX,
|
|
)
|
|
from .data import RestData
|
|
from .schema import CONFIG_SCHEMA, RESOURCE_SCHEMA # noqa: F401
|
|
|
|
_LOGGER = logging.getLogger(__name__)
|
|
|
|
PLATFORMS = [
|
|
Platform.BINARY_SENSOR,
|
|
Platform.NOTIFY,
|
|
Platform.SENSOR,
|
|
Platform.SWITCH,
|
|
]
|
|
|
|
COORDINATOR_AWARE_PLATFORMS = [SENSOR_DOMAIN, BINARY_SENSOR_DOMAIN]
|
|
|
|
|
|
async def async_setup(hass: HomeAssistant, config: ConfigType) -> bool:
|
|
"""Set up the rest platforms."""
|
|
_async_setup_shared_data(hass)
|
|
|
|
async def reload_service_handler(service: ServiceCall) -> None:
|
|
"""Remove all user-defined groups and load new ones from config."""
|
|
conf = None
|
|
with contextlib.suppress(HomeAssistantError):
|
|
conf = await async_integration_yaml_config(hass, DOMAIN)
|
|
if conf is None:
|
|
return
|
|
await async_reload_integration_platforms(hass, DOMAIN, PLATFORMS)
|
|
_async_setup_shared_data(hass)
|
|
await _async_process_config(hass, conf)
|
|
|
|
hass.services.async_register(
|
|
DOMAIN, SERVICE_RELOAD, reload_service_handler, schema=vol.Schema({})
|
|
)
|
|
|
|
return await _async_process_config(hass, config)
|
|
|
|
|
|
@callback
|
|
def _async_setup_shared_data(hass: HomeAssistant) -> None:
|
|
"""Create shared data for platform config and rest coordinators."""
|
|
hass.data[DOMAIN] = {key: [] for key in (REST_DATA, *COORDINATOR_AWARE_PLATFORMS)}
|
|
|
|
|
|
async def _async_process_config(hass: HomeAssistant, config: ConfigType) -> bool:
|
|
"""Process rest configuration."""
|
|
if DOMAIN not in config:
|
|
return True
|
|
|
|
refresh_coroutines: list[Coroutine[Any, Any, None]] = []
|
|
load_coroutines: list[Coroutine[Any, Any, None]] = []
|
|
rest_config: list[ConfigType] = config[DOMAIN]
|
|
for rest_idx, conf in enumerate(rest_config):
|
|
scan_interval: timedelta = conf.get(CONF_SCAN_INTERVAL, DEFAULT_SCAN_INTERVAL)
|
|
resource_template: template.Template | None = conf.get(CONF_RESOURCE_TEMPLATE)
|
|
rest = create_rest_data_from_config(hass, conf)
|
|
coordinator = _rest_coordinator(hass, rest, resource_template, scan_interval)
|
|
refresh_coroutines.append(coordinator.async_refresh())
|
|
hass.data[DOMAIN][REST_DATA].append({REST: rest, COORDINATOR: coordinator})
|
|
|
|
for platform_domain in COORDINATOR_AWARE_PLATFORMS:
|
|
if platform_domain not in conf:
|
|
continue
|
|
|
|
for platform_conf in conf[platform_domain]:
|
|
hass.data[DOMAIN][platform_domain].append(platform_conf)
|
|
platform_idx = len(hass.data[DOMAIN][platform_domain]) - 1
|
|
|
|
load_coroutine = discovery.async_load_platform(
|
|
hass,
|
|
platform_domain,
|
|
DOMAIN,
|
|
{REST_IDX: rest_idx, PLATFORM_IDX: platform_idx},
|
|
config,
|
|
)
|
|
load_coroutines.append(load_coroutine)
|
|
|
|
if refresh_coroutines:
|
|
await asyncio.gather(*refresh_coroutines)
|
|
|
|
if load_coroutines:
|
|
await asyncio.gather(*load_coroutines)
|
|
|
|
return True
|
|
|
|
|
|
async def async_get_config_and_coordinator(
|
|
hass: HomeAssistant, platform_domain: str, discovery_info: DiscoveryInfoType
|
|
) -> tuple[ConfigType, DataUpdateCoordinator[None], RestData]:
|
|
"""Get the config and coordinator for the platform from discovery."""
|
|
shared_data = hass.data[DOMAIN][REST_DATA][discovery_info[REST_IDX]]
|
|
conf: ConfigType = hass.data[DOMAIN][platform_domain][discovery_info[PLATFORM_IDX]]
|
|
coordinator: DataUpdateCoordinator[None] = shared_data[COORDINATOR]
|
|
rest: RestData = shared_data[REST]
|
|
if rest.data is None:
|
|
await coordinator.async_request_refresh()
|
|
return conf, coordinator, rest
|
|
|
|
|
|
def _rest_coordinator(
|
|
hass: HomeAssistant,
|
|
rest: RestData,
|
|
resource_template: template.Template | None,
|
|
update_interval: timedelta,
|
|
) -> DataUpdateCoordinator[None]:
|
|
"""Wrap a DataUpdateCoordinator around the rest object."""
|
|
if resource_template:
|
|
|
|
async def _async_refresh_with_resource_template() -> None:
|
|
rest.set_url(resource_template.async_render(parse_result=False))
|
|
await rest.async_update()
|
|
|
|
update_method = _async_refresh_with_resource_template
|
|
else:
|
|
update_method = rest.async_update
|
|
|
|
return DataUpdateCoordinator(
|
|
hass,
|
|
_LOGGER,
|
|
name="rest data",
|
|
update_method=update_method,
|
|
update_interval=update_interval,
|
|
)
|
|
|
|
|
|
def create_rest_data_from_config(hass: HomeAssistant, config: ConfigType) -> RestData:
|
|
"""Create RestData from config."""
|
|
resource: str | None = config.get(CONF_RESOURCE)
|
|
resource_template: template.Template | None = config.get(CONF_RESOURCE_TEMPLATE)
|
|
method: str = config[CONF_METHOD]
|
|
payload: str | None = config.get(CONF_PAYLOAD)
|
|
verify_ssl: bool = config[CONF_VERIFY_SSL]
|
|
ssl_cipher_list: str = config.get(CONF_SSL_CIPHER_LIST, DEFAULT_SSL_CIPHER_LIST)
|
|
username: str | None = config.get(CONF_USERNAME)
|
|
password: str | None = config.get(CONF_PASSWORD)
|
|
headers: dict[str, str] | None = config.get(CONF_HEADERS)
|
|
params: dict[str, str] | None = config.get(CONF_PARAMS)
|
|
timeout: int = config[CONF_TIMEOUT]
|
|
encoding: str = config[CONF_ENCODING]
|
|
if resource_template is not None:
|
|
resource_template.hass = hass
|
|
resource = resource_template.async_render(parse_result=False)
|
|
|
|
if not resource:
|
|
raise HomeAssistantError("Resource not set for RestData")
|
|
|
|
template.attach(hass, headers)
|
|
template.attach(hass, params)
|
|
|
|
auth: httpx.DigestAuth | tuple[str, str] | None = None
|
|
if username and password:
|
|
if config.get(CONF_AUTHENTICATION) == HTTP_DIGEST_AUTHENTICATION:
|
|
auth = httpx.DigestAuth(username, password)
|
|
else:
|
|
auth = (username, password)
|
|
|
|
return RestData(
|
|
hass,
|
|
method,
|
|
resource,
|
|
encoding,
|
|
auth,
|
|
headers,
|
|
params,
|
|
payload,
|
|
verify_ssl,
|
|
ssl_cipher_list,
|
|
timeout,
|
|
)
|