"""Class to reload platforms.""" from __future__ import annotations import asyncio from collections.abc import Iterable import logging from typing import Any, Literal, overload from homeassistant import config as conf_util from homeassistant.const import SERVICE_RELOAD from homeassistant.core import HomeAssistant, ServiceCall, callback from homeassistant.exceptions import HomeAssistantError from homeassistant.loader import async_get_integration from homeassistant.setup import async_setup_component from .entity import Entity from .entity_component import EntityComponent from .entity_platform import EntityPlatform, async_get_platforms from .service import async_register_admin_service from .typing import ConfigType _LOGGER = logging.getLogger(__name__) PLATFORM_RESET_LOCK = "lock_async_reset_platform_{}" async def async_reload_integration_platforms( hass: HomeAssistant, integration_domain: str, platform_domains: Iterable[str] ) -> None: """Reload an integration's platforms. The platform must support being re-setup. This functionality is only intended to be used for integrations that process Home Assistant data and make this available to other integrations. Examples are template, stats, derivative, utility meter. """ try: unprocessed_conf = await conf_util.async_hass_config_yaml(hass) except HomeAssistantError as err: _LOGGER.error(err) return tasks = [ _resetup_platform(hass, integration_domain, platform_domain, unprocessed_conf) for platform_domain in platform_domains ] await asyncio.gather(*tasks) async def _resetup_platform( hass: HomeAssistant, integration_domain: str, platform_domain: str, unprocessed_config: ConfigType, ) -> None: """Resetup a platform.""" integration = await async_get_integration(hass, platform_domain) conf = await conf_util.async_process_component_and_handle_errors( hass, unprocessed_config, integration ) if not conf: return root_config: dict[str, list[ConfigType]] = {platform_domain: []} # Extract only the config for template, ignore the rest. for p_type, p_config in conf_util.config_per_platform(conf, platform_domain): if p_type != integration_domain: continue root_config[platform_domain].append(p_config) component = await integration.async_get_component() if hasattr(component, "async_reset_platform"): # If the integration has its own way to reset # use this method. async with hass.data.setdefault( PLATFORM_RESET_LOCK.format(platform_domain), asyncio.Lock() ): await component.async_reset_platform(hass, integration_domain) await component.async_setup(hass, root_config) return # If it's an entity platform, we use the entity_platform # async_reset method platform = async_get_platform_without_config_entry( hass, integration_domain, platform_domain ) if platform: await _async_reconfig_platform(platform, root_config[platform_domain]) return if not root_config[platform_domain]: # No config for this platform # and it's not loaded. Nothing to do. return await _async_setup_platform( hass, integration_domain, platform_domain, root_config[platform_domain] ) async def _async_setup_platform( hass: HomeAssistant, integration_domain: str, platform_domain: str, platform_configs: list[dict[str, Any]], ) -> None: """Platform for the first time when new configuration is added.""" if platform_domain not in hass.data: await async_setup_component( hass, platform_domain, {platform_domain: platform_configs} ) return entity_component: EntityComponent[Entity] = hass.data[platform_domain] tasks = [ entity_component.async_setup_platform(integration_domain, p_config) for p_config in platform_configs ] await asyncio.gather(*tasks) async def _async_reconfig_platform( platform: EntityPlatform, platform_configs: list[dict[str, Any]] ) -> None: """Reconfigure an already loaded platform.""" await platform.async_reset() tasks = [platform.async_setup(p_config) for p_config in platform_configs] await asyncio.gather(*tasks) @overload async def async_integration_yaml_config( hass: HomeAssistant, integration_name: str ) -> ConfigType | None: ... @overload async def async_integration_yaml_config( hass: HomeAssistant, integration_name: str, *, raise_on_failure: Literal[True], ) -> ConfigType: ... @overload async def async_integration_yaml_config( hass: HomeAssistant, integration_name: str, *, raise_on_failure: Literal[False] | bool, ) -> ConfigType | None: ... async def async_integration_yaml_config( hass: HomeAssistant, integration_name: str, *, raise_on_failure: bool = False ) -> ConfigType | None: """Fetch the latest yaml configuration for an integration.""" integration = await async_get_integration(hass, integration_name) config = await conf_util.async_hass_config_yaml(hass) return await conf_util.async_process_component_and_handle_errors( hass, config, integration, raise_on_failure=raise_on_failure ) @callback def async_get_platform_without_config_entry( hass: HomeAssistant, integration_name: str, integration_platform_name: str ) -> EntityPlatform | None: """Find an existing platform that is not a config entry.""" for integration_platform in async_get_platforms(hass, integration_name): if integration_platform.config_entry is not None: continue if integration_platform.domain == integration_platform_name: platform: EntityPlatform = integration_platform return platform return None async def async_setup_reload_service( hass: HomeAssistant, domain: str, platforms: Iterable[str] ) -> None: """Create the reload service for the domain.""" if hass.services.has_service(domain, SERVICE_RELOAD): return async def _reload_config(call: ServiceCall) -> None: """Reload the platforms.""" await async_reload_integration_platforms(hass, domain, platforms) hass.bus.async_fire(f"event_{domain}_reloaded", context=call.context) async_register_admin_service(hass, domain, SERVICE_RELOAD, _reload_config) def setup_reload_service( hass: HomeAssistant, domain: str, platforms: Iterable[str] ) -> None: """Sync version of async_setup_reload_service.""" asyncio.run_coroutine_threadsafe( async_setup_reload_service(hass, domain, platforms), hass.loop, ).result()