From 10361e1e8b4ade8c03e0ce53e8ce9a5db94e1ecd Mon Sep 17 00:00:00 2001 From: matbu Date: Thu, 15 Apr 2021 16:17:18 +0200 Subject: [PATCH] Use new CLI for functionnal tests and remove validation.py script Depends-On: https://review.opendev.org/c/openstack/validations-libs/+/782574 Change-Id: Ib2da7a386080094f8733f831cb7c8bfa07c12222 --- roles/validations/defaults/main.yaml | 14 +- roles/validations/tasks/main.yaml | 2 +- roles/validations/tasks/validations.yaml | 41 ++-- setup.cfg | 3 - validations_common/tests/test_validation.py | 126 ---------- validations_common/validation.py | 249 -------------------- 6 files changed, 32 insertions(+), 403 deletions(-) delete mode 100644 validations_common/tests/test_validation.py delete mode 100755 validations_common/validation.py diff --git a/roles/validations/defaults/main.yaml b/roles/validations/defaults/main.yaml index 1b166de..9bd0d52 100644 --- a/roles/validations/defaults/main.yaml +++ b/roles/validations/defaults/main.yaml @@ -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 diff --git a/roles/validations/tasks/main.yaml b/roles/validations/tasks/main.yaml index f46d096..4e8c53d 100644 --- a/roles/validations/tasks/main.yaml +++ b/roles/validations/tasks/main.yaml @@ -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}}" diff --git a/roles/validations/tasks/validations.yaml b/roles/validations/tasks/validations.yaml index 8484cf1..a0e64c3 100644 --- a/roles/validations/tasks/validations.yaml +++ b/roles/validations/tasks/validations.yaml @@ -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" diff --git a/setup.cfg b/setup.cfg index 8f71866..e279978 100644 --- a/setup.cfg +++ b/setup.cfg @@ -26,9 +26,6 @@ classifier = packages = validations_common -scripts = - validations_common/validation.py - data_files = share/ansible/roles = roles/* share/ansible/roles = validations_common/roles/* diff --git a/validations_common/tests/test_validation.py b/validations_common/tests/test_validation.py deleted file mode 100644 index 7f784a6..0000000 --- a/validations_common/tests/test_validation.py +++ /dev/null @@ -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 diff --git a/validations_common/validation.py b/validations_common/validation.py deleted file mode 100755 index 0a3dfe0..0000000 --- a/validations_common/validation.py +++ /dev/null @@ -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='[,,...]', - 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='[,,...]', - 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)