1
0
mirror of https://github.com/home-assistant/core.git synced 2026-05-08 17:49:37 +01:00
Files
core/homeassistant/components/tplink_omada/controller.py
T

135 lines
4.7 KiB
Python

"""Controller for sharing Omada API coordinators between platforms."""
from __future__ import annotations
from collections.abc import Awaitable, Callable
from typing import TYPE_CHECKING
from tplink_omada_client import OmadaSiteClient
from tplink_omada_client.devices import OmadaListDevice, OmadaSwitch
from homeassistant.core import HomeAssistant, callback
if TYPE_CHECKING:
from . import OmadaConfigEntry
from .coordinator import (
OmadaClientsCoordinator,
OmadaDevicesCoordinator,
OmadaGatewayCoordinator,
OmadaSwitchPortCoordinator,
)
class OmadaSiteController:
"""Controller for the Omada SDN site."""
_gateway_coordinator: OmadaGatewayCoordinator | None = None
def __init__(
self,
hass: HomeAssistant,
config_entry: OmadaConfigEntry,
omada_client: OmadaSiteClient,
) -> None:
"""Create the controller."""
self._hass = hass
self._config_entry = config_entry
self._omada_client = omada_client
self._switch_port_coordinators: dict[str, OmadaSwitchPortCoordinator] = {}
self._devices_coordinator = OmadaDevicesCoordinator(
hass, config_entry, omada_client
)
self._clients_coordinator = OmadaClientsCoordinator(
hass, config_entry, omada_client
)
async def initialize_first_refresh(self) -> None:
"""Initialize the all coordinators, and perform first refresh."""
await self._devices_coordinator.async_config_entry_first_refresh()
devices = self._devices_coordinator.data.values()
gateway = next((d for d in devices if d.type == "gateway"), None)
if gateway:
self._gateway_coordinator = OmadaGatewayCoordinator(
self._hass, self._config_entry, self._omada_client, gateway.mac
)
await self._gateway_coordinator.async_config_entry_first_refresh()
await self.clients_coordinator.async_config_entry_first_refresh()
async def async_register_device_entities(
self,
device_filter: Callable[[OmadaListDevice], bool],
entity_callback: Callable[[OmadaListDevice], Awaitable[None]],
) -> None:
"""Register entities for devices matching the given filter.
Allows us to handle newly commissioned devices, and devices that aren't
currently in a queryable state.
Args:
device_filter: Function that returns True if a device should be processed.
entity_callback: Given a discovered Omada device, creates entities for that device.
"""
# Track which devices have been processed already
processed_devices: set[str] = set()
async def _async_register_entities() -> None:
"""Register entities for devices that match the filter."""
devices_to_process = [
device
for device in self._devices_coordinator.data.values()
if device_filter(device) and device.mac not in processed_devices
]
if not devices_to_process:
return
for device in devices_to_process:
processed_devices.add(device.mac)
await entity_callback(device)
@callback
def _handle_devices_update() -> None:
self._config_entry.async_create_task(self._hass, _async_register_entities())
self._config_entry.async_on_unload(
self._devices_coordinator.async_add_listener(_handle_devices_update)
)
# Call once on initial setup
await _async_register_entities()
@property
def omada_client(self) -> OmadaSiteClient:
"""Get the connected client API for the site to manage."""
return self._omada_client
def get_switch_port_coordinator(
self, switch: OmadaSwitch
) -> OmadaSwitchPortCoordinator:
"""Get coordinator for network port information of a given switch."""
if switch.mac not in self._switch_port_coordinators:
self._switch_port_coordinators[switch.mac] = OmadaSwitchPortCoordinator(
self._hass, self._config_entry, self._omada_client, switch
)
return self._switch_port_coordinators[switch.mac]
@property
def gateway_coordinator(self) -> OmadaGatewayCoordinator | None:
"""Gets the coordinator for site's gateway, or None if there is no gateway."""
return self._gateway_coordinator
@property
def devices_coordinator(self) -> OmadaDevicesCoordinator:
"""Gets the coordinator for site's devices."""
return self._devices_coordinator
@property
def clients_coordinator(self) -> OmadaClientsCoordinator:
"""Gets the coordinator for site's clients."""
return self._clients_coordinator