"""The nVent RAYCHEM SENZ integration.""" from __future__ import annotations from datetime import timedelta import logging from aiosenz import SENZAPI, Thermostat from httpx import RequestError import voluptuous as vol from homeassistant.components.application_credentials import ( ClientCredential, async_import_client_credential, ) from homeassistant.config_entries import ConfigEntry from homeassistant.const import CONF_CLIENT_ID, CONF_CLIENT_SECRET, Platform from homeassistant.core import HomeAssistant from homeassistant.exceptions import ConfigEntryNotReady from homeassistant.helpers import ( config_entry_oauth2_flow, config_validation as cv, httpx_client, ) from homeassistant.helpers.typing import ConfigType from homeassistant.helpers.update_coordinator import DataUpdateCoordinator, UpdateFailed from .api import SENZConfigEntryAuth from .const import DOMAIN UPDATE_INTERVAL = timedelta(seconds=30) _LOGGER = logging.getLogger(__name__) CONFIG_SCHEMA = vol.Schema( vol.All( cv.deprecated(DOMAIN), { DOMAIN: vol.Schema( { vol.Required(CONF_CLIENT_ID): cv.string, vol.Required(CONF_CLIENT_SECRET): cv.string, } ) }, ), extra=vol.ALLOW_EXTRA, ) PLATFORMS = [Platform.CLIMATE] SENZDataUpdateCoordinator = DataUpdateCoordinator[dict[str, Thermostat]] async def async_setup(hass: HomeAssistant, config: ConfigType) -> bool: """Set up the SENZ OAuth2 configuration.""" hass.data[DOMAIN] = {} if DOMAIN not in config: return True await async_import_client_credential( hass, DOMAIN, ClientCredential( config[DOMAIN][CONF_CLIENT_ID], config[DOMAIN][CONF_CLIENT_SECRET], ), ) _LOGGER.warning( "Configuration of SENZ integration in YAML is deprecated " "and will be removed in a future release; Your existing OAuth " "Application Credentials have been imported into the UI " "automatically and can be safely removed from your " "configuration.yaml file" ) return True async def async_setup_entry(hass: HomeAssistant, entry: ConfigEntry) -> bool: """Set up SENZ from a config entry.""" implementation = ( await config_entry_oauth2_flow.async_get_config_entry_implementation( hass, entry ) ) session = config_entry_oauth2_flow.OAuth2Session(hass, entry, implementation) auth = SENZConfigEntryAuth(httpx_client.get_async_client(hass), session) senz_api = SENZAPI(auth) async def update_thermostats() -> dict[str, Thermostat]: """Fetch SENZ thermostats data.""" try: thermostats = await senz_api.get_thermostats() except RequestError as err: raise UpdateFailed from err return {thermostat.serial_number: thermostat for thermostat in thermostats} try: account = await senz_api.get_account() except RequestError as err: raise ConfigEntryNotReady from err coordinator = SENZDataUpdateCoordinator( hass, _LOGGER, name=account.username, update_interval=UPDATE_INTERVAL, update_method=update_thermostats, ) await coordinator.async_config_entry_first_refresh() hass.data[DOMAIN][entry.entry_id] = coordinator hass.config_entries.async_setup_platforms(entry, PLATFORMS) return True async def async_unload_entry(hass: HomeAssistant, entry: ConfigEntry) -> bool: """Unload a config entry.""" if unload_ok := await hass.config_entries.async_unload_platforms(entry, PLATFORMS): hass.data[DOMAIN].pop(entry.entry_id) return unload_ok