feat: add DknCoordinator
This commit is contained in:
@@ -0,0 +1,71 @@
|
|||||||
|
"""DataUpdateCoordinator for DKN Cloud NA."""
|
||||||
|
|
||||||
|
from __future__ import annotations
|
||||||
|
|
||||||
|
import asyncio
|
||||||
|
from datetime import timedelta
|
||||||
|
from typing import Any
|
||||||
|
|
||||||
|
from homeassistant.config_entries import ConfigEntry
|
||||||
|
from homeassistant.core import HomeAssistant
|
||||||
|
from homeassistant.exceptions import ConfigEntryAuthFailed
|
||||||
|
from homeassistant.helpers.update_coordinator import DataUpdateCoordinator, UpdateFailed
|
||||||
|
|
||||||
|
from .api import DknAuthError, DknCloudNaClient, DknConnectionError
|
||||||
|
from .const import DEFAULT_SCAN_INTERVAL, DOMAIN, LOGGER
|
||||||
|
|
||||||
|
|
||||||
|
class DknCoordinator(DataUpdateCoordinator[dict[str, dict[str, Any]]]):
|
||||||
|
"""Coordinator that polls all installations and exposes a flat device map.
|
||||||
|
|
||||||
|
``data`` is ``{mac_address: device_dict}`` where device_dict matches the
|
||||||
|
shape returned by DknCloudNaClient.fetch_installations() device entries.
|
||||||
|
|
||||||
|
The coordinator owns the client instance used across all platforms.
|
||||||
|
Per-device asyncio.Lock objects for write serialization are stored in
|
||||||
|
``hass.data[DOMAIN][entry_id]["device_locks"]``.
|
||||||
|
"""
|
||||||
|
|
||||||
|
client: DknCloudNaClient
|
||||||
|
|
||||||
|
def __init__(
|
||||||
|
self,
|
||||||
|
hass: HomeAssistant,
|
||||||
|
entry: ConfigEntry,
|
||||||
|
client: DknCloudNaClient,
|
||||||
|
) -> None:
|
||||||
|
scan_interval = int(entry.options.get("scan_interval", DEFAULT_SCAN_INTERVAL))
|
||||||
|
super().__init__(
|
||||||
|
hass,
|
||||||
|
LOGGER,
|
||||||
|
name=DOMAIN,
|
||||||
|
update_interval=timedelta(seconds=scan_interval),
|
||||||
|
)
|
||||||
|
self.client = client
|
||||||
|
self._entry = entry
|
||||||
|
|
||||||
|
async def _async_update_data(self) -> dict[str, dict[str, Any]]:
|
||||||
|
"""Fetch all installations and flatten into {mac: device_dict}."""
|
||||||
|
try:
|
||||||
|
installations = await self.client.fetch_installations()
|
||||||
|
except DknAuthError as err:
|
||||||
|
# 401 — trigger the reauth UI and mark entities unavailable.
|
||||||
|
raise ConfigEntryAuthFailed("Token invalid or expired") from err
|
||||||
|
except DknConnectionError as err:
|
||||||
|
raise UpdateFailed(f"Cannot reach DKN Cloud NA: {err}") from err
|
||||||
|
except asyncio.CancelledError:
|
||||||
|
raise
|
||||||
|
except Exception as err: # noqa: BLE001
|
||||||
|
raise UpdateFailed(f"Unexpected error: {type(err).__name__}") from err
|
||||||
|
|
||||||
|
devices: dict[str, dict[str, Any]] = {}
|
||||||
|
for installation in installations or []:
|
||||||
|
inst_id = installation.get("_id", "")
|
||||||
|
for device in installation.get("devices", []):
|
||||||
|
mac = str(device.get("mac") or "").strip().lower()
|
||||||
|
if not mac:
|
||||||
|
continue
|
||||||
|
device["_installation_id"] = inst_id
|
||||||
|
devices[mac] = device
|
||||||
|
|
||||||
|
return devices
|
||||||
Reference in New Issue
Block a user