"""Support for Mikrotik routers as device tracker.""" import logging from homeassistant.components.device_tracker import ( DOMAIN as DEVICE_TRACKER, DeviceScanner, ) from homeassistant.util import slugify from homeassistant.const import CONF_METHOD from .const import ( HOSTS, MIKROTIK, CONF_ARP_PING, MIKROTIK_SERVICES, CAPSMAN, WIRELESS, DHCP, ARP, ATTR_DEVICE_TRACKER, ) _LOGGER = logging.getLogger(__name__) def get_scanner(hass, config): """Validate the configuration and return MikrotikScanner.""" for host in hass.data[MIKROTIK][HOSTS]: if DEVICE_TRACKER not in hass.data[MIKROTIK][HOSTS][host]: continue hass.data[MIKROTIK][HOSTS][host].pop(DEVICE_TRACKER, None) api = hass.data[MIKROTIK][HOSTS][host]["api"] config = hass.data[MIKROTIK][HOSTS][host]["config"] hostname = api.get_hostname() scanner = MikrotikScanner(api, host, hostname, config) return scanner if scanner.success_init else None class MikrotikScanner(DeviceScanner): """This class queries a Mikrotik device.""" def __init__(self, api, host, hostname, config): """Initialize the scanner.""" self.api = api self.config = config self.host = host self.hostname = hostname self.method = config.get(CONF_METHOD) self.arp_ping = config.get(CONF_ARP_PING) self.dhcp = None self.devices_arp = {} self.devices_dhcp = {} self.device_tracker = None self.success_init = self.api.connected() def get_extra_attributes(self, device): """ Get extra attributes of a device. Some known extra attributes that may be returned in the device tuple include MAC address (mac), network device (dev), IP address (ip), reachable status (reachable), associated router (host), hostname if known (hostname) among others. """ return self.device_tracker.get(device) or {} def get_device_name(self, device): """Get name for a device.""" host = self.device_tracker.get(device, {}) return host.get("host_name") def scan_devices(self): """Scan for new devices and return a list with found device MACs.""" self.update_device_tracker() return list(self.device_tracker) def get_method(self): """Determine the device tracker polling method.""" if self.method: _LOGGER.debug( "Mikrotik %s: Manually selected polling method %s", self.host, self.method, ) return self.method capsman = self.api.command(MIKROTIK_SERVICES[CAPSMAN]) if not capsman: _LOGGER.debug( "Mikrotik %s: Not a CAPsMAN controller. " "Trying local wireless interfaces", (self.host), ) else: return CAPSMAN wireless = self.api.command(MIKROTIK_SERVICES[WIRELESS]) if not wireless: _LOGGER.info( "Mikrotik %s: Wireless adapters not found. Try to " "use DHCP lease table as presence tracker source. " "Please decrease lease time as much as possible", self.host, ) return DHCP return WIRELESS def update_device_tracker(self): """Update device_tracker from Mikrotik API.""" self.device_tracker = {} if not self.method: self.method = self.get_method() data = self.api.command(MIKROTIK_SERVICES[self.method]) if data is None: return if self.method != DHCP: dhcp = self.api.command(MIKROTIK_SERVICES[DHCP]) if dhcp is not None: self.devices_dhcp = load_mac(dhcp) arp = self.api.command(MIKROTIK_SERVICES[ARP]) self.devices_arp = load_mac(arp) for device in data: mac = device.get("mac-address") if self.method == DHCP: if "active-address" not in device: continue if self.arp_ping and self.devices_arp: if mac not in self.devices_arp: continue ip_address = self.devices_arp[mac]["address"] interface = self.devices_arp[mac]["interface"] if not self.do_arp_ping(ip_address, interface): continue attrs = {} if mac in self.devices_dhcp and "host-name" in self.devices_dhcp[mac]: hostname = self.devices_dhcp[mac].get("host-name") if hostname: attrs["host_name"] = hostname if self.devices_arp and mac in self.devices_arp: attrs["ip_address"] = self.devices_arp[mac].get("address") for attr in ATTR_DEVICE_TRACKER: if attr in device and device[attr] is not None: attrs[slugify(attr)] = device[attr] attrs["scanner_type"] = self.method attrs["scanner_host"] = self.host attrs["scanner_hostname"] = self.hostname self.device_tracker[mac] = attrs def do_arp_ping(self, ip_address, interface): """Attempt to arp ping MAC address via interface.""" params = { "arp-ping": "yes", "interval": "100ms", "count": 3, "interface": interface, "address": ip_address, } cmd = "/ping" data = self.api.command(cmd, params) if data is not None: status = 0 for result in data: if "status" in result: _LOGGER.debug( "Mikrotik %s arp_ping error: %s", self.host, result["status"] ) status += 1 if status == len(data): return None return data 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.pop("mac-address") mac_devices[mac] = device return mac_devices