Merge "Use new CLI for functionnal tests and remove validation.py script"
This commit is contained in:
commit
2d1e6ce819
@ -4,27 +4,27 @@ zuul_work_virtualenv: "{{ ansible_user_dir }}/.venv"
|
||||
ansible_dir: "{{ zuul_work_virtualenv }}/share/ansible/"
|
||||
validation_dir: "{{ zuul_work_virtualenv }}/share/ansible/validation-playbooks"
|
||||
vf_log_dir: "/var/log/validations"
|
||||
val_exec: "source {{ zuul_work_virtualenv }}/bin/activate; validation.py"
|
||||
val_exec: "source {{ zuul_work_virtualenv }}/bin/activate; validation"
|
||||
command:
|
||||
- output: "{{ log_dir }}/run.log"
|
||||
action: "run"
|
||||
command: >-
|
||||
{{ val_exec }} run --validation check-ftype,512e
|
||||
--validation-dir {{ validation_dir }}
|
||||
--ansible-base-dir {{ ansible_dir }}
|
||||
--output-log {{ log_dir }}/run.log
|
||||
- output: "{{ log_dir }}/run-group.log"
|
||||
action: "run"
|
||||
command: >-
|
||||
{{ val_exec }} run --group prep --validation-dir {{ validation_dir }}
|
||||
--ansible-base-dir {{ ansible_dir }}
|
||||
--output-log {{ log_dir }}/run-group.log
|
||||
--extra-vars minimal_ram_gb=7
|
||||
- output: "{{ log_dir }}/list.log"
|
||||
action: "list"
|
||||
command: >-
|
||||
{{ val_exec }} list --validation-dir {{ validation_dir }}
|
||||
--ansible-base-dir {{ ansible_dir }}
|
||||
--output-log {{ log_dir }}/list.log
|
||||
{{ val_exec }} list --validation-dir {{ validation_dir }} -f json > {{ log_dir }}/list.log 2>&1
|
||||
- output: "{{ log_dir }}/show.log"
|
||||
action: "show"
|
||||
command: >-
|
||||
{{ val_exec }} show --validation-dir {{ validation_dir }}
|
||||
--ansible-base-dir {{ ansible_dir }}
|
||||
--output-log {{ log_dir }}/show.log
|
||||
{{ val_exec }} show --validation-dir {{ validation_dir }} check-ram -f json > {{ log_dir }}/show.log 2>&1
|
||||
|
@ -26,5 +26,5 @@
|
||||
name: ansible
|
||||
virtualenv: "{{ zuul_work_virtualenv }}"
|
||||
|
||||
- include: validations.yaml validation_command="{{ item.command }}" val_output="{{ item.output }}"
|
||||
- include: validations.yaml validation_command="{{ item.command }}" val_output="{{ item.output }}" action="{{ item.action }}"
|
||||
loop: "{{command}}"
|
||||
|
@ -4,23 +4,30 @@
|
||||
cmd: "{{ validation_command }}"
|
||||
executable: /bin/bash
|
||||
|
||||
- name: Get run results
|
||||
register: result
|
||||
shell:
|
||||
cmd: "cat {{ val_output }}"
|
||||
executable: /bin/bash
|
||||
- name: set fact for Validation action
|
||||
set_fact: v_action="{{ action }}"
|
||||
|
||||
- name: Get json data
|
||||
set_fact:
|
||||
jsondata: "{{ result.stdout | from_json }}"
|
||||
- name: Get Run results
|
||||
block:
|
||||
- name: Get run results
|
||||
register: result
|
||||
shell:
|
||||
cmd: "cat {{ val_output }}"
|
||||
executable: /bin/bash
|
||||
|
||||
- name: Get Validations Status
|
||||
set_fact:
|
||||
status: "{{ jsondata | json_query(jsonres) }}"
|
||||
vars:
|
||||
jsonres: 'results[*].Status'
|
||||
- name: Get json data
|
||||
set_fact:
|
||||
jsondata: "{{ result.stdout | from_json }}"
|
||||
|
||||
- fail:
|
||||
msg: "Validation failed: some of the validations has failed."
|
||||
when: item != "PASSED"
|
||||
loop: "{{ status }}"
|
||||
- name: Get Validations Status
|
||||
set_fact:
|
||||
status: "{{ jsondata | json_query(jsonres) }}"
|
||||
vars:
|
||||
jsonres: 'results[*].Status'
|
||||
|
||||
- fail:
|
||||
msg: "Validation failed: some of the validations has failed."
|
||||
when:
|
||||
- item != "PASSED"
|
||||
loop: "{{ status }}"
|
||||
when: v_action == "run"
|
||||
|
@ -27,9 +27,6 @@ classifier =
|
||||
packages =
|
||||
validations_common
|
||||
|
||||
scripts =
|
||||
validations_common/validation.py
|
||||
|
||||
data_files =
|
||||
share/ansible/roles = roles/*
|
||||
share/ansible/roles = validations_common/roles/*
|
||||
|
@ -1,126 +0,0 @@
|
||||
# -*- coding: utf-8 -*-
|
||||
|
||||
# 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.
|
||||
|
||||
"""
|
||||
test_validation
|
||||
----------------------------------
|
||||
|
||||
Tests for `validation` sub module.
|
||||
|
||||
"""
|
||||
import argparse
|
||||
|
||||
from unittest import mock
|
||||
|
||||
from validations_common import validation
|
||||
from validations_common.tests import base
|
||||
|
||||
|
||||
class TestValidationModule(base.TestCase):
|
||||
|
||||
def setUp(self):
|
||||
self.module = mock.MagicMock()
|
||||
|
||||
super(TestValidationModule, self).setUp()
|
||||
|
||||
def test_module_init(self):
|
||||
"""Check for presence of required module attributes.
|
||||
Including the color constants needed for output formatting.
|
||||
"""
|
||||
required_attributes = [
|
||||
'DESCRIPTION',
|
||||
'EPILOG',
|
||||
'RED',
|
||||
'GREEN',
|
||||
'CYAN',
|
||||
'RESET'
|
||||
]
|
||||
|
||||
module_attributes = set(dir(validation))
|
||||
|
||||
self.assertTrue(set(required_attributes).issubset(module_attributes))
|
||||
|
||||
|
||||
class TestCommaListGroupAction(base.TestCase):
|
||||
|
||||
def setUp(self):
|
||||
super(TestCommaListGroupAction, self).setUp()
|
||||
|
||||
@mock.patch('validations_common.validation.setattr')
|
||||
def test_comma_list_group_action(self, mock_setattr):
|
||||
"""As there is not much to test in this class
|
||||
the assertions are made on attributes and inheritance.
|
||||
"""
|
||||
action = validation._CommaListGroupAction(
|
||||
dest='foo',
|
||||
option_strings='')
|
||||
|
||||
action('foo', 'bar', 'fizz,buzz')
|
||||
|
||||
self.assertEqual(type(action).__mro__[1], argparse.Action)
|
||||
self.assertEqual(action.dest, 'foo')
|
||||
self.assertEqual(action.option_strings, '')
|
||||
mock_setattr.assert_called_once_with('bar', 'foo', ['fizz', 'buzz'])
|
||||
|
||||
|
||||
class TestCommaListAction(base.TestCase):
|
||||
|
||||
def setUp(self):
|
||||
super(TestCommaListAction, self).setUp()
|
||||
|
||||
@mock.patch('validations_common.validation.setattr')
|
||||
def test_comma_list_action(self, mock_setattr):
|
||||
"""As there is not much to test in this class
|
||||
the assertions are made on attributes and inheritance.
|
||||
"""
|
||||
action = validation._CommaListAction(
|
||||
dest='foo',
|
||||
option_strings='')
|
||||
|
||||
action('foo', 'bar', 'fizz,buzz')
|
||||
|
||||
self.assertEqual(type(action).__mro__[1], argparse.Action)
|
||||
self.assertEqual(action.dest, 'foo')
|
||||
self.assertEqual(action.option_strings, '')
|
||||
mock_setattr.assert_called_once_with('bar', 'foo', ['fizz', 'buzz'])
|
||||
|
||||
|
||||
class TestValidation(base.TestCase):
|
||||
|
||||
def setUp(self):
|
||||
super(TestValidation, self).setUp()
|
||||
|
||||
def test_validation_init(self):
|
||||
pass
|
||||
|
||||
def test_validation_parser(self):
|
||||
pass
|
||||
|
||||
def test_validation_print_dict_table(self):
|
||||
pass
|
||||
|
||||
def test_validation_print_tuple_table(self):
|
||||
pass
|
||||
|
||||
def test_validation_write_output(self):
|
||||
pass
|
||||
|
||||
def test_validation_take_action_run(self):
|
||||
pass
|
||||
|
||||
def test_validation_take_action_list(self):
|
||||
pass
|
||||
|
||||
def test_validation_take_action_show(self):
|
||||
pass
|
@ -1,249 +0,0 @@
|
||||
#!/usr/bin/env 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.
|
||||
"""
|
||||
Uses `validations_libs`_.
|
||||
|
||||
.. _validations_libs: https://opendev.org/openstack/validations-libs
|
||||
"""
|
||||
import argparse
|
||||
import json
|
||||
import logging
|
||||
import sys
|
||||
|
||||
from prettytable import PrettyTable
|
||||
|
||||
from validations_libs.validation_actions import ValidationActions
|
||||
from validations_libs import constants
|
||||
|
||||
DESCRIPTION = "Run, show or list Validations."
|
||||
EPILOG = "Example: ./validation run --validation check-ftype,512e"
|
||||
# PrettyTable
|
||||
RED = "\033[1;31m"
|
||||
GREEN = "\033[0;32m"
|
||||
CYAN = "\033[36m"
|
||||
RESET = "\033[0;0m"
|
||||
|
||||
|
||||
class _CommaListGroupAction(argparse.Action):
|
||||
def __call__(self, parser, namespace, values, option_string=None):
|
||||
setattr(namespace, self.dest, values.split(','))
|
||||
|
||||
|
||||
class _CommaListAction(argparse.Action):
|
||||
def __call__(self, parser, namespace, values, option_string=None):
|
||||
setattr(namespace, self.dest, values.split(','))
|
||||
|
||||
|
||||
class ValidationsFormatter(argparse.ArgumentDefaultsHelpFormatter, argparse.RawTextHelpFormatter):
|
||||
"""
|
||||
Composite class inheriting from both ArgumentDefaultsHelpFormatter
|
||||
and RawTextHelpFormatter.
|
||||
Thus allowing for both more precise help output management
|
||||
and automatic printing of default arg values.
|
||||
"""
|
||||
pass
|
||||
|
||||
|
||||
class Validation(argparse.ArgumentParser):
|
||||
"""Validation client implementation class"""
|
||||
|
||||
log = logging.getLogger(__name__ + ".Validation")
|
||||
|
||||
def __init__(self, description=DESCRIPTION, epilog=EPILOG):
|
||||
"""Init validation paser"""
|
||||
super(Validation, self).__init__(
|
||||
description=DESCRIPTION,
|
||||
epilog=EPILOG,
|
||||
formatter_class=ValidationsFormatter)
|
||||
|
||||
def parser(self, parser):
|
||||
"""Argument parser for validation"""
|
||||
parser.add_argument('action',
|
||||
choices=['run', 'list', 'show'],
|
||||
help=(
|
||||
'Validation Action: \n'
|
||||
' run:\n launches validations specified by '
|
||||
'the --group or --validation args. \n'
|
||||
' list:\n prints list of available validations,'
|
||||
'including their groups. \n'
|
||||
' show:\n prints list of validations executions and related info.'
|
||||
))
|
||||
parser.add_argument('--inventory', '-i', type=str,
|
||||
default="localhost",
|
||||
help=(
|
||||
"Either a path of the Ansible inventory file, \n"
|
||||
"or a comma-separated list of hosts. \n"
|
||||
))
|
||||
parser.add_argument('--extra-vars', action='store',
|
||||
nargs='+',
|
||||
help="Extra ansible variables")
|
||||
parser.add_argument('--validation', '-v',
|
||||
metavar='<validation_id>[,<validation_id>,...]',
|
||||
dest="validation_name",
|
||||
action=_CommaListAction,
|
||||
default=[],
|
||||
help=("Run specific validations, \n"
|
||||
"if more than one validation is required \n"
|
||||
"separate the names with commas: \n"
|
||||
"--validation check-ftype,512e | \n"
|
||||
"--validation 512e \n"))
|
||||
parser.add_argument('--group', '-g',
|
||||
metavar='<group>[,<group>,...]',
|
||||
action=_CommaListGroupAction,
|
||||
default=[],
|
||||
help=("Run specific group of validations, \n"
|
||||
"if more than one group is required \n"
|
||||
"separate the group names with commas: \n"
|
||||
"--group pre-upgrade,prep | \n"
|
||||
"--group openshift-on-openstack \n"))
|
||||
parser.add_argument('--quiet', action='store_true',
|
||||
help=("Run Ansible in silent mode. \n"))
|
||||
parser.add_argument('--validation-dir', dest='validation_dir',
|
||||
default=constants.ANSIBLE_VALIDATION_DIR,
|
||||
help=("Path where the validation playbooks "
|
||||
"are located. \n"))
|
||||
parser.add_argument('--ansible-base-dir', dest='ansible_base_dir',
|
||||
default=constants.DEFAULT_VALIDATIONS_BASEDIR,
|
||||
help=("Path where the ansible roles, library \n"
|
||||
"and plugins are located. \n"))
|
||||
parser.add_argument('--output-log', dest='output_log',
|
||||
default=None,
|
||||
help=("Path where the run result will be stored. \n"))
|
||||
|
||||
return parser.parse_args()
|
||||
|
||||
def _print_dict_table(self, data):
|
||||
"""Print table from python dict with PrettyTable"""
|
||||
t = PrettyTable(border=True, header=True, padding_width=1)
|
||||
# Set Field name by getting the result dict keys
|
||||
try:
|
||||
t.field_names = data[0].keys()
|
||||
t.align = 'l'
|
||||
except KeyError:
|
||||
raise KeyError()
|
||||
for r in data:
|
||||
if r.get('Status_by_Host'):
|
||||
h = []
|
||||
for host in r['Status_by_Host'].split(', '):
|
||||
_name, _status = host.split(',')
|
||||
color = (GREEN if _status == 'PASSED' else RED)
|
||||
_name = '{}{}{}'.format(color, _name, RESET)
|
||||
h.append(_name)
|
||||
r['Status_by_Host'] = ', '.join(h)
|
||||
if r.get('Status'):
|
||||
status = r.get('Status')
|
||||
color = (CYAN if status in ['starting', 'running']
|
||||
else GREEN if status == 'PASSED' else RED)
|
||||
r['Status'] = '{}{}{}'.format(color, status, RESET)
|
||||
t.add_row(r.values())
|
||||
print(t)
|
||||
|
||||
def _print_tuple_table(self, data, status_col=None):
|
||||
"""Print table from python Tuple with PrettyTable"""
|
||||
if isinstance(data, tuple):
|
||||
t = PrettyTable(border=True, header=True, padding_width=1)
|
||||
try:
|
||||
t.field_names = data[0]
|
||||
t.align = 'l'
|
||||
except KeyError:
|
||||
raise KeyError()
|
||||
for r in data[1]:
|
||||
if status_col:
|
||||
_result = list(r)
|
||||
try:
|
||||
_status = _result[status_col]
|
||||
color = (GREEN if _status == 'PASSED' else RED)
|
||||
_result[status_col] = '{}{}{}'.format(color,
|
||||
_status,
|
||||
RESET)
|
||||
except ValueError:
|
||||
logging.warning('No status found.')
|
||||
t.add_row(_result)
|
||||
else:
|
||||
t.add_row(r)
|
||||
print(t)
|
||||
else:
|
||||
raise TypeError(
|
||||
(
|
||||
"data must be of 'tuple' type. "
|
||||
"Instead {} was provided."
|
||||
).format(type(data))
|
||||
)
|
||||
|
||||
def _write_output(self, output_log, results):
|
||||
"""Write output log file as Json format"""
|
||||
with open(output_log, 'w') as output:
|
||||
output.write(json.dumps({'results': results}, indent=4,
|
||||
sort_keys=True))
|
||||
|
||||
def take_action(self, parsed_args):
|
||||
"""Take validation action"""
|
||||
# Get parameters:
|
||||
action = parsed_args.action
|
||||
inventory = parsed_args.inventory
|
||||
group = parsed_args.group
|
||||
validation_name = parsed_args.validation_name
|
||||
quiet = parsed_args.quiet
|
||||
validation_dir = parsed_args.validation_dir
|
||||
ansible_base_dir = parsed_args.ansible_base_dir
|
||||
extra_vars = parsed_args.extra_vars
|
||||
if extra_vars:
|
||||
try:
|
||||
extra_vars = dict(e.split("=") for e in parsed_args.extra_vars)
|
||||
except ValueError as error:
|
||||
msg = "extra vars option should be formed as: KEY=VALUE."
|
||||
raise RuntimeError(msg) from error
|
||||
v_actions = ValidationActions(validation_path=validation_dir,
|
||||
group=group)
|
||||
if 'run' in action:
|
||||
try:
|
||||
results = v_actions.run_validations(
|
||||
inventory=inventory,
|
||||
group=group,
|
||||
validation_name=validation_name,
|
||||
base_dir=ansible_base_dir,
|
||||
extra_vars=extra_vars,
|
||||
quiet=quiet)
|
||||
except RuntimeError as e:
|
||||
sys.exit(e)
|
||||
if results:
|
||||
if parsed_args.output_log:
|
||||
self._write_output(parsed_args.output_log, results)
|
||||
else:
|
||||
self._print_dict_table(results)
|
||||
elif 'list' in action:
|
||||
results = v_actions.list_validations()
|
||||
if results:
|
||||
if parsed_args.output_log:
|
||||
self._write_output(parsed_args.output_log, results)
|
||||
else:
|
||||
self._print_tuple_table(results)
|
||||
elif 'show' in action:
|
||||
results = v_actions.show_history(validation_name)
|
||||
if results:
|
||||
if parsed_args.output_log:
|
||||
self._write_output(parsed_args.output_log, results)
|
||||
else:
|
||||
self._print_tuple_table(data=results, status_col=2)
|
||||
else:
|
||||
msg = "Unknown Action: {}".format(action)
|
||||
raise RuntimeError(msg)
|
||||
|
||||
|
||||
if __name__ == "__main__":
|
||||
validation = Validation()
|
||||
args = validation.parser(validation)
|
||||
validation.take_action(args)
|
Loading…
Reference in New Issue
Block a user