2020-02-10 21:16:04 +00:00
|
|
|
"""Code to handle a Dynalite bridge."""
|
2021-03-17 22:43:55 +00:00
|
|
|
from __future__ import annotations
|
2020-02-10 21:16:04 +00:00
|
|
|
|
2021-09-29 14:15:36 +00:00
|
|
|
from collections.abc import Callable
|
2021-07-20 16:28:31 +00:00
|
|
|
from types import MappingProxyType
|
2021-09-29 14:15:36 +00:00
|
|
|
from typing import Any
|
2020-04-02 18:26:36 +00:00
|
|
|
|
2020-08-06 18:40:54 +00:00
|
|
|
from dynalite_devices_lib.dynalite_devices import (
|
|
|
|
CONF_AREA as dyn_CONF_AREA,
|
|
|
|
CONF_PRESET as dyn_CONF_PRESET,
|
|
|
|
NOTIFICATION_PACKET,
|
|
|
|
NOTIFICATION_PRESET,
|
|
|
|
DynaliteBaseDevice,
|
|
|
|
DynaliteDevices,
|
|
|
|
DynaliteNotification,
|
|
|
|
)
|
2020-02-10 21:16:04 +00:00
|
|
|
|
2020-04-02 18:26:36 +00:00
|
|
|
from homeassistant.const import CONF_HOST
|
|
|
|
from homeassistant.core import HomeAssistant, callback
|
2020-02-21 22:29:59 +00:00
|
|
|
from homeassistant.helpers.dispatcher import async_dispatcher_send
|
2020-02-10 21:16:04 +00:00
|
|
|
|
2021-03-02 20:43:59 +00:00
|
|
|
from .const import ATTR_AREA, ATTR_HOST, ATTR_PACKET, ATTR_PRESET, LOGGER, PLATFORMS
|
2020-04-02 18:26:36 +00:00
|
|
|
from .convert_config import convert_config
|
|
|
|
|
2020-02-10 21:16:04 +00:00
|
|
|
|
|
|
|
class DynaliteBridge:
|
|
|
|
"""Manages a single Dynalite bridge."""
|
|
|
|
|
2021-03-17 22:43:55 +00:00
|
|
|
def __init__(self, hass: HomeAssistant, config: dict[str, Any]) -> None:
|
2020-02-10 21:16:04 +00:00
|
|
|
"""Initialize the system based on host parameter."""
|
|
|
|
self.hass = hass
|
2021-07-20 16:28:31 +00:00
|
|
|
self.async_add_devices: dict[str, Callable] = {}
|
|
|
|
self.waiting_devices: dict[str, list[str]] = {}
|
2020-02-21 22:29:59 +00:00
|
|
|
self.host = config[CONF_HOST]
|
2020-02-10 21:16:04 +00:00
|
|
|
# Configure the dynalite devices
|
|
|
|
self.dynalite_devices = DynaliteDevices(
|
2020-03-05 06:05:39 +00:00
|
|
|
new_device_func=self.add_devices_when_registered,
|
|
|
|
update_device_func=self.update_device,
|
2020-08-06 18:40:54 +00:00
|
|
|
notification_func=self.handle_notification,
|
2020-02-10 21:16:04 +00:00
|
|
|
)
|
2021-07-20 16:28:31 +00:00
|
|
|
self.dynalite_devices.configure(config)
|
2020-02-10 21:16:04 +00:00
|
|
|
|
2020-04-02 18:26:36 +00:00
|
|
|
async def async_setup(self) -> bool:
|
2020-02-10 21:16:04 +00:00
|
|
|
"""Set up a Dynalite bridge."""
|
|
|
|
# Configure the dynalite devices
|
2020-02-28 06:05:55 +00:00
|
|
|
LOGGER.debug("Setting up bridge - host %s", self.host)
|
2020-02-21 22:29:59 +00:00
|
|
|
return await self.dynalite_devices.async_setup()
|
2020-02-10 21:16:04 +00:00
|
|
|
|
2021-07-20 16:28:31 +00:00
|
|
|
def reload_config(self, config: MappingProxyType[str, Any]) -> None:
|
2020-02-28 06:05:55 +00:00
|
|
|
"""Reconfigure a bridge when config changes."""
|
2020-03-01 21:44:24 +00:00
|
|
|
LOGGER.debug("Reloading bridge - host %s, config %s", self.host, config)
|
2020-04-02 18:26:36 +00:00
|
|
|
self.dynalite_devices.configure(convert_config(config))
|
2020-02-28 06:05:55 +00:00
|
|
|
|
2021-03-17 22:43:55 +00:00
|
|
|
def update_signal(self, device: DynaliteBaseDevice | None = None) -> str:
|
2020-02-21 22:29:59 +00:00
|
|
|
"""Create signal to use to trigger entity update."""
|
|
|
|
if device:
|
|
|
|
signal = f"dynalite-update-{self.host}-{device.unique_id}"
|
|
|
|
else:
|
|
|
|
signal = f"dynalite-update-{self.host}"
|
|
|
|
return signal
|
2020-02-10 21:16:04 +00:00
|
|
|
|
|
|
|
@callback
|
2021-03-17 22:43:55 +00:00
|
|
|
def update_device(self, device: DynaliteBaseDevice | None = None) -> None:
|
2020-02-10 21:16:04 +00:00
|
|
|
"""Call when a device or all devices should be updated."""
|
2020-04-24 16:30:45 +00:00
|
|
|
if not device:
|
2020-02-10 21:16:04 +00:00
|
|
|
# This is used to signal connection or disconnection, so all devices may become available or not.
|
2020-02-21 22:29:59 +00:00
|
|
|
log_string = (
|
2020-04-02 18:26:36 +00:00
|
|
|
"Connected" if self.dynalite_devices.connected else "Disconnected"
|
2020-02-21 22:29:59 +00:00
|
|
|
)
|
|
|
|
LOGGER.info("%s to dynalite host", log_string)
|
|
|
|
async_dispatcher_send(self.hass, self.update_signal())
|
2020-02-10 21:16:04 +00:00
|
|
|
else:
|
2020-02-21 22:29:59 +00:00
|
|
|
async_dispatcher_send(self.hass, self.update_signal(device))
|
|
|
|
|
2020-08-06 18:40:54 +00:00
|
|
|
@callback
|
|
|
|
def handle_notification(self, notification: DynaliteNotification) -> None:
|
|
|
|
"""Handle a notification from the platform and issue events."""
|
|
|
|
if notification.notification == NOTIFICATION_PACKET:
|
|
|
|
self.hass.bus.async_fire(
|
|
|
|
"dynalite_packet",
|
|
|
|
{
|
|
|
|
ATTR_HOST: self.host,
|
|
|
|
ATTR_PACKET: notification.data[NOTIFICATION_PACKET],
|
|
|
|
},
|
|
|
|
)
|
|
|
|
if notification.notification == NOTIFICATION_PRESET:
|
|
|
|
self.hass.bus.async_fire(
|
|
|
|
"dynalite_preset",
|
|
|
|
{
|
|
|
|
ATTR_HOST: self.host,
|
|
|
|
ATTR_AREA: notification.data[dyn_CONF_AREA],
|
|
|
|
ATTR_PRESET: notification.data[dyn_CONF_PRESET],
|
|
|
|
},
|
|
|
|
)
|
|
|
|
|
2020-02-10 21:16:04 +00:00
|
|
|
@callback
|
2020-04-02 18:26:36 +00:00
|
|
|
def register_add_devices(self, platform: str, async_add_devices: Callable) -> None:
|
2020-02-10 21:16:04 +00:00
|
|
|
"""Add an async_add_entities for a category."""
|
2020-03-01 21:44:24 +00:00
|
|
|
self.async_add_devices[platform] = async_add_devices
|
|
|
|
if platform in self.waiting_devices:
|
|
|
|
self.async_add_devices[platform](self.waiting_devices[platform])
|
2020-02-10 21:16:04 +00:00
|
|
|
|
2021-03-17 22:43:55 +00:00
|
|
|
def add_devices_when_registered(self, devices: list[DynaliteBaseDevice]) -> None:
|
2020-02-21 22:29:59 +00:00
|
|
|
"""Add the devices to HA if the add devices callback was registered, otherwise queue until it is."""
|
2021-03-02 20:43:59 +00:00
|
|
|
for platform in PLATFORMS:
|
2020-03-01 21:44:24 +00:00
|
|
|
platform_devices = [
|
|
|
|
device for device in devices if device.category == platform
|
|
|
|
]
|
|
|
|
if platform in self.async_add_devices:
|
|
|
|
self.async_add_devices[platform](platform_devices)
|
|
|
|
else: # handle it later when it is registered
|
|
|
|
if platform not in self.waiting_devices:
|
|
|
|
self.waiting_devices[platform] = []
|
|
|
|
self.waiting_devices[platform].extend(platform_devices)
|