summaryrefslogtreecommitdiffstats
path: root/test/runner
diff options
context:
space:
mode:
authorMatt Clay <matt@mystile.com>2016-11-30 06:21:53 +0100
committerGitHub <noreply@github.com>2016-11-30 06:21:53 +0100
commit6bbd92e422fcba608cd00ddd38a4c1f03dca301a (patch)
tree6cd5b7cacb78ddadb41ffe45b5f4ea9c7a1cac79 /test/runner
parentbump submodule refs (diff)
downloadansible-6bbd92e422fcba608cd00ddd38a4c1f03dca301a.tar.xz
ansible-6bbd92e422fcba608cd00ddd38a4c1f03dca301a.zip
Initial ansible-test implementation. (#18556)
Diffstat (limited to 'test/runner')
l---------test/runner/ansible-test1
l---------test/runner/injector/ansible1
l---------test/runner/injector/ansible-console1
l---------test/runner/injector/ansible-doc1
l---------test/runner/injector/ansible-galaxy1
l---------test/runner/injector/ansible-playbook1
l---------test/runner/injector/ansible-pull1
l---------test/runner/injector/ansible-vault1
l---------test/runner/injector/cover1
l---------test/runner/injector/cover21
l---------test/runner/injector/cover2.41
l---------test/runner/injector/cover2.61
l---------test/runner/injector/cover2.71
l---------test/runner/injector/cover31
l---------test/runner/injector/cover3.51
-rwxr-xr-xtest/runner/injector/injector.py184
l---------test/runner/injector/pytest1
l---------test/runner/injector/runner1
l---------test/runner/injector/runner21
l---------test/runner/injector/runner2.41
l---------test/runner/injector/runner2.61
l---------test/runner/injector/runner2.71
l---------test/runner/injector/runner31
l---------test/runner/injector/runner3.51
-rw-r--r--test/runner/lib/__init__.py0
-rw-r--r--test/runner/lib/ansible_util.py34
-rw-r--r--test/runner/lib/changes.py165
-rw-r--r--test/runner/lib/classification.py326
-rw-r--r--test/runner/lib/core_ci.py340
-rw-r--r--test/runner/lib/cover.py148
-rw-r--r--test/runner/lib/delegation.py331
-rw-r--r--test/runner/lib/executor.py1253
-rw-r--r--test/runner/lib/git.py76
-rw-r--r--test/runner/lib/http.py122
-rw-r--r--test/runner/lib/manage_ci.py142
-rw-r--r--test/runner/lib/pytar.py69
-rw-r--r--test/runner/lib/target.py530
-rw-r--r--test/runner/lib/thread.py48
-rw-r--r--test/runner/lib/util.py415
-rwxr-xr-xtest/runner/reorganize-tests.sh235
-rw-r--r--test/runner/requirements/ansible-test.txt1
-rw-r--r--test/runner/requirements/constraints.txt2
-rw-r--r--test/runner/requirements/coverage.txt1
-rw-r--r--test/runner/requirements/integration.txt8
-rw-r--r--test/runner/requirements/sanity.txt5
-rw-r--r--test/runner/requirements/units.txt11
-rw-r--r--test/runner/requirements/windows-integration.txt4
-rw-r--r--test/runner/setup/docker.sh18
-rw-r--r--test/runner/setup/remote.sh69
-rwxr-xr-xtest/runner/test.py446
-rw-r--r--test/runner/tox.ini9
51 files changed, 5015 insertions, 0 deletions
diff --git a/test/runner/ansible-test b/test/runner/ansible-test
new file mode 120000
index 0000000000..9465664311
--- /dev/null
+++ b/test/runner/ansible-test
@@ -0,0 +1 @@
+test.py \ No newline at end of file
diff --git a/test/runner/injector/ansible b/test/runner/injector/ansible
new file mode 120000
index 0000000000..1f9d09cbf2
--- /dev/null
+++ b/test/runner/injector/ansible
@@ -0,0 +1 @@
+injector.py \ No newline at end of file
diff --git a/test/runner/injector/ansible-console b/test/runner/injector/ansible-console
new file mode 120000
index 0000000000..1f9d09cbf2
--- /dev/null
+++ b/test/runner/injector/ansible-console
@@ -0,0 +1 @@
+injector.py \ No newline at end of file
diff --git a/test/runner/injector/ansible-doc b/test/runner/injector/ansible-doc
new file mode 120000
index 0000000000..1f9d09cbf2
--- /dev/null
+++ b/test/runner/injector/ansible-doc
@@ -0,0 +1 @@
+injector.py \ No newline at end of file
diff --git a/test/runner/injector/ansible-galaxy b/test/runner/injector/ansible-galaxy
new file mode 120000
index 0000000000..1f9d09cbf2
--- /dev/null
+++ b/test/runner/injector/ansible-galaxy
@@ -0,0 +1 @@
+injector.py \ No newline at end of file
diff --git a/test/runner/injector/ansible-playbook b/test/runner/injector/ansible-playbook
new file mode 120000
index 0000000000..1f9d09cbf2
--- /dev/null
+++ b/test/runner/injector/ansible-playbook
@@ -0,0 +1 @@
+injector.py \ No newline at end of file
diff --git a/test/runner/injector/ansible-pull b/test/runner/injector/ansible-pull
new file mode 120000
index 0000000000..1f9d09cbf2
--- /dev/null
+++ b/test/runner/injector/ansible-pull
@@ -0,0 +1 @@
+injector.py \ No newline at end of file
diff --git a/test/runner/injector/ansible-vault b/test/runner/injector/ansible-vault
new file mode 120000
index 0000000000..1f9d09cbf2
--- /dev/null
+++ b/test/runner/injector/ansible-vault
@@ -0,0 +1 @@
+injector.py \ No newline at end of file
diff --git a/test/runner/injector/cover b/test/runner/injector/cover
new file mode 120000
index 0000000000..1f9d09cbf2
--- /dev/null
+++ b/test/runner/injector/cover
@@ -0,0 +1 @@
+injector.py \ No newline at end of file
diff --git a/test/runner/injector/cover2 b/test/runner/injector/cover2
new file mode 120000
index 0000000000..1f9d09cbf2
--- /dev/null
+++ b/test/runner/injector/cover2
@@ -0,0 +1 @@
+injector.py \ No newline at end of file
diff --git a/test/runner/injector/cover2.4 b/test/runner/injector/cover2.4
new file mode 120000
index 0000000000..1f9d09cbf2
--- /dev/null
+++ b/test/runner/injector/cover2.4
@@ -0,0 +1 @@
+injector.py \ No newline at end of file
diff --git a/test/runner/injector/cover2.6 b/test/runner/injector/cover2.6
new file mode 120000
index 0000000000..1f9d09cbf2
--- /dev/null
+++ b/test/runner/injector/cover2.6
@@ -0,0 +1 @@
+injector.py \ No newline at end of file
diff --git a/test/runner/injector/cover2.7 b/test/runner/injector/cover2.7
new file mode 120000
index 0000000000..1f9d09cbf2
--- /dev/null
+++ b/test/runner/injector/cover2.7
@@ -0,0 +1 @@
+injector.py \ No newline at end of file
diff --git a/test/runner/injector/cover3 b/test/runner/injector/cover3
new file mode 120000
index 0000000000..1f9d09cbf2
--- /dev/null
+++ b/test/runner/injector/cover3
@@ -0,0 +1 @@
+injector.py \ No newline at end of file
diff --git a/test/runner/injector/cover3.5 b/test/runner/injector/cover3.5
new file mode 120000
index 0000000000..1f9d09cbf2
--- /dev/null
+++ b/test/runner/injector/cover3.5
@@ -0,0 +1 @@
+injector.py \ No newline at end of file
diff --git a/test/runner/injector/injector.py b/test/runner/injector/injector.py
new file mode 100755
index 0000000000..a0c068e148
--- /dev/null
+++ b/test/runner/injector/injector.py
@@ -0,0 +1,184 @@
+#!/usr/bin/env python
+"""Code coverage wrapper."""
+
+from __future__ import absolute_import, print_function
+
+import errno
+import os
+import sys
+import pipes
+import logging
+import getpass
+
+logger = logging.getLogger('injector') # pylint: disable=locally-disabled, invalid-name
+
+
+def main():
+ """Main entry point."""
+ formatter = logging.Formatter('%(asctime)s ' + str(os.getpid()) + ' %(levelname)s %(message)s')
+ log_name = 'ansible-test-coverage.%s.log' % getpass.getuser()
+ self_dir = os.path.dirname(os.path.abspath(__file__))
+
+ handler = logging.FileHandler(os.path.join('/tmp', log_name))
+ handler.setFormatter(formatter)
+ logger.addHandler(handler)
+
+ handler = logging.FileHandler(os.path.abspath(os.path.join(self_dir, '..', 'logs', log_name)))
+ handler.setFormatter(formatter)
+ logger.addHandler(handler)
+
+ logger.setLevel(logging.DEBUG)
+
+ try:
+ logger.debug('Self: %s', __file__)
+ logger.debug('Arguments: %s', ' '.join(pipes.quote(c) for c in sys.argv))
+
+ if os.path.basename(__file__).startswith('runner'):
+ args, env = runner()
+ elif os.path.basename(__file__).startswith('cover'):
+ args, env = cover()
+ else:
+ args, env = injector()
+
+ logger.debug('Run command: %s', ' '.join(pipes.quote(c) for c in args))
+
+ try:
+ cwd = os.getcwd()
+ except OSError as ex:
+ if ex.errno != errno.EACCES:
+ raise
+ cwd = None
+
+ logger.debug('Working directory: %s', cwd or '?')
+
+ for key in sorted(env.keys()):
+ logger.debug('%s=%s', key, env[key])
+
+ os.execvpe(args[0], args, env)
+ except Exception as ex:
+ logger.fatal(ex)
+ raise
+
+
+def injector():
+ """
+ :rtype: list[str], dict[str, str]
+ """
+ self_dir = os.path.dirname(os.path.abspath(__file__))
+ command = os.path.basename(__file__)
+ mode = os.environ.get('ANSIBLE_TEST_COVERAGE')
+ version = os.environ.get('ANSIBLE_TEST_PYTHON_VERSION', '')
+ executable = find_executable(command)
+
+ if mode in ('coverage', 'version'):
+ if mode == 'coverage':
+ args, env = coverage_command(self_dir, version)
+ args += [executable]
+ tool = 'cover'
+ else:
+ interpreter = find_executable('python' + version)
+ args, env = [interpreter, executable], os.environ.copy()
+ tool = 'runner'
+
+ if command in ('ansible', 'ansible-playbook', 'ansible-pull'):
+ interpreter = find_executable(tool + version)
+ args += ['--extra-vars', 'ansible_python_interpreter=' + interpreter]
+ else:
+ args, env = [executable], os.environ.copy()
+
+ args += sys.argv[1:]
+
+ return args, env
+
+
+def runner():
+ """
+ :rtype: list[str], dict[str, str]
+ """
+ command = os.path.basename(__file__)
+ version = command.replace('runner', '')
+
+ interpreter = find_executable('python' + version)
+ args, env = [interpreter], os.environ.copy()
+
+ args += sys.argv[1:]
+
+ return args, env
+
+
+def cover():
+ """
+ :rtype: list[str], dict[str, str]
+ """
+ self_dir = os.path.dirname(os.path.abspath(__file__))
+ command = os.path.basename(__file__)
+ version = command.replace('cover', '')
+
+ if len(sys.argv) > 1:
+ executable = sys.argv[1]
+ else:
+ executable = ''
+
+ if os.path.basename(executable).startswith('ansible_module_'):
+ args, env = coverage_command(self_dir, version)
+ else:
+ interpreter = find_executable('python' + version)
+ args, env = [interpreter], os.environ.copy()
+
+ args += sys.argv[1:]
+
+ return args, env
+
+
+def coverage_command(self_dir, version):
+ """
+ :type self_dir: str
+ :type version: str
+ :rtype: list[str], dict[str, str]
+ """
+ executable = 'coverage'
+
+ if version:
+ executable += '-%s' % version
+
+ args = [
+ find_executable(executable),
+ 'run',
+ '--append',
+ '--rcfile',
+ os.path.join(self_dir, '.coveragerc'),
+ ]
+
+ env = os.environ.copy()
+ env['COVERAGE_FILE'] = os.path.abspath(os.path.join(self_dir, '..', 'output', 'coverage'))
+
+ return args, env
+
+
+def find_executable(executable):
+ """
+ :type executable: str
+ :rtype: str
+ """
+ self = os.path.abspath(__file__)
+ path = os.environ.get('PATH', os.defpath)
+ seen_dirs = set()
+
+ for path_dir in path.split(os.pathsep):
+ if path_dir in seen_dirs:
+ continue
+
+ seen_dirs.add(path_dir)
+ candidate = os.path.abspath(os.path.join(path_dir, executable))
+
+ if candidate == self:
+ continue
+
+ if os.path.exists(candidate) and os.access(candidate, os.F_OK | os.X_OK):
+ return candidate
+
+ raise Exception('Executable "%s" not found in path: %s' % (executable, path))
+
+
+if __name__ == '__main__':
+ main()
diff --git a/test/runner/injector/pytest b/test/runner/injector/pytest
new file mode 120000
index 0000000000..1f9d09cbf2
--- /dev/null
+++ b/test/runner/injector/pytest
@@ -0,0 +1 @@
+injector.py \ No newline at end of file
diff --git a/test/runner/injector/runner b/test/runner/injector/runner
new file mode 120000
index 0000000000..1f9d09cbf2
--- /dev/null
+++ b/test/runner/injector/runner
@@ -0,0 +1 @@
+injector.py \ No newline at end of file
diff --git a/test/runner/injector/runner2 b/test/runner/injector/runner2
new file mode 120000
index 0000000000..1f9d09cbf2
--- /dev/null
+++ b/test/runner/injector/runner2
@@ -0,0 +1 @@
+injector.py \ No newline at end of file
diff --git a/test/runner/injector/runner2.4 b/test/runner/injector/runner2.4
new file mode 120000
index 0000000000..1f9d09cbf2
--- /dev/null
+++ b/test/runner/injector/runner2.4
@@ -0,0 +1 @@
+injector.py \ No newline at end of file
diff --git a/test/runner/injector/runner2.6 b/test/runner/injector/runner2.6
new file mode 120000
index 0000000000..1f9d09cbf2
--- /dev/null
+++ b/test/runner/injector/runner2.6
@@ -0,0 +1 @@
+injector.py \ No newline at end of file
diff --git a/test/runner/injector/runner2.7 b/test/runner/injector/runner2.7
new file mode 120000
index 0000000000..1f9d09cbf2
--- /dev/null
+++ b/test/runner/injector/runner2.7
@@ -0,0 +1 @@
+injector.py \ No newline at end of file
diff --git a/test/runner/injector/runner3 b/test/runner/injector/runner3
new file mode 120000
index 0000000000..1f9d09cbf2
--- /dev/null
+++ b/test/runner/injector/runner3
@@ -0,0 +1 @@
+injector.py \ No newline at end of file
diff --git a/test/runner/injector/runner3.5 b/test/runner/injector/runner3.5
new file mode 120000
index 0000000000..1f9d09cbf2
--- /dev/null
+++ b/test/runner/injector/runner3.5
@@ -0,0 +1 @@
+injector.py \ No newline at end of file
diff --git a/test/runner/lib/__init__.py b/test/runner/lib/__init__.py
new file mode 100644
index 0000000000..e69de29bb2
--- /dev/null
+++ b/test/runner/lib/__init__.py
diff --git a/test/runner/lib/ansible_util.py b/test/runner/lib/ansible_util.py
new file mode 100644
index 0000000000..ae74db7408
--- /dev/null
+++ b/test/runner/lib/ansible_util.py
@@ -0,0 +1,34 @@
+"""Miscellaneous utility functions and classes specific to ansible cli tools."""
+
+from __future__ import absolute_import, print_function
+
+import os
+
+from lib.util import common_environment
+
+
+def ansible_environment(args):
+ """
+ :type args: CommonConfig
+ :rtype: dict[str, str]
+ """
+ env = common_environment()
+ path = env['PATH']
+
+ ansible_path = os.path.join(os.getcwd(), 'bin')
+
+ if not path.startswith(ansible_path + os.pathsep):
+ path = ansible_path + os.pathsep + path
+
+ ansible = dict(
+ ANSIBLE_FORCE_COLOR='%s' % 'true' if args.color else 'false',
+ ANSIBLE_DEPRECATION_WARNINGS='false',
+ ANSIBLE_CONFIG='/dev/null',
+ PYTHONPATH=os.path.abspath('lib'),
+ PAGER='/bin/cat',
+ PATH=path,
+ )
+
+ env.update(ansible)
+
+ return env
diff --git a/test/runner/lib/changes.py b/test/runner/lib/changes.py
new file mode 100644
index 0000000000..019b51e0f7
--- /dev/null
+++ b/test/runner/lib/changes.py
@@ -0,0 +1,165 @@
+"""Detect changes in Ansible code."""
+
+from __future__ import absolute_import, print_function
+
+import re
+import os
+
+from lib.util import (
+ ApplicationError,
+ SubprocessError,
+ MissingEnvironmentVariable,
+ CommonConfig,
+)
+
+from lib.http import (
+ HttpClient,
+ urlencode,
+)
+
+from lib.git import (
+ Git,
+)
+
+
+class InvalidBranch(ApplicationError):
+ """Exception for invalid branch specification."""
+ def __init__(self, branch, reason):
+ """
+ :type branch: str
+ :type reason: str
+ """
+ message = 'Invalid branch: %s\n%s' % (branch, reason)
+
+ super(InvalidBranch, self).__init__(message)
+
+ self.branch = branch
+
+
+class ChangeDetectionNotSupported(ApplicationError):
+ """Exception for cases where change detection is not supported."""
+ def __init__(self, message):
+ """
+ :type message: str
+ """
+ super(ChangeDetectionNotSupported, self).__init__(message)
+
+
+class ShippableChanges(object):
+ """Change information for Shippable build."""
+ def __init__(self, args, git):
+ """
+ :type args: CommonConfig
+ :type git: Git
+ """
+ self.args = args
+
+ try:
+ self.branch = os.environ['BRANCH']
+ self.is_pr = os.environ['IS_PULL_REQUEST'] == 'true'
+ self.is_tag = os.environ['IS_GIT_TAG'] == 'true'
+ self.commit = os.environ['COMMIT']
+ self.project_id = os.environ['PROJECT_ID']
+ except KeyError as ex:
+ raise MissingEnvironmentVariable(name=ex.args[0])
+
+ if self.is_tag:
+ raise ChangeDetectionNotSupported('Change detection is not supported for tags.')
+
+ if self.is_pr:
+ self.paths = sorted(git.get_diff_names([self.branch]))
+ else:
+ merge_runs = self.get_merge_runs(self.project_id, self.branch)
+ last_successful_commit = self.get_last_successful_commit(merge_runs)
+ self.paths = sorted(git.get_diff_names([last_successful_commit, self.commit]))
+
+ def get_merge_runs(self, project_id, branch):
+ """
+ :type project_id: str
+ :type branch: str
+ :rtype: list[dict]
+ """
+ params = dict(
+ isPullRequest='false',
+ projectIds=project_id,
+ branch=branch,
+ )
+
+ client = HttpClient(self.args, always=True)
+ response = client.get('https://api.shippable.com/runs?%s' % urlencode(params))
+ return response.json()
+
+ @staticmethod
+ def get_last_successful_commit(merge_runs):
+ """
+ :type merge_runs: list[dict]
+ :rtype: str
+ """
+ merge_runs = sorted(merge_runs, key=lambda r: r['createdAt'])
+ known_commits = set()
+ last_successful_commit = None
+
+ for merge_run in merge_runs:
+ commit_sha = merge_run['commitSha']
+ if commit_sha not in known_commits:
+ known_commits.add(commit_sha)
+ if merge_run['statusCode'] == 30:
+ last_successful_commit = commit_sha
+
+ return last_successful_commit
+
+
+class LocalChanges(object):
+ """Change information for local work."""
+ def __init__(self, args, git):
+ """
+ :type args: CommonConfig
+ :type git: Git
+ """
+ self.args = args
+ self.current_branch = git.get_branch()
+
+ if self.is_official_branch(self.current_branch):
+ raise InvalidBranch(branch=self.current_branch,
+ reason='Current branch is not a feature branch.')
+
+ self.fork_branch = None
+ self.fork_point = None
+
+ self.local_branches = sorted(git.get_branches())
+ self.official_branches = sorted([b for b in self.local_branches if self.is_official_branch(b)])
+
+ for self.fork_branch in self.official_branches:
+ try:
+ self.fork_point = git.get_branch_fork_point(self.fork_branch)
+ break
+ except SubprocessError:
+ pass
+
+ if self.fork_point is None:
+ raise ApplicationError('Unable to auto-detect fork branch and fork point.')
+
+ # tracked files (including unchanged)
+ self.tracked = sorted(git.get_file_names(['--cached']))
+ # untracked files (except ignored)
+ self.untracked = sorted(git.get_file_names(['--others', '--exclude-standard']))
+ # tracked changes (including deletions) committed since the branch was forked
+ self.committed = sorted(git.get_diff_names([self.fork_point, 'HEAD']))
+ # tracked changes (including deletions) which are staged
+ self.staged = sorted(git.get_diff_names(['--cached']))
+ # tracked changes (including deletions) which are not staged
+ self.unstaged = sorted(git.get_diff_names([]))
+
+ @staticmethod
+ def is_official_branch(name):
+ """
+ :type name: str
+ :rtype: bool
+ """
+ if name == 'devel':
+ return True
+
+ if re.match(r'^stable-[0-9]+\.[0-9]+$', name):
+ return True
+
+ return False
diff --git a/test/runner/lib/classification.py b/test/runner/lib/classification.py
new file mode 100644
index 0000000000..d89737d1e6
--- /dev/null
+++ b/test/runner/lib/classification.py
@@ -0,0 +1,326 @@
+"""Classify changes in Ansible code."""
+
+from __future__ import absolute_import, print_function
+
+import os
+
+from lib.target import (
+ walk_module_targets,
+ walk_integration_targets,
+ walk_units_targets,
+ walk_compile_targets,
+)
+
+from lib.util import (
+ display,
+)
+
+
+def categorize_changes(paths, verbose_command=None):
+ """
+ :type paths: list[str]
+ :type verbose_command: str
+ :rtype paths: dict[str, list[str]]
+ """
+ mapper = PathMapper()
+
+ commands = {
+ 'sanity': set(),
+ 'compile': set(),
+ 'units': set(),
+ 'integration': set(),
+ 'windows-integration': set(),
+ 'network-integration': set(),
+ }
+
+ 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() # 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.keys() 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:
+ if any(t == 'all' for t in commands[command]):
+ commands[command] = set(['all'])
+
+ commands = dict((c, sorted(commands[c])) for c in commands.keys() if commands[c])
+
+ return commands
+
+
+class PathMapper(object):
+ """Map file paths to test commands and targets."""
+ def __init__(self):
+ 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.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(t.path for t in self.units_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.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)
+
+ 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 'sanity' not in result:
+ result['sanity'] = path
+
+ return result
+
+ def _classify(self, path):
+ """
+ :type path: str
+ :rtype: dict[str, str] | None
+ """
+ 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('docs-api/'):
+ return minimal
+
+ if path.startswith('docsite/'):
+ return minimal
+
+ if path.startswith('examples/'):
+ return minimal
+
+ if path.startswith('hacking/'):
+ return minimal
+
+ if path.startswith('lib/ansible/modules/'):
+ module = self.module_names_by_path.get(path)
+
+ if module:
+ return {
+ 'units': module if module in self.units_modules else None,
+ 'integration': self.posix_integration_by_module.get(module) if ext == '.py' else None,
+ 'windows-integration': self.windows_integration_by_module.get(module) if ext == '.ps1' else None,
+ 'network-integration': self.network_integration_by_module.get(module),
+ }
+
+ return minimal
+
+ if path.startswith('lib/ansible/module_utils/'):
+ if ext == '.ps1':
+ return {
+ 'windows-integration': 'all',
+ }
+
+ if ext == '.py':
+ return {
+ 'integration': 'all',
+ 'network-integration': 'all',
+ 'units': 'all',
+ }
+
+ if path.startswith('lib/ansible/plugins/connection/'):
+ if name == '__init__':
+ return {
+ 'integration': 'all',
+ 'windows-integration': 'all',
+ 'network-integration': 'all',
+ 'units': 'test/units/plugins/connection/',
+ }
+
+ if name == 'winrm':
+ return {
+ 'windows-integration': 'all',
+ 'units': 'test/units/plugins/connection/',
+ }
+
+ if name == 'local':
+ return {
+ 'integration': 'all',
+ 'network-integration': 'all',
+ 'units': 'test/units/plugins/connections/',
+ }
+
+ if 'connection_%s' % name in self.integration_targets_by_name:
+ return {
+ 'integration': 'connection_%s' % name,
+ }
+
+ return minimal
+
+ if path.startswith('lib/ansible/utils/module_docs_fragments/'):
+ return {
+ 'sanity': 'all',
+ }
+
+ if path.startswith('lib/ansible/'):
+ return all_tests() # broad impact, run all tests
+
+ if path.startswith('packaging/'):
+ return minimal
+
+ if path.startswith('test/compile/'):
+ return {
+ 'compile': 'all',
+ }
+
+ if path.startswith('test/results/'):
+ return minimal
+
+ if path.startswith('test/integration/roles/'):
+ return minimal
+
+ if path.startswith('test/integration/targets/'):
+ target = self.integration_targets_by_name[path.split('/')[3]]
+
+ if 'hidden/' in target.aliases:
+ return {
+ 'integration': 'all',
+ 'windows-integration': 'all',
+ 'network-integration': 'all',
+ }
+
+ 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/'):
+ return {
+ 'integration': 'all',
+ 'windows-integration': 'all',
+ 'network-integration': 'all',
+ }
+
+ if path.startswith('test/samples/'):
+ return minimal
+
+ 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,
+ }
+
+ return {
+ 'units': os.path.dirname(path),
+ }
+
+ if path.startswith('test/runner/'):
+ return all_tests() # test infrastructure, run all tests
+
+ if path.startswith('test/utils/shippable/'):
+ return all_tests() # 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',
+ 'setup.py',
+ ):
+ return minimal
+
+ if path in (
+ 'shippable.yml',
+ '.coveragerc',
+ ):
+ return all_tests() # test infrastructure, 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():
+ """
+ :rtype: dict[str, str]
+ """
+ return {
+ 'sanity': 'all',
+ 'compile': 'all',
+ 'units': 'all',
+ 'integration': 'all',
+ 'windows-integration': 'all',
+ 'network-integration': 'all',
+ }
diff --git a/test/runner/lib/core_ci.py b/test/runner/lib/core_ci.py
new file mode 100644
index 0000000000..6aaae057f0
--- /dev/null
+++ b/test/runner/lib/core_ci.py
@@ -0,0 +1,340 @@
+"""Access Ansible Core CI remote services."""
+
+from __future__ import absolute_import, print_function
+
+import json
+import os
+import traceback
+import uuid
+import errno
+import time
+
+from lib.http import (
+ HttpClient,
+ HttpResponse,
+ HttpError,
+)
+
+from lib.util import (
+ ApplicationError,
+ run_command,
+ make_dirs,
+ CommonConfig,
+ display,
+ is_shippable,
+)
+
+
+class AnsibleCoreCI(object):
+ """Client for Ansible Core CI services."""
+ def __init__(self, args, platform, version, stage='prod', persist=True, name=None):
+ """
+ :type args: CommonConfig
+ :type platform: str
+ :type version: str
+ :type stage: str
+ :type persist: bool
+ :type name: str
+ """
+ self.args = args
+ self.platform = platform
+ self.version = version
+ self.stage = stage
+ self.client = HttpClient(args)
+ self.connection = None
+ self.instance_id = None
+ self.name = name if name else '%s-%s' % (self.platform, self.version)
+
+ if self.platform == 'windows':
+ self.ssh_key = None
+ self.endpoint = 'https://14blg63h2i.execute-api.us-east-1.amazonaws.com'
+ self.port = 5986
+ elif self.platform == 'freebsd':
+ self.ssh_key = SshKey(args)
+ self.endpoint = 'https://14blg63h2i.execute-api.us-east-1.amazonaws.com'
+ self.port = 22
+ elif self.platform == 'osx':
+ self.ssh_key = SshKey(args)
+ self.endpoint = 'https://osx.testing.ansible.com'
+ self.port = None
+ else:
+ raise ApplicationError('Unsupported platform: %s' % platform)
+
+ self.path = os.path.expanduser('~/.ansible/test/instances/%s-%s' % (self.name, self.stage))
+
+ if persist and self._load():
+ try:
+ display.info('Checking existing %s/%s instance %s.' % (self.platform, self.version, self.instance_id),
+ verbosity=1)
+
+ self.connection = self.get()
+
+ display.info('Loaded existing %s/%s instance %s.' % (self.platform, self.version, self.instance_id),
+ verbosity=1)
+ except HttpError as ex:
+ if ex.status != 404:
+ raise
+
+ self._clear()
+
+ display.info('Cleared stale %s/%s instance %s.' % (self.platform, self.version, self.instance_id),
+ verbosity=1)
+
+ self.instance_id = None
+ else:
+ self.instance_id = None
+ self._clear()
+
+ if self.instance_id:
+ self.started = True
+ else:
+ self.started = False
+ self.instance_id = str(uuid.uuid4())
+
+ display.info('Initializing new %s/%s instance %s.' % (self.platform, self.version, self.instance_id),
+ verbosity=1)
+
+ def start(self):
+ """Start instance."""
+ if is_shippable():
+ self.start_shippable()
+ else:
+ self.start_remote()
+
+ def start_remote(self):
+ """Start instance for remote development/testing."""
+ with open(os.path.expanduser('~/.ansible-core-ci.key'), 'r') as key_fd:
+ auth_key = key_fd.read().strip()
+
+ self._start(dict(
+ remote=dict(
+ key=auth_key,
+ nonce=None,
+ ),
+ ))
+
+ def start_shippable(self):
+ """Start instance on Shippable."""
+ self._start(dict(
+ shippable=dict(
+ run_id=os.environ['SHIPPABLE_BUILD_ID'],
+ job_number=int(os.environ['SHIPPABLE_JOB_NUMBER']),
+ ),
+ ))
+
+ def stop(self):
+ """Stop instance."""
+ if not self.started:
+ display.info('Skipping invalid %s/%s instance %s.' % (self.platform, self.version, self.instance_id),
+ verbosity=1)
+ return
+
+ response = self.client.delete(self._uri)
+
+ if response.status_code == 404:
+ self._clear()
+ display.info('Cleared invalid %s/%s instance %s.' % (self.platform, self.version, self.instance_id),
+ verbosity=1)
+ return
+
+ if response.status_code == 200:
+ self._clear()
+ display.info('Stopped running %s/%s instance %s.' % (self.platform, self.version, self.instance_id),
+ verbosity=1)
+ return
+
+ raise self._create_http_error(response)
+
+ def get(self):
+ """
+ Get instance connection information.
+ :rtype: InstanceConnection
+ """
+ if not self.started:
+ display.info('Skipping invalid %s/%s instance %s.' % (self.platform, self.version, self.instance_id),
+ verbosity=1)
+ return None
+
+ if self.connection and self.connection.running:
+ return self.connection
+
+ response = self.client.get(self._uri)
+
+ if response.status_code != 200:
+ raise self._create_http_error(response)
+
+ if self.args.explain:
+ self.connection = InstanceConnection(
+ running=True,
+ hostname='cloud.example.com',
+ port=self.port or 12345,
+ username='username',
+ password='password' if self.platform == 'windows' else None,
+ )
+ else:
+ response_json = response.json()
+
+ status = response_json['status']
+ con = response_json['connection']
+
+ self.connection = InstanceConnection(
+ running=status == 'running',
+ hostname=con['hostname'],
+ port=int(con.get('port', self.port)),
+ username=con['username'],
+ password=con.get('password'),
+ )
+
+ status = 'running' if self.connection.running else 'starting'
+
+ display.info('Retrieved %s %s/%s instance %s.' % (status, self.platform, self.version, self.instance_id),
+ verbosity=1)
+
+ return self.connection
+
+ def wait(self):
+ """Wait for the instance to become ready."""
+ for _ in range(1, 90):
+ if self.get().running:
+ return
+ time.sleep(10)
+
+ raise ApplicationError('Timeout waiting for %s/%s instance %s.' %
+ (self.platform, self.version, self.instance_id))
+
+ @property
+ def _uri(self):
+ return '%s/%s/jobs/%s' % (self.endpoint, self.stage, self.instance_id)
+
+ def _start(self, auth):
+ """Start instance."""
+ if self.started:
+ display.info('Skipping started %s/%s instance %s.' % (self.platform, self.version, self.instance_id),
+ verbosity=1)
+ return
+
+ data = dict(
+ config=dict(
+ platform=self.platform,
+ version=self.version,
+ public_key=self.ssh_key.pub_contents if self.ssh_key else None,
+ query=False,
+ )
+ )
+
+ data.update(dict(auth=auth))
+
+ headers = {
+ 'Content-Type': 'application/json',
+ }
+
+ response = self.client.put(self._uri, data=json.dumps(data), headers=headers)
+
+ if response.status_code != 200:
+ raise self._create_http_error(response)
+
+ self.started = True
+ self._save()
+
+ display.info('Started %s/%s instance %s.' % (self.platform, self.version, self.instance_id),
+ verbosity=1)
+
+ def _clear(self):
+ """Clear instance information."""
+ try:
+ self.connection = None
+ os.remove(self.path)
+ except OSError as ex:
+ if ex.errno != errno.ENOENT:
+ raise
+
+ def _load(self):
+ """Load instance information."""
+ try:
+ with open(self.path, 'r') as instance_fd:
+ self.instance_id = instance_fd.read()
+ self.started = True
+ except IOError as ex:
+ if ex.errno != errno.ENOENT:
+ raise
+ self.instance_id = None
+
+ return self.instance_id
+
+ def _save(self):
+ """Save instance information."""
+ if self.args.explain:
+ return
+
+ make_dirs(os.path.dirname(self.path))
+
+ with open(self.path, 'w') as instance_fd:
+ instance_fd.write(self.instance_id)
+
+ @staticmethod
+ def _create_http_error(response):
+ """
+ :type response: HttpResponse
+ :rtype: ApplicationError
+ """
+ response_json = response.json()
+ stack_trace = ''
+
+ if 'message' in response_json:
+ message = response_json['message']
+ elif 'errorMessage' in response_json:
+ message = response_json['errorMessage'].strip()
+ if 'stackTrace' in response_json:
+ trace = '\n'.join([x.rstrip() for x in traceback.format_list(response_json['stackTrace'])])
+ stack_trace = ('\nTraceback (from remote server):\n%s' % trace)
+ else:
+ message = str(response_json)
+
+ return HttpError(response.status_code, '%s%s' % (message, stack_trace))
+
+
+class SshKey(object):
+ """Container for SSH key used to connect to remote instances."""
+ def __init__(self, args):
+ """
+ :type args: CommonConfig
+ """
+ tmp = os.path.expanduser('~/.ansible/test/')
+
+ self.key = os.path.join(tmp, 'id_rsa')
+ self.pub = os.path.join(tmp, 'id_rsa.pub')
+
+ if not os.path.isfile(self.pub):
+ if not args.explain:
+ make_dirs(tmp)
+
+ run_command(args, ['ssh-keygen', '-q', '-t', 'rsa', '-N', '', '-f', self.key])
+
+ if args.explain:
+ self.pub_contents = None
+ else:
+ with open(self.pub, 'r') as pub_fd:
+ self.pub_contents = pub_fd.read().strip()
+
+
+class InstanceConnection(object):
+ """Container for remote instance status and connection details."""
+ def __init__(self, running, hostname, port, username, password):
+ """
+ :type running: bool
+ :type hostname: str
+ :type port: int
+ :type username: str
+ :type password: str | None
+ """
+ self.running = running
+ self.hostname = hostname
+ self.port = port
+ self.username = username
+ self.password = password
+
+ def __str__(self):
+ if self.password:
+ return '%s:%s [%s:%s]' % (self.hostname, self.port, self.username, self.password)
+
+ return '%s:%s [%s]' % (self.hostname, self.port, self.username)
diff --git a/test/runner/lib/cover.py b/test/runner/lib/cover.py
new file mode 100644
index 0000000000..493ff72b93
--- /dev/null
+++ b/test/runner/lib/cover.py
@@ -0,0 +1,148 @@
+"""Code coverage utilities."""
+
+from __future__ import absolute_import, print_function
+
+import os
+import re
+
+from lib.target import walk_module_targets
+from lib.util import display, ApplicationError, run_command
+from lib.executor import EnvironmentConfig, Delegate, install_command_requirements
+
+COVERAGE_DIR = 'test/results/coverage'
+COVERAGE_FILE = os.path.join(COVERAGE_DIR, 'coverage')
+
+
+def command_coverage_combine(args):
+ """Patch paths in coverage files and merge into a single file.
+ :type args: CoverageConfig
+ """
+ coverage = initialize_coverage(args)
+
+ modules = dict((t.module, t.path) for t in list(walk_module_targets()))
+
+ coverage_files = [os.path.join(COVERAGE_DIR, f) for f in os.listdir(COVERAGE_DIR)
+ if f.startswith('coverage') and f != 'coverage']
+
+ arc_data = {}
+
+ ansible_path = os.path.abspath('lib/ansible/') + '/'
+ root_path = os.getcwd() + '/'
+
+ for coverage_file in coverage_files:
+ original = coverage.CoverageData()
+
+ if os.path.getsize(coverage_file) == 0:
+ display.warning('Empty coverage file: %s' % coverage_file)
+ continue
+
+ try:
+ original.read_file(coverage_file)
+ except Exception as ex: # pylint: disable=locally-disabled, broad-except
+ display.error(str(ex))
+ continue
+
+ for filename in original.measured_files():
+ arcs = original.arcs(filename)
+
+ if '/ansible_modlib.zip/ansible/' in filename:
+ new_name = re.sub('^.*/ansible_modlib.zip/ansible/', ansible_path, filename)
+ display.info('%s -> %s' % (filename, new_name), verbosity=3)
+ filename = new_name
+ elif '/ansible_module_' in filename:
+ module = re.sub('^.*/ansible_module_(?P<module>.*).py$', '\\g<module>', filename)
+ new_name = os.path.abspath(modules[module])
+ display.info('%s -> %s' % (filename, new_name), verbosity=3)
+ filename = new_name
+ elif filename.startswith('/root/ansible/'):
+ new_name = re.sub('^/.*?/ansible/', root_path, filename)
+ display.info('%s -> %s' % (filename, new_name), verbosity=3)
+ filename = new_name
+
+ if filename not in arc_data:
+ arc_data[filename] = []
+
+ arc_data[filename] += arcs
+
+ updated = coverage.CoverageData()
+
+ for filename in arc_data:
+ if not os.path.isfile(filename):
+ display.warning('Invalid coverage path: %s' % filename)
+ continue
+
+ updated.add_arcs({filename: arc_data[filename]})
+
+ if not args.explain:
+ updated.write_file(COVERAGE_FILE)
+
+
+def command_coverage_report(args):
+ """
+ :type args: CoverageConfig
+ """
+ command_coverage_combine(args)
+ run_command(args, ['coverage', 'report'])
+
+
+def command_coverage_html(args):
+ """
+ :type args: CoverageConfig
+ """
+ command_coverage_combine(args)
+ run_command(args, ['coverage', 'html', '-d', 'test/results/reports/coverage'])
+
+
+def command_coverage_xml(args):
+ """
+ :type args: CoverageConfig
+ """
+ command_coverage_combine(args)
+ run_command(args, ['coverage', 'xml', '-o', 'test/results/reports/coverage.xml'])
+
+
+def command_coverage_erase(args):
+ """
+ :type args: CoverageConfig
+ """
+ initialize_coverage(args)
+
+ for name in os.listdir(COVERAGE_DIR):
+ if not name.startswith('coverage'):
+ continue
+
+ path = os.path.join(COVERAGE_DIR, name)
+
+ if not args.explain:
+ os.remove(path)
+
+
+def initialize_coverage(args):
+ """
+ :type args: CoverageConfig
+ :rtype: coverage
+ """
+ if args.delegate:
+ raise Delegate()
+
+ if args.requirements:
+ install_command_requirements(args)
+
+ try:
+ import coverage
+ except ImportError:
+ coverage = None
+
+ if not coverage:
+ raise ApplicationError('You must install the "coverage" python module to use this command.')
+
+ return coverage
+
+
+class CoverageConfig(EnvironmentConfig):
+ """Configuration for the coverage command."""
+ def __init__(self, args):
+ """
+ :type args: any
+ """
+ super(CoverageConfig, self).__init__(args, 'coverage')
diff --git a/test/runner/lib/delegation.py b/test/runner/lib/delegation.py
new file mode 100644
index 0000000000..88dcf9b7fe
--- /dev/null
+++ b/test/runner/lib/delegation.py
@@ -0,0 +1,331 @@
+"""Delegate test execution to another environment."""
+
+from __future__ import absolute_import, print_function
+
+import os
+import sys
+
+import lib.pytar
+import lib.thread
+
+from lib.executor import (
+ SUPPORTED_PYTHON_VERSIONS,
+ EnvironmentConfig,
+ IntegrationConfig,
+ ShellConfig,
+ TestConfig,
+ create_shell_command,
+)
+
+from lib.core_ci import (
+ AnsibleCoreCI,
+)
+
+from lib.manage_ci import (
+ ManagePosixCI,
+)
+
+from lib.util import (
+ ApplicationError,
+ run_command,
+)
+
+BUFFER_SIZE = 256 * 256
+
+
+def delegate(args, exclude, require):
+ """
+ :type args: EnvironmentConfig
+ :type exclude: list[str]
+ :type require: list[str]
+ """
+ if args.tox:
+ delegate_tox(args, exclude, require)
+ return True
+
+ if args.docker:
+ delegate_docker(args, exclude, require)
+ return True
+
+ if args.remote:
+ delegate_remote(args, exclude, require)
+ return True
+
+ return False
+
+
+def delegate_tox(args, exclude, require):
+ """
+ :type args: EnvironmentConfig
+ :type exclude: list[str]
+ :type require: list[str]
+ """
+ if args.python:
+ versions = args.python,
+
+ if args.python not in SUPPORTED_PYTHON_VERSIONS:
+ raise ApplicationError('tox does not support Python version %s' % args.python)
+ else:
+ versions = SUPPORTED_PYTHON_VERSIONS
+
+ options = {
+ '--tox': args.tox_args,
+ }
+
+ for version in versions:
+ tox = ['tox', '-c', 'test/runner/tox.ini', '-e', 'py' + version.replace('.', ''), '--']
+ cmd = generate_command(args, os.path.abspath('test/runner/test.py'), options, exclude, require)
+
+ if not args.python:
+ cmd += ['--python', version]
+
+ run_command(args, tox + cmd)
+
+
+def delegate_docker(args, exclude, require):
+ """
+ :type args: EnvironmentConfig
+ :type exclude: list[str]
+ :type require: list[str]
+ """
+ util_image = args.docker_util
+ test_image = args.docker
+ privileged = args.docker_privileged
+
+ util_id = None
+ test_id = None
+
+ options = {
+ '--docker': 1,
+ '--docker-privileged': 0,
+ '--docker-util': 1,
+ }
+
+ cmd = generate_command(args, '/root/ansible/test/runner/test.py', options, exclude, require)
+
+ if isinstance(args, IntegrationConfig):
+ if not args.allow_destructive:
+ cmd.append('--allow-destructive')
+
+ if not args.explain:
+ lib.pytar.create_tarfile('/tmp/ansible.tgz', '.', lib.pytar.ignore)
+
+ try:
+ if util_image:
+ util_id, _ = run_command(args, [
+ 'docker', 'run', '--detach',
+ util_image,
+ ], capture=True)
+
+ if args.explain:
+ util_id = 'util_id'
+ else:
+ util_id = util_id.strip()
+ else:
+ util_id = None
+
+ test_cmd = [
+ 'docker', 'run', '--detach',
+ '--volume', '/sys/fs/cgroup:/sys/fs/cgroup:ro',
+ '--privileged=%s' % str(privileged).lower(),
+ ]
+
+ if util_id:
+ test_cmd += [
+ '--link', '%s:ansible.http.tests' % util_id,
+ '--link', '%s:sni1.ansible.http.tests' % util_id,
+ '--link', '%s:sni2.ansible.http.tests' % util_id,
+ '--link', '%s:fail.ansible.http.tests' % util_id,
+ '--env', 'HTTPTESTER=1',
+ ]
+
+ test_id, _ = run_command(args, test_cmd + [test_image], capture=True)
+
+ if args.explain:
+ test_id = 'test_id'
+ else:
+ test_id = test_id.strip()
+
+ # write temporary files to /root since /tmp isn't ready immediately on container start
+ docker_put(args, test_id, 'test/runner/setup/docker.sh', '/root/docker.sh')
+
+ run_command(args,
+ ['docker', 'exec', test_id, '/bin/bash', '/root/docker.sh'])
+
+ docker_put(args, test_id, '/tmp/ansible.tgz', '/root/ansible.tgz')
+
+ run_command(args,
+ ['docker', 'exec', test_id, 'mkdir', '/root/ansible'])
+
+ run_command(args,
+ ['docker', 'exec', test_id, 'tar', 'oxzf', '/root/ansible.tgz', '--directory', '/root/ansible'])
+
+ try:
+ command = ['docker', 'exec']
+
+ if isinstance(args, ShellConfig):
+ command.append('-it')
+
+ run_command(args, command + [test_id] + cmd)
+ finally:
+ run_command(args,
+ ['docker', 'exec', test_id,
+ 'tar', 'czf', '/root/results.tgz', '--directory', '/root/ansible/test', 'results'])
+
+ docker_get(args, test_id, '/root/results.tgz', '/tmp/results.tgz')
+
+ run_command(args,
+ ['tar', 'oxzf', '/tmp/results.tgz', '-C', 'test'])
+ finally:
+ if util_id:
+ run_command(args,
+ ['docker', 'rm', '-f', util_id],
+ capture=True)
+
+ if test_id:
+ run_command(args,
+ ['docker', 'rm', '-f', test_id],
+ capture=True)
+
+
+def docker_put(args, container_id, src, dst):
+ """
+ :type args: EnvironmentConfig
+ :type container_id: str
+ :type src: str
+ :type dst: str
+ """
+ # avoid 'docker cp' due to a bug which causes 'docker rm' to fail
+ cmd = ['docker', 'exec', '-i', container_id, 'dd', 'of=%s' % dst, 'bs=%s' % BUFFER_SIZE]
+
+ with open(src, 'rb') as src_fd:
+ run_command(args, cmd, stdin=src_fd, capture=True)
+
+
+def docker_get(args, container_id, src, dst):
+ """
+ :type args: EnvironmentConfig
+ :type container_id: str
+ :type src: str
+ :type dst: str
+ """
+ # avoid 'docker cp' due to a bug which causes 'docker rm' to fail
+ cmd = ['docker', 'exec', '-i', container_id, 'dd', 'if=%s' % src, 'bs=%s' % BUFFER_SIZE]
+
+ with open(dst, 'wb') as dst_fd:
+ run_command(args, cmd, stdout=dst_fd, capture=True)
+
+
+def delegate_remote(args, exclude, require):
+ """
+ :type args: EnvironmentConfig
+ :type exclude: list[str]
+ :type require: list[str]
+ """
+ parts = args.remote.split('/', 1)
+
+ platform = parts[0]
+ version = parts[1]
+
+ core_ci = AnsibleCoreCI(args, platform, version, stage=args.remote_stage)
+
+ try:
+ core_ci.start()
+ core_ci.wait()
+
+ options = {
+ '--remote': 1,
+ }
+
+ cmd = generate_command(args, 'ansible/test/runner/test.py', options, exclude, require)
+
+ if isinstance(args, IntegrationConfig):
+ if not args.allow_destructive:
+ cmd.append('--allow-destructive')
+
+ manage = ManagePosixCI(core_ci)
+ manage.setup()
+
+ try:
+ manage.ssh(cmd)
+ finally:
+ manage.ssh('rm -rf /tmp/results && cp -a ansible/test/results /tmp/results')
+ manage.download('/tmp/results', 'test')
+ finally:
+ pass
+
+
+def generate_command(args, path, options, exclude, require):
+ """
+ :type args: EnvironmentConfig
+ :type path: str
+ :type options: dict[str, int]
+ :type exclude: list[str]
+ :type require: list[str]
+ :return: list[str]
+ """
+ options['--color'] = 1
+
+ cmd = [path]
+ cmd += list(filter_options(args, sys.argv[1:], options, exclude, require))
+ cmd += ['--color', 'yes' if args.color else 'no']
+
+ if args.requirements:
+ cmd += ['--requirements']
+
+ if isinstance(args, ShellConfig):
+ cmd = create_shell_command(cmd)
+
+ return cmd
+
+
+def filter_options(args, argv, options, exclude, require):
+ """
+ :type args: EnvironmentConfig
+ :type argv: list[str]
+ :type options: dict[str, int]
+ :type exclude: list[str]
+ :type require: list[str]
+ :rtype: collections.Iterable[str]
+ """
+ options = options.copy()
+
+ options['--requirements'] = 0
+
+ if isinstance(args, TestConfig):
+ options.update({
+ '--changed': 0,
+ '--tracked': 0,
+ '--untracked': 0,
+ '--ignore-committed': 0,
+ '--ignore-staged': 0,
+ '--ignore-unstaged': 0,
+ '--changed-from': 1,
+ '--changed-path': 1,
+ })
+
+ remaining = 0
+
+ for arg in argv:
+ if not arg.startswith('-') and remaining:
+ remaining -= 1
+ continue
+
+ remaining = 0
+
+ parts = arg.split('=', 1)
+ key = parts[0]
+
+ if key in options:
+ remaining = options[key] - len(parts) + 1
+ continue
+
+ yield arg
+
+ for target in exclude:
+ yield '--exclude'
+ yield target
+
+ for target in require:
+ yield '--require'
+ yield target
diff --git a/test/runner/lib/executor.py b/test/runner/lib/executor.py
new file mode 100644
index 0000000000..2140704b69
--- /dev/null
+++ b/test/runner/lib/executor.py
@@ -0,0 +1,1253 @@
+"""Execute Ansible tests."""
+
+from __future__ import absolute_import, print_function
+
+import glob
+import os
+import tempfile
+import sys
+import time
+import textwrap
+import functools
+import shutil
+import stat
+import random
+import pipes
+import string
+import atexit
+
+import lib.pytar
+import lib.thread
+
+from lib.core_ci import (
+ AnsibleCoreCI,
+)
+
+from lib.manage_ci import (
+ ManageWindowsCI,
+)
+
+from lib.util import (
+ CommonConfig,
+ ApplicationWarning,
+ ApplicationError,
+ SubprocessError,
+ display,
+ run_command,
+ deepest_path,
+ common_environment,
+ remove_tree,
+ make_dirs,
+ is_shippable,
+)
+
+from lib.ansible_util import (
+ ansible_environment,
+)
+
+from lib.target import (
+ IntegrationTarget,
+ walk_external_targets,
+ walk_internal_targets,
+ walk_posix_integration_targets,
+ walk_network_integration_targets,
+ walk_windows_integration_targets,
+ walk_units_targets,
+ walk_compile_targets,
+ walk_sanity_targets,
+)
+
+from lib.changes import (
+ ShippableChanges,
+ LocalChanges,
+)
+
+from lib.git import (
+ Git,
+)
+
+from lib.classification import (
+ categorize_changes,
+)
+
+SUPPORTED_PYTHON_VERSIONS = (
+ '2.6',
+ '2.7',
+ '3.5',
+)
+
+COMPILE_PYTHON_VERSIONS = tuple(sorted(SUPPORTED_PYTHON_VERSIONS + ('2.4',)))
+
+coverage_path = '' # pylint: disable=locally-disabled, invalid-name
+
+
+def create_shell_command(command):
+ """
+ :type command: list[str]
+ :rtype: list[str]
+ """
+ optional_vars = (
+ 'TERM',
+ )
+
+ cmd = ['/usr/bin/env']
+ cmd += ['%s=%s' % (var, os.environ[var]) for var in optional_vars if var in os.environ]
+ cmd += command
+
+ return cmd
+
+
+def install_command_requirements(args):
+ """
+ :type args: EnvironmentConfig
+ """
+ generate_egg_info(args)
+
+ if not args.requirements:
+ return
+
+ cmd = generate_pip_install(args.command)
+
+ if not cmd:
+ return
+
+ if isinstance(args, TestConfig):
+ if args.coverage:
+ cmd += ['coverage']
+
+ try:
+ run_command(args, cmd)
+ except SubprocessError as ex:
+ if ex.status != 2:
+ raise
+
+ # If pip is too old it won't understand the arguments we passed in, so we'll need to upgrade it.
+
+ # Installing "coverage" on ubuntu 16.04 fails with the error:
+ # AttributeError: 'Requirement' object has no attribute 'project_name'
+ # See: https://bugs.launchpad.net/ubuntu/xenial/+source/python-pip/+bug/1626258
+ # Upgrading pip works around the issue.
+ run_command(args, ['pip', 'install', '--upgrade', 'pip'])
+ run_command(args, cmd)
+
+
+def generate_egg_info(args):
+ """
+ :type args: EnvironmentConfig
+ """
+ if os.path.isdir('lib/ansible.egg-info'):
+ return
+
+ run_command(args, ['python', 'setup.py', 'egg_info'], capture=args.verbosity < 3)
+
+
+def generate_pip_install(command):
+ """
+ :type command: str
+ :return: list[str] | None
+ """
+ constraints = 'test/runner/requirements/constraints.txt'
+ requirements = 'test/runner/requirements/%s.txt' % command
+
+ if not os.path.exists(requirements) or not os.path.getsize(requirements):
+ return None
+
+ return ['pip', 'install', '--disable-pip-version-check', '-r', requirements, '-c', constraints]
+
+
+def command_shell(args):
+ """
+ :type args: ShellConfig
+ """
+ if args.delegate:
+ raise Delegate()
+
+ install_command_requirements(args)
+
+ cmd = create_shell_command(['bash', '-i'])
+ run_command(args, cmd)
+
+
+def command_posix_integration(args):
+ """
+ :type args: PosixIntegrationConfig
+ """
+ internal_targets = command_integration_filter(args, walk_posix_integration_targets())
+ command_integration_filtered(args, internal_targets)
+
+
+def command_network_integration(args):
+ """
+ :type args: NetworkIntegrationConfig
+ """
+ internal_targets = command_integration_filter(args, walk_network_integration_targets())
+ command_integration_filtered(args, internal_targets)
+
+
+def command_windows_integration(args):
+ """
+ :type args: WindowsIntegrationConfig
+ """
+ internal_targets = command_integration_filter(args, walk_windows_integration_targets())
+
+ if args.windows:
+ instances = [] # type: list [lib.thread.WrappedThread]
+
+ for version in args.windows:
+ instance = lib.thread.WrappedThread(functools.partial(windows_run, args, version))
+ instance.daemon = True
+ instance.start()
+ instances.append(instance)
+
+ install_command_requirements(args)
+
+ while any(instance.is_alive() for instance in instances):
+ time.sleep(1)
+
+ remotes = [instance.wait_for_result() for instance in instances]
+ inventory = windows_inventory(remotes)
+
+ if not args.explain:
+ with open('test/integration/inventory.winrm', 'w') as inventory_fd:
+ inventory_fd.write(inventory)
+ else:
+ install_command_requirements(args)
+
+ try:
+ command_integration_filtered(args, internal_targets)
+ finally:
+ pass
+
+
+def windows_run(args, version):
+ """
+ :type args: WindowsIntegrationConfig
+ :type version: str
+ :rtype: AnsibleCoreCI
+ """
+ core_ci = AnsibleCoreCI(args, 'windows', version, stage=args.remote_stage)
+ core_ci.start()
+ core_ci.wait()
+
+ manage = ManageWindowsCI(core_ci)
+ manage.wait()
+
+ return core_ci
+
+
+def windows_inventory(remotes):
+ """
+ :type remotes: list[AnsibleCoreCI]
+ :rtype: str
+ """
+ hosts = ['%s ansible_host=%s ansible_user=%s ansible_password="%s" ansible_port=%s' %
+ (
+ remote.name.replace('/', '_'),
+ remote.connection.hostname,
+ remote.connection.username,
+ remote.connection.password,
+ remote.connection.port,
+ )
+ for remote in remotes]
+
+ template = """
+ [windows]
+ %s
+
+ [windows:vars]
+ ansible_connection=winrm
+ ansible_winrm_server_cert_validation=ignore
+
+ # support winrm connection tests (temporary solution, does not support testing enable/disable of pipelining)
+ [winrm:children]
+ windows
+
+ # support winrm binary module tests (temporary solution)
+ [testhost_binary_modules:children]
+ windows
+ """
+
+ template = textwrap.dedent(template)
+ inventory = template % ('\n'.join(hosts))
+
+ return inventory
+
+
+def command_integration_filter(args, targets):
+ """
+ :type args: IntegrationConfig
+ :type targets: collections.Iterable[IntegrationTarget]
+ :rtype: tuple[IntegrationTarget]
+ """
+ targets = tuple(targets)
+ changes = get_changes_filter(args)
+ require = (args.require or []) + changes
+ exclude = (args.exclude or [])
+
+ internal_targets = walk_internal_targets(targets, args.include, exclude, require)
+ environment_exclude = get_integration_filter(args, internal_targets)
+
+ if environment_exclude:
+ exclude += environment_exclude
+ internal_targets = walk_internal_targets(targets, args.include, exclude, require)
+
+ if not internal_targets:
+ raise AllTargetsSkipped()
+
+ if args.start_at and not any(t.name == args.start_at for t in internal_targets):
+ raise ApplicationError('Start at target matches nothing: %s' % args.start_at)
+
+ if args.delegate:
+ raise Delegate(require=changes, exclude=exclude)
+
+ install_command_requirements(args)
+
+ return internal_targets
+
+
+def command_integration_filtered(args, targets):
+ """
+ :type args: IntegrationConfig
+ :type targets: tuple[IntegrationTarget]
+ """
+ found = False
+
+ targets_iter = iter(targets)
+
+ test_dir = os.path.expanduser('~/ansible_testing')
+
+ if not args.explain:
+ remove_tree(test_dir)
+ make_dirs(test_dir)
+
+ if any('needs/ssh/' in target.aliases for target in targets):
+ max_tries = 20
+ display.info('SSH service required for tests. Checking to make sure we can connect.')
+ for i in range(1, max_tries + 1):
+ try:
+ run_command(args, ['ssh', '-o', 'BatchMode=yes', 'localhost', 'id'], capture=True)
+ display.info('SSH service responded.')
+ break
+ except SubprocessError as ex:
+ if i == max_tries:
+ raise ex
+ seconds = 3
+ display.warning('SSH service not responding. Waiting %d second(s) before checking again.' % seconds)
+ time.sleep(seconds)
+
+ start_at_task = args.start_at_task
+
+ for target in targets_iter:
+ if args.start_at and not found:
+ found = target.name == args.start_at
+
+ if not found:
+ continue
+
+ tries = 2 if args.retry_on_error else 1
+ verbosity = args.verbosity
+
+ try:
+ while tries:
+ tries -= 1
+
+ try:
+ if target.script_path:
+ command_integration_script(args, target)
+ else:
+ command_integration_role(args, target, start_at_task)
+ start_at_task = None
+ break
+ except SubprocessError:
+ if not tries:
+ raise
+
+ display.warning('Retrying test target "%s" with maximum verbosity.' % target.name)
+ display.verbosity = args.verbosity = 6
+ except:
+ display.notice('To resume at this test target, use the option: --start-at %s' % target.name)
+
+ next_target = next(targets_iter, None)
+
+ if next_target:
+ display.notice('To resume after this test target, use the option: --start-at %s' % next_target.name)
+
+ raise
+ finally:
+ display.verbosity = args.verbosity = verbosity
+
+
+def integration_environment(args):
+ """
+ :type args: IntegrationConfig
+ :rtype: dict[str, str]
+ """
+ env = ansible_environment(args)
+
+ integration = dict(
+ JUNIT_OUTPUT_DIR=os.path.abspath('test/results/junit'),
+ ANSIBLE_CALLBACK_WHITELIST='junit',
+ )
+
+ env.update(integration)
+
+ return env
+
+
+def command_integration_script(args, target):
+ """
+ :type args: IntegrationConfig
+ :type target: IntegrationTarget
+ """
+ display.info('Running %s integration test script' % target.name)
+
+ cmd = ['./%s' % os.path.basename(target.script_path)]
+
+ if args.verbosity:
+ cmd.append('-' + ('v' * args.verbosity))
+
+ env = integration_environment(args)
+ cwd = target.path
+
+ intercept_command(args, cmd, env=env, cwd=cwd)
+
+
+def command_integration_role(args, target, start_at_task):
+ """
+ :type args: IntegrationConfig
+ :type target: IntegrationTarget
+ :type start_at_task: str
+ """
+ display.info('Running %s integration test role' % target.name)
+
+ vars_file = 'integration_config.yml'
+
+ if 'windows/' in target.aliases:
+ inventory = 'inventory.winrm'
+ hosts = 'windows'
+ gather_facts = False
+ elif 'network/' in target.aliases:
+ inventory = 'inventory.network'
+ hosts = target.name[:target.name.find('_')]
+ gather_facts = False
+ else:
+ inventory = 'inventory'
+ hosts = 'testhost'
+ gather_facts = True
+
+ playbook = '''
+- hosts: %s
+ gather_facts: %s
+ roles:
+ - { role: %s }
+ ''' % (hosts, gather_facts, target.name)
+
+ with tempfile.NamedTemporaryFile(dir='test/integration', prefix='%s-' % target.name, suffix='.yml') as pb_fd:
+ pb_fd.write(playbook.encode('utf-8'))
+ pb_fd.flush()
+
+ filename = os.path.basename(pb_fd.name)
+
+ display.info('>>> Playbook: %s\n%s' % (filename, playbook.strip()), verbosity=3)
+
+ cmd = ['ansible-playbook', filename, '-i', inventory, '-e', '@%s' % vars_file]
+
+ if start_at_task:
+ cmd += ['--start-at-task', start_at_task]
+
+ if args.verbosity:
+ cmd.append('-' + ('v' * args.verbosity))
+
+ env = integration_environment(args)
+ cwd = 'test/integration'
+
+ env['ANSIBLE_ROLES_PATH'] = os.path.abspath('test/integration/targets')
+
+ intercept_command(args, cmd, env=env, cwd=cwd)
+
+
+def command_units(args):
+ """
+ :type args: UnitsConfig
+ """
+ changes = get_changes_filter(args)
+ require = (args.require or []) + changes
+ include, exclude = walk_external_targets(walk_units_targets(), args.include, args.exclude, require)
+
+ if not include:
+ raise AllTargetsSkipped()
+
+ if args.delegate:
+ raise Delegate(require=changes)
+
+ install_command_requirements(args)
+
+ version_commands = []
+
+ for version in SUPPORTED_PYTHON_VERSIONS:
+ # run all versions unless version given, in which case run only that version
+ if args.python and version != args.python:
+ continue
+
+ env = ansible_environment(args)
+
+ cmd = [
+ 'pytest',
+ '-r', 'a',
+ '--color',
+ 'yes' if args.color else 'no',
+ '--junit-xml',
+ 'test/results/junit/python%s-units.xml' % version,
+ ]
+
+ if args.collect_only:
+ cmd.append('--collect-only')
+
+ if args.verbosity:
+ cmd.append('-' + ('v' * args.verbosity))
+
+ if exclude:
+ cmd += ['--ignore=%s' % target.path for target in exclude]
+
+ cmd += [target.path for target in include]
+
+ version_commands.append((version, cmd, env))
+
+ for version, command, env in version_commands:
+ display.info('Unit test with Python %s' % version)
+ intercept_command(args, command, env=env, python_version=version)
+
+
+def command_compile(args):
+ """
+ :type args: CompileConfig
+ """
+ changes = get_changes_filter(args)
+ require = (args.require or []) + changes
+ include, exclude = walk_external_targets(walk_compile_targets(), args.include, args.exclude, require)
+
+ if not include:
+ raise AllTargetsSkipped()
+
+ if args.delegate:
+ raise Delegate(require=changes)
+
+ install_command_requirements(args)
+
+ version_commands = []
+
+ for version in COMPILE_PYTHON_VERSIONS:
+ # run all versions unless version given, in which case run only that version
+ if args.python and version != args.python:
+ continue
+
+ # optional list of regex patterns to exclude from tests
+ skip_file = 'test/compile/python%s-skip.txt' % version
+
+ if os.path.exists(skip_file):
+ with open(skip_file, 'r') as skip_fd:
+ skip_paths = skip_fd.read().splitlines()
+ else:
+ skip_paths = []
+
+ # augment file exclusions
+ skip_paths += [e.path for e in exclude]
+ skip_paths.append('/.tox/')
+
+ skip_paths = sorted(skip_paths)
+
+ python = 'python%s' % version
+ cmd = [python, '-m', 'compileall', '-fq']
+
+ if skip_paths:
+ cmd += ['-x', '|'.join(skip_paths)]
+
+ cmd += [target.path for target in include]
+
+ version_commands.append((version, cmd))
+
+ for version, command in version_commands:
+ display.info('Compile with Python %s' % version)
+ run_command(args, command)
+
+
+def command_sanity(args):
+ """
+ :type args: SanityConfig
+ """
+ changes = get_changes_filter(args)
+ require = (args.require or []) + changes
+ targets = SanityTargets(args.include, args.exclude, require)
+
+ if not targets.include:
+ raise AllTargetsSkipped()
+
+ if args.delegate:
+ raise Delegate(require=changes)
+
+ install_command_requirements(args)
+
+ tests = SANITY_TESTS
+
+ if args.test:
+ tests = [t for t in tests if t.name in args.test]
+
+ if args.skip_test:
+ tests = [t for t in tests if t.name not in args.skip_test]
+
+ for test in tests:
+ if args.list_tests:
+ display.info(test.name)
+ continue
+
+ if test.intercept:
+ versions = SUPPORTED_PYTHON_VERSIONS
+ else:
+ versions = None,
+
+ for version in versions:
+ if args.python and version and version != args.python:
+ continue
+
+ display.info('Sanity check using %s%s' % (test.name, ' with Python %s' % version if version else ''))
+
+ if test.intercept:
+ test.func(args, targets, python_version=version)
+ else:
+ test.func(args, targets)
+
+
+def command_sanity_code_smell(args, _):
+ """
+ :type args: SanityConfig
+ :type _: SanityTargets
+ """
+ with open('test/sanity/code-smell/skip.txt', 'r') as skip_fd:
+ skip_tests = skip_fd.read().splitlines()
+
+ tests = glob.glob('test/sanity/code-smell/*')
+ tests = sorted(p for p in tests if os.access(p, os.X_OK) and os.path.basename(p) not in skip_tests)
+
+ for test in tests:
+ display.info('Code smell check using %s' % os.path.basename(test))
+ run_command(args, [test])
+
+
+def command_sanity_validate_modules(args, targets):
+ """
+ :type args: SanityConfig
+ :type targets: SanityTargets
+ """
+ env = ansible_environment(args)
+
+ paths = [deepest_path(i.path, 'lib/ansible/modules/') for i in targets.include_external]
+ paths = sorted(set(p for p in paths if p))
+
+ if not paths:
+ display.info('No tests applicable.', verbosity=1)
+ return
+
+ cmd = ['test/sanity/validate-modules/validate-modules'] + paths
+
+ with open('test/sanity/validate-modules/skip.txt', 'r') as skip_fd:
+ skip_paths = skip_fd.read().splitlines()
+
+ skip_paths += [e.path for e in targets.exclude_external]
+
+ if skip_paths:
+ cmd += ['--exclude', '^(%s)' % '|'.join(skip_paths)]
+
+ run_command(args, cmd, env=env)
+
+
+def command_sanity_shellcheck(args, targets):
+ """
+ :type args: SanityConfig
+ :type targets: SanityTargets
+ """
+ with open('test/sanity/shellcheck/skip.txt', 'r') as skip_fd:
+ skip_paths = set(skip_fd.read().splitlines())
+
+ paths = sorted(i.path for i in targets.include if os.path.splitext(i.path)[1] == '.sh' and i.path not in skip_paths)
+
+ if not paths:
+ display.info('No tests applicable.', verbosity=1)
+ return
+
+ run_command(args, ['shellcheck'] + paths)
+
+
+def command_sanity_yamllint(args, targets):
+ """
+ :type args: SanityConfig
+ :type targets: SanityTargets
+ """
+ paths = sorted(i.path for i in targets.include if os.path.splitext(i.path)[1] in ('.yml', '.yaml'))
+
+ if not paths:
+ display.info('No tests applicable.', verbosity=1)
+ return
+
+ run_command(args, ['yamllint'] + paths)
+
+
+def command_sanity_ansible_doc(args, targets, python_version):
+ """
+ :type args: SanityConfig
+ :type targets: SanityTargets
+ :type python_version: str
+ """
+ with open('test/sanity/ansible-doc/skip.txt', 'r') as skip_fd:
+ skip_modules = set(skip_fd.read().splitlines())
+
+ modules = sorted(set(m for i in targets.include_external for m in i.modules) -
+ set(m for i in targets.exclude_external for m in i.modules) -
+ skip_modules)
+
+ if not modules:
+ display.info('No tests applicable.', verbosity=1)
+ return
+
+ env = ansible_environment(args)
+ cmd = ['ansible-doc'] + modules
+
+ stdout, stderr = intercept_command(args, cmd, env=env, capture=True, python_version=python_version)
+
+ if stderr:
+ # consider any output on stderr an error, even though the return code is zero
+ raise SubprocessError(cmd, stderr=stderr)
+
+ if stdout:
+ display.info(stdout.strip(), verbosity=3)
+
+
+def intercept_command(args, cmd, capture=False, env=None, data=None, cwd=None, python_version=None):
+ """
+ :type args: TestConfig
+ :type cmd: collections.Iterable[str]
+ :type capture: bool
+ :type env: dict[str, str] | None
+ :type data: str | None
+ :type cwd: str | None
+ :type python_version: str | None
+ :rtype: str | None, str | None
+ """
+ if not env:
+ env = common_environment()
+
+ cmd = list(cmd)
+ escaped_cmd = ' '.join(pipes.quote(c) for c in cmd)
+ inject_path = get_coverage_path(args)
+
+ env['PATH'] = inject_path + os.pathsep + env['PATH']
+ env['ANSIBLE_TEST_COVERAGE'] = 'coverage' if args.coverage else 'version'
+ env['ANSIBLE_TEST_PYTHON_VERSION'] = python_version or args.python_version
+ env['ANSIBLE_TEST_CMD'] = escaped_cmd
+
+ return run_command(args, cmd, capture=capture, env=env, data=data, cwd=cwd)
+
+
+def get_coverage_path(args):
+ """
+ :type args: TestConfig
+ :rtype: str
+ """
+ global coverage_path # pylint: disable=locally-disabled, global-statement, invalid-name
+
+ if coverage_path:
+ return os.path.join(coverage_path, 'coverage')
+
+ prefix = 'ansible-test-coverage-'
+ tmp_dir = '/tmp'
+
+ if args.explain:
+ return os.path.join(tmp_dir, '%stmp' % prefix, 'coverage')
+
+ src = os.path.abspath(os.path.join(os.getcwd(), 'test/runner/injector/'))
+
+ coverage_path = tempfile.mkdtemp('', prefix, dir=tmp_dir)
+ os.chmod(coverage_path, stat.S_IRWXU | stat.S_IRGRP | stat.S_IXGRP | stat.S_IROTH | stat.S_IXOTH)
+
+ shutil.copytree(src, os.path.join(coverage_path, 'coverage'))
+ shutil.copy('.coveragerc', os.path.join(coverage_path, 'coverage', '.coveragerc'))
+
+ for directory in 'output', 'logs':
+ os.mkdir(os.path.join(coverage_path, directory))
+ os.chmod(os.path.join(coverage_path, directory), stat.S_IRWXU | stat.S_IRWXG | stat.S_IRWXO)
+
+ atexit.register(cleanup_coverage_dir)
+
+ return os.path.join(coverage_path, 'coverage')
+
+
+def cleanup_coverage_dir():
+ """Copy over coverage data from temporary directory and purge temporary directory."""
+ output_dir = os.path.join(coverage_path, 'output')
+
+ for filename in os.listdir(output_dir):
+ src = os.path.join(output_dir, filename)
+ dst = os.path.join(os.getcwd(), 'test', 'results', 'coverage')
+ shutil.copy(src, dst)
+
+ logs_dir = os.path.join(coverage_path, 'logs')
+
+ for filename in os.listdir(logs_dir):
+ random_suffix = ''.join(random.choice(string.ascii_letters + string.digits) for _ in range(8))
+ new_name = '%s.%s.log' % (os.path.splitext(os.path.basename(filename))[0], random_suffix)
+ src = os.path.join(logs_dir, filename)
+ dst = os.path.join(os.getcwd(), 'test', 'results', 'logs', new_name)
+ shutil.copy(src, dst)
+
+ shutil.rmtree(coverage_path)
+
+
+def get_changes_filter(args):
+ """
+ :type args: TestConfig
+ :rtype: list[str]
+ """
+ paths = detect_changes(args)
+
+ if paths is None:
+ return [] # change detection not enabled, do not filter targets
+
+ if not paths:
+ raise NoChangesDetected()
+
+ commands = categorize_changes(paths, args.command)
+
+ targets = commands.get(args.command)
+
+ if targets is None:
+ raise NoTestsForChanges()
+
+ if targets == ['all']:
+ return [] # changes require testing all targets, do not filter targets
+
+ return targets
+
+
+def detect_changes(args):
+ """
+ :type args: TestConfig
+ :rtype: list[str] | None
+ """
+ if is_shippable():
+ display.info('Shippable detected, collecting parameters from environment.')
+ paths = detect_changes_shippable(args)
+ elif args.changed_from or args.changed_path:
+ paths = args.changed_path or []
+ if args.changed_from:
+ with open(args.changed_from, 'r') as changes_fd:
+ paths += changes_fd.read().splitlines()
+ elif args.changed:
+ paths = detect_changes_local(args)
+ else:
+ return None # change detection not enabled
+
+ display.info('Detected changes in %d file(s).' % len(paths))
+
+ for path in paths:
+ display.info(path, verbosity=1)
+
+ return paths
+
+
+def detect_changes_shippable(args):
+ """Initialize change detection on Shippable.
+ :type args: CommonConfig
+ :rtype: list[str]
+ """
+ git = Git(args)
+ result = ShippableChanges(args, git)
+
+ if result.is_pr:
+ job_type = 'pull request'
+ elif result.is_tag:
+ job_type = 'tag'
+ else:
+ job_type = 'merge commit'
+
+ display.info('Processing %s for branch %s commit %s' % (job_type, result.branch, result.commit))
+
+ return result.paths
+
+
+def detect_changes_local(args):
+ """
+ :type args: TestConfig
+ :rtype: list[str]
+ """
+ git = Git(args)
+ result = LocalChanges(args, git)
+
+ display.info('Detected branch %s forked from %s at commit %s' % (
+ result.current_branch, result.fork_branch, result.fork_point))
+
+ if result.untracked and not args.untracked:
+ display.warning('Ignored %s untracked file(s). Use --untracked to include them.' %
+ len(result.untracked))
+
+ if result.committed and not args.committed:
+ display.warning('Ignored %s committed change(s). Omit --ignore-committed to include them.' %
+ len(result.committed))
+
+ if result.staged and not args.staged:
+ display.warning('Ignored %s staged change(s). Omit --ignore-staged to include them.' %
+ len(result.staged))
+
+ if result.unstaged and not args.unstaged:
+ display.warning('Ignored %s unstaged change(s). Omit --ignore-unstaged to include them.' %
+ len(result.unstaged))
+
+ names = set()
+
+ if args.tracked:
+ names |= set(result.tracked)
+ if args.untracked:
+ names |= set(result.untracked)
+ if args.committed:
+ names |= set(result.committed)
+ if args.staged:
+ names |= set(result.staged)
+ if args.unstaged:
+ names |= set(result.unstaged)
+
+ return sorted(names)
+
+
+def docker_qualify_image(name):
+ """
+ :type name: str | None
+ :rtype: str | None
+ """
+ if not name or any((c in name) for c in ('/', ':')):
+ return name
+
+ return 'ansible/ansible:%s' % name
+
+
+def get_integration_filter(args, targets):
+ """
+ :type args: IntegrationConfig
+ :type targets: tuple[IntegrationTarget]
+ :rtype: list[str]
+ """
+ if args.tox:
+ # tox has the same exclusions as the local environment
+ return get_integration_local_filter(args, targets)
+
+ if args.docker:
+ return get_integration_docker_filter(args, targets)
+
+ if args.remote:
+ return get_integration_remote_filter(args, targets)
+
+ return get_integration_local_filter(args, targets)
+
+
+def get_integration_local_filter(args, targets):
+ """
+ :type args: IntegrationConfig
+ :type targets: tuple[IntegrationTarget]
+ :rtype: list[str]
+ """
+ exclude = []
+
+ if os.getuid() != 0:
+ skip = 'needs/root/'
+ skipped = [target.name for target in targets if skip in target.aliases]
+ if skipped:
+ exclude.append(skip)
+ display.warning('Excluding tests marked "%s" which require running as root: %s'
+ % (skip.rstrip('/'), ', '.join(skipped)))
+
+ # consider explicit testing of destructive as though --allow-destructive was given
+ include_destructive = any(target.startswith('destructive/') for target in args.include)
+
+ if not args.allow_destructive and not include_destructive:
+ skip = 'destructive/'
+ skipped = [target.name for target in targets if skip in target.aliases]
+ if skipped:
+ exclude.append(skip)
+ display.warning('Excluding tests marked "%s" which require --allow-destructive to run locally: %s'
+ % (skip.rstrip('/'), ', '.join(skipped)))
+
+ return exclude
+
+
+def get_integration_docker_filter(args, targets):
+ """
+ :type args: IntegrationConfig
+ :type targets: tuple[IntegrationTarget]
+ :rtype: list[str]
+ """
+ exclude = []
+
+ if not args.docker_privileged:
+ skip = 'needs/privileged/'
+ skipped = [target.name for target in targets if skip in target.aliases]
+ if skipped:
+ exclude.append(skip)
+ display.warning('Excluding tests marked "%s" which require --docker-privileged to run under docker: %s'
+ % (skip.rstrip('/'), ', '.join(skipped)))
+
+ if args.docker.endswith('py3'):
+ skip = 'skip/python3/'
+ skipped = [target.name for target in targets if skip in target.aliases]
+ if skipped:
+ exclude.append(skip)
+ display.warning('Excluding tests marked "%s" which are not yet supported on python 3: %s'
+ % (skip.rstrip('/'), ', '.join(skipped)))
+
+ return exclude
+
+
+def get_integration_remote_filter(args, targets):
+ """
+ :type args: IntegrationConfig
+ :type targets: tuple[IntegrationTarget]
+ :rtype: list[str]
+ """
+ parts = args.remote.split('/', 1)
+
+ platform = parts[0]
+
+ exclude = []
+
+ skip = 'skip/%s/' % platform
+ skipped = [target.name for target in targets if skip in target.aliases]
+ if skipped:
+ exclude.append(skip)
+ display.warning('Excluding tests marked "%s" which are not yet supported on %s: %s'
+ % (skip.rstrip('/'), platform, ', '.join(skipped)))
+
+ return exclude
+
+
+class NoChangesDetected(ApplicationWarning):
+ """Exception when change detection was performed, but no changes were found."""
+ def __init__(self):
+ super(NoChangesDetected, self).__init__('No changes detected.')
+
+
+class NoTestsForChanges(ApplicationWarning):
+ """Exception when changes detected, but no tests trigger as a result."""
+ def __init__(self):
+ super(NoTestsForChanges, self).__init__('No tests found for detected changes.')
+
+
+class SanityTargets(object):
+ """Sanity test target information."""
+ def __init__(self, include, exclude, require):
+ """
+ :type include: list[str]
+ :type exclude: list[str]
+ :type require: list[str]
+ """
+ self.all = not include
+ self.targets = tuple(sorted(walk_sanity_targets()))
+ self.include = walk_internal_targets(self.targets, include, exclude, require)
+ self.include_external, self.exclude_external = walk_external_targets(self.targets, include, exclude, require)
+
+
+class SanityTest(object):
+ """Sanity test base class."""
+ def __init__(self, name):
+ self.name = name
+
+
+class SanityFunc(SanityTest):
+ """Sanity test function information."""
+ def __init__(self, name, func, intercept=True):
+ """
+ :type name: str
+ :type func: (SanityConfig, SanityTargets) -> None
+ :type intercept: bool
+ """
+ super(SanityFunc, self).__init__(name)
+
+ self.func = func
+ self.intercept = intercept
+
+
+class EnvironmentConfig(CommonConfig):
+ """Configuration common to all commands which execute in an environment."""
+ def __init__(self, args, command):
+ """
+ :type args: any
+ """
+ super(EnvironmentConfig, self).__init__(args)
+
+ self.command = command
+
+ self.local = args.local is True
+
+ if args.tox is True or args.tox is False or args.tox is None:
+ self.tox = args.tox is True
+ self.tox_args = 0
+ self.python = args.python if 'python' in args else None # type: str
+ else:
+ self.tox = True
+ self.tox_args = 1
+ self.python = args.tox # type: str
+
+ self.docker = docker_qualify_image(args.docker) # type: str
+ self.remote = args.remote # type: str
+
+ self.docker_privileged = args.docker_privileged if 'docker_privileged' in args else False # type: bool
+ self.docker_util = docker_qualify_image(args.docker_util if 'docker_util' in args else None) # type: str | None
+
+ self.remote_stage = args.remote_stage # type: str
+
+ self.requirements = args.requirements # type: bool
+
+ self.python_version = self.python or '.'.join(str(i) for i in sys.version_info[:2])
+
+ self.delegate = self.tox or self.docker or self.remote
+
+ if self.delegate:
+ self.requirements = True
+
+
+class TestConfig(EnvironmentConfig):
+ """Configuration common to all test commands."""
+ def __init__(self, args, command):
+ """
+ :type args: any
+ :type command: str
+ """
+ super(TestConfig, self).__init__(args, command)
+
+ self.coverage = args.coverage # type: bool
+ self.include = args.include # type: list [str]
+ self.exclude = args.exclude # type: list [str]
+ self.require = args.require # type: list [str]
+
+ self.changed = args.changed # type: bool
+ self.tracked = args.tracked # type: bool
+ self.untracked = args.untracked # type: bool
+ self.committed = args.committed # type: bool
+ self.staged = args.staged # type: bool
+ self.unstaged = args.unstaged # type: bool
+ self.changed_from = args.changed_from # type: str
+ self.changed_path = args.changed_path # type: list [str]
+
+
+class ShellConfig(EnvironmentConfig):
+ """Configuration for the shell command."""
+ def __init__(self, args):
+ """
+ :type args: any
+ """
+ super(ShellConfig, self).__init__(args, 'shell')
+
+
+class SanityConfig(TestConfig):
+ """Configuration for the sanity command."""
+ def __init__(self, args):
+ """
+ :type args: any
+ """
+ super(SanityConfig, self).__init__(args, 'sanity')
+
+ self.test = args.test # type: list [str]
+ self.skip_test = args.skip_test # type: list [str]
+ self.list_tests = args.list_tests # type: bool
+
+
+class IntegrationConfig(TestConfig):
+ """Configuration for the integration command."""
+ def __init__(self, args, command):
+ """
+ :type args: any
+ :type command: str
+ """
+ super(IntegrationConfig, self).__init__(args, command)
+
+ self.start_at = args.start_at # type: str
+ self.start_at_task = args.start_at_task # type: str
+ self.allow_destructive = args.allow_destructive if 'allow_destructive' in args else False # type: bool
+ self.retry_on_error = args.retry_on_error # type: bool
+
+
+class PosixIntegrationConfig(IntegrationConfig):
+ """Configuration for the posix integration command."""
+
+ def __init__(self, args):
+ """
+ :type args: any
+ """
+ super(PosixIntegrationConfig, self).__init__(args, 'integration')
+
+
+class WindowsIntegrationConfig(IntegrationConfig):
+ """Configuration for the windows integration command."""
+
+ def __init__(self, args):
+ """
+ :type args: any
+ """
+ super(WindowsIntegrationConfig, self).__init__(args, 'windows-integration')
+
+ self.windows = args.windows # type: list [str]
+
+
+class NetworkIntegrationConfig(IntegrationConfig):
+ """Configuration for the network integration command."""
+
+ def __init__(self, args):
+ """
+ :type args: any
+ """
+ super(NetworkIntegrationConfig, self).__init__(args, 'network-integration')
+
+
+class UnitsConfig(TestConfig):
+ """Configuration for the units command."""
+ def __init__(self, args):
+ """
+ :type args: any
+ """
+ super(UnitsConfig, self).__init__(args, 'units')
+
+ self.collect_only = args.collect_only # type: bool
+
+
+class CompileConfig(TestConfig):
+ """Configuration for the compile command."""
+ def __init__(self, args):
+ """
+ :type args: any
+ """
+ super(CompileConfig, self).__init__(args, 'compile')
+
+
+class Delegate(Exception):
+ """Trigger command delegation."""
+ def __init__(self, exclude=None, require=None):
+ """
+ :type exclude: list[str] | None
+ :type require: list[str] | None
+ """
+ super(Delegate, self).__init__()
+
+ self.exclude = exclude or []
+ self.require = require or []
+
+
+class AllTargetsSkipped(ApplicationWarning):
+ """All targets skipped."""
+ def __init__(self):
+ super(AllTargetsSkipped, self).__init__('All targets skipped.')
+
+
+SANITY_TESTS = (
+ # tests which ignore include/exclude (they're so fast it doesn't matter)
+ SanityFunc('code-smell', command_sanity_code_smell, intercept=False),
+ # tests which honor include/exclude
+ SanityFunc('shellcheck', command_sanity_shellcheck, intercept=False),
+ SanityFunc('yamllint', command_sanity_yamllint, intercept=False),
+ SanityFunc('validate-modules', command_sanity_validate_modules, intercept=False),
+ SanityFunc('ansible-doc', command_sanity_ansible_doc),
+)
diff --git a/test/runner/lib/git.py b/test/runner/lib/git.py
new file mode 100644
index 0000000000..b732a5dbe1
--- /dev/null
+++ b/test/runner/lib/git.py
@@ -0,0 +1,76 @@
+"""Wrapper around git command-line tools."""
+
+from __future__ import absolute_import, print_function
+
+from lib.util import (
+ CommonConfig,
+ run_command,
+)
+
+
+class Git(object):
+ """Wrapper around git command-line tools."""
+ def __init__(self, args):
+ """
+ :type args: CommonConfig
+ """
+ self.args = args
+ self.git = 'git'
+
+ def get_diff_names(self, args):
+ """
+ :type args: list[str]
+ :rtype: list[str]
+ """
+ cmd = ['diff', '--name-only', '--no-renames', '-z'] + args
+ return self.run_git_split(cmd, '\0')
+
+ def get_file_names(self, args):
+ """
+ :type args: list[str]
+ :rtype: list[str]
+ """
+ cmd = ['ls-files', '-z'] + args
+ return self.run_git_split(cmd, '\0')
+
+ def get_branches(self):
+ """
+ :rtype: list[str]
+ """
+ cmd = ['for-each-ref', 'refs/heads/', '--format', '%(refname:strip=2)']
+ return self.run_git_split(cmd)
+
+ def get_branch(self):
+ """
+ :rtype: str
+ """
+ cmd = ['symbolic-ref', '--short', 'HEAD']
+ return self.run_git(cmd).strip()
+
+ def get_branch_fork_point(self, branch):
+ """
+ :type branch: str
+ :rtype: str
+ """
+ cmd = ['merge-base', '--fork-point', branch]
+ return self.run_git(cmd).strip()
+
+ def run_git_split(self, cmd, separator=None):
+ """
+ :type cmd: list[str]
+ :param separator: str | None
+ :rtype: list[str]
+ """
+ output = self.run_git(cmd).strip(separator)
+
+ if len(output) == 0:
+ return []
+
+ return output.split(separator)
+
+ def run_git(self, cmd):
+ """
+ :type cmd: list[str]
+ :rtype: str
+ """
+ return run_command(self.args, [self.git] + cmd, capture=True, always=True)[0]
diff --git a/test/runner/lib/http.py b/test/runner/lib/http.py
new file mode 100644
index 0000000000..df82166efc
--- /dev/null
+++ b/test/runner/lib/http.py
@@ -0,0 +1,122 @@
+"""
+Primitive replacement for requests to avoid extra dependency.
+Avoids use of urllib2 due to lack of SNI support.
+"""
+
+from __future__ import absolute_import, print_function
+
+import json
+
+try:
+ from urllib import urlencode
+except ImportError:
+ # noinspection PyCompatibility,PyUnresolvedReferences,PyUnresolvedReferences
+ from urllib.parse import urlencode # pylint: disable=locally-disabled, import-error, no-name-in-module
+
+from lib.util import (
+ CommonConfig,
+ ApplicationError,
+ run_command,
+)
+
+
+class HttpClient(object):
+ """Make HTTP requests via curl."""
+ def __init__(self, args, always=False):
+ """
+ :type args: CommonConfig
+ :type always: bool
+ """
+ self.args = args
+ self.always = always
+
+ def get(self, url):
+ """
+ :type url: str
+ :rtype: HttpResponse
+ """
+ return self.request('GET', url)
+
+ def delete(self, url):
+ """
+ :type url: str
+ :rtype: HttpResponse
+ """
+ return self.request('DELETE', url)
+
+ def put(self, url, data=None, headers=None):
+ """
+ :type url: str
+ :type data: str | None
+ :type headers: dict[str, str] | None
+ :rtype: HttpResponse
+ """
+ return self.request('PUT', url, data, headers)
+
+ def request(self, method, url, data=None, headers=None):
+ """
+ :type method: str
+ :type url: str
+ :type data: str | None
+ :type headers: dict[str, str] | None
+ :rtype: HttpResponse
+ """
+ cmd = ['curl', '-s', '-S', '-i', '-X', method]
+
+ if headers is None:
+ headers = {}
+
+ headers['Expect'] = '' # don't send expect continue header
+
+ for header in headers.keys():
+ cmd += ['-H', '%s: %s' % (header, headers[header])]
+
+ if data is not None:
+ cmd += ['-d', data]
+
+ cmd += [url]
+
+ stdout, _ = run_command(self.args, cmd, capture=True, always=self.always)
+
+ if self.args.explain and not self.always:
+ return HttpResponse(200, '')
+
+ header, body = stdout.split('\r\n\r\n', 1)
+
+ response_headers = header.split('\r\n')
+ first_line = response_headers[0]
+ http_response = first_line.split(' ')
+ status_code = int(http_response[1])
+
+ return HttpResponse(status_code, body)
+
+
+class HttpResponse(object):
+ """HTTP response from curl."""
+ def __init__(self, status_code, response):
+ """
+ :type status_code: int
+ :type response: str
+ """
+ self.status_code = status_code
+ self.response = response
+
+ def json(self):
+ """
+ :rtype: any
+ """
+ try:
+ return json.loads(self.response)
+ except ValueError:
+ raise HttpError(self.status_code, 'Cannot parse response as JSON:\n%s' % self.response)
+
+
+class HttpError(ApplicationError):
+ """HTTP response as an error."""
+ def __init__(self, status, message):
+ """
+ :type status: int
+ :type message: str
+ """
+ super(HttpError, self).__init__('%s: %s' % (status, message))
+ self.status = status
diff --git a/test/runner/lib/manage_ci.py b/test/runner/lib/manage_ci.py
new file mode 100644
index 0000000000..85839cd644
--- /dev/null
+++ b/test/runner/lib/manage_ci.py
@@ -0,0 +1,142 @@
+"""Access Ansible Core CI remote services."""
+
+from __future__ import absolute_import, print_function
+
+import pipes
+
+from time import sleep
+
+import lib.pytar
+
+from lib.util import (
+ SubprocessError,
+ ApplicationError,
+ run_command,
+)
+
+from lib.core_ci import (
+ AnsibleCoreCI,
+)
+
+from lib.ansible_util import (
+ ansible_environment,
+)
+
+
+class ManageWindowsCI(object):
+ """Manage access to a Windows instance provided by Ansible Core CI."""
+ def __init__(self, core_ci):
+ """
+ :type core_ci: AnsibleCoreCI
+ """
+ self.core_ci = core_ci
+
+ def wait(self):
+ """Wait for instance to respond to ansible ping."""
+ extra_vars = [
+ 'ansible_connection=winrm',
+ 'ansible_host=%s' % self.core_ci.connection.hostname,
+ 'ansible_user=%s' % self.core_ci.connection.username,
+ 'ansible_password=%s' % self.core_ci.connection.password,
+ 'ansible_port=%s' % self.core_ci.connection.port,
+ 'ansible_winrm_server_cert_validation=ignore',
+ ]
+
+ name = 'windows_%s' % self.core_ci.version
+
+ env = ansible_environment(self.core_ci.args)
+ cmd = ['ansible', '-m', 'win_ping', '-i', '%s,' % name, name, '-e', ' '.join(extra_vars)]
+
+ for _ in range(1, 90):
+ try:
+ run_command(self.core_ci.args, cmd, env=env)
+ return
+ except SubprocessError:
+ sleep(10)
+ continue
+
+ raise ApplicationError('Timeout waiting for %s/%s instance %s.' %
+ (self.core_ci.platform, self.core_ci.version, self.core_ci.instance_id))
+
+
+class ManagePosixCI(object):
+ """Manage access to a POSIX instance provided by Ansible Core CI."""
+ def __init__(self, core_ci):
+ """
+ :type core_ci: AnsibleCoreCI
+ """
+ self.core_ci = core_ci
+ self.ssh_args = ['-o', 'BatchMode=yes', '-o', 'StrictHostKeyChecking=no', '-i', self.core_ci.ssh_key.key]
+
+ if self.core_ci.platform == 'freebsd':
+ self.become = ['su', '-l', 'root', '-c']
+ elif self.core_ci.platform == 'osx':
+ self.become = ['sudo', '-in', 'PATH=/usr/local/bin:$PATH']
+
+ def setup(self):
+ """Start instance and wait for it to become ready and respond to an ansible ping."""
+ self.wait()
+ self.configure()
+ self.upload_source()
+
+ def wait(self):
+ """Wait for instance to respond to SSH."""
+ for _ in range(1, 90):
+ try:
+ self.ssh('id')
+ return
+ except SubprocessError:
+ sleep(10)
+ continue
+
+ raise ApplicationError('Timeout waiting for %s/%s instance %s.' %
+ (self.core_ci.platform, self.core_ci.version, self.core_ci.instance_id))
+
+ def configure(self):
+ """Configure remote host for testing."""
+ self.upload('test/runner/setup/remote.sh', '/tmp')
+ self.ssh('chmod +x /tmp/remote.sh && /tmp/remote.sh %s' % self.core_ci.platform)
+
+ def upload_source(self):
+ """Upload and extract source."""
+ if not self.core_ci.args.explain:
+ lib.pytar.create_tarfile('/tmp/ansible.tgz', '.', lib.pytar.ignore)
+
+ self.upload('/tmp/ansible.tgz', '/tmp')
+ self.ssh('rm -rf ~/ansible && mkdir ~/ansible && cd ~/ansible && tar oxzf /tmp/ansible.tgz')
+
+ def download(self, remote, local):
+ """
+ :type remote: str
+ :type local: str
+ """
+ self.scp('%s@%s:%s' % (self.core_ci.connection.username, self.core_ci.connection.hostname, remote), local)
+
+ def upload(self, local, remote):
+ """
+ :type local: str
+ :type remote: str
+ """
+ self.scp(local, '%s@%s:%s' % (self.core_ci.connection.username, self.core_ci.connection.hostname, remote))
+
+ def ssh(self, command):
+ """
+ :type command: str | list[str]
+ """
+ if isinstance(command, list):
+ command = ' '.join(pipes.quote(c) for c in command)
+
+ run_command(self.core_ci.args,
+ ['ssh', '-tt', '-q'] + self.ssh_args +
+ ['-p', str(self.core_ci.connection.port),
+ '%s@%s' % (self.core_ci.connection.username, self.core_ci.connection.hostname)] +
+ self.become + [pipes.quote(command)])
+
+ def scp(self, src, dst):
+ """
+ :type src: str
+ :type dst: str
+ """
+ run_command(self.core_ci.args,
+ ['scp'] + self.ssh_args +
+ ['-P', str(self.core_ci.connection.port), '-q', '-r', src, dst])
diff --git a/test/runner/lib/pytar.py b/test/runner/lib/pytar.py
new file mode 100644
index 0000000000..2f10b18133
--- /dev/null
+++ b/test/runner/lib/pytar.py
@@ -0,0 +1,69 @@
+"""Python native TGZ creation."""
+
+from __future__ import absolute_import, print_function
+
+import tarfile
+import os
+
+# improve performance by disabling uid/gid lookups
+tarfile.pwd = None
+tarfile.grp = None
+
+# To reduce archive time and size, ignore non-versioned files which are large or numerous.
+# Also ignore miscellaneous git related files since the .git directory is ignored.
+
+IGNORE_DIRS = (
+ '.tox',
+ '.git',
+ '.idea',
+ '__pycache__',
+ 'ansible.egg-info',
+)
+
+IGNORE_FILES = (
+ '.gitignore',
+ '.gitdir',
+)
+
+IGNORE_EXTENSIONS = (
+ '.pyc',
+ '.retry',
+)
+
+
+def ignore(item):
+ """
+ :type item: tarfile.TarInfo
+ :rtype: tarfile.TarInfo | None
+ """
+ filename = os.path.basename(item.path)
+ name, ext = os.path.splitext(filename)
+ dirs = os.path.split(item.path)
+
+ if not item.isdir():
+ if item.path.startswith('./test/results/'):
+ return None
+
+ if item.path.startswith('./docsite/') and filename.endswith('_module.rst'):
+ return None
+
+ if name in IGNORE_FILES:
+ return None
+
+ if ext in IGNORE_EXTENSIONS:
+ return None
+
+ if any(d in IGNORE_DIRS for d in dirs):
+ return None
+
+ return item
+
+
+def create_tarfile(dst_path, src_path, tar_filter):
+ """
+ :type dst_path: str
+ :type src_path: str
+ :type tar_filter: (tarfile.TarInfo) -> tarfile.TarInfo | None
+ """
+ with tarfile.TarFile.gzopen(dst_path, mode='w', compresslevel=4) as tar:
+ tar.add(src_path, filter=tar_filter)
diff --git a/test/runner/lib/target.py b/test/runner/lib/target.py
new file mode 100644
index 0000000000..1ea925e5cc
--- /dev/null
+++ b/test/runner/lib/target.py
@@ -0,0 +1,530 @@
+"""Test target identification, iteration and inclusion/exclusion."""
+
+from __future__ import absolute_import, print_function
+
+import os
+import re
+import errno
+import itertools
+import abc
+
+from lib.util import ApplicationError
+
+MODULE_EXTENSIONS = '.py', '.ps1'
+
+
+def find_target_completion(target_func, prefix):
+ """
+ :type target_func: () -> collections.Iterable[CompletionTarget]
+ :type prefix: unicode
+ :rtype: list[str]
+ """
+ try:
+ targets = target_func()
+ prefix = prefix.encode()
+ short = os.environ.get('COMP_TYPE') == '63' # double tab completion from bash
+ matches = walk_completion_targets(targets, prefix, short)
+ return matches
+ except Exception as ex: # pylint: disable=locally-disabled, broad-except
+ return [str(ex)]
+
+
+def walk_completion_targets(targets, prefix, short=False):
+ """
+ :type targets: collections.Iterable[CompletionTarget]
+ :type prefix: str
+ :type short: bool
+ :rtype: tuple[str]
+ """
+ aliases = set(alias for target in targets for alias in target.aliases)
+
+ if prefix.endswith('/') and prefix in aliases:
+ aliases.remove(prefix)
+
+ matches = [alias for alias in aliases if alias.startswith(prefix) and '/' not in alias[len(prefix):-1]]
+
+ if short:
+ offset = len(os.path.dirname(prefix))
+ if offset:
+ offset += 1
+ relative_matches = [match[offset:] for match in matches if len(match) > offset]
+ if len(relative_matches) > 1:
+ matches = relative_matches
+
+ return tuple(sorted(matches))
+
+
+def walk_internal_targets(targets, includes=None, excludes=None, requires=None):
+ """
+ :type targets: collections.Iterable[T <= CompletionTarget]
+ :type includes: list[str]
+ :type excludes: list[str]
+ :type requires: list[str]
+ :rtype: tuple[T <= CompletionTarget]
+ """
+ targets = tuple(targets)
+
+ include_targets = sorted(filter_targets(targets, includes, errors=True, directories=False), key=lambda t: t.name)
+
+ if requires:
+ require_targets = set(filter_targets(targets, requires, errors=True, directories=False))
+ include_targets = [target for target in include_targets if target in require_targets]
+
+ if excludes:
+ list(filter_targets(targets, excludes, errors=True, include=False, directories=False))
+
+ internal_targets = set(filter_targets(include_targets, excludes, errors=False, include=False, directories=False))
+ return tuple(sorted(internal_targets, key=lambda t: t.name))
+
+
+def walk_external_targets(targets, includes=None, excludes=None, requires=None):
+ """
+ :type targets: collections.Iterable[CompletionTarget]
+ :type includes: list[str]
+ :type excludes: list[str]
+ :type requires: list[str]
+ :rtype: tuple[CompletionTarget], tuple[CompletionTarget]
+ """
+ targets = tuple(targets)
+
+ if requires:
+ include_targets = list(filter_targets(targets, includes, errors=True, directories=False))
+ require_targets = set(filter_targets(targets, requires, errors=True, directories=False))
+ includes = [target.name for target in include_targets if target in require_targets]
+
+ if includes:
+ include_targets = sorted(filter_targets(targets, includes, errors=True), key=lambda t: t.name)
+ else:
+ include_targets = []
+ else:
+ include_targets = sorted(filter_targets(targets, includes, errors=True), key=lambda t: t.name)
+
+ if excludes:
+ exclude_targets = sorted(filter_targets(targets, excludes, errors=True), key=lambda t: t.name)
+ else:
+ exclude_targets = []
+
+ previous = None
+ include = []
+ for target in include_targets:
+ if isinstance(previous, DirectoryTarget) and isinstance(target, DirectoryTarget) \
+ and previous.name == target.name:
+ previous.modules = tuple(set(previous.modules) | set(target.modules))
+ else:
+ include.append(target)
+ previous = target
+
+ previous = None
+ exclude = []
+ for target in exclude_targets:
+ if isinstance(previous, DirectoryTarget) and isinstance(target, DirectoryTarget) \
+ and previous.name == target.name:
+ previous.modules = tuple(set(previous.modules) | set(target.modules))
+ else:
+ exclude.append(target)
+ previous = target
+
+ return tuple(include), tuple(exclude)
+
+
+def filter_targets(targets, patterns, include=True, directories=True, errors=True):
+ """
+ :type targets: collections.Iterable[CompletionTarget]
+ :type patterns: list[str]
+ :type include: bool
+ :type directories: bool
+ :type errors: bool
+ :rtype: collections.Iterable[CompletionTarget]
+ """
+ unmatched = set(patterns or ())
+
+ for target in targets:
+ matched_directories = set()
+ match = False
+
+ if patterns:
+ for alias in target.aliases:
+ for pattern in patterns:
+ if re.match('^%s$' % pattern, alias):
+ match = True
+
+ try:
+ unmatched.remove(pattern)
+ except KeyError:
+ pass
+
+ if alias.endswith('/'):
+ if target.base_path and len(target.base_path) > len(alias):
+ matched_directories.add(target.base_path)
+ else:
+ matched_directories.add(alias)
+ elif include:
+ match = True
+ if not target.base_path:
+ matched_directories.add('.')
+ for alias in target.aliases:
+ if alias.endswith('/'):
+ if target.base_path and len(target.base_path) > len(alias):
+ matched_directories.add(target.base_path)
+ else:
+ matched_directories.add(alias)
+
+ if match != include:
+ continue
+
+ if directories and matched_directories:
+ yield DirectoryTarget(sorted(matched_directories, key=len)[0], target.modules)
+ else:
+ yield target
+
+ if errors:
+ if unmatched:
+ raise TargetPatternsNotMatched(unmatched)
+
+
+def walk_module_targets():
+ """
+ :rtype: collections.Iterable[TestTarget]
+ """
+ path = 'lib/ansible/modules'
+
+ for target in walk_test_targets(path, path + '/', extensions=MODULE_EXTENSIONS):
+ if not target.module:
+ continue
+
+ yield target
+
+
+def walk_units_targets():
+ """
+ :rtype: collections.Iterable[TestTarget]
+ """
+ return walk_test_targets(path='test/units', module_path='test/units/modules/', extensions=('.py',), prefix='test_')
+
+
+def walk_compile_targets():
+ """
+ :rtype: collections.Iterable[TestTarget]
+ """
+ return walk_test_targets(module_path='lib/ansible/modules/', extensions=('.py',))
+
+
+def walk_sanity_targets():
+ """
+ :rtype: collections.Iterable[TestTarget]
+ """
+ return walk_test_targets(module_path='lib/ansible/modules/')
+
+
+def walk_posix_integration_targets():
+ """
+ :rtype: collections.Iterable[IntegrationTarget]
+ """
+ for target in walk_integration_targets():
+ if 'posix/' in target.aliases:
+ yield target
+
+
+def walk_network_integration_targets():
+ """
+ :rtype: collections.Iterable[IntegrationTarget]
+ """
+ for target in walk_integration_targets():
+ if 'network/' in target.aliases:
+ yield target
+
+
+def walk_windows_integration_targets():
+ """
+ :rtype: collections.Iterable[IntegrationTarget]
+ """
+ for target in walk_integration_targets():
+ if 'windows/' in target.aliases:
+ yield target
+
+
+def walk_integration_targets():
+ """
+ :rtype: collections.Iterable[IntegrationTarget]
+ """
+ path = 'test/integration/targets'
+ modules = frozenset(t.module for t in walk_module_targets())
+ paths = sorted(os.path.join(path, p) for p in os.listdir(path))
+ prefixes = load_integration_prefixes()
+
+ for path in paths:
+ yield IntegrationTarget(path, modules, prefixes)
+
+
+def load_integration_prefixes():
+ """
+ :rtype: dict[str, str]
+ """
+ path = 'test/integration'
+ names = sorted(f for f in os.listdir(path) if os.path.splitext(f)[0] == 'target-prefixes')
+ prefixes = {}
+
+ for name in names:
+ prefix = os.path.splitext(name)[1][1:]
+ with open(os.path.join(path, name), 'r') as prefix_fd:
+ prefixes.update(dict((k, prefix) for k in prefix_fd.read().splitlines()))
+
+ return prefixes
+
+
+def walk_test_targets(path=None, module_path=None, extensions=None, prefix=None):
+ """
+ :type path: str | None
+ :type module_path: str | None
+ :type extensions: tuple[str] | None
+ :type prefix: str | None
+ :rtype: collections.Iterable[TestTarget]
+ """
+ for root, _, file_names in os.walk(path or '.', topdown=False):
+ if root.endswith('/__pycache__'):
+ continue
+
+ if path is None:
+ root = root[2:]
+
+ if root.startswith('.'):
+ continue
+
+ for file_name in file_names:
+ name, ext = os.path.splitext(os.path.basename(file_name))
+
+ if name.startswith('.'):
+ continue
+
+ if extensions and ext not in extensions:
+ continue
+
+ if prefix and not name.startswith(prefix):
+ continue
+
+ yield TestTarget(os.path.join(root, file_name), module_path, prefix, path)
+
+
+class CompletionTarget(object):
+ """Command-line argument completion target base class."""
+ __metaclass__ = abc.ABCMeta
+
+ def __init__(self):
+ self.name = None
+ self.path = None
+ self.base_path = None
+ self.modules = tuple()
+ self.aliases = tuple()
+
+ def __eq__(self, other):
+ if isinstance(other, CompletionTarget):
+ return self.__repr__() == other.__repr__()
+ else:
+ return False
+
+ def __ne__(self, other):
+ return not self.__eq__(other)
+
+ def __lt__(self, other):
+ return self.name.__lt__(other.name)
+
+ def __gt__(self, other):
+ return self.name.__gt__(other.name)
+
+ def __hash__(self):
+ return hash(self.__repr__())
+
+ def __repr__(self):
+ if self.modules:
+ return '%s (%s)' % (self.name, ', '.join(self.modules))
+
+ return self.name
+
+
+class DirectoryTarget(CompletionTarget):
+ """Directory target."""
+ def __init__(self, path, modules):
+ """
+ :type path: str
+ :type modules: tuple[str]
+ """
+ super(DirectoryTarget, self).__init__()
+
+ self.name = path
+ self.path = path
+ self.modules = modules
+
+
+class TestTarget(CompletionTarget):
+ """Generic test target."""
+ def __init__(self, path, module_path, module_prefix, base_path):
+ """
+ :type path: str
+ :type module_path: str | None
+ :type module_prefix: str | None
+ :type base_path: str
+ """
+ super(TestTarget, self).__init__()
+
+ self.name = path
+ self.path = path
+ self.base_path = base_path + '/' if base_path else None
+
+ name, ext = os.path.splitext(os.path.basename(self.path))
+
+ if module_path and path.startswith(module_path) and name != '__init__' and ext in MODULE_EXTENSIONS:
+ self.module = name[len(module_prefix or ''):].lstrip('_')
+ self.modules = self.module,
+ else:
+ self.module = None
+ self.modules = tuple()
+
+ aliases = [self.path, self.module]
+ parts = self.path.split('/')
+
+ for i in range(1, len(parts)):
+ alias = '%s/' % '/'.join(parts[:i])
+ aliases.append(alias)
+
+ aliases = [a for a in aliases if a]
+
+ self.aliases = tuple(sorted(aliases))
+
+
+class IntegrationTarget(CompletionTarget):
+ """Integration test target."""
+ non_posix = frozenset((
+ 'network',
+ 'windows',
+ ))
+
+ categories = frozenset(non_posix | frozenset((
+ 'posix',
+ 'module',
+ 'needs',
+ 'skip',
+ )))
+
+ def __init__(self, path, modules, prefixes):
+ """
+ :type path: str
+ :type modules: frozenset[str]
+ :type prefixes: dict[str, str]
+ """
+ super(IntegrationTarget, self).__init__()
+
+ self.name = os.path.basename(path)
+ self.path = path
+
+ # script_path and type
+
+ contents = sorted(os.listdir(path))
+
+ runme_files = tuple(c for c in contents if os.path.splitext(c)[0] == 'runme')
+ test_files = tuple(c for c in contents if os.path.splitext(c)[0] == 'test')
+
+ self.script_path = None
+
+ if runme_files:
+ self.type = 'script'
+ self.script_path = os.path.join(path, runme_files[0])
+ elif test_files:
+ self.type = 'special'
+ elif os.path.isdir(os.path.join(path, 'tasks')):
+ self.type = 'role'
+ else:
+ self.type = 'unknown'
+
+ # static_aliases
+
+ try:
+ with open(os.path.join(path, 'aliases'), 'r') as aliases_file:
+ static_aliases = tuple(aliases_file.read().splitlines())
+ except IOError as ex:
+ if ex.errno != errno.ENOENT:
+ raise
+ static_aliases = tuple()
+
+ # modules
+
+ if self.name in modules:
+ module = self.name
+ elif self.name.startswith('win_') and self.name[4:] in modules:
+ module = self.name[4:]
+ else:
+ module = None
+
+ self.modules = tuple(sorted(a for a in static_aliases + tuple([module]) if a in modules))
+
+ # groups
+
+ groups = [self.type]
+ groups += [a for a in static_aliases if a not in modules]
+ groups += ['module/%s' % m for m in self.modules]
+
+ if not self.modules:
+ groups.append('non_module')
+
+ if 'destructive' not in groups:
+ groups.append('non_destructive')
+
+ if '_' in self.name:
+ prefix = self.name[:self.name.find('_')]
+ else:
+ prefix = None
+
+ if prefix in prefixes:
+ group = prefixes[prefix]
+
+ if group != prefix:
+ group = '%s/%s' % (group, prefix)
+
+ groups.append(group)
+
+ if self.name.startswith('win_'):
+ groups.append('windows')
+
+ if self.name.startswith('connection_'):
+ groups.append('connection')
+
+ if self.name.startswith('setup_') or self.name.startswith('prepare_'):
+ groups.append('hidden')
+
+ if self.type not in ('script', 'role'):
+ groups.append('hidden')
+
+ for group in itertools.islice(groups, 0, len(groups)):
+ if '/' in group:
+ parts = group.split('/')
+ for i in range(1, len(parts)):
+ groups.append('/'.join(parts[:i]))
+
+ if not any(g in self.non_posix for g in groups):
+ groups.append('posix')
+
+ # aliases
+
+ aliases = [self.name] + \
+ ['%s/' % g for g in groups] + \
+ ['%s/%s' % (g, self.name) for g in groups if g not in self.categories]
+
+ if 'hidden/' in aliases:
+ aliases = ['hidden/'] + ['hidden/%s' % a for a in aliases if not a.startswith('hidden/')]
+
+ self.aliases = tuple(sorted(set(aliases)))
+
+
+class TargetPatternsNotMatched(ApplicationError):
+ """One or more targets were not matched when a match was required."""
+ def __init__(self, patterns):
+ """
+ :type patterns: set[str]
+ """
+ self.patterns = sorted(patterns)
+
+ if len(patterns) > 1:
+ message = 'Target patterns not matched:\n%s' % '\n'.join(self.patterns)
+ else:
+ message = 'Target pattern not matched: %s' % self.patterns[0]
+
+ super(TargetPatternsNotMatched, self).__init__(message)
diff --git a/test/runner/lib/thread.py b/test/runner/lib/thread.py
new file mode 100644
index 0000000000..cf7d6a36c4
--- /dev/null
+++ b/test/runner/lib/thread.py
@@ -0,0 +1,48 @@
+"""Python threading tools."""
+
+from __future__ import absolute_import, print_function
+
+import threading
+import sys
+
+try:
+ # noinspection PyPep8Naming
+ import Queue as queue
+except ImportError:
+ # noinspection PyUnresolvedReferences
+ import queue # pylint: disable=locally-disabled, import-error
+
+
+class WrappedThread(threading.Thread):
+ """Wrapper around Thread which captures results and exceptions."""
+ def __init__(self, action):
+ """
+ :type action: () -> any
+ """
+ super(WrappedThread, self).__init__()
+ self._result = queue.Queue()
+ self.action = action
+
+ def run(self):
+ """
+ Run action and capture results or exception.
+ Do not override. Do not call directly. Executed by the start() method.
+ """
+ # noinspection PyBroadException
+ try:
+ self._result.put((self.action(), None))
+ except: # pylint: disable=locally-disabled, bare-except
+ self._result.put((None, sys.exc_info()))
+
+ def wait_for_result(self):
+ """
+ Wait for thread to exit and return the result or raise an exception.
+ :rtype: any
+ """
+ result, exception = self._result.get()
+ if exception:
+ if sys.version_info[0] > 2:
+ raise exception[0](exception[1]).with_traceback(exception[2])
+ # noinspection PyRedundantParentheses
+ exec('raise exception[0], exception[1], exception[2]') # pylint: disable=locally-disabled, exec-used
+ return result
diff --git a/test/runner/lib/util.py b/test/runner/lib/util.py
new file mode 100644
index 0000000000..f7544c4b19
--- /dev/null
+++ b/test/runner/lib/util.py
@@ -0,0 +1,415 @@
+"""Miscellaneous utility functions and classes."""
+
+from __future__ import absolute_import, print_function
+
+import errno
+import os
+import pipes
+import shutil
+import subprocess
+import sys
+import time
+
+
+def is_shippable():
+ """
+ :rtype: bool
+ """
+ return os.environ.get('SHIPPABLE') == 'true'
+
+
+def remove_file(path):
+ """
+ :type path: str
+ """
+ if os.path.isfile(path):
+ os.remove(path)
+
+
+def find_executable(executable, cwd=None, path=None, required=True):
+ """
+ :type executable: str
+ :type cwd: str
+ :type path: str
+ :type required: bool | str
+ :rtype: str | None
+ """
+ match = None
+ real_cwd = os.getcwd()
+
+ if not cwd:
+ cwd = real_cwd
+
+ if os.path.dirname(executable):
+ target = os.path.join(cwd, executable)
+ if os.path.exists(target) and os.access(target, os.F_OK | os.X_OK):
+ match = executable
+ else:
+ if path is None:
+ path = os.environ.get('PATH', os.defpath)
+
+ if path:
+ path_dirs = path.split(os.pathsep)
+ seen_dirs = set()
+
+ for path_dir in path_dirs:
+ if path_dir in seen_dirs:
+ continue
+
+ seen_dirs.add(path_dir)
+
+ if os.path.abspath(path_dir) == real_cwd:
+ path_dir = cwd
+
+ candidate = os.path.join(path_dir, executable)
+
+ if os.path.exists(candidate) and os.access(candidate, os.F_OK | os.X_OK):
+ match = candidate
+ break
+
+ if not match and required:
+ message = 'Required program "%s" not found.' % executable
+
+ if required != 'warning':
+ raise ApplicationError(message)
+
+ display.warning(message)
+
+ return match
+
+
+def run_command(args, cmd, capture=False, env=None, data=None, cwd=None, always=False, stdin=None, stdout=None):
+ """
+ :type args: CommonConfig
+ :type cmd: collections.Iterable[str]
+ :type capture: bool
+ :type env: dict[str, str] | None
+ :type data: str | None
+ :type cwd: str | None
+ :type always: bool
+ :type stdin: file | None
+ :type stdout: file | None
+ :rtype: str | None, str | None
+ """
+ explain = args.explain and not always
+ return raw_command(cmd, capture=capture, env=env, data=data, cwd=cwd, explain=explain, stdin=stdin, stdout=stdout)
+
+
+def raw_command(cmd, capture=False, env=None, data=None, cwd=None, explain=False, stdin=None, stdout=None):
+ """
+ :type cmd: collections.Iterable[str]
+ :type capture: bool
+ :type env: dict[str, str] | None
+ :type data: str | None
+ :type cwd: str | None
+ :type explain: bool
+ :type stdin: file | None
+ :type stdout: file | None
+ :rtype: str | None, str | None
+ """
+ if not cwd:
+ cwd = os.getcwd()
+
+ if not env:
+ env = common_environment()
+
+ cmd = list(cmd)
+
+ escaped_cmd = ' '.join(pipes.quote(c) for c in cmd)
+
+ display.info('Run command: %s' % escaped_cmd, verbosity=1)
+ display.info('Working directory: %s' % cwd, verbosity=2)
+
+ program = find_executable(cmd[0], cwd=cwd, path=env['PATH'], required='warning')
+
+ if program:
+ display.info('Program found: %s' % program, verbosity=2)
+
+ for key in sorted(env.keys()):
+ display.info('%s=%s' % (key, env[key]), verbosity=2)
+
+ if explain:
+ return None, None
+
+ communicate = False
+
+ if stdin is not None:
+ data = None
+ communicate = True
+ elif data is not None:
+ stdin = subprocess.PIPE
+ communicate = True
+
+ if stdout:
+ communicate = True
+
+ if capture:
+ stdout = stdout or subprocess.PIPE
+ stderr = subprocess.PIPE
+ communicate = True
+ else:
+ stderr = None
+
+ start = time.time()
+
+ try:
+ process = subprocess.Popen(cmd, env=env, stdin=stdin, stdout=stdout, stderr=stderr, cwd=cwd)
+ except OSError as ex:
+ if ex.errno == errno.ENOENT:
+ raise ApplicationError('Required program "%s" not found.' % cmd[0])
+ raise
+
+ if communicate:
+ stdout, stderr = process.communicate(data)
+ else:
+ process.wait()
+ stdout, stderr = None, None
+
+ status = process.returncode
+ runtime = time.time() - start
+
+ display.info('Command exited with status %s after %s seconds.' % (status, runtime), verbosity=4)
+
+ if status == 0:
+ return stdout, stderr
+
+ raise SubprocessError(cmd, status, stdout, stderr, runtime)
+
+
+def common_environment():
+ """Common environment used for executing all programs."""
+ env = dict(
+ LC_ALL='en_US.UTF-8',
+ PATH=os.environ.get('PATH', os.defpath),
+ )
+
+ required = (
+ 'HOME',
+ )
+
+ optional = (
+ 'HTTPTESTER',
+ )
+
+ env.update(pass_vars(required=required, optional=optional))
+
+ return env
+
+
+def pass_vars(required=None, optional=None):
+ """
+ :type required: collections.Iterable[str]
+ :type optional: collections.Iterable[str]
+ :rtype: dict[str, str]
+ """
+ env = {}
+
+ for name in required:
+ if name not in os.environ:
+ raise MissingEnvironmentVariable(name)
+ env[name] = os.environ[name]
+
+ for name in optional:
+ if name not in os.environ:
+ continue
+ env[name] = os.environ[name]
+
+ return env
+
+
+def deepest_path(path_a, path_b):
+ """Return the deepest of two paths, or None if the paths are unrelated.
+ :type path_a: str
+ :type path_b: str
+ :return: str | None
+ """
+ if path_a == '.':
+ path_a = ''
+
+ if path_b == '.':
+ path_b = ''
+
+ if path_a.startswith(path_b):
+ return path_a or '.'
+
+ if path_b.startswith(path_a):
+ return path_b or '.'
+
+ return None
+
+
+def remove_tree(path):
+ """
+ :type path: str
+ """
+ try:
+ shutil.rmtree(path)
+ except OSError as ex:
+ if ex.errno != errno.ENOENT:
+ raise
+
+
+def make_dirs(path):
+ """
+ :type path: str
+ """
+ try:
+ os.makedirs(path)
+ except OSError as ex:
+ if ex.errno != errno.EEXIST:
+ raise
+
+
+class Display(object):
+ """Manages color console output."""
+ clear = '\033[0m'
+ red = '\033[31m'
+ green = '\033[32m'
+ yellow = '\033[33m'
+ blue = '\033[34m'
+ purple = '\033[35m'
+ cyan = '\033[36m'
+
+ verbosity_colors = {
+ 0: None,
+ 1: green,
+ 2: blue,
+ 3: cyan,
+ }
+
+ def __init__(self):
+ self.verbosity = 0
+ self.color = True
+ self.warnings = []
+
+ def __warning(self, message):
+ """
+ :type message: str
+ """
+ self.print_message('WARNING: %s' % message, color=self.purple, fd=sys.stderr)
+
+ def review_warnings(self):
+ """Review all warnings which previously occurred."""
+ if not self.warnings:
+ return
+
+ self.__warning('Reviewing previous %d warning(s):' % len(self.warnings))
+
+ for warning in self.warnings:
+ self.__warning(warning)
+
+ def warning(self, message):
+ """
+ :type message: str
+ """
+ self.__warning(message)
+ self.warnings.append(message)
+
+ def notice(self, message):
+ """
+ :type message: str
+ """
+ self.print_message('NOTICE: %s' % message, color=self.purple, fd=sys.stderr)
+
+ def error(self, message):
+ """
+ :type message: str
+ """
+ self.print_message('ERROR: %s' % message, color=self.red, fd=sys.stderr)
+
+ def info(self, message, verbosity=0):
+ """
+ :type message: str
+ :type verbosity: int
+ """
+ if self.verbosity >= verbosity:
+ color = self.verbosity_colors.get(verbosity, self.yellow)
+ self.print_message(message, color=color)
+
+ def print_message(self, message, color=None, fd=sys.stdout): # pylint: disable=locally-disabled, invalid-name
+ """
+ :type message: str
+ :type color: str | None
+ :type fd: file
+ """
+ if color and self.color:
+ # convert color resets in message to desired color
+ message = message.replace(self.clear, color)
+ message = '%s%s%s' % (color, message, self.clear)
+
+ print(message, file=fd)
+ fd.flush()
+
+
+class ApplicationError(Exception):
+ """General application error."""
+ def __init__(self, message=None):
+ """
+ :type message: str | None
+ """
+ super(ApplicationError, self).__init__(message)
+
+
+class ApplicationWarning(Exception):
+ """General application warning which interrupts normal program flow."""
+ def __init__(self, message=None):
+ """
+ :type message: str | None
+ """
+ super(ApplicationWarning, self).__init__(message)
+
+
+class SubprocessError(ApplicationError):
+ """Error resulting from failed subprocess execution."""
+ def __init__(self, cmd, status=0, stdout=None, stderr=None, runtime=None):
+ """
+ :type cmd: list[str]
+ :type status: int
+ :type stdout: str | None
+ :type stderr: str | None
+ :type runtime: float | None
+ """
+ message = 'Command "%s" returned exit status %s.\n' % (' '.join(pipes.quote(c) for c in cmd), status)
+
+ if stderr:
+ message += '>>> Standard Error\n'
+ message += '%s%s\n' % (stderr.strip(), Display.clear)
+
+ if stdout:
+ message += '>>> Standard Output\n'
+ message += '%s%s\n' % (stdout.strip(), Display.clear)
+
+ message = message.strip()
+
+ super(SubprocessError, self).__init__(message)
+
+ self.cmd = cmd
+ self.status = status
+ self.stdout = stdout
+ self.stderr = stderr
+ self.runtime = runtime
+
+
+class MissingEnvironmentVariable(ApplicationError):
+ """Error caused by missing environment variable."""
+ def __init__(self, name):
+ """
+ :type name: str
+ """
+ super(MissingEnvironmentVariable, self).__init__('Missing environment variable: %s' % name)
+
+ self.name = name
+
+
+class CommonConfig(object):
+ """Configuration common to all commands."""
+ def __init__(self, args):
+ """
+ :type args: any
+ """
+ self.color = args.color # type: bool
+ self.explain = args.explain # type: bool
+ self.verbosity = args.verbosity # type: int
+
+
+display = Display() # pylint: disable=locally-disabled, invalid-name
diff --git a/test/runner/reorganize-tests.sh b/test/runner/reorganize-tests.sh
new file mode 100755
index 0000000000..1d6daf8986
--- /dev/null
+++ b/test/runner/reorganize-tests.sh
@@ -0,0 +1,235 @@
+#!/usr/bin/env bash
+
+set -eu
+
+source_root=$(python -c "from os import path; print(path.abspath(path.join(path.dirname('$0'), '..', '..')))")
+
+cd "${source_root}"
+
+# Convert existing compile skip files to match the unified repository layout.
+
+mkdir -p test/compile
+
+rm -f test/compile/*.txt
+
+for type in core extras; do
+ sed "s|^|/lib/ansible/modules/${type}|" \
+ < "lib/ansible/modules/${type}/test/utils/shippable/sanity-skip-python24.txt" \
+ >> "test/compile/python2.4-skip.txt"
+done
+
+# Existing skip files are only for modules.
+# Add missing skip entries for core code.
+
+cat << EOF >> test/compile/python2.4-skip.txt
+/lib/ansible/modules/__init__.py
+/lib/ansible/module_utils/a10.py
+/lib/ansible/module_utils/rax.py
+/lib/ansible/module_utils/openstack.py
+/lib/ansible/module_utils/cloud.py
+/lib/ansible/module_utils/ec2.py
+/lib/ansible/module_utils/gce.py
+/lib/ansible/module_utils/lxd.py
+/lib/ansible/module_utils/docker_common.py
+/lib/ansible/module_utils/azure_rm_common.py
+/lib/ansible/module_utils/vca.py
+/lib/ansible/module_utils/vmware.py
+/lib/ansible/module_utils/gcp.py
+/lib/ansible/module_utils/gcdns.py
+/lib/ansible/vars/
+/lib/ansible/utils/
+/lib/ansible/template/
+/lib/ansible/plugins/
+/lib/ansible/playbook/
+/lib/ansible/parsing/
+/lib/ansible/inventory/
+/lib/ansible/galaxy/
+/lib/ansible/executor/
+/lib/ansible/errors/
+/lib/ansible/compat/
+/lib/ansible/config/
+/lib/ansible/cli/
+/lib/ansible/constants.py
+/lib/ansible/release.py
+/lib/ansible/__init__.py
+/hacking/
+/contrib/
+/docsite/
+/test/
+EOF
+
+cat << EOF >> test/compile/python2.6-skip.txt
+/contrib/inventory/vagrant.py
+/hacking/dump_playbook_attributes.py
+EOF
+
+cat << EOF >> test/compile/python3.5-skip.txt
+/test/samples/multi.py
+/examples/scripts/uptime.py
+EOF
+
+for path in test/compile/*.txt; do
+ sort -o "${path}" "${path}"
+done
+
+# Not all scripts pass shellcheck yet.
+
+mkdir -p test/sanity/shellcheck
+
+cat << EOF > test/sanity/shellcheck/skip.txt
+test/sanity/code-smell/boilerplate.sh
+EOF
+
+sort -o test/sanity/shellcheck/skip.txt test/sanity/shellcheck/skip.txt
+
+# Add skip list for code-smell scripts.
+# These scripts don't pass, so we can't run them in CI.
+
+cat << EOF > test/sanity/code-smell/skip.txt
+inappropriately-private.sh
+EOF
+
+# Add skip list for validate-modules.
+# Some of these exclusions are temporary, others belong in validate-modules.
+cat << EOF > test/sanity/validate-modules/skip.txt
+lib/ansible/modules/core/utilities/logic/async_status.py
+lib/ansible/modules/core/utilities/helper/_fireball.py
+lib/ansible/modules/core/utilities/helper/_accelerate.py
+lib/ansible/modules/core/test
+lib/ansible/modules/core/.github
+lib/ansible/modules/extras/test
+lib/ansible/modules/extras/.github
+EOF
+
+# Remove existing aliases from previous script runs.
+rm -f test/integration/targets/*/aliases
+
+# Map destructive/ targets to integration tests.
+targets=$(grep 'role:' "test/integration/destructive.yml" \
+ | sed 's/^.* role: //; s/[ ,].*$//;')
+
+for target in ${targets}; do
+ alias='destructive'
+ echo "target: ${target}, alias: ${alias}"
+ echo "${alias}" >> "test/integration/targets/${target}/aliases"
+done
+
+# Map destructive/non_destructive targets to posix groups for integration tests.
+# This will allow re-balancing of posix tests on Shippable independently of destructive/non_destructive targets.
+for type in destructive non_destructive; do
+ targets=$(grep 'role:' "test/integration/${type}.yml" \
+ | sed 's/^.* role: //; s/[ ,].*$//;')
+
+ if [ "${type}" = "destructive" ]; then
+ group="posix/ci/group1"
+ else
+ group="posix/ci/group2"
+ fi
+
+ for target in ${targets}; do
+ echo "target: ${target}, group: ${group}"
+ echo "${group}" >> "test/integration/targets/${target}/aliases"
+ done
+done
+
+# Add aliases to integration tests.
+targets=$(grep 'role:' test/integration/{destructive,non_destructive}.yml \
+ | sed 's/^.* role: //; s/[ ,].*$//;')
+
+for target in ${targets}; do
+ aliases=$(grep -h "role: *${target}[ ,]" test/integration/{destructive,non_destructive}.yml \
+ | sed 's/when:[^,]*//;' \
+ | sed 's/^.*tags:[ []*//g; s/[]}].*$//g; s/ //g; s/,/ /g; s/test_//g;')
+
+ for alias in ${aliases}; do
+ if [ "${target}" != "${alias}" ]; then
+ # convert needs_ prefixed aliases to groups
+ alias="${alias//needs_/needs\/}"
+
+ echo "target: ${target}, alias: ${alias}"
+ echo "${alias}" >> "test/integration/targets/${target}/aliases"
+ fi
+ done
+done
+
+# Map test_win_group* targets to windows groups for integration tests.
+for type in test_win_group1 test_win_group2 test_win_group3; do
+ targets=$(grep 'role:' "test/integration/${type}.yml" \
+ | sed 's/^.* role: //; s/[ ,].*$//;')
+
+ group=$(echo "${type}" | sed 's/^test_win_/windows_/; s/_/\/ci\//;')
+
+ for target in ${targets}; do
+ echo "target: ${target}, group: ${group}"
+ echo "${group}" >> "test/integration/targets/${target}/aliases"
+ done
+done
+
+# Add additional windows tests to appropriate groups.
+echo 'windows/ci/group2' >> test/integration/targets/binary_modules_winrm/aliases
+echo 'windows/ci/group3' >> test/integration/targets/connection_winrm/aliases
+
+# Add posix/ci/group3 for posix tests which are not already grouped for ci.
+group="posix/ci/group3"
+for target in test/integration/targets/*; do
+ target=$(basename "${target}")
+ if [[ "${target}" =~ (setup|prepare)_ ]]; then
+ continue
+ fi
+ if [ -f "test/integration/targets/${target}/test.sh" ]; then
+ continue
+ fi
+ if [ -f "test/integration/targets/${target}/aliases" ]; then
+ if grep -q -P "^(windows|posix)/" "test/integration/targets/${target}/aliases"; then
+ continue
+ fi
+ fi
+ if [[ "${target}" =~ _ ]]; then
+ prefix="${target//_*/}"
+ if grep -q --line-regex "${prefix}" test/integration/target-prefixes.*; then
+ continue
+ fi
+ fi
+ echo "target: ${target}, group: ${group}"
+ echo "${group}" >> "test/integration/targets/${target}/aliases"
+done
+
+# Add skip aliases for python3.
+sed 's/^test_//' test/utils/shippable/python3-test-tag-blacklist.txt | while IFS= read -r target; do
+ echo "skip/python3" >> "test/integration/targets/${target}/aliases"
+done
+
+# Add skip aliases for tests which don't pass yet on osx/freebsd.
+for target in service postgresql mysql_db mysql_user mysql_variables uri get_url async_extra_data; do
+ echo "skip/osx" >> "test/integration/targets/${target}/aliases"
+ echo "skip/freebsd" >> "test/integration/targets/${target}/aliases"
+done
+
+# Add skip aliases for tests which don't pass yet on osx.
+for target in gathering_facts iterators git; do
+ echo "skip/osx" >> "test/integration/targets/${target}/aliases"
+done
+
+# Add needs/root entries as required.
+for target in connection_chroot authorized_key copy template unarchive; do
+ echo "needs/root" >> "test/integration/targets/${target}/aliases"
+done
+
+# Add needs/ssh entries as required.
+for target in async_extra_data connection_ssh connection_paramiko_ssh; do
+ echo "needs/ssh" >> "test/integration/targets/${target}/aliases"
+done
+
+# Add missing alias for windows async_status.
+echo "async_status" >> test/integration/targets/win_async_wrapper/aliases
+
+# Remove connection tests from CI groups which aren't supported yet.
+for connection in docker jail libvirt_lxc lxc lxd; do
+ target="connection_${connection}"
+ sed -i '/^posix\/ci\/.*$/d' "test/integration/targets/${target}/aliases"
+done
+
+# Sort aliases.
+for file in test/integration/targets/*/aliases; do
+ sort -o "${file}" "${file}"
+done
diff --git a/test/runner/requirements/ansible-test.txt b/test/runner/requirements/ansible-test.txt
new file mode 100644
index 0000000000..d08d921a32
--- /dev/null
+++ b/test/runner/requirements/ansible-test.txt
@@ -0,0 +1 @@
+argparse ; python_version < '2.7'
diff --git a/test/runner/requirements/constraints.txt b/test/runner/requirements/constraints.txt
new file mode 100644
index 0000000000..ec8e8a069b
--- /dev/null
+++ b/test/runner/requirements/constraints.txt
@@ -0,0 +1,2 @@
+coverage >= 4.2
+pywinrm >= 0.2.1 # 0.1.1 required, but 0.2.1 provides better performance
diff --git a/test/runner/requirements/coverage.txt b/test/runner/requirements/coverage.txt
new file mode 100644
index 0000000000..4ebc8aea50
--- /dev/null
+++ b/test/runner/requirements/coverage.txt
@@ -0,0 +1 @@
+coverage
diff --git a/test/runner/requirements/integration.txt b/test/runner/requirements/integration.txt
new file mode 100644
index 0000000000..84ca1ec94a
--- /dev/null
+++ b/test/runner/requirements/integration.txt
@@ -0,0 +1,8 @@
+jinja2
+jmespath
+junit-xml
+ordereddict ; python_version < '2.7'
+paramiko
+passlib
+pycrypto
+pyyaml
diff --git a/test/runner/requirements/sanity.txt b/test/runner/requirements/sanity.txt
new file mode 100644
index 0000000000..1a86edd464
--- /dev/null
+++ b/test/runner/requirements/sanity.txt
@@ -0,0 +1,5 @@
+jinja2
+mock
+pylint
+voluptuous
+yamllint
diff --git a/test/runner/requirements/units.txt b/test/runner/requirements/units.txt
new file mode 100644
index 0000000000..885050a7b6
--- /dev/null
+++ b/test/runner/requirements/units.txt
@@ -0,0 +1,11 @@
+boto3
+jinja2
+mock
+nose
+passlib
+pycrypto
+pytest
+python-memcached
+pyyaml
+redis
+unittest2 ; python_version < '2.7'
diff --git a/test/runner/requirements/windows-integration.txt b/test/runner/requirements/windows-integration.txt
new file mode 100644
index 0000000000..d6fcc566fc
--- /dev/null
+++ b/test/runner/requirements/windows-integration.txt
@@ -0,0 +1,4 @@
+jinja2
+junit-xml
+pywinrm
+pyyaml
diff --git a/test/runner/setup/docker.sh b/test/runner/setup/docker.sh
new file mode 100644
index 0000000000..fa5788e8e7
--- /dev/null
+++ b/test/runner/setup/docker.sh
@@ -0,0 +1,18 @@
+#!/bin/sh
+
+set -eu
+
+# Support images with only python3 installed.
+if [ ! -f /usr/bin/python ] && [ -f /usr/bin/python3 ]; then
+ ln -s /usr/bin/python3 /usr/bin/python
+fi
+if [ ! -f /usr/bin/pip ] && [ -f /usr/bin/pip3 ]; then
+ ln -s /usr/bin/pip3 /usr/bin/pip
+fi
+
+# Improve prompts on remote host for interactive use.
+cat << EOF > ~/.bashrc
+alias ls='ls --color=auto'
+export PS1='\[\e]0;\u@\h: \w\a\]\[\033[01;32m\]\u@\h\[\033[00m\]:\[\033[01;34m\]\w\[\033[00m\]\$ '
+cd ~/ansible/
+EOF
diff --git a/test/runner/setup/remote.sh b/test/runner/setup/remote.sh
new file mode 100644
index 0000000000..13eb1cbc74
--- /dev/null
+++ b/test/runner/setup/remote.sh
@@ -0,0 +1,69 @@
+#!/bin/sh
+
+set -eu
+
+platform="$1"
+
+env
+
+cd ~/
+
+if [ "${platform}" = "freebsd" ]; then
+ pkg install -y curl
+
+ if [ ! -f bootstrap.sh ]; then
+ curl "https://raw.githubusercontent.com/mattclay/ansible-hacking/master/bootstrap.sh" -o bootstrap.sh -#
+ fi
+
+ chmod +x bootstrap.sh
+ ./bootstrap.sh pip -y -q
+
+ pkg install -y \
+ bash \
+ devel/ruby-gems \
+ gtar \
+ mercurial \
+ rsync \
+ ruby \
+ subversion \
+ sudo \
+ zip
+fi
+
+pip install virtualenv
+
+# Tests assume loopback addresses other than 127.0.0.1 will work.
+# Add aliases for loopback addresses used by tests.
+
+for i in 3 4 254; do
+ ifconfig lo0 alias "127.0.0.${i}" up
+done
+
+ifconfig lo0
+
+# Since tests run as root, we also need to be able to ssh to localhost as root.
+sed -i '' 's/^# *PermitRootLogin.*$/PermitRootLogin yes/;' /etc/ssh/sshd_config
+
+if [ "${platform}" = "freebsd" ]; then
+ # Restart sshd for configuration changes and loopback aliases to work.
+ service sshd restart
+fi
+
+# Generate our ssh key and add it to our authorized_keys file.
+# We also need to add localhost's server keys to known_hosts.
+
+if [ ! -f "${HOME}/.ssh/id_rsa.pub" ]; then
+ ssh-keygen -q -t rsa -N '' -f "${HOME}/.ssh/id_rsa"
+ cp "${HOME}/.ssh/id_rsa.pub" "${HOME}/.ssh/authorized_keys"
+ for key in /etc/ssh/ssh_host_*_key.pub; do
+ pk=$(cat "${key}")
+ echo "localhost ${pk}" >> "${HOME}/.ssh/known_hosts"
+ done
+fi
+
+# Improve prompts on remote host for interactive use.
+cat << EOF > ~/.bashrc
+alias ls='ls -G'
+export PS1='\[\e]0;\u@\h: \w\a\]\[\033[01;32m\]\u@\h\[\033[00m\]:\[\033[01;34m\]\w\[\033[00m\]\$ '
+cd ~/ansible/
+EOF
diff --git a/test/runner/test.py b/test/runner/test.py
new file mode 100755
index 0000000000..b9589b6c35
--- /dev/null
+++ b/test/runner/test.py
@@ -0,0 +1,446 @@
+#!/usr/bin/env python
+# PYTHON_ARGCOMPLETE_OK
+"""Test runner for all Ansible tests."""
+
+from __future__ import absolute_import, print_function
+
+import errno
+import os
+import sys
+
+from lib.util import (
+ ApplicationError,
+ display,
+ raw_command,
+)
+
+from lib.delegation import (
+ delegate,
+)
+
+from lib.executor import (
+ command_posix_integration,
+ command_network_integration,
+ command_windows_integration,
+ command_units,
+ command_compile,
+ command_sanity,
+ command_shell,
+ SANITY_TESTS,
+ SUPPORTED_PYTHON_VERSIONS,
+ COMPILE_PYTHON_VERSIONS,
+ PosixIntegrationConfig,
+ WindowsIntegrationConfig,
+ NetworkIntegrationConfig,
+ SanityConfig,
+ UnitsConfig,
+ CompileConfig,
+ ShellConfig,
+ ApplicationWarning,
+ Delegate,
+ generate_pip_install,
+)
+
+from lib.target import (
+ find_target_completion,
+ walk_posix_integration_targets,
+ walk_network_integration_targets,
+ walk_windows_integration_targets,
+ walk_units_targets,
+ walk_compile_targets,
+ walk_sanity_targets,
+)
+
+import lib.cover
+
+
+def main():
+ """Main program function."""
+ try:
+ git_root = os.path.abspath(os.path.join(os.path.dirname(os.path.abspath(__file__)), '..', '..'))
+ os.chdir(git_root)
+ args = parse_args()
+ config = args.config(args)
+ display.verbosity = config.verbosity
+ display.color = config.color
+
+ try:
+ args.func(config)
+ except Delegate as ex:
+ delegate(config, ex.exclude, ex.require)
+
+ display.review_warnings()
+ except ApplicationWarning as ex:
+ display.warning(str(ex))
+ exit(0)
+ except ApplicationError as ex:
+ display.error(str(ex))
+ exit(1)
+ except KeyboardInterrupt:
+ exit(2)
+ except IOError as ex:
+ if ex.errno == errno.EPIPE:
+ exit(3)
+ raise
+
+
+def parse_args():
+ """Parse command line arguments."""
+ try:
+ import argparse
+ except ImportError:
+ if '--requirements' not in sys.argv:
+ raise
+ raw_command(generate_pip_install('ansible-test'))
+ import argparse
+
+ try:
+ import argcomplete
+ except ImportError:
+ argcomplete = None
+
+ if argcomplete:
+ epilog = 'Tab completion available using the "argcomplete" python package.'
+ else:
+ epilog = 'Install the "argcomplete" python package to enable tab completion.'
+
+ parser = argparse.ArgumentParser(epilog=epilog)
+
+ common = argparse.ArgumentParser(add_help=False)
+
+ common.add_argument('-e', '--explain',
+ action='store_true',
+ help='explain commands that would be executed')
+
+ common.add_argument('-v', '--verbose',
+ dest='verbosity',
+ action='count',
+ default=0,
+ help='display more output')
+
+ common.add_argument('--color',
+ metavar='COLOR',
+ nargs='?',
+ help='generate color output: %(choices)s',
+ choices=('yes', 'no', 'auto'),
+ const='yes',
+ default='auto')
+
+ test = argparse.ArgumentParser(add_help=False, parents=[common])
+
+ test.add_argument('include',
+ metavar='TARGET',
+ nargs='*',
+ help='test the specified target').completer = complete_target
+
+ test.add_argument('--exclude',
+ metavar='TARGET',
+ action='append',
+ help='exclude the specified target').completer = complete_target
+
+ test.add_argument('--require',
+ metavar='TARGET',
+ action='append',
+ help='require the specified target').completer = complete_target
+
+ test.add_argument('--coverage',
+ action='store_true',
+ help='analyze code coverage when running tests')
+
+ add_changes(test, argparse)
+ add_environments(test)
+
+ integration = argparse.ArgumentParser(add_help=False, parents=[test])
+
+ integration.add_argument('--python',
+ metavar='VERSION',
+ choices=SUPPORTED_PYTHON_VERSIONS,
+ help='python version: %s' % ', '.join(SUPPORTED_PYTHON_VERSIONS))
+
+ integration.add_argument('--start-at',
+ metavar='TARGET',
+ help='start at the specified target').completer = complete_target
+
+ integration.add_argument('--start-at-task',
+ metavar='TASK',
+ help='start at the specified task')
+
+ integration.add_argument('--allow-destructive',
+ action='store_true',
+ help='allow destructive tests (--local and --tox only)')
+
+ integration.add_argument('--retry-on-error',
+ action='store_true',
+ help='retry failed test with increased verbosity')
+
+ subparsers = parser.add_subparsers(metavar='COMMAND')
+ subparsers.required = True # work-around for python 3 bug which makes subparsers optional
+
+ posix_integration = subparsers.add_parser('integration',
+ parents=[integration],
+ help='posix integration tests')
+
+ posix_integration.set_defaults(func=command_posix_integration,
+ targets=walk_posix_integration_targets,
+ config=PosixIntegrationConfig)
+
+ add_extra_docker_options(posix_integration)
+
+ network_integration = subparsers.add_parser('network-integration',
+ parents=[integration],
+ help='network integration tests')
+
+ network_integration.set_defaults(func=command_network_integration,
+ targets=walk_network_integration_targets,
+ config=NetworkIntegrationConfig)
+
+ windows_integration = subparsers.add_parser('windows-integration',
+ parents=[integration],
+ help='windows integration tests')
+
+ windows_integration.set_defaults(func=command_windows_integration,
+ targets=walk_windows_integration_targets,
+ config=WindowsIntegrationConfig)
+
+ windows_integration.add_argument('--windows',
+ metavar='VERSION',
+ action='append',
+ help='windows version')
+
+ units = subparsers.add_parser('units',
+ parents=[test],
+ help='unit tests')
+
+ units.set_defaults(func=command_units,
+ targets=walk_units_targets,
+ config=UnitsConfig)
+
+ units.add_argument('--python',
+ metavar='VERSION',
+ choices=SUPPORTED_PYTHON_VERSIONS,
+ help='python version: %s' % ', '.join(SUPPORTED_PYTHON_VERSIONS))
+
+ units.add_argument('--collect-only',
+ action='store_true',
+ help='collect tests but do not execute them')
+
+ compiler = subparsers.add_parser('compile',
+ parents=[test],
+ help='compile tests')
+
+ compiler.set_defaults(func=command_compile,
+ targets=walk_compile_targets,
+ config=CompileConfig)
+
+ compiler.add_argument('--python',
+ metavar='VERSION',
+ choices=COMPILE_PYTHON_VERSIONS,
+ help='python version: %s' % ', '.join(COMPILE_PYTHON_VERSIONS))
+
+ sanity = subparsers.add_parser('sanity',
+ parents=[test],
+ help='sanity tests')
+
+ sanity.set_defaults(func=command_sanity,
+ targets=walk_sanity_targets,
+ config=SanityConfig)
+
+ sanity.add_argument('--test',
+ metavar='TEST',
+ action='append',
+ choices=[t.name for t in SANITY_TESTS],
+ help='tests to run')
+
+ sanity.add_argument('--skip-test',
+ metavar='TEST',
+ action='append',
+ choices=[t.name for t in SANITY_TESTS],
+ help='tests to skip')
+
+ sanity.add_argument('--list-tests',
+ action='store_true',
+ help='list available tests')
+
+ sanity.add_argument('--python',
+ metavar='VERSION',
+ choices=SUPPORTED_PYTHON_VERSIONS,
+ help='python version: %s' % ', '.join(SUPPORTED_PYTHON_VERSIONS))
+
+ shell = subparsers.add_parser('shell',
+ parents=[common],
+ help='open an interactive shell')
+
+ shell.set_defaults(func=command_shell,
+ config=ShellConfig)
+
+ add_environments(shell, tox_version=True)
+ add_extra_docker_options(shell)
+
+ coverage_common = argparse.ArgumentParser(add_help=False, parents=[common])
+
+ add_environments(coverage_common, tox_version=True, tox_only=True)
+
+ coverage = subparsers.add_parser('coverage',
+ help='code coverage management and reporting')
+
+ coverage_subparsers = coverage.add_subparsers(metavar='COMMAND')
+ coverage_subparsers.required = True # work-around for python 3 bug which makes subparsers optional
+
+ coverage_combine = coverage_subparsers.add_parser('combine',
+ parents=[coverage_common],
+ help='combine coverage data and rewrite remote paths')
+
+ coverage_combine.set_defaults(func=lib.cover.command_coverage_combine,
+ config=lib.cover.CoverageConfig)
+
+ coverage_erase = coverage_subparsers.add_parser('erase',
+ parents=[coverage_common],
+ help='erase coverage data files')
+
+ coverage_erase.set_defaults(func=lib.cover.command_coverage_erase,
+ config=lib.cover.CoverageConfig)
+
+ coverage_report = coverage_subparsers.add_parser('report',
+ parents=[coverage_common],
+ help='generate console coverage report')
+
+ coverage_report.set_defaults(func=lib.cover.command_coverage_report,
+ config=lib.cover.CoverageConfig)
+
+ coverage_html = coverage_subparsers.add_parser('html',
+ parents=[coverage_common],
+ help='generate html coverage report')
+
+ coverage_html.set_defaults(func=lib.cover.command_coverage_html,
+ config=lib.cover.CoverageConfig)
+
+ coverage_xml = coverage_subparsers.add_parser('xml',
+ parents=[coverage_common],
+ help='generate xml coverage report')
+
+ coverage_xml.set_defaults(func=lib.cover.command_coverage_xml,
+ config=lib.cover.CoverageConfig)
+
+ if argcomplete:
+ argcomplete.autocomplete(parser, always_complete_options=False, validator=lambda i, k: True)
+
+ args = parser.parse_args()
+
+ if args.explain and not args.verbosity:
+ args.verbosity = 1
+
+ if args.color == 'yes':
+ args.color = True
+ elif args.color == 'no':
+ args.color = False
+ else:
+ args.color = sys.stdout.isatty()
+
+ return args
+
+
+def add_changes(parser, argparse):
+ """
+ :type parser: argparse.ArgumentParser
+ :type argparse: argparse
+ """
+ parser.add_argument('--changed', action='store_true', help='limit targets based on changes')
+
+ changes = parser.add_argument_group(title='change detection arguments')
+
+ changes.add_argument('--tracked', action='store_true', help=argparse.SUPPRESS)
+ changes.add_argument('--untracked', action='store_true', help='include untracked files')
+ changes.add_argument('--ignore-committed', dest='committed', action='store_false', help='exclude committed files')
+ changes.add_argument('--ignore-staged', dest='staged', action='store_false', help='exclude staged files')
+ changes.add_argument('--ignore-unstaged', dest='unstaged', action='store_false', help='exclude unstaged files')
+
+ changes.add_argument('--changed-from', metavar='PATH', help=argparse.SUPPRESS)
+ changes.add_argument('--changed-path', metavar='PATH', action='append', help=argparse.SUPPRESS)
+
+
+def add_environments(parser, tox_version=False, tox_only=False):
+ """
+ :type parser: argparse.ArgumentParser
+ :type tox_version: bool
+ :type tox_only: bool
+ """
+ parser.add_argument('--requirements',
+ action='store_true',
+ help='install command requirements')
+
+ environments = parser.add_mutually_exclusive_group()
+
+ environments.add_argument('--local',
+ action='store_true',
+ help='run from the local environment')
+
+ if tox_version:
+ environments.add_argument('--tox',
+ metavar='VERSION',
+ nargs='?',
+ default=None,
+ const='.'.join(str(i) for i in sys.version_info[:2]),
+ choices=SUPPORTED_PYTHON_VERSIONS,
+ help='run from a tox virtualenv: %s' % ', '.join(SUPPORTED_PYTHON_VERSIONS))
+ else:
+ environments.add_argument('--tox',
+ action='store_true',
+ help='run from a tox virtualenv')
+
+ if tox_only:
+ environments.set_defaults(
+ docker=None,
+ remote=None,
+ remote_stage=None,
+ )
+
+ return
+
+ environments.add_argument('--docker',
+ metavar='IMAGE',
+ nargs='?',
+ default=None,
+ const='ubuntu1604',
+ help='run from a docker container')
+
+ environments.add_argument('--remote',
+ metavar='PLATFORM',
+ default=None,
+ help='run from a remote instance')
+
+ remote = parser.add_argument_group(title='remote arguments')
+
+ remote.add_argument('--remote-stage',
+ metavar='STAGE',
+ help='remote stage to use: %(choices)s',
+ choices=['prod', 'dev'],
+ default='prod')
+
+
+def add_extra_docker_options(parser):
+ """
+ :type parser: argparse.ArgumentParser
+ """
+ docker = parser.add_argument_group(title='docker arguments')
+
+ docker.add_argument('--docker-util',
+ metavar='IMAGE',
+ default='httptester',
+ help='docker utility image to provide test services')
+
+ docker.add_argument('--docker-privileged',
+ action='store_true',
+ help='run docker container in privileged mode')
+
+
+def complete_target(prefix, parsed_args, **_):
+ """
+ :type prefix: unicode
+ :type parsed_args: any
+ :rtype: list[str]
+ """
+ return find_target_completion(parsed_args.targets, prefix)
+
+
+if __name__ == '__main__':
+ main()
diff --git a/test/runner/tox.ini b/test/runner/tox.ini
new file mode 100644
index 0000000000..6399a524da
--- /dev/null
+++ b/test/runner/tox.ini
@@ -0,0 +1,9 @@
+[tox]
+skipsdist = True
+minversion = 2.5.0
+
+[testenv]
+changedir = {toxinidir}/../../
+commands = {posargs}
+passenv = HOME
+args_are_paths = False