core/homeassistant/components/peblar/coordinator.py

185 lines
5.9 KiB
Python

"""Data update coordinator for Peblar EV chargers."""
from __future__ import annotations
from collections.abc import Callable, Coroutine
from dataclasses import dataclass
from datetime import timedelta
from typing import Any, Concatenate
from peblar import (
Peblar,
PeblarApi,
PeblarAuthenticationError,
PeblarConnectionError,
PeblarError,
PeblarEVInterface,
PeblarMeter,
PeblarSystem,
PeblarSystemInformation,
PeblarUserConfiguration,
PeblarVersions,
)
from homeassistant.config_entries import ConfigEntry, ConfigEntryState
from homeassistant.core import HomeAssistant
from homeassistant.exceptions import ConfigEntryAuthFailed
from homeassistant.helpers.update_coordinator import DataUpdateCoordinator, UpdateFailed
from .const import DOMAIN, LOGGER
@dataclass(kw_only=True)
class PeblarRuntimeData:
"""Class to hold runtime data."""
data_coordinator: PeblarDataUpdateCoordinator
system_information: PeblarSystemInformation
user_configuration_coordinator: PeblarUserConfigurationDataUpdateCoordinator
version_coordinator: PeblarVersionDataUpdateCoordinator
type PeblarConfigEntry = ConfigEntry[PeblarRuntimeData]
@dataclass(kw_only=True, frozen=True)
class PeblarVersionInformation:
"""Class to hold version information."""
current: PeblarVersions
available: PeblarVersions
@dataclass(kw_only=True)
class PeblarData:
"""Class to hold active charging related information of Peblar.
This is data that needs to be polled and updated at a relatively high
frequency in order for this integration to function correctly.
All this data is updated at the same time by a single coordinator.
"""
ev: PeblarEVInterface
meter: PeblarMeter
system: PeblarSystem
def _coordinator_exception_handler[
_DataUpdateCoordinatorT: PeblarDataUpdateCoordinator
| PeblarVersionDataUpdateCoordinator
| PeblarUserConfigurationDataUpdateCoordinator,
**_P,
](
func: Callable[Concatenate[_DataUpdateCoordinatorT, _P], Coroutine[Any, Any, Any]],
) -> Callable[Concatenate[_DataUpdateCoordinatorT, _P], Coroutine[Any, Any, Any]]:
"""Handle exceptions within the update handler of a coordinator."""
async def handler(
self: _DataUpdateCoordinatorT, *args: _P.args, **kwargs: _P.kwargs
) -> Any:
try:
return await func(self, *args, **kwargs)
except PeblarAuthenticationError as error:
if self.config_entry and self.config_entry.state is ConfigEntryState.LOADED:
# This is not the first refresh, so let's reload
# the config entry to ensure we trigger a re-authentication
# flow (or recover in case of API token changes).
self.hass.config_entries.async_schedule_reload(
self.config_entry.entry_id
)
raise ConfigEntryAuthFailed(
translation_domain=DOMAIN,
translation_key="authentication_error",
) from error
except PeblarConnectionError as error:
raise UpdateFailed(
translation_domain=DOMAIN,
translation_key="communication_error",
translation_placeholders={"error": str(error)},
) from error
except PeblarError as error:
raise UpdateFailed(
translation_domain=DOMAIN,
translation_key="unknown_error",
translation_placeholders={"error": str(error)},
) from error
return handler
class PeblarVersionDataUpdateCoordinator(
DataUpdateCoordinator[PeblarVersionInformation]
):
"""Class to manage fetching Peblar version information."""
def __init__(
self, hass: HomeAssistant, entry: PeblarConfigEntry, peblar: Peblar
) -> None:
"""Initialize the coordinator."""
self.peblar = peblar
super().__init__(
hass,
LOGGER,
config_entry=entry,
name=f"Peblar {entry.title} version",
update_interval=timedelta(hours=2),
)
@_coordinator_exception_handler
async def _async_update_data(self) -> PeblarVersionInformation:
"""Fetch data from the Peblar device."""
return PeblarVersionInformation(
current=await self.peblar.current_versions(),
available=await self.peblar.available_versions(),
)
class PeblarDataUpdateCoordinator(DataUpdateCoordinator[PeblarData]):
"""Class to manage fetching Peblar active data."""
def __init__(
self, hass: HomeAssistant, entry: PeblarConfigEntry, api: PeblarApi
) -> None:
"""Initialize the coordinator."""
self.api = api
super().__init__(
hass,
LOGGER,
config_entry=entry,
name=f"Peblar {entry.title} meter",
update_interval=timedelta(seconds=10),
)
@_coordinator_exception_handler
async def _async_update_data(self) -> PeblarData:
"""Fetch data from the Peblar device."""
return PeblarData(
ev=await self.api.ev_interface(),
meter=await self.api.meter(),
system=await self.api.system(),
)
class PeblarUserConfigurationDataUpdateCoordinator(
DataUpdateCoordinator[PeblarUserConfiguration]
):
"""Class to manage fetching Peblar user configuration data."""
def __init__(
self, hass: HomeAssistant, entry: PeblarConfigEntry, peblar: Peblar
) -> None:
"""Initialize the coordinator."""
self.peblar = peblar
super().__init__(
hass,
LOGGER,
config_entry=entry,
name=f"Peblar {entry.title} user configuration",
update_interval=timedelta(minutes=5),
)
@_coordinator_exception_handler
async def _async_update_data(self) -> PeblarUserConfiguration:
"""Fetch data from the Peblar device."""
return await self.peblar.user_configuration()