Run Validations with ThreadPoolExecutor
The validations are sequentially executed through ansible-playbook and could take some time to complete. This patch adds support of running all the validations in parallel through a ThreadPoolExecutor in order to save processing time. Without this patch, it takes almost ~5min to run all the validations but only ~1min10 with the ThreadPoolExecutor. - Add --worker/-w argument to give the maximum number of threads that can be used to execute the given validations - Python 2.7: use futures backport instead, ThreadPoolExecutor is default - Use six in tripleo_validator.py for python 3 compatibility Change-Id: Ia805a556bc26700a3eb520ed72e90b37546901b8 Signed-off-by: Gael Chamoulaud <gchamoul@redhat.com>
This commit is contained in:
parent
44c76f8c38
commit
fdf79cf4f3
@ -25,6 +25,7 @@ extras==1.0.0
|
|||||||
fasteners==0.7.0
|
fasteners==0.7.0
|
||||||
fixtures==3.0.0
|
fixtures==3.0.0
|
||||||
flake8==2.5.5
|
flake8==2.5.5
|
||||||
|
futures==3.0.0
|
||||||
futurist==1.2.0
|
futurist==1.2.0
|
||||||
gitdb==0.6.4
|
gitdb==0.6.4
|
||||||
GitPython==1.0.1
|
GitPython==1.0.1
|
||||||
|
@ -18,3 +18,4 @@ osc-lib>=1.8.0 # Apache-2.0
|
|||||||
websocket-client>=0.44.0 # LGPLv2+
|
websocket-client>=0.44.0 # LGPLv2+
|
||||||
tripleo-common>=10.7.0 # Apache-2.0
|
tripleo-common>=10.7.0 # Apache-2.0
|
||||||
cryptography>=2.1 # BSD/Apache-2.0
|
cryptography>=2.1 # BSD/Apache-2.0
|
||||||
|
futures>=3.0.0;python_version=='2.7' or python_version=='2.6' # BSD
|
||||||
|
@ -18,8 +18,10 @@ import json
|
|||||||
import logging
|
import logging
|
||||||
import os
|
import os
|
||||||
import pwd
|
import pwd
|
||||||
|
import six
|
||||||
import sys
|
import sys
|
||||||
|
|
||||||
|
from concurrent.futures import ThreadPoolExecutor
|
||||||
from osc_lib.command import command
|
from osc_lib.command import command
|
||||||
from osc_lib.i18n import _
|
from osc_lib.i18n import _
|
||||||
|
|
||||||
@ -212,6 +214,16 @@ class TripleOValidatorRun(command.Command):
|
|||||||
help=_("Execute the validations using Mistral")
|
help=_("Execute the validations using Mistral")
|
||||||
)
|
)
|
||||||
|
|
||||||
|
parser.add_argument(
|
||||||
|
'--workers', '-w',
|
||||||
|
metavar='N',
|
||||||
|
dest='workers',
|
||||||
|
default=1,
|
||||||
|
type=int,
|
||||||
|
help=_("The maximum number of threads that can "
|
||||||
|
"be used to execute the given validations")
|
||||||
|
)
|
||||||
|
|
||||||
extra_vars_group = parser.add_mutually_exclusive_group(required=False)
|
extra_vars_group = parser.add_mutually_exclusive_group(required=False)
|
||||||
|
|
||||||
extra_vars_group.add_argument(
|
extra_vars_group.add_argument(
|
||||||
@ -302,6 +314,23 @@ class TripleOValidatorRun(command.Command):
|
|||||||
out.get('validation_name'),
|
out.get('validation_name'),
|
||||||
oooutils.indent(out.get('stdout'))))
|
oooutils.indent(out.get('stdout'))))
|
||||||
|
|
||||||
|
def _run_ansible(self, logger, plan, workdir, log_path_dir, playbook,
|
||||||
|
inventory, retries, output_callback, extra_vars,
|
||||||
|
python_interpreter, gathering_policy):
|
||||||
|
rc, output = oooutils.run_ansible_playbook(
|
||||||
|
logger=logger,
|
||||||
|
plan=plan,
|
||||||
|
workdir=workdir,
|
||||||
|
log_path_dir=log_path_dir,
|
||||||
|
playbook=playbook,
|
||||||
|
inventory=inventory,
|
||||||
|
retries=retries,
|
||||||
|
output_callback=output_callback,
|
||||||
|
extra_vars=extra_vars,
|
||||||
|
python_interpreter=python_interpreter,
|
||||||
|
gathering_policy=gathering_policy)
|
||||||
|
return rc, output
|
||||||
|
|
||||||
def _run_validator_run(self, parsed_args):
|
def _run_validator_run(self, parsed_args):
|
||||||
clients = self.app.client_manager
|
clients = self.app.client_manager
|
||||||
LOG = logging.getLogger(__name__ + ".ValidationsRunAnsible")
|
LOG = logging.getLogger(__name__ + ".ValidationsRunAnsible")
|
||||||
@ -343,10 +372,11 @@ class TripleOValidatorRun(command.Command):
|
|||||||
|
|
||||||
failed_val = False
|
failed_val = False
|
||||||
|
|
||||||
for playbook in playbooks:
|
with ThreadPoolExecutor(max_workers=parsed_args.workers) as executor:
|
||||||
try:
|
|
||||||
LOG.debug(_('Running the validations with Ansible'))
|
LOG.debug(_('Running the validations with Ansible'))
|
||||||
rc, output = oooutils.run_ansible_playbook(
|
tasks_exec = {
|
||||||
|
executor.submit(
|
||||||
|
self._run_ansible,
|
||||||
logger=LOG,
|
logger=LOG,
|
||||||
plan=parsed_args.plan,
|
plan=parsed_args.plan,
|
||||||
workdir=constants.ANSIBLE_VALIDATION_DIR,
|
workdir=constants.ANSIBLE_VALIDATION_DIR,
|
||||||
@ -357,13 +387,18 @@ class TripleOValidatorRun(command.Command):
|
|||||||
output_callback='validation_output',
|
output_callback='validation_output',
|
||||||
extra_vars=extra_vars_input,
|
extra_vars=extra_vars_input,
|
||||||
python_interpreter=python_interpreter,
|
python_interpreter=python_interpreter,
|
||||||
gathering_policy='explicit')
|
gathering_policy='explicit'): playbook
|
||||||
print('[SUCCESS] - {}\n{}'.format(playbook,
|
for playbook in playbooks
|
||||||
oooutils.indent(output)))
|
}
|
||||||
|
|
||||||
|
for tk, pl in six.iteritems(tasks_exec):
|
||||||
|
try:
|
||||||
|
rc, output = tk.result()
|
||||||
|
print('[SUCCESS] - {}\n{}'.format(pl, oooutils.indent(output)))
|
||||||
except Exception as e:
|
except Exception as e:
|
||||||
failed_val = True
|
failed_val = True
|
||||||
LOG.error('[FAILED] - {}\n{}'.format(
|
LOG.error('[FAILED] - {}\n{}'.format(
|
||||||
playbook, oooutils.indent(e.args[0])))
|
pl, oooutils.indent(e.args[0])))
|
||||||
|
|
||||||
LOG.debug(_('Removing static tripleo ansible inventory file'))
|
LOG.debug(_('Removing static tripleo ansible inventory file'))
|
||||||
oooutils.cleanup_tripleo_ansible_inventory_file(
|
oooutils.cleanup_tripleo_ansible_inventory_file(
|
||||||
|
Loading…
Reference in New Issue
Block a user