346 lines
11 KiB
Python
346 lines
11 KiB
Python
"""Represent the Netgear router and its devices."""
|
|
from __future__ import annotations
|
|
|
|
from abc import abstractmethod
|
|
import asyncio
|
|
from datetime import timedelta
|
|
import logging
|
|
from typing import Any
|
|
|
|
from pynetgear import Netgear
|
|
|
|
from homeassistant.config_entries import ConfigEntry
|
|
from homeassistant.const import (
|
|
CONF_HOST,
|
|
CONF_PASSWORD,
|
|
CONF_PORT,
|
|
CONF_SSL,
|
|
CONF_USERNAME,
|
|
)
|
|
from homeassistant.core import HomeAssistant, callback
|
|
from homeassistant.helpers import device_registry as dr
|
|
from homeassistant.helpers.device_registry import CONNECTION_NETWORK_MAC, format_mac
|
|
from homeassistant.helpers.entity import DeviceInfo
|
|
from homeassistant.helpers.update_coordinator import (
|
|
CoordinatorEntity,
|
|
DataUpdateCoordinator,
|
|
)
|
|
from homeassistant.util import dt as dt_util
|
|
|
|
from .const import (
|
|
CONF_CONSIDER_HOME,
|
|
DEFAULT_CONSIDER_HOME,
|
|
DEFAULT_NAME,
|
|
DOMAIN,
|
|
MODE_ROUTER,
|
|
MODELS_V2,
|
|
)
|
|
from .errors import CannotLoginException
|
|
|
|
_LOGGER = logging.getLogger(__name__)
|
|
|
|
|
|
def get_api(
|
|
password: str,
|
|
host: str = None,
|
|
username: str = None,
|
|
port: int = None,
|
|
ssl: bool = False,
|
|
) -> Netgear:
|
|
"""Get the Netgear API and login to it."""
|
|
api: Netgear = Netgear(password, host, username, port, ssl)
|
|
|
|
if not api.login_try_port():
|
|
raise CannotLoginException
|
|
|
|
return api
|
|
|
|
|
|
class NetgearRouter:
|
|
"""Representation of a Netgear router."""
|
|
|
|
def __init__(self, hass: HomeAssistant, entry: ConfigEntry) -> None:
|
|
"""Initialize a Netgear router."""
|
|
assert entry.unique_id
|
|
self.hass = hass
|
|
self.entry = entry
|
|
self.entry_id = entry.entry_id
|
|
self.unique_id = entry.unique_id
|
|
self._host: str = entry.data[CONF_HOST]
|
|
self._port: int = entry.data[CONF_PORT]
|
|
self._ssl: bool = entry.data[CONF_SSL]
|
|
self._username = entry.data.get(CONF_USERNAME)
|
|
self._password = entry.data[CONF_PASSWORD]
|
|
|
|
self._info = None
|
|
self.model = ""
|
|
self.mode = MODE_ROUTER
|
|
self.device_name = ""
|
|
self.firmware_version = ""
|
|
self.hardware_version = ""
|
|
self.serial_number = ""
|
|
|
|
self.method_version = 1
|
|
consider_home_int = entry.options.get(
|
|
CONF_CONSIDER_HOME, DEFAULT_CONSIDER_HOME.total_seconds()
|
|
)
|
|
self._consider_home = timedelta(seconds=consider_home_int)
|
|
|
|
self._api: Netgear = None
|
|
self._api_lock = asyncio.Lock()
|
|
|
|
self.devices: dict[str, Any] = {}
|
|
|
|
def _setup(self) -> bool:
|
|
"""Set up a Netgear router sync portion."""
|
|
self._api = get_api(
|
|
self._password,
|
|
self._host,
|
|
self._username,
|
|
self._port,
|
|
self._ssl,
|
|
)
|
|
|
|
self._info = self._api.get_info()
|
|
if self._info is None:
|
|
return False
|
|
|
|
self.device_name = self._info.get("DeviceName", DEFAULT_NAME)
|
|
self.model = self._info.get("ModelName")
|
|
self.firmware_version = self._info.get("Firmwareversion")
|
|
self.hardware_version = self._info.get("Hardwareversion")
|
|
self.serial_number = self._info["SerialNumber"]
|
|
self.mode = self._info.get("DeviceMode", MODE_ROUTER)
|
|
|
|
for model in MODELS_V2:
|
|
if self.model.startswith(model):
|
|
self.method_version = 2
|
|
|
|
if self.method_version == 2 and self.mode == MODE_ROUTER:
|
|
if not self._api.get_attached_devices_2():
|
|
_LOGGER.error(
|
|
"Netgear Model '%s' in MODELS_V2 list, but failed to get attached devices using V2",
|
|
self.model,
|
|
)
|
|
self.method_version = 1
|
|
|
|
return True
|
|
|
|
async def async_setup(self) -> bool:
|
|
"""Set up a Netgear router."""
|
|
async with self._api_lock:
|
|
if not await self.hass.async_add_executor_job(self._setup):
|
|
return False
|
|
|
|
# set already known devices to away instead of unavailable
|
|
if self.mode == MODE_ROUTER:
|
|
device_registry = dr.async_get(self.hass)
|
|
devices = dr.async_entries_for_config_entry(device_registry, self.entry_id)
|
|
for device_entry in devices:
|
|
if device_entry.via_device_id is None:
|
|
continue # do not add the router itself
|
|
|
|
device_mac = dict(device_entry.connections)[dr.CONNECTION_NETWORK_MAC]
|
|
self.devices[device_mac] = {
|
|
"mac": device_mac,
|
|
"name": device_entry.name,
|
|
"active": False,
|
|
"last_seen": dt_util.utcnow() - timedelta(days=365),
|
|
"device_model": None,
|
|
"device_type": None,
|
|
"type": None,
|
|
"link_rate": None,
|
|
"signal": None,
|
|
"ip": None,
|
|
"ssid": None,
|
|
"conn_ap_mac": None,
|
|
"allow_or_block": None,
|
|
}
|
|
|
|
return True
|
|
|
|
async def async_get_attached_devices(self) -> list:
|
|
"""Get the devices connected to the router."""
|
|
if self.method_version == 1:
|
|
async with self._api_lock:
|
|
return await self.hass.async_add_executor_job(
|
|
self._api.get_attached_devices
|
|
)
|
|
|
|
async with self._api_lock:
|
|
return await self.hass.async_add_executor_job(
|
|
self._api.get_attached_devices_2
|
|
)
|
|
|
|
async def async_update_device_trackers(self, now=None) -> bool:
|
|
"""Update Netgear devices."""
|
|
new_device = False
|
|
ntg_devices = await self.async_get_attached_devices()
|
|
now = dt_util.utcnow()
|
|
|
|
if ntg_devices is None:
|
|
return new_device
|
|
|
|
if _LOGGER.isEnabledFor(logging.DEBUG):
|
|
_LOGGER.debug("Netgear scan result: \n%s", ntg_devices)
|
|
|
|
for ntg_device in ntg_devices:
|
|
device_mac = format_mac(ntg_device.mac)
|
|
|
|
if not self.devices.get(device_mac):
|
|
new_device = True
|
|
|
|
# ntg_device is a namedtuple from the collections module that needs conversion to a dict through ._asdict method
|
|
self.devices[device_mac] = ntg_device._asdict()
|
|
self.devices[device_mac]["mac"] = device_mac
|
|
self.devices[device_mac]["last_seen"] = now
|
|
|
|
for device in self.devices.values():
|
|
device["active"] = now - device["last_seen"] <= self._consider_home
|
|
|
|
if new_device:
|
|
_LOGGER.debug("Netgear tracker: new device found")
|
|
|
|
return new_device
|
|
|
|
async def async_get_traffic_meter(self) -> dict[str, Any] | None:
|
|
"""Get the traffic meter data of the router."""
|
|
async with self._api_lock:
|
|
return await self.hass.async_add_executor_job(self._api.get_traffic_meter)
|
|
|
|
async def async_get_speed_test(self) -> dict[str, Any] | None:
|
|
"""Perform a speed test and get the results from the router."""
|
|
async with self._api_lock:
|
|
return await self.hass.async_add_executor_job(
|
|
self._api.get_new_speed_test_result
|
|
)
|
|
|
|
async def async_allow_block_device(self, mac: str, allow_block: str) -> None:
|
|
"""Allow or block a device connected to the router."""
|
|
async with self._api_lock:
|
|
await self.hass.async_add_executor_job(
|
|
self._api.allow_block_device, mac, allow_block
|
|
)
|
|
|
|
async def async_reboot(self) -> None:
|
|
"""Reboot the router."""
|
|
async with self._api_lock:
|
|
await self.hass.async_add_executor_job(self._api.reboot)
|
|
|
|
@property
|
|
def port(self) -> int:
|
|
"""Port used by the API."""
|
|
return self._api.port
|
|
|
|
@property
|
|
def ssl(self) -> bool:
|
|
"""SSL used by the API."""
|
|
return self._api.ssl
|
|
|
|
|
|
class NetgearBaseEntity(CoordinatorEntity):
|
|
"""Base class for a device connected to a Netgear router."""
|
|
|
|
def __init__(
|
|
self, coordinator: DataUpdateCoordinator, router: NetgearRouter, device: dict
|
|
) -> None:
|
|
"""Initialize a Netgear device."""
|
|
super().__init__(coordinator)
|
|
self._router = router
|
|
self._device = device
|
|
self._mac = device["mac"]
|
|
self._name = self.get_device_name()
|
|
self._device_name = self._name
|
|
self._active = device["active"]
|
|
|
|
def get_device_name(self):
|
|
"""Return the name of the given device or the MAC if we don't know."""
|
|
name = self._device["name"]
|
|
if not name or name == "--":
|
|
name = self._mac
|
|
|
|
return name
|
|
|
|
@abstractmethod
|
|
@callback
|
|
def async_update_device(self) -> None:
|
|
"""Update the Netgear device."""
|
|
|
|
@callback
|
|
def _handle_coordinator_update(self) -> None:
|
|
"""Handle updated data from the coordinator."""
|
|
self.async_update_device()
|
|
super()._handle_coordinator_update()
|
|
|
|
@property
|
|
def name(self) -> str:
|
|
"""Return the name."""
|
|
return self._name
|
|
|
|
|
|
class NetgearDeviceEntity(NetgearBaseEntity):
|
|
"""Base class for a device connected to a Netgear router."""
|
|
|
|
def __init__(
|
|
self, coordinator: DataUpdateCoordinator, router: NetgearRouter, device: dict
|
|
) -> None:
|
|
"""Initialize a Netgear device."""
|
|
super().__init__(coordinator, router, device)
|
|
self._unique_id = self._mac
|
|
|
|
@property
|
|
def unique_id(self) -> str:
|
|
"""Return a unique ID."""
|
|
return self._unique_id
|
|
|
|
@property
|
|
def device_info(self) -> DeviceInfo:
|
|
"""Return the device information."""
|
|
return DeviceInfo(
|
|
connections={(CONNECTION_NETWORK_MAC, self._mac)},
|
|
default_name=self._device_name,
|
|
default_model=self._device["device_model"],
|
|
via_device=(DOMAIN, self._router.unique_id),
|
|
)
|
|
|
|
|
|
class NetgearRouterEntity(CoordinatorEntity):
|
|
"""Base class for a Netgear router entity."""
|
|
|
|
def __init__(
|
|
self, coordinator: DataUpdateCoordinator, router: NetgearRouter
|
|
) -> None:
|
|
"""Initialize a Netgear device."""
|
|
super().__init__(coordinator)
|
|
self._router = router
|
|
self._name = router.device_name
|
|
self._unique_id = router.serial_number
|
|
|
|
@abstractmethod
|
|
@callback
|
|
def async_update_device(self) -> None:
|
|
"""Update the Netgear device."""
|
|
|
|
@callback
|
|
def _handle_coordinator_update(self) -> None:
|
|
"""Handle updated data from the coordinator."""
|
|
self.async_update_device()
|
|
super()._handle_coordinator_update()
|
|
|
|
@property
|
|
def unique_id(self) -> str:
|
|
"""Return a unique ID."""
|
|
return self._unique_id
|
|
|
|
@property
|
|
def name(self) -> str:
|
|
"""Return the name."""
|
|
return self._name
|
|
|
|
@property
|
|
def device_info(self) -> DeviceInfo:
|
|
"""Return the device information."""
|
|
return DeviceInfo(
|
|
identifiers={(DOMAIN, self._router.unique_id)},
|
|
)
|