533 lines
21 KiB
Python
533 lines
21 KiB
Python
# Copyright 2020 Red Hat, Inc.
|
|
#
|
|
# Licensed under the Apache License, Version 2.0 (the "License"); you may
|
|
# not use this file except in compliance with the License. You may obtain
|
|
# a copy of the License at
|
|
#
|
|
# http://www.apache.org/licenses/LICENSE-2.0
|
|
#
|
|
# Unless required by applicable law or agreed to in writing, software
|
|
# distributed under the License is distributed on an "AS IS" BASIS, WITHOUT
|
|
# WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the
|
|
# License for the specific language governing permissions and limitations
|
|
# under the License.
|
|
#
|
|
|
|
import ansible_runner
|
|
from validations_libs.logger import getLogger
|
|
import pkg_resources
|
|
import pwd
|
|
import os
|
|
import sys
|
|
import tempfile
|
|
import threading
|
|
import yaml
|
|
|
|
import configparser
|
|
from validations_libs import constants
|
|
from validations_libs import utils
|
|
|
|
LOG = getLogger(__name__ + ".ansible")
|
|
|
|
# NOTE(cloudnull): This is setting the FileExistsError for py2 environments.
|
|
# When we no longer support py2 (centos7) this should be
|
|
# removed.
|
|
try:
|
|
FileExistsError = FileExistsError
|
|
except NameError:
|
|
FileExistsError = OSError
|
|
|
|
try:
|
|
version = pkg_resources.get_distribution("ansible_runner").version
|
|
BACKWARD_COMPAT = (version < '1.4.0')
|
|
except pkg_resources.DistributionNotFound:
|
|
BACKWARD_COMPAT = False
|
|
|
|
|
|
class Ansible:
|
|
"""An Object for encapsulating an Ansible execution"""
|
|
|
|
def __init__(self, uuid=None):
|
|
self.log = getLogger(__name__ + ".Ansible")
|
|
self.uuid = uuid
|
|
|
|
def _playbook_check(self, play, playbook_dir=None):
|
|
"""Check if playbook exist"""
|
|
if not os.path.exists(play):
|
|
play = os.path.join(playbook_dir, play)
|
|
if not os.path.exists(play):
|
|
raise RuntimeError('No such playbook: {}'.format(play))
|
|
self.log.debug('Ansible playbook {} found'.format(play))
|
|
return play
|
|
|
|
def _inventory(self, inventory, ansible_artifact_path):
|
|
"""Handle inventory for Ansible"""
|
|
if inventory:
|
|
if isinstance(inventory, str):
|
|
# check is file path
|
|
if os.path.exists(inventory):
|
|
return os.path.abspath(inventory)
|
|
elif isinstance(inventory, dict):
|
|
inventory = yaml.safe_dump(
|
|
inventory,
|
|
default_flow_style=False
|
|
)
|
|
return ansible_runner.utils.dump_artifact(
|
|
inventory,
|
|
ansible_artifact_path,
|
|
'hosts'
|
|
)
|
|
|
|
def _creates_ansible_fact_dir(self,
|
|
temp_suffix='validations-libs-ansible'):
|
|
"""Creates ansible fact dir"""
|
|
ansible_fact_path = os.path.join(
|
|
tempfile.gettempdir(),
|
|
temp_suffix,
|
|
'fact_cache'
|
|
)
|
|
try:
|
|
os.makedirs(ansible_fact_path)
|
|
return ansible_fact_path
|
|
except FileExistsError:
|
|
self.log.debug(
|
|
'Directory "{}" was not created because it'
|
|
' already exists.'.format(
|
|
ansible_fact_path
|
|
)
|
|
)
|
|
|
|
def _get_extra_vars(self, extra_vars):
|
|
"""Manage extra_vars into a dict"""
|
|
extravars = dict()
|
|
if extra_vars:
|
|
if isinstance(extra_vars, dict):
|
|
extravars.update(extra_vars)
|
|
elif os.path.exists(extra_vars) and os.path.isfile(extra_vars):
|
|
with open(extra_vars) as f:
|
|
extravars.update(yaml.safe_load(f.read()))
|
|
return extravars
|
|
|
|
def _callbacks(self, callback_whitelist, output_callback, envvars={},
|
|
env={}):
|
|
"""Set callbacks"""
|
|
# if output_callback is exported in env, then use it
|
|
if isinstance(envvars, dict):
|
|
env.update(envvars)
|
|
output_callback = env.get('ANSIBLE_STDOUT_CALLBACK', output_callback)
|
|
# TODO(jpodivin) Whitelist was extended with new callback names
|
|
# to prevent issues during transition period.
|
|
# The entries with 'vf_' prefix should be removed afterwards.
|
|
callback_whitelist = ','.join(filter(None, [callback_whitelist,
|
|
output_callback,
|
|
'profile_tasks',
|
|
'vf_validation_json']))
|
|
return callback_whitelist, output_callback
|
|
|
|
def _ansible_env_var(self, output_callback, ssh_user, workdir, connection,
|
|
gathering_policy, module_path, key,
|
|
extra_env_variables, ansible_timeout,
|
|
callback_whitelist, base_dir, python_interpreter,
|
|
env={}, validation_cfg_file=None):
|
|
"""Handle Ansible env var for Ansible config execution"""
|
|
community_roles = ""
|
|
community_library = ""
|
|
community_lookup = ""
|
|
if utils.community_validations_on(validation_cfg_file):
|
|
community_roles = "{}:".format(constants.COMMUNITY_ROLES_DIR)
|
|
community_library = "{}:".format(constants.COMMUNITY_LIBRARY_DIR)
|
|
community_lookup = "{}:".format(constants.COMMUNITY_LOOKUP_DIR)
|
|
|
|
cwd = os.getcwd()
|
|
env['ANSIBLE_SSH_ARGS'] = (
|
|
'-o UserKnownHostsFile={} '
|
|
'-o StrictHostKeyChecking=no '
|
|
'-o ControlMaster=auto '
|
|
'-o ControlPersist=30m '
|
|
'-o ServerAliveInterval=64 '
|
|
'-o ServerAliveCountMax=1024 '
|
|
'-o Compression=no '
|
|
'-o TCPKeepAlive=yes '
|
|
'-o VerifyHostKeyDNS=no '
|
|
'-o ForwardX11=no '
|
|
'-o ForwardAgent=yes '
|
|
'-o PreferredAuthentications=publickey '
|
|
'-T'
|
|
).format(os.devnull)
|
|
|
|
env['ANSIBLE_DISPLAY_FAILED_STDERR'] = True
|
|
env['ANSIBLE_FORKS'] = 36
|
|
env['ANSIBLE_TIMEOUT'] = ansible_timeout
|
|
env['ANSIBLE_GATHER_TIMEOUT'] = 45
|
|
env['ANSIBLE_SSH_RETRIES'] = 3
|
|
env['ANSIBLE_PIPELINING'] = True
|
|
if ssh_user:
|
|
env['ANSIBLE_REMOTE_USER'] = ssh_user
|
|
env['ANSIBLE_STDOUT_CALLBACK'] = output_callback
|
|
env['ANSIBLE_LIBRARY'] = os.path.expanduser(
|
|
'~/.ansible/plugins/modules:'
|
|
'{}:{}:'
|
|
'/usr/share/ansible/plugins/modules:'
|
|
'/usr/share/ceph-ansible/library:'
|
|
'{community_path}'
|
|
'{}/library'.format(
|
|
os.path.join(workdir, 'modules'),
|
|
os.path.join(cwd, 'modules'),
|
|
base_dir,
|
|
community_path=community_library
|
|
)
|
|
)
|
|
env['ANSIBLE_LOOKUP_PLUGINS'] = os.path.expanduser(
|
|
'~/.ansible/plugins/lookup:'
|
|
'{}:{}:'
|
|
'/usr/share/ansible/plugins/lookup:'
|
|
'/usr/share/ceph-ansible/plugins/lookup:'
|
|
'{community_path}'
|
|
'{}/lookup_plugins'.format(
|
|
os.path.join(workdir, 'lookup'),
|
|
os.path.join(cwd, 'lookup'),
|
|
base_dir,
|
|
community_path=community_lookup
|
|
)
|
|
)
|
|
env['ANSIBLE_CALLBACK_PLUGINS'] = os.path.expanduser(
|
|
'~/.ansible/plugins/callback:'
|
|
'{}:{}:'
|
|
'/usr/share/ansible/plugins/callback:'
|
|
'/usr/share/ceph-ansible/plugins/callback:'
|
|
'{}/callback_plugins'.format(
|
|
os.path.join(workdir, 'callback'),
|
|
os.path.join(cwd, 'callback'),
|
|
base_dir
|
|
)
|
|
)
|
|
env['ANSIBLE_ACTION_PLUGINS'] = os.path.expanduser(
|
|
'~/.ansible/plugins/action:'
|
|
'{}:{}:'
|
|
'/usr/share/ansible/plugins/action:'
|
|
'/usr/share/ceph-ansible/plugins/actions:'
|
|
'{}/action_plugins'.format(
|
|
os.path.join(workdir, 'action'),
|
|
os.path.join(cwd, 'action'),
|
|
base_dir
|
|
)
|
|
)
|
|
env['ANSIBLE_FILTER_PLUGINS'] = os.path.expanduser(
|
|
'~/.ansible/plugins/filter:'
|
|
'{}:{}:'
|
|
'/usr/share/ansible/plugins/filter:'
|
|
'/usr/share/ceph-ansible/plugins/filter:'
|
|
'{}/filter_plugins'.format(
|
|
os.path.join(workdir, 'filter'),
|
|
os.path.join(cwd, 'filter'),
|
|
base_dir
|
|
)
|
|
)
|
|
env['ANSIBLE_ROLES_PATH'] = os.path.expanduser(
|
|
'~/.ansible/roles:'
|
|
'{}:{}:'
|
|
'/usr/share/ansible/roles:'
|
|
'/usr/share/ceph-ansible/roles:'
|
|
'/etc/ansible/roles:'
|
|
'{community_path}'
|
|
'{}/roles'.format(
|
|
os.path.join(workdir, 'roles'),
|
|
os.path.join(cwd, 'roles'),
|
|
base_dir,
|
|
community_path=community_roles
|
|
)
|
|
)
|
|
env['ANSIBLE_CALLBACK_WHITELIST'] = callback_whitelist
|
|
env['ANSIBLE_RETRY_FILES_ENABLED'] = False
|
|
env['ANSIBLE_HOST_KEY_CHECKING'] = False
|
|
env['ANSIBLE_TRANSPORT'] = connection
|
|
env['ANSIBLE_CACHE_PLUGIN_TIMEOUT'] = 7200
|
|
|
|
if self.uuid:
|
|
env['ANSIBLE_UUID'] = self.uuid
|
|
|
|
if python_interpreter:
|
|
env['ANSIBLE_PYTHON_INTERPRETER'] = python_interpreter
|
|
elif connection == 'local':
|
|
env['ANSIBLE_PYTHON_INTERPRETER'] = sys.executable
|
|
|
|
if gathering_policy in ('smart', 'explicit', 'implicit'):
|
|
env['ANSIBLE_GATHERING'] = gathering_policy
|
|
|
|
if module_path:
|
|
env['ANSIBLE_LIBRARY'] = ':'.join(
|
|
[env['ANSIBLE_LIBRARY'], module_path]
|
|
)
|
|
|
|
try:
|
|
user_pwd = pwd.getpwuid(int(os.getenv('SUDO_UID', os.getuid())))
|
|
except TypeError:
|
|
home = os.path.expanduser('~')
|
|
else:
|
|
home = user_pwd.pw_dir
|
|
|
|
env['ANSIBLE_LOG_PATH'] = os.path.join(home, 'ansible.log')
|
|
|
|
if key:
|
|
env['ANSIBLE_PRIVATE_KEY_FILE'] = key
|
|
|
|
if extra_env_variables:
|
|
if not isinstance(extra_env_variables, dict):
|
|
msg = "extra_env_variables must be a dict"
|
|
self.log.error(msg)
|
|
raise SystemError(msg)
|
|
else:
|
|
env.update(extra_env_variables)
|
|
|
|
return env
|
|
|
|
def _encode_envvars(self, env):
|
|
"""Encode a hash of values.
|
|
|
|
:param env: A hash of key=value items.
|
|
:type env: `dict`.
|
|
"""
|
|
for key, value in env.items():
|
|
env[key] = str(value)
|
|
else:
|
|
return env
|
|
|
|
def _dump_validation_config(self, config, path, filename='validation.cfg'):
|
|
"""Dump Validation config in artifact directory"""
|
|
parser = configparser.ConfigParser()
|
|
for section_key in config.keys():
|
|
parser.add_section(section_key)
|
|
for item_key in config[section_key].keys():
|
|
parser.set(section_key, item_key,
|
|
str(config[section_key][item_key]))
|
|
with open('{}/{}'.format(path, filename), 'w') as conf:
|
|
parser.write(conf)
|
|
|
|
def _check_ansible_files(self, env):
|
|
# Check directories
|
|
callbacks_path = env.get('ANSIBLE_CALLBACK_PLUGINS', '')
|
|
roles_path = env.get('ANSIBLE_ROLES_PATH', '')
|
|
if not any([path for path
|
|
in callbacks_path.split(':')
|
|
if os.path.exists('%s/vf_validation_json.py' % (path))]):
|
|
raise RuntimeError('Callback vf_validation_json.py not found '
|
|
'in {}'.format(callbacks_path))
|
|
if not any([path for path
|
|
in roles_path.split(':')
|
|
if os.path.exists(path)]):
|
|
raise RuntimeError('roles directory not found '
|
|
'in {}'.format(roles_path))
|
|
|
|
def run(self, playbook, inventory, workdir, playbook_dir=None,
|
|
connection='smart', output_callback=None,
|
|
base_dir=constants.DEFAULT_VALIDATIONS_BASEDIR,
|
|
ssh_user=None, key=None, module_path=None,
|
|
limit_hosts=None, tags=None, skip_tags=None,
|
|
verbosity=0, quiet=False, extra_vars=None,
|
|
gathering_policy='smart',
|
|
extra_env_variables=None, parallel_run=False,
|
|
callback_whitelist=None, ansible_cfg_file=None,
|
|
ansible_timeout=30, ansible_artifact_path=None,
|
|
log_path=None, run_async=False, python_interpreter=None,
|
|
validation_cfg_file=None):
|
|
"""Execute one or multiple Ansible playbooks
|
|
|
|
:param playbook: The Absolute path of the Ansible playbook
|
|
:type playbook: ``string``
|
|
:param inventory: Either proper inventory file or a
|
|
comma-separated list
|
|
:type inventory: ``string``
|
|
:param workdir: The absolute path of the Ansible-runner
|
|
artifacts directory
|
|
:type workdir: ``string``
|
|
:param playbook_dir: The absolute path of the Validations playbooks
|
|
directory
|
|
:type playbook_dir: ``string``
|
|
:param connection: Connection type (local, smart, etc).
|
|
(efaults to 'smart')
|
|
:type connection: String
|
|
:param output_callback: Callback for output format. Defaults to
|
|
'yaml'.
|
|
:type output_callback: ``string``
|
|
:param base_dir: The absolute path of the default validations base
|
|
directory
|
|
:type base_dir: ``string``
|
|
:param ssh_user: User for the ssh connection (Defaults to 'root')
|
|
:type ssh_user: ``string``
|
|
:param key: Private key to use for the ssh connection.
|
|
:type key: ``string``
|
|
:param module_path: Location of the ansible module and library.
|
|
:type module_path: ``string``
|
|
:param limit_hosts: Limit the execution to the hosts.
|
|
:type limit_hosts: ``string``
|
|
:param tags: Run specific tags.
|
|
:type tags: ``string``
|
|
:param skip_tags: Skip specific tags.
|
|
:type skip_tags: ``string``
|
|
:param verbosity: Verbosity level for Ansible execution.
|
|
:type verbosity: ``integer``
|
|
:param quiet: Disable all output (Defaults to False)
|
|
:type quiet: ``boolean``
|
|
:param extra_vars: Set additional variables as a Dict or the absolute
|
|
path of a JSON or YAML file type.
|
|
:type extra_vars: Either a Dict or the absolute path of JSON or YAML
|
|
:param gathering_policy: This setting controls the default policy of
|
|
fact gathering ('smart', 'implicit', 'explicit').
|
|
(Defaults to 'smart')
|
|
:type gathering_facts: ``string``
|
|
:param extra_env_vars: Set additional ansible variables using an
|
|
extravar dictionary.
|
|
:type extra_env_vars: ``dict``
|
|
:param parallel_run: Isolate playbook execution when playbooks are
|
|
to be executed with multi-processing.
|
|
:type parallel_run: ``boolean``
|
|
:param callback_whitelist: Comma separated list of callback plugins.
|
|
Custom output_callback is also whitelisted.
|
|
(Defaults to ``None``)
|
|
:type callback_whitelist: ``list`` or ``string``
|
|
:param ansible_cfg_file: Path to an ansible configuration file. One
|
|
will be generated in the artifact path if
|
|
this option is None.
|
|
:type ansible_cfg_file: ``string``
|
|
:param ansible_timeout: Timeout for ansible connections.
|
|
(Defaults to ``30 minutes``)
|
|
:type ansible_timeout: ``integer``
|
|
:param ansible_artifact_path: The Ansible artifact path
|
|
:type ansible_artifact_path: ``string``
|
|
:param log_path: The absolute path of the validations logs directory
|
|
:type log_path: ``string``
|
|
:param run_async: Enable the Ansible asynchronous mode
|
|
(Defaults to 'False')
|
|
:type run_async: ``boolean``
|
|
:param python_interpreter: Path to the Python interpreter to be
|
|
used for module execution on remote targets,
|
|
or an automatic discovery mode (``auto``,
|
|
``auto_silent`` or the default one
|
|
``auto_legacy``)
|
|
:type python_interpreter: ``string``
|
|
:param validation_cfg_file: A dictionary of configuration for
|
|
Validation loaded from an validation.cfg
|
|
file.
|
|
:type validation_cfg_file: ``dict``
|
|
|
|
:return: A ``tuple`` containing the the absolute path of the executed
|
|
playbook, the return code and the status of the run
|
|
:rtype: ``tuple``
|
|
"""
|
|
if not playbook_dir:
|
|
playbook_dir = workdir
|
|
|
|
if not ansible_artifact_path:
|
|
if log_path:
|
|
ansible_artifact_path = "{}/artifacts/".format(log_path)
|
|
else:
|
|
ansible_artifact_path = \
|
|
constants.VALIDATION_ANSIBLE_ARTIFACT_PATH
|
|
|
|
playbook = self._playbook_check(playbook, playbook_dir)
|
|
self.log.debug(
|
|
'Running Ansible playbook: {},'
|
|
' Working directory: {},'
|
|
' Playbook directory: {}'.format(
|
|
playbook,
|
|
workdir,
|
|
playbook_dir
|
|
)
|
|
)
|
|
|
|
# Get env variables:
|
|
env = {}
|
|
env = os.environ.copy()
|
|
extravars = self._get_extra_vars(extra_vars)
|
|
|
|
if isinstance(callback_whitelist, list):
|
|
callback_whitelist = ','.join(callback_whitelist)
|
|
callback_whitelist, output_callback = self._callbacks(
|
|
callback_whitelist,
|
|
output_callback,
|
|
extra_env_variables,
|
|
env)
|
|
# Set ansible environment variables
|
|
env.update(self._ansible_env_var(output_callback, ssh_user, workdir,
|
|
connection, gathering_policy,
|
|
module_path, key, extra_env_variables,
|
|
ansible_timeout, callback_whitelist,
|
|
base_dir, python_interpreter,
|
|
validation_cfg_file=validation_cfg_file))
|
|
# Check if the callback is present and the roles path
|
|
self._check_ansible_files(env)
|
|
|
|
if 'ANSIBLE_CONFIG' not in env and not ansible_cfg_file:
|
|
ansible_cfg_file = os.path.join(ansible_artifact_path,
|
|
'ansible.cfg')
|
|
ansible_config = configparser.ConfigParser()
|
|
ansible_config.add_section('defaults')
|
|
ansible_config.set('defaults', 'internal_poll_interval', '0.05')
|
|
with open(ansible_cfg_file, 'w') as f:
|
|
ansible_config.write(f)
|
|
env['ANSIBLE_CONFIG'] = ansible_cfg_file
|
|
elif 'ANSIBLE_CONFIG' not in env and ansible_cfg_file:
|
|
env['ANSIBLE_CONFIG'] = ansible_cfg_file
|
|
|
|
if log_path:
|
|
env['VALIDATIONS_LOG_DIR'] = log_path
|
|
|
|
envvars = self._encode_envvars(env=env)
|
|
r_opts = {
|
|
'private_data_dir': workdir,
|
|
'inventory': self._inventory(inventory, ansible_artifact_path),
|
|
'playbook': playbook,
|
|
'verbosity': verbosity,
|
|
'quiet': quiet,
|
|
'extravars': extravars,
|
|
'artifact_dir': workdir,
|
|
'rotate_artifacts': 256,
|
|
'ident': ''
|
|
}
|
|
|
|
if not BACKWARD_COMPAT:
|
|
r_opts.update({
|
|
'project_dir': playbook_dir,
|
|
'fact_cache': ansible_artifact_path,
|
|
'fact_cache_type': 'jsonfile'
|
|
})
|
|
else:
|
|
parallel_run = False
|
|
|
|
if skip_tags:
|
|
r_opts['skip_tags'] = skip_tags
|
|
|
|
if tags:
|
|
r_opts['tags'] = tags
|
|
|
|
if limit_hosts:
|
|
r_opts['limit'] = limit_hosts
|
|
|
|
if parallel_run:
|
|
r_opts['directory_isolation_base_path'] = ansible_artifact_path
|
|
|
|
if validation_cfg_file:
|
|
if 'ansible_runner' in validation_cfg_file.keys():
|
|
r_opts.update(validation_cfg_file['ansible_runner'])
|
|
if 'ansible_environment' in validation_cfg_file.keys():
|
|
envvars.update(validation_cfg_file['ansible_environment'])
|
|
self._dump_validation_config(validation_cfg_file,
|
|
ansible_artifact_path)
|
|
if not BACKWARD_COMPAT:
|
|
r_opts.update({'envvars': envvars})
|
|
|
|
runner_config = ansible_runner.runner_config.RunnerConfig(**r_opts)
|
|
runner_config.prepare()
|
|
runner_config.env['ANSIBLE_STDOUT_CALLBACK'] = \
|
|
envvars['ANSIBLE_STDOUT_CALLBACK']
|
|
if BACKWARD_COMPAT:
|
|
runner_config.env.update(envvars)
|
|
|
|
runner = ansible_runner.Runner(config=runner_config)
|
|
if run_async:
|
|
thr = threading.Thread(target=runner.run)
|
|
thr.start()
|
|
return playbook, runner.rc, runner.status
|
|
status, rc = runner.run()
|
|
return playbook, rc, status
|