mirror of
				https://github.com/ansible-collections/community.general.git
				synced 2025-10-25 05:23:58 -07:00 
			
		
		
		
	* Replace ansible.module_utils._text by ansible.module_utils.common.text.converters.
* Also adjust tests.
(cherry picked from commit fafabed9e6)
		
	
			
		
			
				
	
	
		
			213 lines
		
	
	
	
		
			8.1 KiB
		
	
	
	
		
			Python
		
	
	
	
	
	
			
		
		
	
	
			213 lines
		
	
	
	
		
			8.1 KiB
		
	
	
	
		
			Python
		
	
	
	
	
	
| # Copyright (c) 2017 Ansible Project
 | |
| # GNU General Public License v3.0+ (see COPYING or https://www.gnu.org/licenses/gpl-3.0.txt)
 | |
| 
 | |
| from __future__ import (absolute_import, division, print_function)
 | |
| __metaclass__ = type
 | |
| 
 | |
| DOCUMENTATION = '''
 | |
|     author: Unknown (!UNKNOWN)
 | |
|     name: nmap
 | |
|     short_description: Uses nmap to find hosts to target
 | |
|     description:
 | |
|         - Uses a YAML configuration file with a valid YAML extension.
 | |
|     extends_documentation_fragment:
 | |
|       - constructed
 | |
|       - inventory_cache
 | |
|     requirements:
 | |
|       - nmap CLI installed
 | |
|     options:
 | |
|         plugin:
 | |
|             description: token that ensures this is a source file for the 'nmap' plugin.
 | |
|             required: True
 | |
|             choices: ['nmap', 'community.general.nmap']
 | |
|         address:
 | |
|             description: Network IP or range of IPs to scan, you can use a simple range (10.2.2.15-25) or CIDR notation.
 | |
|             required: True
 | |
|         exclude:
 | |
|             description: list of addresses to exclude
 | |
|             type: list
 | |
|         ports:
 | |
|             description: Enable/disable scanning for open ports
 | |
|             type: boolean
 | |
|             default: True
 | |
|         ipv4:
 | |
|             description: use IPv4 type addresses
 | |
|             type: boolean
 | |
|             default: True
 | |
|         ipv6:
 | |
|             description: use IPv6 type addresses
 | |
|             type: boolean
 | |
|             default: True
 | |
|     notes:
 | |
|         - At least one of ipv4 or ipv6 is required to be True, both can be True, but they cannot both be False.
 | |
|         - 'TODO: add OS fingerprinting'
 | |
| '''
 | |
| EXAMPLES = '''
 | |
| # inventory.config file in YAML format
 | |
| plugin: community.general.nmap
 | |
| strict: False
 | |
| address: 192.168.0.0/24
 | |
| '''
 | |
| 
 | |
| import os
 | |
| import re
 | |
| 
 | |
| from subprocess import Popen, PIPE
 | |
| 
 | |
| from ansible import constants as C
 | |
| from ansible.errors import AnsibleParserError
 | |
| from ansible.module_utils.common.text.converters import to_native, to_text
 | |
| from ansible.plugins.inventory import BaseInventoryPlugin, Constructable, Cacheable
 | |
| from ansible.module_utils.common.process import get_bin_path
 | |
| 
 | |
| 
 | |
| class InventoryModule(BaseInventoryPlugin, Constructable, Cacheable):
 | |
| 
 | |
|     NAME = 'community.general.nmap'
 | |
|     find_host = re.compile(r'^Nmap scan report for ([\w,.,-]+)(?: \(([\w,.,:,\[,\]]+)\))?')
 | |
|     find_port = re.compile(r'^(\d+)/(\w+)\s+(\w+)\s+(\w+)')
 | |
| 
 | |
|     def __init__(self):
 | |
|         self._nmap = None
 | |
|         super(InventoryModule, self).__init__()
 | |
| 
 | |
|     def _populate(self, hosts):
 | |
|         # Use constructed if applicable
 | |
|         strict = self.get_option('strict')
 | |
| 
 | |
|         for host in hosts:
 | |
|             hostname = host['name']
 | |
|             self.inventory.add_host(hostname)
 | |
|             for var, value in host.items():
 | |
|                 self.inventory.set_variable(hostname, var, value)
 | |
| 
 | |
|             # Composed variables
 | |
|             self._set_composite_vars(self.get_option('compose'), host, hostname, strict=strict)
 | |
| 
 | |
|             # Complex groups based on jinja2 conditionals, hosts that meet the conditional are added to group
 | |
|             self._add_host_to_composed_groups(self.get_option('groups'), host, hostname, strict=strict)
 | |
| 
 | |
|             # Create groups based on variable values and add the corresponding hosts to it
 | |
|             self._add_host_to_keyed_groups(self.get_option('keyed_groups'), host, hostname, strict=strict)
 | |
| 
 | |
|     def verify_file(self, path):
 | |
| 
 | |
|         valid = False
 | |
|         if super(InventoryModule, self).verify_file(path):
 | |
|             file_name, ext = os.path.splitext(path)
 | |
| 
 | |
|             if not ext or ext in C.YAML_FILENAME_EXTENSIONS:
 | |
|                 valid = True
 | |
| 
 | |
|         return valid
 | |
| 
 | |
|     def parse(self, inventory, loader, path, cache=True):
 | |
| 
 | |
|         try:
 | |
|             self._nmap = get_bin_path('nmap')
 | |
|         except ValueError as e:
 | |
|             raise AnsibleParserError('nmap inventory plugin requires the nmap cli tool to work: {0}'.format(to_native(e)))
 | |
| 
 | |
|         super(InventoryModule, self).parse(inventory, loader, path, cache=cache)
 | |
| 
 | |
|         self._read_config_data(path)
 | |
| 
 | |
|         cache_key = self.get_cache_key(path)
 | |
| 
 | |
|         # cache may be True or False at this point to indicate if the inventory is being refreshed
 | |
|         # get the user's cache option too to see if we should save the cache if it is changing
 | |
|         user_cache_setting = self.get_option('cache')
 | |
| 
 | |
|         # read if the user has caching enabled and the cache isn't being refreshed
 | |
|         attempt_to_read_cache = user_cache_setting and cache
 | |
|         # update if the user has caching enabled and the cache is being refreshed; update this value to True if the cache has expired below
 | |
|         cache_needs_update = user_cache_setting and not cache
 | |
| 
 | |
|         if attempt_to_read_cache:
 | |
|             try:
 | |
|                 results = self._cache[cache_key]
 | |
|             except KeyError:
 | |
|                 # This occurs if the cache_key is not in the cache or if the cache_key expired, so the cache needs to be updated
 | |
|                 cache_needs_update = True
 | |
| 
 | |
|         if not user_cache_setting or cache_needs_update:
 | |
|             # setup command
 | |
|             cmd = [self._nmap]
 | |
|             if not self._options['ports']:
 | |
|                 cmd.append('-sP')
 | |
| 
 | |
|             if self._options['ipv4'] and not self._options['ipv6']:
 | |
|                 cmd.append('-4')
 | |
|             elif self._options['ipv6'] and not self._options['ipv4']:
 | |
|                 cmd.append('-6')
 | |
|             elif not self._options['ipv6'] and not self._options['ipv4']:
 | |
|                 raise AnsibleParserError('One of ipv4 or ipv6 must be enabled for this plugin')
 | |
| 
 | |
|             if self._options['exclude']:
 | |
|                 cmd.append('--exclude')
 | |
|                 cmd.append(','.join(self._options['exclude']))
 | |
| 
 | |
|             cmd.append(self._options['address'])
 | |
|             try:
 | |
|                 # execute
 | |
|                 p = Popen(cmd, stdout=PIPE, stderr=PIPE)
 | |
|                 stdout, stderr = p.communicate()
 | |
|                 if p.returncode != 0:
 | |
|                     raise AnsibleParserError('Failed to run nmap, rc=%s: %s' % (p.returncode, to_native(stderr)))
 | |
| 
 | |
|                 # parse results
 | |
|                 host = None
 | |
|                 ip = None
 | |
|                 ports = []
 | |
|                 results = []
 | |
| 
 | |
|                 try:
 | |
|                     t_stdout = to_text(stdout, errors='surrogate_or_strict')
 | |
|                 except UnicodeError as e:
 | |
|                     raise AnsibleParserError('Invalid (non unicode) input returned: %s' % to_native(e))
 | |
| 
 | |
|                 for line in t_stdout.splitlines():
 | |
|                     hits = self.find_host.match(line)
 | |
|                     if hits:
 | |
|                         if host is not None and ports:
 | |
|                             results[-1]['ports'] = ports
 | |
| 
 | |
|                         # if dns only shows arpa, just use ip instead as hostname
 | |
|                         if hits.group(1).endswith('.in-addr.arpa'):
 | |
|                             host = hits.group(2)
 | |
|                         else:
 | |
|                             host = hits.group(1)
 | |
| 
 | |
|                         # if no reverse dns exists, just use ip instead as hostname
 | |
|                         if hits.group(2) is not None:
 | |
|                             ip = hits.group(2)
 | |
|                         else:
 | |
|                             ip = hits.group(1)
 | |
| 
 | |
|                         if host is not None:
 | |
|                             # update inventory
 | |
|                             results.append(dict())
 | |
|                             results[-1]['name'] = host
 | |
|                             results[-1]['ip'] = ip
 | |
|                             ports = []
 | |
|                         continue
 | |
| 
 | |
|                     host_ports = self.find_port.match(line)
 | |
|                     if host is not None and host_ports:
 | |
|                         ports.append({'port': host_ports.group(1),
 | |
|                                       'protocol': host_ports.group(2),
 | |
|                                       'state': host_ports.group(3),
 | |
|                                       'service': host_ports.group(4)})
 | |
|                         continue
 | |
| 
 | |
|                 # if any leftovers
 | |
|                 if host and ports:
 | |
|                     results[-1]['ports'] = ports
 | |
| 
 | |
|             except Exception as e:
 | |
|                 raise AnsibleParserError("failed to parse %s: %s " % (to_native(path), to_native(e)))
 | |
| 
 | |
|         if cache_needs_update:
 | |
|             self._cache[cache_key] = results
 | |
| 
 | |
|         self._populate(results)
 |