mirror of
				https://github.com/ansible-collections/community.general.git
				synced 2025-10-24 21:14:00 -07:00 
			
		
		
		
	
		
			
				
	
	
		
			621 lines
		
	
	
	
		
			21 KiB
		
	
	
	
		
			Python
		
	
	
	
	
	
			
		
		
	
	
			621 lines
		
	
	
	
		
			21 KiB
		
	
	
	
		
			Python
		
	
	
	
	
	
| """Classify changes in Ansible code."""
 | |
| 
 | |
| from __future__ import absolute_import, print_function
 | |
| 
 | |
| import os
 | |
| import time
 | |
| 
 | |
| from lib.target import (
 | |
|     walk_module_targets,
 | |
|     walk_integration_targets,
 | |
|     walk_units_targets,
 | |
|     walk_compile_targets,
 | |
|     walk_sanity_targets,
 | |
|     load_integration_prefixes,
 | |
|     analyze_integration_target_dependencies,
 | |
| )
 | |
| 
 | |
| from lib.util import (
 | |
|     display,
 | |
| )
 | |
| 
 | |
| from lib.import_analysis import (
 | |
|     get_python_module_utils_imports,
 | |
| )
 | |
| 
 | |
| from lib.powershell_import_analysis import (
 | |
|     get_powershell_module_utils_imports,
 | |
| )
 | |
| 
 | |
| from lib.config import (
 | |
|     TestConfig,
 | |
|     IntegrationConfig,
 | |
| )
 | |
| 
 | |
| 
 | |
| def categorize_changes(args, paths, verbose_command=None):
 | |
|     """
 | |
|     :type args: TestConfig
 | |
|     :type paths: list[str]
 | |
|     :type verbose_command: str
 | |
|     :rtype paths: dict[str, list[str]]
 | |
|     """
 | |
|     mapper = PathMapper(args)
 | |
| 
 | |
|     commands = {
 | |
|         'sanity': set(),
 | |
|         'compile': set(),
 | |
|         'units': set(),
 | |
|         'integration': set(),
 | |
|         'windows-integration': set(),
 | |
|         'network-integration': set(),
 | |
|     }
 | |
| 
 | |
|     additional_paths = set()
 | |
| 
 | |
|     for path in paths:
 | |
|         if not os.path.exists(path):
 | |
|             continue
 | |
| 
 | |
|         dependent_paths = mapper.get_dependent_paths(path)
 | |
| 
 | |
|         if not dependent_paths:
 | |
|             continue
 | |
| 
 | |
|         display.info('Expanded "%s" to %d dependent file(s):' % (path, len(dependent_paths)), verbosity=1)
 | |
| 
 | |
|         for dependent_path in dependent_paths:
 | |
|             display.info(dependent_path, verbosity=1)
 | |
|             additional_paths.add(dependent_path)
 | |
| 
 | |
|     additional_paths -= set(paths)  # don't count changed paths as additional paths
 | |
| 
 | |
|     if additional_paths:
 | |
|         display.info('Expanded %d changed file(s) into %d additional dependent file(s).' % (len(paths), len(additional_paths)))
 | |
|         paths = sorted(set(paths) | additional_paths)
 | |
| 
 | |
|     display.info('Mapping %d changed file(s) to tests.' % len(paths))
 | |
| 
 | |
|     for path in paths:
 | |
|         tests = mapper.classify(path)
 | |
| 
 | |
|         if tests is None:
 | |
|             display.info('%s -> all' % path, verbosity=1)
 | |
|             tests = all_tests(args)  # not categorized, run all tests
 | |
|             display.warning('Path not categorized: %s' % path)
 | |
|         else:
 | |
|             tests = dict((key, value) for key, value in tests.items() if value)
 | |
| 
 | |
|             if verbose_command:
 | |
|                 result = '%s: %s' % (verbose_command, tests.get(verbose_command) or 'none')
 | |
| 
 | |
|                 # identify targeted integration tests (those which only target a single integration command)
 | |
|                 if 'integration' in verbose_command and tests.get(verbose_command):
 | |
|                     if not any('integration' in command for command in tests if command != verbose_command):
 | |
|                         result += ' (targeted)'
 | |
|             else:
 | |
|                 result = '%s' % tests
 | |
| 
 | |
|             display.info('%s -> %s' % (path, result), verbosity=1)
 | |
| 
 | |
|         for command, target in tests.items():
 | |
|             commands[command].add(target)
 | |
| 
 | |
|     for command in commands:
 | |
|         commands[command].discard('none')
 | |
| 
 | |
|         if any(t == 'all' for t in commands[command]):
 | |
|             commands[command] = set(['all'])
 | |
| 
 | |
|     commands = dict((c, sorted(commands[c])) for c in commands if commands[c])
 | |
| 
 | |
|     return commands
 | |
| 
 | |
| 
 | |
| class PathMapper(object):
 | |
|     """Map file paths to test commands and targets."""
 | |
|     def __init__(self, args):
 | |
|         """
 | |
|         :type args: TestConfig
 | |
|         """
 | |
|         self.args = args
 | |
|         self.integration_all_target = get_integration_all_target(self.args)
 | |
| 
 | |
|         self.integration_targets = list(walk_integration_targets())
 | |
|         self.module_targets = list(walk_module_targets())
 | |
|         self.compile_targets = list(walk_compile_targets())
 | |
|         self.units_targets = list(walk_units_targets())
 | |
|         self.sanity_targets = list(walk_sanity_targets())
 | |
|         self.powershell_targets = [t for t in self.sanity_targets if os.path.splitext(t.path)[1] == '.ps1']
 | |
| 
 | |
|         self.compile_paths = set(t.path for t in self.compile_targets)
 | |
|         self.units_modules = set(t.module for t in self.units_targets if t.module)
 | |
|         self.units_paths = set(a for t in self.units_targets for a in t.aliases)
 | |
|         self.sanity_paths = set(t.path for t in self.sanity_targets)
 | |
| 
 | |
|         self.module_names_by_path = dict((t.path, t.module) for t in self.module_targets)
 | |
|         self.integration_targets_by_name = dict((t.name, t) for t in self.integration_targets)
 | |
|         self.integration_targets_by_alias = dict((a, t) for t in self.integration_targets for a in t.aliases)
 | |
| 
 | |
|         self.posix_integration_by_module = dict((m, t.name) for t in self.integration_targets
 | |
|                                                 if 'posix/' in t.aliases for m in t.modules)
 | |
|         self.windows_integration_by_module = dict((m, t.name) for t in self.integration_targets
 | |
|                                                   if 'windows/' in t.aliases for m in t.modules)
 | |
|         self.network_integration_by_module = dict((m, t.name) for t in self.integration_targets
 | |
|                                                   if 'network/' in t.aliases for m in t.modules)
 | |
| 
 | |
|         self.prefixes = load_integration_prefixes()
 | |
|         self.integration_dependencies = analyze_integration_target_dependencies(self.integration_targets)
 | |
| 
 | |
|         self.python_module_utils_imports = {}  # populated on first use to reduce overhead when not needed
 | |
|         self.powershell_module_utils_imports = {}  # populated on first use to reduce overhead when not needed
 | |
| 
 | |
|     def get_dependent_paths(self, path):
 | |
|         """
 | |
|         :type path: str
 | |
|         :rtype: list[str]
 | |
|         """
 | |
|         ext = os.path.splitext(os.path.split(path)[1])[1]
 | |
| 
 | |
|         if path.startswith('lib/ansible/module_utils/'):
 | |
|             if ext == '.py':
 | |
|                 return self.get_python_module_utils_usage(path)
 | |
| 
 | |
|             if ext == '.psm1':
 | |
|                 return self.get_powershell_module_utils_usage(path)
 | |
| 
 | |
|         if path.startswith('test/integration/targets/'):
 | |
|             return self.get_integration_target_usage(path)
 | |
| 
 | |
|         return []
 | |
| 
 | |
|     def get_python_module_utils_usage(self, path):
 | |
|         """
 | |
|         :type path: str
 | |
|         :rtype: list[str]
 | |
|         """
 | |
|         if path == 'lib/ansible/module_utils/__init__.py':
 | |
|             return []
 | |
| 
 | |
|         if not self.python_module_utils_imports:
 | |
|             display.info('Analyzing python module_utils imports...')
 | |
|             before = time.time()
 | |
|             self.python_module_utils_imports = get_python_module_utils_imports(self.compile_targets)
 | |
|             after = time.time()
 | |
|             display.info('Processed %d python module_utils in %d second(s).' % (len(self.python_module_utils_imports), after - before))
 | |
| 
 | |
|         name = os.path.splitext(path)[0].replace('/', '.')[4:]
 | |
| 
 | |
|         if name.endswith('.__init__'):
 | |
|             name = name[:-9]
 | |
| 
 | |
|         return sorted(self.python_module_utils_imports[name])
 | |
| 
 | |
|     def get_powershell_module_utils_usage(self, path):
 | |
|         """
 | |
|         :type path: str
 | |
|         :rtype: list[str]
 | |
|         """
 | |
|         if not self.powershell_module_utils_imports:
 | |
|             display.info('Analyzing powershell module_utils imports...')
 | |
|             before = time.time()
 | |
|             self.powershell_module_utils_imports = get_powershell_module_utils_imports(self.powershell_targets)
 | |
|             after = time.time()
 | |
|             display.info('Processed %d powershell module_utils in %d second(s).' % (len(self.powershell_module_utils_imports), after - before))
 | |
| 
 | |
|         name = os.path.splitext(os.path.basename(path))[0]
 | |
| 
 | |
|         return sorted(self.powershell_module_utils_imports[name])
 | |
| 
 | |
|     def get_integration_target_usage(self, path):
 | |
|         """
 | |
|         :type path: str
 | |
|         :rtype: list[str]
 | |
|         """
 | |
|         target_name = path.split('/')[3]
 | |
|         dependents = [os.path.join('test/integration/targets/%s/' % target) for target in sorted(self.integration_dependencies.get(target_name, set()))]
 | |
| 
 | |
|         return dependents
 | |
| 
 | |
|     def classify(self, path):
 | |
|         """
 | |
|         :type path: str
 | |
|         :rtype: dict[str, str] | None
 | |
|         """
 | |
|         result = self._classify(path)
 | |
| 
 | |
|         # run all tests when no result given
 | |
|         if result is None:
 | |
|             return None
 | |
| 
 | |
|         # compile path if eligible
 | |
|         if path in self.compile_paths:
 | |
|             result['compile'] = path
 | |
| 
 | |
|         # run sanity on path unless result specified otherwise
 | |
|         if path in self.sanity_paths and 'sanity' not in result:
 | |
|             result['sanity'] = path
 | |
| 
 | |
|         return result
 | |
| 
 | |
|     def _classify(self, path):
 | |
|         """
 | |
|         :type path: str
 | |
|         :rtype: dict[str, str] | None
 | |
|         """
 | |
|         dirname = os.path.dirname(path)
 | |
|         filename = os.path.basename(path)
 | |
|         name, ext = os.path.splitext(filename)
 | |
| 
 | |
|         minimal = {}
 | |
| 
 | |
|         if path.startswith('.github/'):
 | |
|             return minimal
 | |
| 
 | |
|         if path.startswith('bin/'):
 | |
|             return minimal
 | |
| 
 | |
|         if path.startswith('contrib/'):
 | |
|             return {
 | |
|                 'units': 'test/units/contrib/'
 | |
|             }
 | |
| 
 | |
|         if path.startswith('docs/'):
 | |
|             return minimal
 | |
| 
 | |
|         if path.startswith('examples/'):
 | |
|             if path == 'examples/scripts/ConfigureRemotingForAnsible.ps1':
 | |
|                 return {
 | |
|                     'windows-integration': 'connection_winrm',
 | |
|                 }
 | |
| 
 | |
|             return minimal
 | |
| 
 | |
|         if path.startswith('hacking/'):
 | |
|             return minimal
 | |
| 
 | |
|         if path.startswith('lib/ansible/modules/'):
 | |
|             module_name = self.module_names_by_path.get(path)
 | |
| 
 | |
|             if module_name:
 | |
|                 return {
 | |
|                     'units': module_name if module_name in self.units_modules else None,
 | |
|                     'integration': self.posix_integration_by_module.get(module_name) if ext == '.py' else None,
 | |
|                     'windows-integration': self.windows_integration_by_module.get(module_name) if ext == '.ps1' else None,
 | |
|                     'network-integration': self.network_integration_by_module.get(module_name),
 | |
|                 }
 | |
| 
 | |
|             return minimal
 | |
| 
 | |
|         if path.startswith('lib/ansible/module_utils/'):
 | |
|             if ext == '.psm1':
 | |
|                 return minimal  # already expanded using get_dependent_paths
 | |
| 
 | |
|             if ext == '.py':
 | |
|                 return minimal  # already expanded using get_dependent_paths
 | |
| 
 | |
|         if path.startswith('lib/ansible/plugins/connection/'):
 | |
|             if name == '__init__':
 | |
|                 return {
 | |
|                     'integration': self.integration_all_target,
 | |
|                     'windows-integration': self.integration_all_target,
 | |
|                     'network-integration': self.integration_all_target,
 | |
|                     'units': 'test/units/plugins/connection/',
 | |
|                 }
 | |
| 
 | |
|             units_path = 'test/units/plugins/connection/test_%s.py' % name
 | |
| 
 | |
|             if units_path not in self.units_paths:
 | |
|                 units_path = None
 | |
| 
 | |
|             integration_name = 'connection_%s' % name
 | |
| 
 | |
|             if integration_name not in self.integration_targets_by_name:
 | |
|                 integration_name = None
 | |
| 
 | |
|             # entire integration test commands depend on these connection plugins
 | |
| 
 | |
|             if name == 'winrm':
 | |
|                 return {
 | |
|                     'windows-integration': self.integration_all_target,
 | |
|                     'units': units_path,
 | |
|                 }
 | |
| 
 | |
|             if name == 'local':
 | |
|                 return {
 | |
|                     'integration': self.integration_all_target,
 | |
|                     'network-integration': self.integration_all_target,
 | |
|                     'units': units_path,
 | |
|                 }
 | |
| 
 | |
|             if name == 'network_cli':
 | |
|                 return {
 | |
|                     'network-integration': self.integration_all_target,
 | |
|                     'units': units_path,
 | |
|                 }
 | |
| 
 | |
|             # other connection plugins have isolated integration and unit tests
 | |
| 
 | |
|             return {
 | |
|                 'integration': integration_name,
 | |
|                 'units': units_path,
 | |
|             }
 | |
| 
 | |
|         if path.startswith('lib/ansible/plugins/terminal/'):
 | |
|             if ext == '.py':
 | |
|                 if name in self.prefixes and self.prefixes[name] == 'network':
 | |
|                     network_target = 'network/%s/' % name
 | |
| 
 | |
|                     if network_target in self.integration_targets_by_alias:
 | |
|                         return {
 | |
|                             'network-integration': network_target,
 | |
|                             'units': 'all',
 | |
|                         }
 | |
| 
 | |
|                     display.warning('Integration tests for "%s" not found.' % network_target)
 | |
| 
 | |
|                     return {
 | |
|                         'units': 'all',
 | |
|                     }
 | |
| 
 | |
|                 return {
 | |
|                     'network-integration': self.integration_all_target,
 | |
|                     'units': 'all',
 | |
|                 }
 | |
| 
 | |
|         if path.startswith('lib/ansible/utils/module_docs_fragments/'):
 | |
|             return {
 | |
|                 'sanity': 'all',
 | |
|             }
 | |
| 
 | |
|         if path.startswith('lib/ansible/'):
 | |
|             return all_tests(self.args)  # broad impact, run all tests
 | |
| 
 | |
|         if path.startswith('packaging/'):
 | |
|             if path.startswith('packaging/requirements/'):
 | |
|                 if name.startswith('requirements-') and ext == '.txt':
 | |
|                     component = name.split('-', 1)[1]
 | |
| 
 | |
|                     candidates = (
 | |
|                         'cloud/%s/' % component,
 | |
|                     )
 | |
| 
 | |
|                     for candidate in candidates:
 | |
|                         if candidate in self.integration_targets_by_alias:
 | |
|                             return {
 | |
|                                 'integration': candidate,
 | |
|                             }
 | |
| 
 | |
|                 return all_tests(self.args)  # broad impact, run all tests
 | |
| 
 | |
|             return minimal
 | |
| 
 | |
|         if path.startswith('test/cache/'):
 | |
|             return minimal
 | |
| 
 | |
|         if path.startswith('test/compile/'):
 | |
|             return {
 | |
|                 'compile': 'all',
 | |
|             }
 | |
| 
 | |
|         if path.startswith('test/results/'):
 | |
|             return minimal
 | |
| 
 | |
|         if path.startswith('test/legacy/'):
 | |
|             return minimal
 | |
| 
 | |
|         if path.startswith('test/integration/roles/'):
 | |
|             return minimal
 | |
| 
 | |
|         if path.startswith('test/integration/targets/'):
 | |
|             if not os.path.exists(path):
 | |
|                 return minimal
 | |
| 
 | |
|             target = self.integration_targets_by_name[path.split('/')[3]]
 | |
| 
 | |
|             if 'hidden/' in target.aliases:
 | |
|                 if target.type == 'role':
 | |
|                     return minimal  # already expanded using get_dependent_paths
 | |
| 
 | |
|                 return {
 | |
|                     'integration': self.integration_all_target,
 | |
|                     'windows-integration': self.integration_all_target,
 | |
|                     'network-integration': self.integration_all_target,
 | |
|                 }
 | |
| 
 | |
|             return {
 | |
|                 'integration': target.name if 'posix/' in target.aliases else None,
 | |
|                 'windows-integration': target.name if 'windows/' in target.aliases else None,
 | |
|                 'network-integration': target.name if 'network/' in target.aliases else None,
 | |
|             }
 | |
| 
 | |
|         if path.startswith('test/integration/'):
 | |
|             if dirname == 'test/integration':
 | |
|                 if self.prefixes.get(name) == 'network' and ext == '.yaml':
 | |
|                     return minimal  # network integration test playbooks are not used by ansible-test
 | |
| 
 | |
|                 if filename == 'network-all.yaml':
 | |
|                     return minimal  # network integration test playbook not used by ansible-test
 | |
| 
 | |
|                 if filename == 'platform_agnostic.yaml':
 | |
|                     return minimal  # network integration test playbook not used by ansible-test
 | |
| 
 | |
|                 for command in (
 | |
|                         'integration',
 | |
|                         'windows-integration',
 | |
|                         'network-integration',
 | |
|                 ):
 | |
|                     if name == command and ext == '.cfg':
 | |
|                         return {
 | |
|                             command: self.integration_all_target,
 | |
|                         }
 | |
| 
 | |
|                 if name.startswith('cloud-config-'):
 | |
|                     cloud_target = 'cloud/%s/' % name.split('-')[2].split('.')[0]
 | |
| 
 | |
|                     if cloud_target in self.integration_targets_by_alias:
 | |
|                         return {
 | |
|                             'integration': cloud_target,
 | |
|                         }
 | |
| 
 | |
|             return {
 | |
|                 'integration': self.integration_all_target,
 | |
|                 'windows-integration': self.integration_all_target,
 | |
|                 'network-integration': self.integration_all_target,
 | |
|             }
 | |
| 
 | |
|         if path.startswith('test/sanity/'):
 | |
|             return {
 | |
|                 'sanity': 'all',  # test infrastructure, run all sanity checks
 | |
|             }
 | |
| 
 | |
|         if path.startswith('test/units/'):
 | |
|             if path in self.units_paths:
 | |
|                 return {
 | |
|                     'units': path,
 | |
|                 }
 | |
| 
 | |
|             # changes to files which are not unit tests should trigger tests from the nearest parent directory
 | |
| 
 | |
|             test_path = os.path.dirname(path)
 | |
| 
 | |
|             while test_path:
 | |
|                 if test_path + '/' in self.units_paths:
 | |
|                     return {
 | |
|                         'units': test_path + '/',
 | |
|                     }
 | |
| 
 | |
|                 test_path = os.path.dirname(test_path)
 | |
| 
 | |
|         if path.startswith('test/runner/completion/'):
 | |
|             if path == 'test/runner/completion/docker.txt':
 | |
|                 return all_tests(self.args, force=True)  # force all tests due to risk of breaking changes in new test environment
 | |
| 
 | |
|         if path.startswith('test/runner/docker/'):
 | |
|             return minimal  # not used by tests, only used to build the default container
 | |
| 
 | |
|         if path.startswith('test/runner/lib/cloud/'):
 | |
|             cloud_target = 'cloud/%s/' % name
 | |
| 
 | |
|             if cloud_target in self.integration_targets_by_alias:
 | |
|                 return {
 | |
|                     'integration': cloud_target,
 | |
|                 }
 | |
| 
 | |
|             return all_tests(self.args)  # test infrastructure, run all tests
 | |
| 
 | |
|         if path.startswith('test/runner/lib/sanity/'):
 | |
|             return {
 | |
|                 'sanity': 'all',  # test infrastructure, run all sanity checks
 | |
|             }
 | |
| 
 | |
|         if path.startswith('test/runner/requirements/'):
 | |
|             if name in (
 | |
|                     'integration',
 | |
|                     'network-integration',
 | |
|                     'windows-integration',
 | |
|             ):
 | |
|                 return {
 | |
|                     name: self.integration_all_target,
 | |
|                 }
 | |
| 
 | |
|             if name in (
 | |
|                     'sanity',
 | |
|                     'units',
 | |
|             ):
 | |
|                 return {
 | |
|                     name: 'all',
 | |
|                 }
 | |
| 
 | |
|             if name.startswith('integration.cloud.'):
 | |
|                 cloud_target = 'cloud/%s/' % name.split('.')[2]
 | |
| 
 | |
|                 if cloud_target in self.integration_targets_by_alias:
 | |
|                     return {
 | |
|                         'integration': cloud_target,
 | |
|                     }
 | |
| 
 | |
|         if path.startswith('test/runner/'):
 | |
|             if dirname == 'test/runner' and name in (
 | |
|                     'Dockerfile',
 | |
|                     '.dockerignore',
 | |
|             ):
 | |
|                 return minimal  # not used by tests, only used to build the default container
 | |
| 
 | |
|             return all_tests(self.args)  # test infrastructure, run all tests
 | |
| 
 | |
|         if path.startswith('test/utils/shippable/'):
 | |
|             return all_tests(self.args)  # test infrastructure, run all tests
 | |
| 
 | |
|         if path.startswith('test/utils/'):
 | |
|             return minimal
 | |
| 
 | |
|         if path == 'test/README.md':
 | |
|             return minimal
 | |
| 
 | |
|         if path.startswith('ticket_stubs/'):
 | |
|             return minimal
 | |
| 
 | |
|         if '/' not in path:
 | |
|             if path in (
 | |
|                     '.gitattributes',
 | |
|                     '.gitignore',
 | |
|                     '.gitmodules',
 | |
|                     '.mailmap',
 | |
|                     'tox.ini',  # obsolete
 | |
|                     'COPYING',
 | |
|                     'VERSION',
 | |
|                     'Makefile',
 | |
|             ):
 | |
|                 return minimal
 | |
| 
 | |
|             if path in (
 | |
|                     'shippable.yml',
 | |
|                     '.coveragerc',
 | |
|             ):
 | |
|                 return all_tests(self.args)  # test infrastructure, run all tests
 | |
| 
 | |
|             if path == 'setup.py':
 | |
|                 return all_tests(self.args)  # broad impact, run all tests
 | |
| 
 | |
|             if path == '.yamllint':
 | |
|                 return {
 | |
|                     'sanity': 'all',
 | |
|                 }
 | |
| 
 | |
|             if ext in ('.md', '.rst', '.txt', '.xml', '.in'):
 | |
|                 return minimal
 | |
| 
 | |
|         return None  # unknown, will result in fall-back to run all tests
 | |
| 
 | |
| 
 | |
| def all_tests(args, force=False):
 | |
|     """
 | |
|     :type args: TestConfig
 | |
|     :type force: bool
 | |
|     :rtype: dict[str, str]
 | |
|     """
 | |
|     if force:
 | |
|         integration_all_target = 'all'
 | |
|     else:
 | |
|         integration_all_target = get_integration_all_target(args)
 | |
| 
 | |
|     return {
 | |
|         'sanity': 'all',
 | |
|         'compile': 'all',
 | |
|         'units': 'all',
 | |
|         'integration': integration_all_target,
 | |
|         'windows-integration': integration_all_target,
 | |
|         'network-integration': integration_all_target,
 | |
|     }
 | |
| 
 | |
| 
 | |
| def get_integration_all_target(args):
 | |
|     """
 | |
|     :type args: TestConfig
 | |
|     :rtype: str
 | |
|     """
 | |
|     if isinstance(args, IntegrationConfig):
 | |
|         return args.changed_all_target
 | |
| 
 | |
|     return 'all'
 |