196 lines
6.1 KiB
Python
196 lines
6.1 KiB
Python
"""The Keenetic Client class."""
|
|
from __future__ import annotations
|
|
|
|
from datetime import timedelta
|
|
import logging
|
|
from typing import Callable
|
|
|
|
from ndms2_client import Client, ConnectionException, Device, TelnetConnection
|
|
from ndms2_client.client import RouterInfo
|
|
|
|
from homeassistant.config_entries import ConfigEntry
|
|
from homeassistant.const import (
|
|
CONF_HOST,
|
|
CONF_PASSWORD,
|
|
CONF_PORT,
|
|
CONF_SCAN_INTERVAL,
|
|
CONF_USERNAME,
|
|
)
|
|
from homeassistant.core import HomeAssistant
|
|
from homeassistant.exceptions import ConfigEntryNotReady
|
|
from homeassistant.helpers.dispatcher import async_dispatcher_send
|
|
from homeassistant.helpers.event import async_call_later
|
|
import homeassistant.util.dt as dt_util
|
|
|
|
from .const import (
|
|
CONF_CONSIDER_HOME,
|
|
CONF_INCLUDE_ARP,
|
|
CONF_INCLUDE_ASSOCIATED,
|
|
CONF_INTERFACES,
|
|
CONF_TRY_HOTSPOT,
|
|
DOMAIN,
|
|
)
|
|
|
|
_LOGGER = logging.getLogger(__name__)
|
|
|
|
|
|
class KeeneticRouter:
|
|
"""Keenetic client Object."""
|
|
|
|
def __init__(self, hass: HomeAssistant, config_entry: ConfigEntry) -> None:
|
|
"""Initialize the Client."""
|
|
self.hass = hass
|
|
self.config_entry = config_entry
|
|
self._last_devices: dict[str, Device] = {}
|
|
self._router_info: RouterInfo | None = None
|
|
self._connection: TelnetConnection | None = None
|
|
self._client: Client | None = None
|
|
self._cancel_periodic_update: Callable | None = None
|
|
self._available = False
|
|
self._progress = None
|
|
self._tracked_interfaces = set(config_entry.options[CONF_INTERFACES])
|
|
|
|
@property
|
|
def client(self):
|
|
"""Read-only accessor for the client connection."""
|
|
return self._client
|
|
|
|
@property
|
|
def last_devices(self):
|
|
"""Read-only accessor for last_devices."""
|
|
return self._last_devices
|
|
|
|
@property
|
|
def host(self):
|
|
"""Return the host of this hub."""
|
|
return self.config_entry.data[CONF_HOST]
|
|
|
|
@property
|
|
def device_info(self):
|
|
"""Return the host of this hub."""
|
|
return {
|
|
"identifiers": {(DOMAIN, f"router-{self.config_entry.entry_id}")},
|
|
"manufacturer": self.manufacturer,
|
|
"model": self.model,
|
|
"name": self.name,
|
|
"sw_version": self.firmware,
|
|
}
|
|
|
|
@property
|
|
def name(self):
|
|
"""Return the name of the hub."""
|
|
return self._router_info.name if self._router_info else self.host
|
|
|
|
@property
|
|
def model(self):
|
|
"""Return the model of the hub."""
|
|
return self._router_info.model if self._router_info else None
|
|
|
|
@property
|
|
def firmware(self):
|
|
"""Return the firmware of the hub."""
|
|
return self._router_info.fw_version if self._router_info else None
|
|
|
|
@property
|
|
def manufacturer(self):
|
|
"""Return the firmware of the hub."""
|
|
return self._router_info.manufacturer if self._router_info else None
|
|
|
|
@property
|
|
def available(self):
|
|
"""Return if the hub is connected."""
|
|
return self._available
|
|
|
|
@property
|
|
def consider_home_interval(self):
|
|
"""Config entry option defining number of seconds from last seen to away."""
|
|
return timedelta(seconds=self.config_entry.options[CONF_CONSIDER_HOME])
|
|
|
|
@property
|
|
def tracked_interfaces(self):
|
|
"""Tracked interfaces."""
|
|
return self._tracked_interfaces
|
|
|
|
@property
|
|
def signal_update(self):
|
|
"""Event specific per router entry to signal updates."""
|
|
return f"keenetic-update-{self.config_entry.entry_id}"
|
|
|
|
async def request_update(self):
|
|
"""Request an update."""
|
|
if self._progress is not None:
|
|
await self._progress
|
|
return
|
|
|
|
self._progress = self.hass.async_create_task(self.async_update())
|
|
await self._progress
|
|
|
|
self._progress = None
|
|
|
|
async def async_update(self):
|
|
"""Update devices information."""
|
|
await self.hass.async_add_executor_job(self._update_devices)
|
|
async_dispatcher_send(self.hass, self.signal_update)
|
|
|
|
async def async_setup(self):
|
|
"""Set up the connection."""
|
|
self._connection = TelnetConnection(
|
|
self.config_entry.data[CONF_HOST],
|
|
self.config_entry.data[CONF_PORT],
|
|
self.config_entry.data[CONF_USERNAME],
|
|
self.config_entry.data[CONF_PASSWORD],
|
|
)
|
|
self._client = Client(self._connection)
|
|
|
|
try:
|
|
await self.hass.async_add_executor_job(self._update_router_info)
|
|
except ConnectionException as error:
|
|
raise ConfigEntryNotReady from error
|
|
|
|
async def async_update_data(_now):
|
|
await self.request_update()
|
|
self._cancel_periodic_update = async_call_later(
|
|
self.hass,
|
|
self.config_entry.options[CONF_SCAN_INTERVAL],
|
|
async_update_data,
|
|
)
|
|
|
|
await async_update_data(dt_util.utcnow())
|
|
|
|
async def async_teardown(self):
|
|
"""Teardown up the connection."""
|
|
if self._cancel_periodic_update:
|
|
self._cancel_periodic_update()
|
|
self._connection.disconnect()
|
|
|
|
def _update_router_info(self):
|
|
try:
|
|
self._router_info = self._client.get_router_info()
|
|
self._available = True
|
|
except Exception:
|
|
self._available = False
|
|
raise
|
|
|
|
def _update_devices(self):
|
|
"""Get ARP from keenetic router."""
|
|
_LOGGER.debug("Fetching devices from router")
|
|
|
|
try:
|
|
_response = self._client.get_devices(
|
|
try_hotspot=self.config_entry.options[CONF_TRY_HOTSPOT],
|
|
include_arp=self.config_entry.options[CONF_INCLUDE_ARP],
|
|
include_associated=self.config_entry.options[CONF_INCLUDE_ASSOCIATED],
|
|
)
|
|
self._last_devices = {
|
|
dev.mac: dev
|
|
for dev in _response
|
|
if dev.interface in self._tracked_interfaces
|
|
}
|
|
_LOGGER.debug("Successfully fetched data from router: %s", str(_response))
|
|
self._router_info = self._client.get_router_info()
|
|
self._available = True
|
|
|
|
except ConnectionException:
|
|
_LOGGER.error("Error fetching data from router")
|
|
self._available = False
|