"""Plugwise platform for Home Assistant Core.""" from __future__ import annotations import asyncio from datetime import timedelta import logging import async_timeout from plugwise.exceptions import ( InvalidAuthentication, PlugwiseException, XMLDataMissingError, ) from plugwise.smile import Smile from homeassistant.config_entries import ConfigEntry from homeassistant.const import ( ATTR_CONFIGURATION_URL, ATTR_MODEL, ATTR_VIA_DEVICE, CONF_HOST, CONF_PASSWORD, CONF_PORT, CONF_SCAN_INTERVAL, CONF_USERNAME, ) from homeassistant.core import HomeAssistant, callback from homeassistant.exceptions import ConfigEntryNotReady from homeassistant.helpers import device_registry as dr from homeassistant.helpers.aiohttp_client import async_get_clientsession from homeassistant.helpers.entity import DeviceInfo from homeassistant.helpers.update_coordinator import ( CoordinatorEntity, DataUpdateCoordinator, UpdateFailed, ) from .const import ( API, COORDINATOR, DEFAULT_PORT, DEFAULT_SCAN_INTERVAL, DEFAULT_TIMEOUT, DEFAULT_USERNAME, DOMAIN, GATEWAY, PLATFORMS_GATEWAY, PW_TYPE, SENSOR_PLATFORMS, UNDO_UPDATE_LISTENER, ) _LOGGER = logging.getLogger(__name__) async def async_setup_entry_gw(hass: HomeAssistant, entry: ConfigEntry) -> bool: """Set up Plugwise Smiles from a config entry.""" websession = async_get_clientsession(hass, verify_ssl=False) api = Smile( host=entry.data[CONF_HOST], username=entry.data.get(CONF_USERNAME, DEFAULT_USERNAME), password=entry.data[CONF_PASSWORD], port=entry.data.get(CONF_PORT, DEFAULT_PORT), timeout=30, websession=websession, ) try: connected = await api.connect() if not connected: _LOGGER.error("Unable to connect to Smile") raise ConfigEntryNotReady except InvalidAuthentication: _LOGGER.error("Invalid username or Smile ID") return False except PlugwiseException as err: _LOGGER.error("Error while communicating to device %s", api.smile_name) raise ConfigEntryNotReady from err except asyncio.TimeoutError as err: _LOGGER.error("Timeout while connecting to Smile %s", api.smile_name) raise ConfigEntryNotReady from err update_interval = timedelta( seconds=entry.options.get( CONF_SCAN_INTERVAL, DEFAULT_SCAN_INTERVAL[api.smile_type] ) ) async def async_update_data(): """Update data via API endpoint.""" try: async with async_timeout.timeout(DEFAULT_TIMEOUT): await api.full_update_device() return True except XMLDataMissingError as err: raise UpdateFailed("Smile update failed") from err coordinator = DataUpdateCoordinator( hass, _LOGGER, name=f"Smile {api.smile_name}", update_method=async_update_data, update_interval=update_interval, ) await coordinator.async_config_entry_first_refresh() api.get_all_devices() if entry.unique_id is None and api.smile_version[0] != "1.8.0": hass.config_entries.async_update_entry(entry, unique_id=api.smile_hostname) undo_listener = entry.add_update_listener(_update_listener) hass.data.setdefault(DOMAIN, {})[entry.entry_id] = { "api": api, COORDINATOR: coordinator, PW_TYPE: GATEWAY, UNDO_UPDATE_LISTENER: undo_listener, } device_registry = dr.async_get(hass) device_registry.async_get_or_create( config_entry_id=entry.entry_id, identifiers={(DOMAIN, api.gateway_id)}, manufacturer="Plugwise", name=entry.title, model=f"Smile {api.smile_name}", sw_version=api.smile_version[0], ) single_master_thermostat = api.single_master_thermostat() platforms = PLATFORMS_GATEWAY if single_master_thermostat is None: platforms = SENSOR_PLATFORMS hass.config_entries.async_setup_platforms(entry, platforms) return True async def _update_listener(hass: HomeAssistant, entry: ConfigEntry): """Handle options update.""" coordinator = hass.data[DOMAIN][entry.entry_id][COORDINATOR] update_interval = entry.options.get(CONF_SCAN_INTERVAL) if update_interval is None: api = hass.data[DOMAIN][entry.entry_id][API] update_interval = DEFAULT_SCAN_INTERVAL[api.smile_type] coordinator.update_interval = timedelta(seconds=update_interval) async def async_unload_entry_gw(hass: HomeAssistant, entry: ConfigEntry): """Unload a config entry.""" unload_ok = await hass.config_entries.async_unload_platforms( entry, PLATFORMS_GATEWAY ) hass.data[DOMAIN][entry.entry_id][UNDO_UPDATE_LISTENER]() if unload_ok: hass.data[DOMAIN].pop(entry.entry_id) return unload_ok class SmileGateway(CoordinatorEntity): """Represent Smile Gateway.""" def __init__(self, api, coordinator, name, dev_id): """Initialise the gateway.""" super().__init__(coordinator) self._api = api self._name = name self._dev_id = dev_id self._unique_id = None self._model = None self._entity_name = self._name @property def unique_id(self): """Return a unique ID.""" return self._unique_id @property def name(self): """Return the name of the entity, if any.""" return self._name @property def device_info(self) -> DeviceInfo: """Return the device information.""" device_information = DeviceInfo( identifiers={(DOMAIN, self._dev_id)}, name=self._entity_name, manufacturer="Plugwise", ) if entry := self.coordinator.config_entry: device_information[ ATTR_CONFIGURATION_URL ] = f"http://{entry.data[CONF_HOST]}" if self._model is not None: device_information[ATTR_MODEL] = self._model.replace("_", " ").title() if self._dev_id != self._api.gateway_id: device_information[ATTR_VIA_DEVICE] = (DOMAIN, self._api.gateway_id) return device_information async def async_added_to_hass(self): """Subscribe to updates.""" self._async_process_data() self.async_on_remove( self.coordinator.async_add_listener(self._async_process_data) ) @callback def _async_process_data(self): """Interpret and process API data.""" raise NotImplementedError