453 lines
15 KiB
Python
453 lines
15 KiB
Python
"""Support for Netgear routers."""
|
|
from __future__ import annotations
|
|
|
|
from collections.abc import Callable
|
|
from dataclasses import dataclass
|
|
from datetime import date, datetime
|
|
from decimal import Decimal
|
|
import logging
|
|
|
|
from homeassistant.components.sensor import (
|
|
RestoreSensor,
|
|
SensorDeviceClass,
|
|
SensorEntity,
|
|
SensorEntityDescription,
|
|
SensorStateClass,
|
|
)
|
|
from homeassistant.config_entries import ConfigEntry
|
|
from homeassistant.const import (
|
|
PERCENTAGE,
|
|
EntityCategory,
|
|
UnitOfDataRate,
|
|
UnitOfInformation,
|
|
UnitOfTime,
|
|
)
|
|
from homeassistant.core import HomeAssistant, callback
|
|
from homeassistant.helpers.entity_platform import AddEntitiesCallback
|
|
from homeassistant.helpers.typing import StateType
|
|
from homeassistant.helpers.update_coordinator import DataUpdateCoordinator
|
|
|
|
from .const import (
|
|
DOMAIN,
|
|
KEY_COORDINATOR,
|
|
KEY_COORDINATOR_LINK,
|
|
KEY_COORDINATOR_SPEED,
|
|
KEY_COORDINATOR_TRAFFIC,
|
|
KEY_COORDINATOR_UTIL,
|
|
KEY_ROUTER,
|
|
)
|
|
from .entity import NetgearDeviceEntity, NetgearRouterCoordinatorEntity
|
|
from .router import NetgearRouter
|
|
|
|
_LOGGER = logging.getLogger(__name__)
|
|
|
|
SENSOR_TYPES = {
|
|
"type": SensorEntityDescription(
|
|
key="type",
|
|
translation_key="link_type",
|
|
entity_category=EntityCategory.DIAGNOSTIC,
|
|
icon="mdi:lan",
|
|
),
|
|
"link_rate": SensorEntityDescription(
|
|
key="link_rate",
|
|
translation_key="link_rate",
|
|
native_unit_of_measurement="Mbps",
|
|
entity_category=EntityCategory.DIAGNOSTIC,
|
|
icon="mdi:speedometer",
|
|
),
|
|
"signal": SensorEntityDescription(
|
|
key="signal",
|
|
translation_key="signal_strength",
|
|
native_unit_of_measurement=PERCENTAGE,
|
|
entity_category=EntityCategory.DIAGNOSTIC,
|
|
icon="mdi:wifi",
|
|
),
|
|
"ssid": SensorEntityDescription(
|
|
key="ssid",
|
|
translation_key="ssid",
|
|
entity_category=EntityCategory.DIAGNOSTIC,
|
|
icon="mdi:wifi-marker",
|
|
),
|
|
"conn_ap_mac": SensorEntityDescription(
|
|
key="conn_ap_mac",
|
|
translation_key="access_point_mac",
|
|
entity_category=EntityCategory.DIAGNOSTIC,
|
|
icon="mdi:router-network",
|
|
),
|
|
}
|
|
|
|
|
|
@dataclass
|
|
class NetgearSensorEntityDescription(SensorEntityDescription):
|
|
"""Class describing Netgear sensor entities."""
|
|
|
|
value: Callable = lambda data: data
|
|
index: int = 0
|
|
|
|
|
|
SENSOR_TRAFFIC_TYPES = [
|
|
NetgearSensorEntityDescription(
|
|
key="NewTodayUpload",
|
|
translation_key="upload_today",
|
|
entity_category=EntityCategory.DIAGNOSTIC,
|
|
native_unit_of_measurement=UnitOfInformation.MEGABYTES,
|
|
device_class=SensorDeviceClass.DATA_SIZE,
|
|
icon="mdi:upload",
|
|
),
|
|
NetgearSensorEntityDescription(
|
|
key="NewTodayDownload",
|
|
translation_key="download_today",
|
|
entity_category=EntityCategory.DIAGNOSTIC,
|
|
native_unit_of_measurement=UnitOfInformation.MEGABYTES,
|
|
device_class=SensorDeviceClass.DATA_SIZE,
|
|
icon="mdi:download",
|
|
),
|
|
NetgearSensorEntityDescription(
|
|
key="NewYesterdayUpload",
|
|
translation_key="upload_yesterday",
|
|
entity_category=EntityCategory.DIAGNOSTIC,
|
|
native_unit_of_measurement=UnitOfInformation.MEGABYTES,
|
|
device_class=SensorDeviceClass.DATA_SIZE,
|
|
icon="mdi:upload",
|
|
),
|
|
NetgearSensorEntityDescription(
|
|
key="NewYesterdayDownload",
|
|
translation_key="download_yesterday",
|
|
entity_category=EntityCategory.DIAGNOSTIC,
|
|
native_unit_of_measurement=UnitOfInformation.MEGABYTES,
|
|
device_class=SensorDeviceClass.DATA_SIZE,
|
|
icon="mdi:download",
|
|
),
|
|
NetgearSensorEntityDescription(
|
|
key="NewWeekUpload",
|
|
translation_key="upload_week",
|
|
entity_category=EntityCategory.DIAGNOSTIC,
|
|
native_unit_of_measurement=UnitOfInformation.MEGABYTES,
|
|
device_class=SensorDeviceClass.DATA_SIZE,
|
|
icon="mdi:upload",
|
|
index=0,
|
|
value=lambda data: data[0],
|
|
),
|
|
NetgearSensorEntityDescription(
|
|
key="NewWeekUpload",
|
|
translation_key="upload_week_average",
|
|
entity_category=EntityCategory.DIAGNOSTIC,
|
|
native_unit_of_measurement=UnitOfInformation.MEGABYTES,
|
|
device_class=SensorDeviceClass.DATA_SIZE,
|
|
icon="mdi:upload",
|
|
index=1,
|
|
value=lambda data: data[1],
|
|
),
|
|
NetgearSensorEntityDescription(
|
|
key="NewWeekDownload",
|
|
translation_key="download_week",
|
|
entity_category=EntityCategory.DIAGNOSTIC,
|
|
native_unit_of_measurement=UnitOfInformation.MEGABYTES,
|
|
device_class=SensorDeviceClass.DATA_SIZE,
|
|
icon="mdi:download",
|
|
index=0,
|
|
value=lambda data: data[0],
|
|
),
|
|
NetgearSensorEntityDescription(
|
|
key="NewWeekDownload",
|
|
translation_key="download_week_average",
|
|
entity_category=EntityCategory.DIAGNOSTIC,
|
|
native_unit_of_measurement=UnitOfInformation.MEGABYTES,
|
|
device_class=SensorDeviceClass.DATA_SIZE,
|
|
icon="mdi:download",
|
|
index=1,
|
|
value=lambda data: data[1],
|
|
),
|
|
NetgearSensorEntityDescription(
|
|
key="NewMonthUpload",
|
|
translation_key="upload_month",
|
|
entity_category=EntityCategory.DIAGNOSTIC,
|
|
native_unit_of_measurement=UnitOfInformation.MEGABYTES,
|
|
device_class=SensorDeviceClass.DATA_SIZE,
|
|
icon="mdi:upload",
|
|
index=0,
|
|
value=lambda data: data[0],
|
|
),
|
|
NetgearSensorEntityDescription(
|
|
key="NewMonthUpload",
|
|
translation_key="upload_month_average",
|
|
entity_category=EntityCategory.DIAGNOSTIC,
|
|
native_unit_of_measurement=UnitOfInformation.MEGABYTES,
|
|
device_class=SensorDeviceClass.DATA_SIZE,
|
|
icon="mdi:upload",
|
|
index=1,
|
|
value=lambda data: data[1],
|
|
),
|
|
NetgearSensorEntityDescription(
|
|
key="NewMonthDownload",
|
|
translation_key="download_month",
|
|
entity_category=EntityCategory.DIAGNOSTIC,
|
|
native_unit_of_measurement=UnitOfInformation.MEGABYTES,
|
|
device_class=SensorDeviceClass.DATA_SIZE,
|
|
icon="mdi:download",
|
|
index=0,
|
|
value=lambda data: data[0],
|
|
),
|
|
NetgearSensorEntityDescription(
|
|
key="NewMonthDownload",
|
|
translation_key="download_month_average",
|
|
entity_category=EntityCategory.DIAGNOSTIC,
|
|
native_unit_of_measurement=UnitOfInformation.MEGABYTES,
|
|
device_class=SensorDeviceClass.DATA_SIZE,
|
|
icon="mdi:download",
|
|
index=1,
|
|
value=lambda data: data[1],
|
|
),
|
|
NetgearSensorEntityDescription(
|
|
key="NewLastMonthUpload",
|
|
translation_key="upload_last_month",
|
|
entity_category=EntityCategory.DIAGNOSTIC,
|
|
native_unit_of_measurement=UnitOfInformation.MEGABYTES,
|
|
device_class=SensorDeviceClass.DATA_SIZE,
|
|
icon="mdi:upload",
|
|
index=0,
|
|
value=lambda data: data[0],
|
|
),
|
|
NetgearSensorEntityDescription(
|
|
key="NewLastMonthUpload",
|
|
translation_key="upload_last_month_average",
|
|
entity_category=EntityCategory.DIAGNOSTIC,
|
|
native_unit_of_measurement=UnitOfInformation.MEGABYTES,
|
|
device_class=SensorDeviceClass.DATA_SIZE,
|
|
icon="mdi:upload",
|
|
index=1,
|
|
value=lambda data: data[1],
|
|
),
|
|
NetgearSensorEntityDescription(
|
|
key="NewLastMonthDownload",
|
|
translation_key="download_last_month",
|
|
entity_category=EntityCategory.DIAGNOSTIC,
|
|
native_unit_of_measurement=UnitOfInformation.MEGABYTES,
|
|
device_class=SensorDeviceClass.DATA_SIZE,
|
|
icon="mdi:download",
|
|
index=0,
|
|
value=lambda data: data[0],
|
|
),
|
|
NetgearSensorEntityDescription(
|
|
key="NewLastMonthDownload",
|
|
translation_key="download_last_month_average",
|
|
entity_category=EntityCategory.DIAGNOSTIC,
|
|
native_unit_of_measurement=UnitOfInformation.MEGABYTES,
|
|
device_class=SensorDeviceClass.DATA_SIZE,
|
|
icon="mdi:download",
|
|
index=1,
|
|
value=lambda data: data[1],
|
|
),
|
|
]
|
|
|
|
SENSOR_SPEED_TYPES = [
|
|
NetgearSensorEntityDescription(
|
|
key="NewOOKLAUplinkBandwidth",
|
|
translation_key="uplink_bandwidth",
|
|
entity_category=EntityCategory.DIAGNOSTIC,
|
|
native_unit_of_measurement=UnitOfDataRate.MEGABITS_PER_SECOND,
|
|
device_class=SensorDeviceClass.DATA_RATE,
|
|
icon="mdi:upload",
|
|
),
|
|
NetgearSensorEntityDescription(
|
|
key="NewOOKLADownlinkBandwidth",
|
|
translation_key="downlink_bandwidth",
|
|
entity_category=EntityCategory.DIAGNOSTIC,
|
|
native_unit_of_measurement=UnitOfDataRate.MEGABITS_PER_SECOND,
|
|
device_class=SensorDeviceClass.DATA_RATE,
|
|
icon="mdi:download",
|
|
),
|
|
NetgearSensorEntityDescription(
|
|
key="AveragePing",
|
|
translation_key="average_ping",
|
|
entity_category=EntityCategory.DIAGNOSTIC,
|
|
native_unit_of_measurement=UnitOfTime.MILLISECONDS,
|
|
icon="mdi:wan",
|
|
),
|
|
]
|
|
|
|
SENSOR_UTILIZATION = [
|
|
NetgearSensorEntityDescription(
|
|
key="NewCPUUtilization",
|
|
translation_key="cpu_utilization",
|
|
entity_category=EntityCategory.DIAGNOSTIC,
|
|
native_unit_of_measurement=PERCENTAGE,
|
|
icon="mdi:cpu-64-bit",
|
|
state_class=SensorStateClass.MEASUREMENT,
|
|
),
|
|
NetgearSensorEntityDescription(
|
|
key="NewMemoryUtilization",
|
|
translation_key="memory_utilization",
|
|
entity_category=EntityCategory.DIAGNOSTIC,
|
|
native_unit_of_measurement=PERCENTAGE,
|
|
icon="mdi:memory",
|
|
state_class=SensorStateClass.MEASUREMENT,
|
|
),
|
|
]
|
|
|
|
SENSOR_LINK_TYPES = [
|
|
NetgearSensorEntityDescription(
|
|
key="NewEthernetLinkStatus",
|
|
translation_key="ethernet_link_status",
|
|
entity_category=EntityCategory.DIAGNOSTIC,
|
|
icon="mdi:ethernet",
|
|
),
|
|
]
|
|
|
|
|
|
async def async_setup_entry(
|
|
hass: HomeAssistant, entry: ConfigEntry, async_add_entities: AddEntitiesCallback
|
|
) -> None:
|
|
"""Set up device tracker for Netgear component."""
|
|
router = hass.data[DOMAIN][entry.entry_id][KEY_ROUTER]
|
|
coordinator = hass.data[DOMAIN][entry.entry_id][KEY_COORDINATOR]
|
|
coordinator_traffic = hass.data[DOMAIN][entry.entry_id][KEY_COORDINATOR_TRAFFIC]
|
|
coordinator_speed = hass.data[DOMAIN][entry.entry_id][KEY_COORDINATOR_SPEED]
|
|
coordinator_utilization = hass.data[DOMAIN][entry.entry_id][KEY_COORDINATOR_UTIL]
|
|
coordinator_link = hass.data[DOMAIN][entry.entry_id][KEY_COORDINATOR_LINK]
|
|
|
|
# Router entities
|
|
router_entities = []
|
|
|
|
for description in SENSOR_TRAFFIC_TYPES:
|
|
router_entities.append(
|
|
NetgearRouterSensorEntity(coordinator_traffic, router, description)
|
|
)
|
|
|
|
for description in SENSOR_SPEED_TYPES:
|
|
router_entities.append(
|
|
NetgearRouterSensorEntity(coordinator_speed, router, description)
|
|
)
|
|
|
|
for description in SENSOR_UTILIZATION:
|
|
router_entities.append(
|
|
NetgearRouterSensorEntity(coordinator_utilization, router, description)
|
|
)
|
|
|
|
for description in SENSOR_LINK_TYPES:
|
|
router_entities.append(
|
|
NetgearRouterSensorEntity(coordinator_link, router, description)
|
|
)
|
|
|
|
async_add_entities(router_entities)
|
|
|
|
# Entities per network device
|
|
tracked = set()
|
|
sensors = ["type", "link_rate", "signal"]
|
|
if router.method_version == 2:
|
|
sensors.extend(["ssid", "conn_ap_mac"])
|
|
|
|
@callback
|
|
def new_device_callback() -> None:
|
|
"""Add new devices if needed."""
|
|
if not coordinator.data:
|
|
return
|
|
|
|
new_entities = []
|
|
|
|
for mac, device in router.devices.items():
|
|
if mac in tracked:
|
|
continue
|
|
|
|
new_entities.extend(
|
|
[
|
|
NetgearSensorEntity(coordinator, router, device, attribute)
|
|
for attribute in sensors
|
|
]
|
|
)
|
|
tracked.add(mac)
|
|
|
|
async_add_entities(new_entities)
|
|
|
|
entry.async_on_unload(coordinator.async_add_listener(new_device_callback))
|
|
|
|
coordinator.data = True
|
|
new_device_callback()
|
|
|
|
|
|
class NetgearSensorEntity(NetgearDeviceEntity, SensorEntity):
|
|
"""Representation of a device connected to a Netgear router."""
|
|
|
|
_attr_entity_registry_enabled_default = False
|
|
|
|
def __init__(
|
|
self,
|
|
coordinator: DataUpdateCoordinator,
|
|
router: NetgearRouter,
|
|
device: dict,
|
|
attribute: str,
|
|
) -> None:
|
|
"""Initialize a Netgear device."""
|
|
super().__init__(coordinator, router, device)
|
|
self._attribute = attribute
|
|
self.entity_description = SENSOR_TYPES[attribute]
|
|
self._attr_unique_id = f"{self._mac}-{attribute}"
|
|
self._state = device.get(attribute)
|
|
|
|
@property
|
|
def native_value(self):
|
|
"""Return the state of the sensor."""
|
|
return self._state
|
|
|
|
@callback
|
|
def async_update_device(self) -> None:
|
|
"""Update the Netgear device."""
|
|
self._device = self._router.devices[self._mac]
|
|
self._active = self._device["active"]
|
|
if self._device.get(self._attribute) is not None:
|
|
self._state = self._device[self._attribute]
|
|
|
|
|
|
class NetgearRouterSensorEntity(NetgearRouterCoordinatorEntity, RestoreSensor):
|
|
"""Representation of a device connected to a Netgear router."""
|
|
|
|
_attr_entity_registry_enabled_default = False
|
|
entity_description: NetgearSensorEntityDescription
|
|
|
|
def __init__(
|
|
self,
|
|
coordinator: DataUpdateCoordinator,
|
|
router: NetgearRouter,
|
|
entity_description: NetgearSensorEntityDescription,
|
|
) -> None:
|
|
"""Initialize a Netgear device."""
|
|
super().__init__(coordinator, router)
|
|
self.entity_description = entity_description
|
|
self._attr_unique_id = f"{router.serial_number}-{entity_description.key}-{entity_description.index}"
|
|
|
|
self._value: StateType | date | datetime | Decimal = None
|
|
self.async_update_device()
|
|
|
|
@property
|
|
def native_value(self):
|
|
"""Return the state of the sensor."""
|
|
return self._value
|
|
|
|
async def async_added_to_hass(self) -> None:
|
|
"""Handle entity which will be added."""
|
|
await super().async_added_to_hass()
|
|
if self.coordinator.data is None:
|
|
sensor_data = await self.async_get_last_sensor_data()
|
|
if sensor_data is not None:
|
|
self._value = sensor_data.native_value
|
|
else:
|
|
await self.coordinator.async_request_refresh()
|
|
|
|
@callback
|
|
def async_update_device(self) -> None:
|
|
"""Update the Netgear device."""
|
|
if self.coordinator.data is None:
|
|
return
|
|
|
|
data = self.coordinator.data.get(self.entity_description.key)
|
|
if data is None:
|
|
self._value = None
|
|
_LOGGER.debug(
|
|
"key '%s' not in Netgear router response '%s'",
|
|
self.entity_description.key,
|
|
data,
|
|
)
|
|
return
|
|
|
|
self._value = self.entity_description.value(data)
|