Improve loading UniFi switch entities (#80910)

pull/80968/head
Robert Svensson 2022-10-25 19:26:56 +02:00 committed by GitHub
parent 870a5b6f37
commit 13e2bb1e22
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
3 changed files with 82 additions and 40 deletions

View File

@ -42,3 +42,8 @@ DEFAULT_TRACK_WIRED_CLIENTS = True
DEFAULT_DETECTION_TIME = 300 DEFAULT_DETECTION_TIME = 300
ATTR_MANUFACTURER = "Ubiquiti Networks" ATTR_MANUFACTURER = "Ubiquiti Networks"
BLOCK_SWITCH = "block"
DPI_SWITCH = "dpi"
POE_SWITCH = "poe"
OUTLET_SWITCH = "outlet"

View File

@ -37,6 +37,7 @@ import homeassistant.util.dt as dt_util
from .const import ( from .const import (
ATTR_MANUFACTURER, ATTR_MANUFACTURER,
BLOCK_SWITCH,
CONF_ALLOW_BANDWIDTH_SENSORS, CONF_ALLOW_BANDWIDTH_SENSORS,
CONF_ALLOW_UPTIME_SENSORS, CONF_ALLOW_UPTIME_SENSORS,
CONF_BLOCK_CLIENT, CONF_BLOCK_CLIENT,
@ -61,10 +62,10 @@ from .const import (
DOMAIN as UNIFI_DOMAIN, DOMAIN as UNIFI_DOMAIN,
LOGGER, LOGGER,
PLATFORMS, PLATFORMS,
POE_SWITCH,
UNIFI_WIRELESS_CLIENTS, UNIFI_WIRELESS_CLIENTS,
) )
from .errors import AuthenticationRequired, CannotConnect from .errors import AuthenticationRequired, CannotConnect
from .switch import BLOCK_SWITCH, POE_SWITCH
RETRY_TIMER = 15 RETRY_TIMER = 15
CHECK_HEARTBEAT_INTERVAL = timedelta(seconds=1) CHECK_HEARTBEAT_INTERVAL = timedelta(seconds=1)

View File

@ -4,11 +4,17 @@ Support for controlling power supply of clients which are powered over Ethernet
Support for controlling network access of clients selected in option flow. Support for controlling network access of clients selected in option flow.
Support for controlling deep packet inspection (DPI) restriction groups. Support for controlling deep packet inspection (DPI) restriction groups.
""" """
from __future__ import annotations
import asyncio import asyncio
from typing import Any from collections.abc import Callable
from dataclasses import dataclass
from typing import Any, Generic, TypeVar
from aiounifi.interfaces.api_handlers import ItemEvent from aiounifi.interfaces.api_handlers import ItemEvent
from aiounifi.interfaces.dpi_restriction_groups import DPIRestrictionGroups
from aiounifi.interfaces.outlets import Outlets
from aiounifi.interfaces.ports import Ports
from aiounifi.models.api import SOURCE_EVENT from aiounifi.models.api import SOURCE_EVENT
from aiounifi.models.client import ClientBlockRequest from aiounifi.models.client import ClientBlockRequest
from aiounifi.models.device import ( from aiounifi.models.device import (
@ -31,17 +37,34 @@ from homeassistant.helpers.entity import DeviceInfo, EntityCategory
from homeassistant.helpers.entity_platform import AddEntitiesCallback from homeassistant.helpers.entity_platform import AddEntitiesCallback
from homeassistant.helpers.restore_state import RestoreEntity from homeassistant.helpers.restore_state import RestoreEntity
from .const import ATTR_MANUFACTURER, DOMAIN as UNIFI_DOMAIN from .const import (
ATTR_MANUFACTURER,
BLOCK_SWITCH,
DOMAIN as UNIFI_DOMAIN,
DPI_SWITCH,
OUTLET_SWITCH,
POE_SWITCH,
)
from .controller import UniFiController
from .unifi_client import UniFiClient from .unifi_client import UniFiClient
BLOCK_SWITCH = "block"
DPI_SWITCH = "dpi"
POE_SWITCH = "poe"
OUTLET_SWITCH = "outlet"
CLIENT_BLOCKED = (EventKey.WIRED_CLIENT_BLOCKED, EventKey.WIRELESS_CLIENT_BLOCKED) CLIENT_BLOCKED = (EventKey.WIRED_CLIENT_BLOCKED, EventKey.WIRELESS_CLIENT_BLOCKED)
CLIENT_UNBLOCKED = (EventKey.WIRED_CLIENT_UNBLOCKED, EventKey.WIRELESS_CLIENT_UNBLOCKED) CLIENT_UNBLOCKED = (EventKey.WIRED_CLIENT_UNBLOCKED, EventKey.WIRELESS_CLIENT_UNBLOCKED)
T = TypeVar("T")
@dataclass
class UnifiEntityLoader(Generic[T]):
"""Validate and load entities from different UniFi handlers."""
config_option_fn: Callable[[UniFiController], bool]
entity_cls: type[UnifiDPIRestrictionSwitch] | type[UnifiOutletSwitch] | type[
UnifiPoePortSwitch
] | type[UnifiDPIRestrictionSwitch]
handler_fn: Callable[[UniFiController], T]
value_fn: Callable[[T, str], bool | None]
async def async_setup_entry( async def async_setup_entry(
hass: HomeAssistant, hass: HomeAssistant,
@ -52,7 +75,7 @@ async def async_setup_entry(
Switches are controlling network access and switch ports with POE. Switches are controlling network access and switch ports with POE.
""" """
controller = hass.data[UNIFI_DOMAIN][config_entry.entry_id] controller: UniFiController = hass.data[UNIFI_DOMAIN][config_entry.entry_id]
controller.entities[DOMAIN] = { controller.entities[DOMAIN] = {
BLOCK_SWITCH: set(), BLOCK_SWITCH: set(),
POE_SWITCH: set(), POE_SWITCH: set(),
@ -105,42 +128,33 @@ async def async_setup_entry(
known_poe_clients.clear() known_poe_clients.clear()
@callback @callback
def async_add_outlet_switch(_: ItemEvent, obj_id: str) -> None: def async_load_entities(loader: UnifiEntityLoader) -> None:
"""Add power outlet switch from UniFi controller.""" """Load and subscribe to UniFi devices."""
if not controller.api.outlets[obj_id].has_relay: entities: list[SwitchEntity] = []
return api_handler = loader.handler_fn(controller)
async_add_entities([UnifiOutletSwitch(obj_id, controller)])
controller.api.ports.subscribe(async_add_outlet_switch, ItemEvent.ADDED)
for index in controller.api.outlets:
async_add_outlet_switch(ItemEvent.ADDED, index)
def async_add_dpi_switch(_: ItemEvent, obj_id: str) -> None:
"""Add DPI switch from UniFi controller."""
if (
not controller.option_dpi_restrictions
or not controller.api.dpi_groups[obj_id].dpiapp_ids
):
return
async_add_entities([UnifiDPIRestrictionSwitch(obj_id, controller)])
controller.api.ports.subscribe(async_add_dpi_switch, ItemEvent.ADDED)
for dpi_group_id in controller.api.dpi_groups:
async_add_dpi_switch(ItemEvent.ADDED, dpi_group_id)
@callback @callback
def async_add_poe_switch(_: ItemEvent, obj_id: str) -> None: def async_create_entity(event: ItemEvent, obj_id: str) -> None:
"""Add port PoE switch from UniFi controller.""" """Create UniFi entity."""
if not controller.api.ports[obj_id].port_poe: if not loader.config_option_fn(controller) or not loader.value_fn(
api_handler, obj_id
):
return return
async_add_entities([UnifiPoePortSwitch(obj_id, controller)])
controller.api.ports.subscribe(async_add_poe_switch, ItemEvent.ADDED) entity = loader.entity_cls(obj_id, controller)
if event == ItemEvent.ADDED:
async_add_entities(entities)
return
entities.append(entity)
for port_idx in controller.api.ports: for obj_id in api_handler:
async_add_poe_switch(ItemEvent.ADDED, port_idx) async_create_entity(ItemEvent.CHANGED, obj_id)
async_add_entities(entities)
api_handler.subscribe(async_create_entity, ItemEvent.ADDED)
for unifi_loader in UNIFI_LOADERS:
async_load_entities(unifi_loader)
@callback @callback
@ -640,3 +654,25 @@ class UnifiPoePortSwitch(SwitchEntity):
await self.controller.api.request( await self.controller.api.request(
DeviceSetPoePortModeRequest.create(device, self._index, "off") DeviceSetPoePortModeRequest.create(device, self._index, "off")
) )
UNIFI_LOADERS: tuple[UnifiEntityLoader, ...] = (
UnifiEntityLoader[DPIRestrictionGroups](
config_option_fn=lambda controller: controller.option_dpi_restrictions,
entity_cls=UnifiDPIRestrictionSwitch,
handler_fn=lambda controller: controller.api.dpi_groups,
value_fn=lambda handler, index: bool(handler[index].dpiapp_ids),
),
UnifiEntityLoader[Outlets](
config_option_fn=lambda controller: True,
entity_cls=UnifiOutletSwitch,
handler_fn=lambda controller: controller.api.outlets,
value_fn=lambda handler, index: handler[index].has_relay,
),
UnifiEntityLoader[Ports](
config_option_fn=lambda controller: True,
entity_cls=UnifiPoePortSwitch,
handler_fn=lambda controller: controller.api.ports,
value_fn=lambda handler, index: handler[index].port_poe,
),
)