"""The nuki component.""" from collections import defaultdict from datetime import timedelta import logging import async_timeout from pynuki import NukiBridge, NukiLock, NukiOpener from pynuki.bridge import InvalidCredentialsException from pynuki.device import NukiDevice from requests.exceptions import RequestException from homeassistant import exceptions from homeassistant.config_entries import ConfigEntry from homeassistant.const import CONF_HOST, CONF_PORT, CONF_TOKEN, Platform from homeassistant.core import HomeAssistant from homeassistant.helpers import entity_registry as er from homeassistant.helpers.update_coordinator import ( CoordinatorEntity, DataUpdateCoordinator, UpdateFailed, ) from .const import ( DATA_BRIDGE, DATA_COORDINATOR, DATA_LOCKS, DATA_OPENERS, DEFAULT_TIMEOUT, DOMAIN, ERROR_STATES, ) from .helpers import parse_id _LOGGER = logging.getLogger(__name__) PLATFORMS = [Platform.BINARY_SENSOR, Platform.LOCK] UPDATE_INTERVAL = timedelta(seconds=30) def _get_bridge_devices(bridge: NukiBridge) -> tuple[list[NukiLock], list[NukiOpener]]: return bridge.locks, bridge.openers def _update_devices(devices: list[NukiDevice]) -> dict[str, set[str]]: """ Update the Nuki devices. Returns: A dict with the events to be fired. The event type is the key and the device ids are the value """ events: dict[str, set[str]] = defaultdict(set) for device in devices: for level in (False, True): try: if isinstance(device, NukiOpener): last_ring_action_state = device.ring_action_state device.update(level) if not last_ring_action_state and device.ring_action_state: events["ring"].add(device.nuki_id) else: device.update(level) except RequestException: continue if device.state not in ERROR_STATES: break return events async def async_setup_entry(hass: HomeAssistant, entry: ConfigEntry) -> bool: """Set up the Nuki entry.""" hass.data.setdefault(DOMAIN, {}) # Migration of entry unique_id if isinstance(entry.unique_id, int): new_id = parse_id(entry.unique_id) params = {"unique_id": new_id} if entry.title == entry.unique_id: params["title"] = new_id hass.config_entries.async_update_entry(entry, **params) try: bridge = await hass.async_add_executor_job( NukiBridge, entry.data[CONF_HOST], entry.data[CONF_TOKEN], entry.data[CONF_PORT], True, DEFAULT_TIMEOUT, ) locks, openers = await hass.async_add_executor_job(_get_bridge_devices, bridge) except InvalidCredentialsException as err: raise exceptions.ConfigEntryAuthFailed from err except RequestException as err: raise exceptions.ConfigEntryNotReady from err async def async_update_data(): """Fetch data from Nuki bridge.""" try: # Note: asyncio.TimeoutError and aiohttp.ClientError are already # handled by the data update coordinator. async with async_timeout.timeout(10): events = await hass.async_add_executor_job( _update_devices, locks + openers ) except InvalidCredentialsException as err: raise UpdateFailed(f"Invalid credentials for Bridge: {err}") from err except RequestException as err: raise UpdateFailed(f"Error communicating with Bridge: {err}") from err ent_reg = er.async_get(hass) for event, device_ids in events.items(): for device_id in device_ids: entity_id = ent_reg.async_get_entity_id( Platform.LOCK, DOMAIN, device_id ) event_data = { "entity_id": entity_id, "type": event, } hass.bus.async_fire("nuki_event", event_data) coordinator = DataUpdateCoordinator( hass, _LOGGER, # Name of the data. For logging purposes. name="nuki devices", update_method=async_update_data, # Polling interval. Will only be polled if there are subscribers. update_interval=UPDATE_INTERVAL, ) hass.data[DOMAIN][entry.entry_id] = { DATA_COORDINATOR: coordinator, DATA_BRIDGE: bridge, DATA_LOCKS: locks, DATA_OPENERS: openers, } # Fetch initial data so we have data when entities subscribe await coordinator.async_refresh() await hass.config_entries.async_forward_entry_setups(entry, PLATFORMS) return True async def async_unload_entry(hass: HomeAssistant, entry: ConfigEntry) -> bool: """Unload the Nuki entry.""" unload_ok = await hass.config_entries.async_unload_platforms(entry, PLATFORMS) if unload_ok: hass.data[DOMAIN].pop(entry.entry_id) return unload_ok class NukiEntity(CoordinatorEntity): """An entity using CoordinatorEntity. The CoordinatorEntity class provides: should_poll async_update async_added_to_hass available """ def __init__( self, coordinator: DataUpdateCoordinator[None], nuki_device: NukiDevice ) -> None: """Pass coordinator to CoordinatorEntity.""" super().__init__(coordinator) self._nuki_device = nuki_device