Migrate nmap_tracker to use different nmap lib
This commit is contained in:
parent
81190be7ba
commit
ffac067be8
1 changed files with 34 additions and 56 deletions
|
@ -19,6 +19,11 @@ hosts
|
||||||
*Required
|
*Required
|
||||||
The IP addresses to scan in the network-prefix notation (192.168.1.1/24) or
|
The IP addresses to scan in the network-prefix notation (192.168.1.1/24) or
|
||||||
the range notation (192.168.1.1-255).
|
the range notation (192.168.1.1-255).
|
||||||
|
|
||||||
|
home_interval
|
||||||
|
*Optional
|
||||||
|
Number of minutes it will not scan devices that it found in previous results.
|
||||||
|
This is to save battery.
|
||||||
"""
|
"""
|
||||||
import logging
|
import logging
|
||||||
from datetime import timedelta
|
from datetime import timedelta
|
||||||
|
@ -26,13 +31,6 @@ from collections import namedtuple
|
||||||
import subprocess
|
import subprocess
|
||||||
import re
|
import re
|
||||||
|
|
||||||
try:
|
|
||||||
from libnmap.process import NmapProcess
|
|
||||||
from libnmap.parser import NmapParser, NmapParserException
|
|
||||||
LIB_LOADED = True
|
|
||||||
except ImportError:
|
|
||||||
LIB_LOADED = False
|
|
||||||
|
|
||||||
import homeassistant.util.dt as dt_util
|
import homeassistant.util.dt as dt_util
|
||||||
from homeassistant.const import CONF_HOSTS
|
from homeassistant.const import CONF_HOSTS
|
||||||
from homeassistant.helpers import validate_config
|
from homeassistant.helpers import validate_config
|
||||||
|
@ -47,7 +45,7 @@ _LOGGER = logging.getLogger(__name__)
|
||||||
# interval in minutes to exclude devices from a scan while they are home
|
# interval in minutes to exclude devices from a scan while they are home
|
||||||
CONF_HOME_INTERVAL = "home_interval"
|
CONF_HOME_INTERVAL = "home_interval"
|
||||||
|
|
||||||
REQUIREMENTS = ['python-libnmap==0.6.1']
|
REQUIREMENTS = ['python-nmap==0.4.1']
|
||||||
|
|
||||||
|
|
||||||
def get_scanner(hass, config):
|
def get_scanner(hass, config):
|
||||||
|
@ -56,10 +54,6 @@ def get_scanner(hass, config):
|
||||||
_LOGGER):
|
_LOGGER):
|
||||||
return None
|
return None
|
||||||
|
|
||||||
if not LIB_LOADED:
|
|
||||||
_LOGGER.error("Error while importing dependency python-libnmap.")
|
|
||||||
return False
|
|
||||||
|
|
||||||
scanner = NmapDeviceScanner(config[DOMAIN])
|
scanner = NmapDeviceScanner(config[DOMAIN])
|
||||||
|
|
||||||
return scanner if scanner.success_init else None
|
return scanner if scanner.success_init else None
|
||||||
|
@ -76,7 +70,7 @@ def _arp(ip_address):
|
||||||
if match:
|
if match:
|
||||||
return match.group(0)
|
return match.group(0)
|
||||||
_LOGGER.info("No MAC address found for %s", ip_address)
|
_LOGGER.info("No MAC address found for %s", ip_address)
|
||||||
return ''
|
return None
|
||||||
|
|
||||||
|
|
||||||
class NmapDeviceScanner(object):
|
class NmapDeviceScanner(object):
|
||||||
|
@ -89,8 +83,7 @@ class NmapDeviceScanner(object):
|
||||||
minutes = convert(config.get(CONF_HOME_INTERVAL), int, 0)
|
minutes = convert(config.get(CONF_HOME_INTERVAL), int, 0)
|
||||||
self.home_interval = timedelta(minutes=minutes)
|
self.home_interval = timedelta(minutes=minutes)
|
||||||
|
|
||||||
self.success_init = True
|
self.success_init = self._update_info()
|
||||||
self._update_info()
|
|
||||||
_LOGGER.info("nmap scanner initialized")
|
_LOGGER.info("nmap scanner initialized")
|
||||||
|
|
||||||
def scan_devices(self):
|
def scan_devices(self):
|
||||||
|
@ -112,43 +105,16 @@ class NmapDeviceScanner(object):
|
||||||
else:
|
else:
|
||||||
return None
|
return None
|
||||||
|
|
||||||
def _parse_results(self, stdout):
|
|
||||||
""" Parses results from an nmap scan.
|
|
||||||
Returns True if successful, False otherwise. """
|
|
||||||
try:
|
|
||||||
results = NmapParser.parse(stdout)
|
|
||||||
now = dt_util.now()
|
|
||||||
self.last_results = []
|
|
||||||
for host in results.hosts:
|
|
||||||
if host.is_up():
|
|
||||||
if host.hostnames:
|
|
||||||
name = host.hostnames[0]
|
|
||||||
else:
|
|
||||||
name = host.ipv4
|
|
||||||
if host.mac:
|
|
||||||
mac = host.mac
|
|
||||||
else:
|
|
||||||
mac = _arp(host.ipv4)
|
|
||||||
if mac:
|
|
||||||
device = Device(mac.upper(), name, host.ipv4, now)
|
|
||||||
self.last_results.append(device)
|
|
||||||
_LOGGER.info("nmap scan successful")
|
|
||||||
return True
|
|
||||||
except NmapParserException as parse_exc:
|
|
||||||
_LOGGER.error("failed to parse nmap results: %s", parse_exc.msg)
|
|
||||||
self.last_results = []
|
|
||||||
return False
|
|
||||||
|
|
||||||
@Throttle(MIN_TIME_BETWEEN_SCANS)
|
@Throttle(MIN_TIME_BETWEEN_SCANS)
|
||||||
def _update_info(self):
|
def _update_info(self):
|
||||||
""" Scans the network for devices.
|
""" Scans the network for devices.
|
||||||
Returns boolean if scanning successful. """
|
Returns boolean if scanning successful. """
|
||||||
if not self.success_init:
|
|
||||||
return False
|
|
||||||
|
|
||||||
_LOGGER.info("Scanning")
|
_LOGGER.info("Scanning")
|
||||||
|
|
||||||
options = "-F --host-timeout 5"
|
from nmap import PortScanner, PortScannerError
|
||||||
|
scanner = PortScanner()
|
||||||
|
|
||||||
|
options = "-sP --host-timeout 5"
|
||||||
exclude_targets = set()
|
exclude_targets = set()
|
||||||
if self.home_interval:
|
if self.home_interval:
|
||||||
now = dt_util.now()
|
now = dt_util.now()
|
||||||
|
@ -159,14 +125,26 @@ class NmapDeviceScanner(object):
|
||||||
target_list = [t.ip for t in exclude_targets]
|
target_list = [t.ip for t in exclude_targets]
|
||||||
options += " --exclude {}".format(",".join(target_list))
|
options += " --exclude {}".format(",".join(target_list))
|
||||||
|
|
||||||
nmap = NmapProcess(targets=self.hosts, options=options)
|
try:
|
||||||
|
result = scanner.scan(hosts=self.hosts, arguments=options)
|
||||||
nmap.run()
|
except PortScannerError:
|
||||||
|
|
||||||
if nmap.rc == 0:
|
|
||||||
if self._parse_results(nmap.stdout):
|
|
||||||
self.last_results.extend(exclude_targets)
|
|
||||||
else:
|
|
||||||
self.last_results = []
|
|
||||||
_LOGGER.error(nmap.stderr)
|
|
||||||
return False
|
return False
|
||||||
|
|
||||||
|
now = dt_util.now()
|
||||||
|
self.last_results = []
|
||||||
|
for ip, info in result['scan'].items():
|
||||||
|
if info['status']['state'] != 'up':
|
||||||
|
continue
|
||||||
|
name = info['hostnames'][0] if info['hostnames'] else ip
|
||||||
|
# Mac address only returned if nmap ran as root
|
||||||
|
mac = info['addresses'].get('mac')
|
||||||
|
if mac is None:
|
||||||
|
mac = _arp(ip)
|
||||||
|
if mac is None:
|
||||||
|
continue
|
||||||
|
device = Device(mac.upper(), name, ip, now)
|
||||||
|
self.last_results.append(device)
|
||||||
|
self.last_results.extend(exclude_targets)
|
||||||
|
|
||||||
|
_LOGGER.info("nmap scan successful")
|
||||||
|
return True
|
||||||
|
|
Loading…
Add table
Reference in a new issue