"""The Mikrotik router class.""" from datetime import timedelta import logging import socket import ssl import librouteros from librouteros.login import plain as login_plain, token as login_token from homeassistant.const import CONF_HOST, CONF_PASSWORD, CONF_USERNAME, CONF_VERIFY_SSL from homeassistant.exceptions import ConfigEntryNotReady from homeassistant.helpers.dispatcher import async_dispatcher_send from homeassistant.util import slugify import homeassistant.util.dt as dt_util from .const import ( ARP, ATTR_DEVICE_TRACKER, ATTR_FIRMWARE, ATTR_MODEL, ATTR_SERIAL_NUMBER, CAPSMAN, CONF_ARP_PING, CONF_DETECTION_TIME, CONF_FORCE_DHCP, DEFAULT_DETECTION_TIME, DHCP, IDENTITY, INFO, IS_CAPSMAN, IS_WIRELESS, MIKROTIK_SERVICES, NAME, PLATFORMS, WIRELESS, ) from .errors import CannotConnect, LoginError _LOGGER = logging.getLogger(__name__) class Device: """Represents a network device.""" def __init__(self, mac, params): """Initialize the network device.""" self._mac = mac self._params = params self._last_seen = None self._attrs = {} self._wireless_params = None @property def name(self): """Return device name.""" return self._params.get("host-name", self.mac) @property def ip_address(self): """Return device primary ip address.""" return self._params.get("address") @property def mac(self): """Return device mac.""" return self._mac @property def last_seen(self): """Return device last seen.""" return self._last_seen @property def attrs(self): """Return device attributes.""" attr_data = self._wireless_params if self._wireless_params else self._params for attr in ATTR_DEVICE_TRACKER: if attr in attr_data: self._attrs[slugify(attr)] = attr_data[attr] self._attrs["ip_address"] = self._params.get("active-address") return self._attrs def update(self, wireless_params=None, params=None, active=False): """Update Device params.""" if wireless_params: self._wireless_params = wireless_params if params: self._params = params if active: self._last_seen = dt_util.utcnow() class MikrotikData: """Handle all communication with the Mikrotik API.""" def __init__(self, hass, config_entry, api): """Initialize the Mikrotik Client.""" self.hass = hass self.config_entry = config_entry self.api = api self._host = self.config_entry.data[CONF_HOST] self.all_devices = {} self.devices = {} self.available = True self.support_capsman = False self.support_wireless = False self.hostname = None self.model = None self.firmware = None self.serial_number = None @staticmethod def load_mac(devices=None): """Load dictionary using MAC address as key.""" if not devices: return None mac_devices = {} for device in devices: if "mac-address" in device: mac = device["mac-address"] mac_devices[mac] = device return mac_devices @property def arp_enabled(self): """Return arp_ping option setting.""" return self.config_entry.options[CONF_ARP_PING] @property def force_dhcp(self): """Return force_dhcp option setting.""" return self.config_entry.options[CONF_FORCE_DHCP] def get_info(self, param): """Return device model name.""" cmd = IDENTITY if param == NAME else INFO data = self.command(MIKROTIK_SERVICES[cmd]) return ( data[0].get(param) # pylint: disable=unsubscriptable-object if data else None ) def get_hub_details(self): """Get Hub info.""" self.hostname = self.get_info(NAME) self.model = self.get_info(ATTR_MODEL) self.firmware = self.get_info(ATTR_FIRMWARE) self.serial_number = self.get_info(ATTR_SERIAL_NUMBER) self.support_capsman = bool(self.command(MIKROTIK_SERVICES[IS_CAPSMAN])) self.support_wireless = bool(self.command(MIKROTIK_SERVICES[IS_WIRELESS])) def connect_to_hub(self): """Connect to hub.""" try: self.api = get_api(self.hass, self.config_entry.data) self.available = True return True except (LoginError, CannotConnect): self.available = False return False def get_list_from_interface(self, interface): """Get devices from interface.""" result = self.command(MIKROTIK_SERVICES[interface]) return self.load_mac(result) if result else {} def restore_device(self, mac): """Restore a missing device after restart.""" self.devices[mac] = Device(mac, self.all_devices[mac]) def update_devices(self): """Get list of devices with latest status.""" arp_devices = {} device_list = {} wireless_devices = {} try: self.all_devices = self.get_list_from_interface(DHCP) if self.support_capsman: _LOGGER.debug("Hub is a CAPSman manager") device_list = wireless_devices = self.get_list_from_interface(CAPSMAN) elif self.support_wireless: _LOGGER.debug("Hub supports wireless Interface") device_list = wireless_devices = self.get_list_from_interface(WIRELESS) if not device_list or self.force_dhcp: device_list = self.all_devices _LOGGER.debug("Falling back to DHCP for scanning devices") if self.arp_enabled: _LOGGER.debug("Using arp-ping to check devices") arp_devices = self.get_list_from_interface(ARP) # get new hub firmware version if updated self.firmware = self.get_info(ATTR_FIRMWARE) except (CannotConnect, socket.timeout, OSError): self.available = False return if not device_list: return for mac, params in device_list.items(): if mac not in self.devices: self.devices[mac] = Device(mac, self.all_devices.get(mac, {})) else: self.devices[mac].update(params=self.all_devices.get(mac, {})) if mac in wireless_devices: # if wireless is supported then wireless_params are params self.devices[mac].update( wireless_params=wireless_devices[mac], active=True ) continue # for wired devices or when forcing dhcp check for active-address if not params.get("active-address"): self.devices[mac].update(active=False) continue # ping check the rest of active devices if arp ping is enabled active = True if self.arp_enabled and mac in arp_devices: active = self.do_arp_ping( params.get("active-address"), arp_devices[mac].get("interface") ) self.devices[mac].update(active=active) def do_arp_ping(self, ip_address, interface): """Attempt to arp ping MAC address via interface.""" _LOGGER.debug("pinging - %s", ip_address) params = { "arp-ping": "yes", "interval": "100ms", "count": 3, "interface": interface, "address": ip_address, } cmd = "/ping" data = self.command(cmd, params) if data is not None: status = 0 for result in data: # pylint: disable=not-an-iterable if "status" in result: status += 1 if status == len(data): _LOGGER.debug( "Mikrotik %s - %s arp_ping timed out", ip_address, interface ) return False return True def command(self, cmd, params=None): """Retrieve data from Mikrotik API.""" try: _LOGGER.info("Running command %s", cmd) if params: response = list(self.api(cmd=cmd, **params)) else: response = list(self.api(cmd=cmd)) except ( librouteros.exceptions.ConnectionClosed, OSError, socket.timeout, ) as api_error: _LOGGER.error("Mikrotik %s connection error %s", self._host, api_error) raise CannotConnect from api_error except librouteros.exceptions.ProtocolError as api_error: _LOGGER.warning( "Mikrotik %s failed to retrieve data. cmd=[%s] Error: %s", self._host, cmd, api_error, ) return None return response if response else None def update(self): """Update device_tracker from Mikrotik API.""" if (not self.available or not self.api) and not self.connect_to_hub(): return _LOGGER.debug("updating network devices for host: %s", self._host) self.update_devices() class MikrotikHub: """Mikrotik Hub Object.""" def __init__(self, hass, config_entry): """Initialize the Mikrotik Client.""" self.hass = hass self.config_entry = config_entry self._mk_data = None self.progress = None @property def host(self): """Return the host of this hub.""" return self.config_entry.data[CONF_HOST] @property def hostname(self): """Return the hostname of the hub.""" return self._mk_data.hostname @property def model(self): """Return the model of the hub.""" return self._mk_data.model @property def firmware(self): """Return the firmware of the hub.""" return self._mk_data.firmware @property def serial_num(self): """Return the serial number of the hub.""" return self._mk_data.serial_number @property def available(self): """Return if the hub is connected.""" return self._mk_data.available @property def option_detection_time(self): """Config entry option defining number of seconds from last seen to away.""" return timedelta(seconds=self.config_entry.options[CONF_DETECTION_TIME]) @property def signal_update(self): """Event specific per Mikrotik entry to signal updates.""" return f"mikrotik-update-{self.host}" @property def api(self): """Represent Mikrotik data object.""" return self._mk_data async def async_add_options(self): """Populate default options for Mikrotik.""" if not self.config_entry.options: data = dict(self.config_entry.data) options = { CONF_ARP_PING: data.pop(CONF_ARP_PING, False), CONF_FORCE_DHCP: data.pop(CONF_FORCE_DHCP, False), CONF_DETECTION_TIME: data.pop( CONF_DETECTION_TIME, DEFAULT_DETECTION_TIME ), } self.hass.config_entries.async_update_entry( self.config_entry, data=data, options=options ) 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 Mikrotik devices information.""" await self.hass.async_add_executor_job(self._mk_data.update) async_dispatcher_send(self.hass, self.signal_update) async def async_setup(self): """Set up the Mikrotik hub.""" try: api = await self.hass.async_add_executor_job( get_api, self.hass, self.config_entry.data ) except CannotConnect as api_error: raise ConfigEntryNotReady from api_error except LoginError: return False self._mk_data = MikrotikData(self.hass, self.config_entry, api) await self.async_add_options() await self.hass.async_add_executor_job(self._mk_data.get_hub_details) await self.hass.async_add_executor_job(self._mk_data.update) self.hass.config_entries.async_setup_platforms(self.config_entry, PLATFORMS) return True def get_api(hass, entry): """Connect to Mikrotik hub.""" _LOGGER.debug("Connecting to Mikrotik hub [%s]", entry[CONF_HOST]) _login_method = (login_plain, login_token) kwargs = {"login_methods": _login_method, "port": entry["port"], "encoding": "utf8"} if entry[CONF_VERIFY_SSL]: ssl_context = ssl.create_default_context() ssl_context.check_hostname = False ssl_context.verify_mode = ssl.CERT_NONE _ssl_wrapper = ssl_context.wrap_socket kwargs["ssl_wrapper"] = _ssl_wrapper try: api = librouteros.connect( entry[CONF_HOST], entry[CONF_USERNAME], entry[CONF_PASSWORD], **kwargs, ) _LOGGER.debug("Connected to %s successfully", entry[CONF_HOST]) return api except ( librouteros.exceptions.LibRouterosError, OSError, socket.timeout, ) as api_error: _LOGGER.error("Mikrotik %s error: %s", entry[CONF_HOST], api_error) if "invalid user name or password" in str(api_error): raise LoginError from api_error raise CannotConnect from api_error