163 lines
4.8 KiB
Python
163 lines
4.8 KiB
Python
"""Common code for GogoGate2 component."""
|
|
from __future__ import annotations
|
|
|
|
from collections.abc import Awaitable
|
|
from datetime import timedelta
|
|
import logging
|
|
from typing import Callable, NamedTuple
|
|
|
|
from gogogate2_api import AbstractGateApi, GogoGate2Api, ISmartGateApi
|
|
from gogogate2_api.common import AbstractDoor, get_door_by_id
|
|
|
|
from homeassistant.config_entries import ConfigEntry
|
|
from homeassistant.const import (
|
|
CONF_DEVICE,
|
|
CONF_IP_ADDRESS,
|
|
CONF_PASSWORD,
|
|
CONF_USERNAME,
|
|
)
|
|
from homeassistant.core import HomeAssistant
|
|
from homeassistant.helpers.debounce import Debouncer
|
|
from homeassistant.helpers.update_coordinator import (
|
|
CoordinatorEntity,
|
|
DataUpdateCoordinator,
|
|
UpdateFailed,
|
|
)
|
|
|
|
from .const import DATA_UPDATE_COORDINATOR, DEVICE_TYPE_ISMARTGATE, DOMAIN, MANUFACTURER
|
|
|
|
_LOGGER = logging.getLogger(__name__)
|
|
|
|
|
|
class StateData(NamedTuple):
|
|
"""State data for a cover entity."""
|
|
|
|
config_unique_id: str
|
|
unique_id: str | None
|
|
door: AbstractDoor | None
|
|
|
|
|
|
class DeviceDataUpdateCoordinator(DataUpdateCoordinator):
|
|
"""Manages polling for state changes from the device."""
|
|
|
|
def __init__(
|
|
self,
|
|
hass: HomeAssistant,
|
|
logger: logging.Logger,
|
|
api: AbstractGateApi,
|
|
*,
|
|
name: str,
|
|
update_interval: timedelta,
|
|
update_method: Callable[[], Awaitable] | None = None,
|
|
request_refresh_debouncer: Debouncer | None = None,
|
|
):
|
|
"""Initialize the data update coordinator."""
|
|
DataUpdateCoordinator.__init__(
|
|
self,
|
|
hass,
|
|
logger,
|
|
name=name,
|
|
update_interval=update_interval,
|
|
update_method=update_method,
|
|
request_refresh_debouncer=request_refresh_debouncer,
|
|
)
|
|
self.api = api
|
|
|
|
|
|
class GoGoGate2Entity(CoordinatorEntity):
|
|
"""Base class for gogogate2 entities."""
|
|
|
|
def __init__(
|
|
self,
|
|
config_entry: ConfigEntry,
|
|
data_update_coordinator: DeviceDataUpdateCoordinator,
|
|
door: AbstractDoor,
|
|
unique_id: str,
|
|
) -> None:
|
|
"""Initialize gogogate2 base entity."""
|
|
super().__init__(data_update_coordinator)
|
|
self._config_entry = config_entry
|
|
self._door = door
|
|
self._unique_id = unique_id
|
|
|
|
@property
|
|
def unique_id(self) -> str | None:
|
|
"""Return a unique ID."""
|
|
return self._unique_id
|
|
|
|
def _get_door(self) -> AbstractDoor:
|
|
door = get_door_by_id(self._door.door_id, self.coordinator.data)
|
|
self._door = door or self._door
|
|
return self._door
|
|
|
|
@property
|
|
def device_info(self):
|
|
"""Device info for the controller."""
|
|
data = self.coordinator.data
|
|
return {
|
|
"identifiers": {(DOMAIN, self._config_entry.unique_id)},
|
|
"name": self._config_entry.title,
|
|
"manufacturer": MANUFACTURER,
|
|
"model": data.model,
|
|
"sw_version": data.firmwareversion,
|
|
}
|
|
|
|
|
|
def get_data_update_coordinator(
|
|
hass: HomeAssistant, config_entry: ConfigEntry
|
|
) -> DeviceDataUpdateCoordinator:
|
|
"""Get an update coordinator."""
|
|
hass.data.setdefault(DOMAIN, {})
|
|
hass.data[DOMAIN].setdefault(config_entry.entry_id, {})
|
|
config_entry_data = hass.data[DOMAIN][config_entry.entry_id]
|
|
|
|
if DATA_UPDATE_COORDINATOR not in config_entry_data:
|
|
api = get_api(config_entry.data)
|
|
|
|
async def async_update_data():
|
|
try:
|
|
return await api.async_info()
|
|
except Exception as exception:
|
|
raise UpdateFailed(
|
|
f"Error communicating with API: {exception}"
|
|
) from exception
|
|
|
|
config_entry_data[DATA_UPDATE_COORDINATOR] = DeviceDataUpdateCoordinator(
|
|
hass,
|
|
_LOGGER,
|
|
api,
|
|
# Name of the data. For logging purposes.
|
|
name="gogogate2",
|
|
update_method=async_update_data,
|
|
# Polling interval. Will only be polled if there are subscribers.
|
|
update_interval=timedelta(seconds=5),
|
|
)
|
|
|
|
return config_entry_data[DATA_UPDATE_COORDINATOR]
|
|
|
|
|
|
def cover_unique_id(config_entry: ConfigEntry, door: AbstractDoor) -> str:
|
|
"""Generate a cover entity unique id."""
|
|
return f"{config_entry.unique_id}_{door.door_id}"
|
|
|
|
|
|
def sensor_unique_id(
|
|
config_entry: ConfigEntry, door: AbstractDoor, sensor_type: str
|
|
) -> str:
|
|
"""Generate a cover entity unique id."""
|
|
return f"{config_entry.unique_id}_{door.door_id}_{sensor_type}"
|
|
|
|
|
|
def get_api(config_data: dict) -> AbstractGateApi:
|
|
"""Get an api object for config data."""
|
|
gate_class = GogoGate2Api
|
|
|
|
if config_data[CONF_DEVICE] == DEVICE_TYPE_ISMARTGATE:
|
|
gate_class = ISmartGateApi
|
|
|
|
return gate_class(
|
|
config_data[CONF_IP_ADDRESS],
|
|
config_data[CONF_USERNAME],
|
|
config_data[CONF_PASSWORD],
|
|
)
|