"""The Overkiz (by Somfy) integration.""" from __future__ import annotations import asyncio from collections import defaultdict from dataclasses import dataclass import logging from aiohttp import ClientError, ServerDisconnectedError from pyoverkiz.client import OverkizClient from pyoverkiz.const import SUPPORTED_SERVERS from pyoverkiz.exceptions import ( BadCredentialsException, MaintenanceException, TooManyRequestsException, ) from pyoverkiz.models import Device, Scenario from homeassistant.config_entries import ConfigEntry from homeassistant.const import CONF_PASSWORD, CONF_USERNAME, Platform from homeassistant.core import HomeAssistant from homeassistant.exceptions import ConfigEntryNotReady from homeassistant.helpers import device_registry as dr from homeassistant.helpers.aiohttp_client import async_create_clientsession from .const import ( CONF_HUB, DOMAIN, OVERKIZ_DEVICE_TO_PLATFORM, PLATFORMS, UPDATE_INTERVAL, UPDATE_INTERVAL_ALL_ASSUMED_STATE, ) from .coordinator import OverkizDataUpdateCoordinator _LOGGER = logging.getLogger(__name__) @dataclass class HomeAssistantOverkizData: """Overkiz data stored in the Home Assistant data object.""" coordinator: OverkizDataUpdateCoordinator platforms: defaultdict[Platform, list[Device | Scenario]] async def async_setup_entry(hass: HomeAssistant, entry: ConfigEntry) -> bool: """Set up Overkiz from a config entry.""" username = entry.data[CONF_USERNAME] password = entry.data[CONF_PASSWORD] server = SUPPORTED_SERVERS[entry.data[CONF_HUB]] # To allow users with multiple accounts/hubs, we create a new session so they have separate cookies session = async_create_clientsession(hass) client = OverkizClient( username=username, password=password, session=session, server=server ) try: await client.login() setup, scenarios = await asyncio.gather( *[ client.get_setup(), client.get_scenarios(), ] ) except BadCredentialsException: _LOGGER.error("Invalid authentication") return False except TooManyRequestsException as exception: raise ConfigEntryNotReady("Too many requests, try again later") from exception except (TimeoutError, ClientError, ServerDisconnectedError) as exception: raise ConfigEntryNotReady("Failed to connect") from exception except MaintenanceException as exception: raise ConfigEntryNotReady("Server is down for maintenance") from exception coordinator = OverkizDataUpdateCoordinator( hass, _LOGGER, name="device events", client=client, devices=setup.devices, places=setup.root_place, update_interval=UPDATE_INTERVAL, config_entry_id=entry.entry_id, ) await coordinator.async_config_entry_first_refresh() if coordinator.is_stateless: _LOGGER.debug( "All devices have an assumed state. Update interval has been reduced to: %s", UPDATE_INTERVAL_ALL_ASSUMED_STATE, ) coordinator.update_interval = UPDATE_INTERVAL_ALL_ASSUMED_STATE platforms: defaultdict[Platform, list[Device | Scenario]] = defaultdict(list) hass.data.setdefault(DOMAIN, {})[entry.entry_id] = HomeAssistantOverkizData( coordinator=coordinator, platforms=platforms, ) # Map Overkiz entities to Home Assistant platform platforms[Platform.SCENE] = scenarios for device in coordinator.data.values(): _LOGGER.debug( "The following device has been retrieved. Report an issue if not supported correctly (%s)", device, ) if platform := OVERKIZ_DEVICE_TO_PLATFORM.get( device.widget ) or OVERKIZ_DEVICE_TO_PLATFORM.get(device.ui_class): platforms[platform].append(device) hass.config_entries.async_setup_platforms(entry, PLATFORMS) device_registry = dr.async_get(hass) for gateway in setup.gateways: _LOGGER.debug("Added gateway (%s)", gateway) device_registry.async_get_or_create( config_entry_id=entry.entry_id, identifiers={(DOMAIN, gateway.id)}, model=gateway.sub_type.beautify_name if gateway.sub_type else None, manufacturer=server.manufacturer, name=gateway.type.beautify_name, sw_version=gateway.connectivity.protocol_version, configuration_url=server.configuration_url, ) 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