"""Represent the Netgear router and its devices.""" from __future__ import annotations from abc import abstractmethod import asyncio from datetime import timedelta import logging 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, 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.""" self.hass = hass self.entry = entry self.entry_id = entry.entry_id self.unique_id = entry.unique_id self._host = entry.data.get(CONF_HOST) self._port = entry.data.get(CONF_PORT) self._ssl = entry.data.get(CONF_SSL) self._username = entry.data.get(CONF_USERNAME) self._password = entry.data[CONF_PASSWORD] self._info = None self.model = "" 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 = {} def _setup(self) -> None: """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"] for model in MODELS_V2: if self.model.startswith(model): self.method_version = 2 if self.method_version == 2: 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.""" if not await self.hass.async_add_executor_job(self._setup): return False # set already known devices to away instead of unavailable 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).get(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, } 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) -> None: """Update Netgear devices.""" new_device = False ntg_devices = await self.async_get_attached_devices() now = dt_util.utcnow() if ntg_devices is None: return 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) -> 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) @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)}, )