David Kranz | b9d9750 | 2013-05-01 15:55:04 -0400 | [diff] [blame] | 1 | #!/usr/bin/env python |
| 2 | |
David Kranz | b9d9750 | 2013-05-01 15:55:04 -0400 | [diff] [blame] | 3 | # Copyright 2013 Quanta Research Cambridge, Inc. |
| 4 | # |
| 5 | # Licensed under the Apache License, Version 2.0 (the "License"); |
| 6 | # you may not use this file except in compliance with the License. |
| 7 | # You may obtain a copy of the License at |
| 8 | # |
| 9 | # http://www.apache.org/licenses/LICENSE-2.0 |
| 10 | # |
| 11 | # Unless required by applicable law or agreed to in writing, software |
| 12 | # distributed under the License is distributed on an "AS IS" BASIS, |
| 13 | # WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. |
| 14 | # See the License for the specific language governing permissions and |
| 15 | # limitations under the License. |
| 16 | |
| 17 | import argparse |
Marc Koderer | b060441 | 2013-09-02 09:43:40 +0200 | [diff] [blame] | 18 | import inspect |
Marc Koderer | 888ddc4 | 2013-07-23 16:13:07 +0200 | [diff] [blame] | 19 | import sys |
Will | 16a778a | 2014-01-12 21:49:14 -0500 | [diff] [blame] | 20 | try: |
| 21 | from unittest import loader |
| 22 | except ImportError: |
| 23 | # unittest in python 2.6 does not contain loader, so uses unittest2 |
| 24 | from unittest2 import loader |
Marc Koderer | f863b3b | 2015-12-18 09:21:13 +0100 | [diff] [blame] | 25 | import traceback |
Tin Lam | 336aca7 | 2016-03-07 18:51:44 -0600 | [diff] [blame] | 26 | import warnings |
Marc Koderer | 32221b8e | 2013-08-23 13:57:50 +0200 | [diff] [blame] | 27 | |
Masayuki Igawa | e5c7028 | 2015-10-30 12:18:58 +0900 | [diff] [blame] | 28 | from cliff import command |
Doug Hellmann | 503a048 | 2015-03-11 21:20:04 +0000 | [diff] [blame] | 29 | from oslo_log import log as logging |
Matthew Treinish | 2190551 | 2015-07-13 10:33:35 -0400 | [diff] [blame] | 30 | from oslo_serialization import jsonutils as json |
Matthew Treinish | 96e9e88 | 2014-06-09 18:37:19 -0400 | [diff] [blame] | 31 | from testtools import testsuite |
| 32 | |
Marc Koderer | 211ca96 | 2013-12-05 12:20:21 +0100 | [diff] [blame] | 33 | from tempest.stress import driver |
Marc Koderer | 16648e9 | 2013-10-05 10:49:13 +0200 | [diff] [blame] | 34 | |
| 35 | LOG = logging.getLogger(__name__) |
| 36 | |
Marc Koderer | 32221b8e | 2013-08-23 13:57:50 +0200 | [diff] [blame] | 37 | |
Marc Koderer | b060441 | 2013-09-02 09:43:40 +0200 | [diff] [blame] | 38 | def discover_stress_tests(path="./", filter_attr=None, call_inherited=False): |
Ken'ichi Ohmichi | 2e2ee19 | 2015-11-19 09:48:27 +0000 | [diff] [blame] | 39 | """Discovers all tempest tests and create action out of them""" |
Marc Koderer | 16648e9 | 2013-10-05 10:49:13 +0200 | [diff] [blame] | 40 | LOG.info("Start test discovery") |
Marc Koderer | 32221b8e | 2013-08-23 13:57:50 +0200 | [diff] [blame] | 41 | tests = [] |
| 42 | testloader = loader.TestLoader() |
| 43 | list = testloader.discover(path) |
Masayuki Igawa | 7e9eb54 | 2014-02-17 15:03:44 +0900 | [diff] [blame] | 44 | for func in (testsuite.iterate_tests(list)): |
Marc Koderer | 16648e9 | 2013-10-05 10:49:13 +0200 | [diff] [blame] | 45 | attrs = [] |
Marc Koderer | 32221b8e | 2013-08-23 13:57:50 +0200 | [diff] [blame] | 46 | try: |
| 47 | method_name = getattr(func, '_testMethodName') |
| 48 | full_name = "%s.%s.%s" % (func.__module__, |
| 49 | func.__class__.__name__, |
| 50 | method_name) |
| 51 | test_func = getattr(func, method_name) |
| 52 | # NOTE(mkoderer): this contains a list of all type attributes |
| 53 | attrs = getattr(test_func, "__testtools_attrs") |
| 54 | except Exception: |
| 55 | next |
| 56 | if 'stress' in attrs: |
Matthew Treinish | c795b9e | 2014-06-09 17:01:10 -0400 | [diff] [blame] | 57 | if filter_attr is not None and filter_attr not in attrs: |
Marc Koderer | 32221b8e | 2013-08-23 13:57:50 +0200 | [diff] [blame] | 58 | continue |
| 59 | class_setup_per = getattr(test_func, "st_class_setup_per") |
| 60 | |
| 61 | action = {'action': |
| 62 | "tempest.stress.actions.unit_test.UnitTest", |
| 63 | 'kwargs': {"test_method": full_name, |
| 64 | "class_setup_per": class_setup_per |
| 65 | } |
| 66 | } |
Marc Koderer | b060441 | 2013-09-02 09:43:40 +0200 | [diff] [blame] | 67 | if (not call_inherited and |
| 68 | getattr(test_func, "st_allow_inheritance") is not True): |
| 69 | class_structure = inspect.getmro(test_func.im_class) |
| 70 | if test_func.__name__ not in class_structure[0].__dict__: |
| 71 | continue |
Marc Koderer | 32221b8e | 2013-08-23 13:57:50 +0200 | [diff] [blame] | 72 | tests.append(action) |
| 73 | return tests |
David Kranz | b9d9750 | 2013-05-01 15:55:04 -0400 | [diff] [blame] | 74 | |
David Kranz | b9d9750 | 2013-05-01 15:55:04 -0400 | [diff] [blame] | 75 | |
Masayuki Igawa | e5c7028 | 2015-10-30 12:18:58 +0900 | [diff] [blame] | 76 | class TempestRunStress(command.Command): |
| 77 | |
Tin Lam | 336aca7 | 2016-03-07 18:51:44 -0600 | [diff] [blame] | 78 | @staticmethod |
| 79 | def display_deprecation_warning(): |
| 80 | warnings.simplefilter('once', category=DeprecationWarning) |
| 81 | warnings.warn( |
| 82 | 'Stress tests are deprecated and will be removed from Tempest ' |
| 83 | 'in the Newton release.', |
| 84 | DeprecationWarning) |
| 85 | warnings.resetwarnings() |
| 86 | |
Masayuki Igawa | e5c7028 | 2015-10-30 12:18:58 +0900 | [diff] [blame] | 87 | def get_parser(self, prog_name): |
Tin Lam | 336aca7 | 2016-03-07 18:51:44 -0600 | [diff] [blame] | 88 | self.display_deprecation_warning() |
Masayuki Igawa | e5c7028 | 2015-10-30 12:18:58 +0900 | [diff] [blame] | 89 | pa = super(TempestRunStress, self).get_parser(prog_name) |
| 90 | pa = add_arguments(pa) |
| 91 | return pa |
| 92 | |
| 93 | def take_action(self, pa): |
Marc Koderer | f863b3b | 2015-12-18 09:21:13 +0100 | [diff] [blame] | 94 | try: |
| 95 | action(pa) |
| 96 | except Exception: |
| 97 | LOG.exception("Failure in the stress test framework") |
| 98 | traceback.print_exc() |
| 99 | raise |
Matthew Treinish | 55e29b4 | 2014-05-07 01:04:17 -0400 | [diff] [blame] | 100 | |
Marc Koderer | 876af15 | 2015-12-17 09:49:01 +0100 | [diff] [blame] | 101 | def get_description(self): |
| 102 | return 'Run tempest stress tests' |
| 103 | |
Matthew Treinish | 55e29b4 | 2014-05-07 01:04:17 -0400 | [diff] [blame] | 104 | |
Masayuki Igawa | e5c7028 | 2015-10-30 12:18:58 +0900 | [diff] [blame] | 105 | def add_arguments(parser): |
| 106 | parser.add_argument('-d', '--duration', default=300, type=int, |
| 107 | help="Duration of test in secs") |
| 108 | parser.add_argument('-s', '--serial', action='store_true', |
| 109 | help="Trigger running tests serially") |
| 110 | parser.add_argument('-S', '--stop', action='store_true', |
| 111 | default=False, help="Stop on first error") |
| 112 | parser.add_argument('-n', '--number', type=int, |
| 113 | help="How often an action is executed for each " |
| 114 | "process") |
| 115 | group = parser.add_mutually_exclusive_group(required=True) |
| 116 | group.add_argument('-a', '--all', action='store_true', |
| 117 | help="Execute all stress tests") |
| 118 | parser.add_argument('-T', '--type', |
| 119 | help="Filters tests of a certain type (e.g. gate)") |
| 120 | parser.add_argument('-i', '--call-inherited', action='store_true', |
| 121 | default=False, |
| 122 | help="Call also inherited function with stress " |
| 123 | "attribute") |
| 124 | group.add_argument('-t', "--tests", nargs='?', |
| 125 | help="Name of the file with test description") |
| 126 | return parser |
| 127 | |
| 128 | |
| 129 | def action(ns): |
Marc Koderer | 888ddc4 | 2013-07-23 16:13:07 +0200 | [diff] [blame] | 130 | result = 0 |
Marc Koderer | 32221b8e | 2013-08-23 13:57:50 +0200 | [diff] [blame] | 131 | if not ns.all: |
| 132 | tests = json.load(open(ns.tests, 'r')) |
| 133 | else: |
Marc Koderer | b060441 | 2013-09-02 09:43:40 +0200 | [diff] [blame] | 134 | tests = discover_stress_tests(filter_attr=ns.type, |
| 135 | call_inherited=ns.call_inherited) |
Marc Koderer | 32221b8e | 2013-08-23 13:57:50 +0200 | [diff] [blame] | 136 | |
Matthew Treinish | 8e663fc | 2013-07-16 10:55:22 -0400 | [diff] [blame] | 137 | if ns.serial: |
David Kranz | 71fee47 | 2014-10-13 15:13:02 -0400 | [diff] [blame] | 138 | # Duration is total time |
| 139 | duration = ns.duration / len(tests) |
Matthew Treinish | 8e663fc | 2013-07-16 10:55:22 -0400 | [diff] [blame] | 140 | for test in tests: |
Marc Koderer | 888ddc4 | 2013-07-23 16:13:07 +0200 | [diff] [blame] | 141 | step_result = driver.stress_openstack([test], |
David Kranz | 71fee47 | 2014-10-13 15:13:02 -0400 | [diff] [blame] | 142 | duration, |
Marc Koderer | 3414d73 | 2013-07-31 08:36:36 +0200 | [diff] [blame] | 143 | ns.number, |
| 144 | ns.stop) |
| 145 | # NOTE(mkoderer): we just save the last result code |
Marc Koderer | 888ddc4 | 2013-07-23 16:13:07 +0200 | [diff] [blame] | 146 | if (step_result != 0): |
| 147 | result = step_result |
Attila Fazekas | f2a8bbd | 2014-02-23 17:19:12 +0100 | [diff] [blame] | 148 | if ns.stop: |
| 149 | return result |
Matthew Treinish | 8e663fc | 2013-07-16 10:55:22 -0400 | [diff] [blame] | 150 | else: |
Attila Fazekas | f2a8bbd | 2014-02-23 17:19:12 +0100 | [diff] [blame] | 151 | result = driver.stress_openstack(tests, |
| 152 | ns.duration, |
| 153 | ns.number, |
| 154 | ns.stop) |
Marc Koderer | 888ddc4 | 2013-07-23 16:13:07 +0200 | [diff] [blame] | 155 | return result |
David Kranz | b9d9750 | 2013-05-01 15:55:04 -0400 | [diff] [blame] | 156 | |
| 157 | |
Masayuki Igawa | e5c7028 | 2015-10-30 12:18:58 +0900 | [diff] [blame] | 158 | def main(): |
Tin Lam | 336aca7 | 2016-03-07 18:51:44 -0600 | [diff] [blame] | 159 | TempestRunStress.display_deprecation_warning() |
Masayuki Igawa | e5c7028 | 2015-10-30 12:18:58 +0900 | [diff] [blame] | 160 | parser = argparse.ArgumentParser(description='Run stress tests') |
| 161 | pa = add_arguments(parser) |
| 162 | ns = pa.parse_args() |
| 163 | return action(ns) |
| 164 | |
| 165 | |
Marc Koderer | 888ddc4 | 2013-07-23 16:13:07 +0200 | [diff] [blame] | 166 | if __name__ == "__main__": |
Marc Koderer | 16648e9 | 2013-10-05 10:49:13 +0200 | [diff] [blame] | 167 | try: |
Matthew Treinish | 55e29b4 | 2014-05-07 01:04:17 -0400 | [diff] [blame] | 168 | sys.exit(main()) |
Marc Koderer | 16648e9 | 2013-10-05 10:49:13 +0200 | [diff] [blame] | 169 | except Exception: |
| 170 | LOG.exception("Failure in the stress test framework") |
Marc Koderer | f863b3b | 2015-12-18 09:21:13 +0100 | [diff] [blame] | 171 | traceback.print_exc() |
Marc Koderer | 16648e9 | 2013-10-05 10:49:13 +0200 | [diff] [blame] | 172 | sys.exit(1) |