Migrate nmap_tracker to use different nmap lib

This commit is contained in:
Paulus Schoutsen 2015-08-31 00:29:41 -07:00
parent 81190be7ba
commit ffac067be8

View file

@ -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