1# Copyright 2017, The Android Open Source Project 2# 3# Licensed under the Apache License, Version 2.0 (the "License"); 4# you may not use this file except in compliance with the License. 5# You may obtain a copy of the License at 6# 7# http://www.apache.org/licenses/LICENSE-2.0 8# 9# Unless required by applicable law or agreed to in writing, software 10# distributed under the License is distributed on an "AS IS" BASIS, 11# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. 12# See the License for the specific language governing permissions and 13# limitations under the License. 14 15""" 16Aggregates test runners, groups tests by test runners and kicks off tests. 17""" 18 19# pylint: disable=line-too-long 20# pylint: disable=import-outside-toplevel 21 22import itertools 23import time 24import traceback 25 26from typing import Any, Dict, List 27 28from atest import atest_error 29from atest import bazel_mode 30from atest import constants 31from atest import module_info 32from atest import result_reporter 33from atest import atest_utils 34 35from atest.atest_enum import ExitCode 36from atest.metrics import metrics 37from atest.metrics import metrics_utils 38from atest.test_finders import test_info 39from atest.test_runners import atest_tf_test_runner 40from atest.test_runners import roboleaf_test_runner 41from atest.test_runners import robolectric_test_runner 42from atest.test_runners import suite_plan_test_runner 43from atest.test_runners import vts_tf_test_runner 44 45_TEST_RUNNERS = { 46 atest_tf_test_runner.AtestTradefedTestRunner.NAME: atest_tf_test_runner.AtestTradefedTestRunner, 47 robolectric_test_runner.RobolectricTestRunner.NAME: robolectric_test_runner.RobolectricTestRunner, 48 suite_plan_test_runner.SuitePlanTestRunner.NAME: suite_plan_test_runner.SuitePlanTestRunner, 49 vts_tf_test_runner.VtsTradefedTestRunner.NAME: vts_tf_test_runner.VtsTradefedTestRunner, 50 bazel_mode.BazelTestRunner.NAME: bazel_mode.BazelTestRunner, 51 roboleaf_test_runner.RoboleafTestRunner.NAME: roboleaf_test_runner.RoboleafTestRunner, 52} 53 54 55def _get_test_runners(): 56 """Returns the test runners. 57 58 If external test runners are defined outside atest, they can be try-except 59 imported into here. 60 61 Returns: 62 Dict of test runner name to test runner class. 63 """ 64 test_runners_dict = _TEST_RUNNERS 65 # Example import of example test runner: 66 try: 67 from test_runners import example_test_runner 68 test_runners_dict[example_test_runner.ExampleTestRunner.NAME] = example_test_runner.ExampleTestRunner 69 except ImportError: 70 pass 71 return test_runners_dict 72 73 74def group_tests_by_test_runners(test_infos): 75 """Group the test_infos by test runners 76 77 Args: 78 test_infos: List of TestInfo. 79 80 Returns: 81 List of tuples (test runner, tests). 82 """ 83 tests_by_test_runner = [] 84 test_runner_dict = _get_test_runners() 85 key = lambda x: x.test_runner 86 sorted_test_infos = sorted(list(test_infos), key=key) 87 for test_runner, tests in itertools.groupby(sorted_test_infos, key): 88 # groupby returns a grouper object, we want to operate on a list. 89 tests = list(tests) 90 test_runner_class = test_runner_dict.get(test_runner) 91 if test_runner_class is None: 92 raise atest_error.UnknownTestRunnerError('Unknown Test Runner %s' % 93 test_runner) 94 tests_by_test_runner.append((test_runner_class, tests)) 95 return tests_by_test_runner 96 97 98def get_test_runner_reqs(mod_info: module_info.ModuleInfo, 99 test_infos: List[test_info.TestInfo], 100 extra_args: Dict[str, Any]=None): 101 """Returns the requirements for all test runners specified in the tests. 102 103 Args: 104 mod_info: ModuleInfo object. 105 test_infos: List of TestInfo. 106 extra_args: Dict of extra args for test runners to use. 107 108 Returns: 109 Set of build targets required by the test runners. 110 """ 111 unused_result_dir = '' 112 test_runner_build_req = set() 113 for test_runner, tests in group_tests_by_test_runners(test_infos): 114 test_runner_build_req |= test_runner( 115 unused_result_dir, 116 mod_info=mod_info, 117 extra_args=extra_args or {}, 118 ).get_test_runner_build_reqs(tests) 119 return test_runner_build_req 120 121 122# pylint: disable=too-many-locals 123def run_all_tests(results_dir, test_infos, extra_args, mod_info, 124 delay_print_summary=False): 125 """Run the given tests. 126 127 Args: 128 results_dir: String directory to store atest results. 129 test_infos: List of TestInfo. 130 extra_args: Dict of extra args for test runners to use. 131 mod_info: ModuleInfo object. 132 133 Returns: 134 0 if tests succeed, non-zero otherwise. 135 """ 136 reporter = result_reporter.ResultReporter( 137 collect_only=extra_args.get(constants.COLLECT_TESTS_ONLY), 138 flakes_info=extra_args.get(constants.FLAKES_INFO)) 139 reporter.print_starting_text() 140 tests_ret_code = ExitCode.SUCCESS 141 for test_runner, tests in group_tests_by_test_runners(test_infos): 142 test_name = ' '.join([test.test_name for test in tests]) 143 test_start = time.time() 144 is_success = True 145 ret_code = ExitCode.TEST_FAILURE 146 stacktrace = '' 147 try: 148 test_runner = test_runner( 149 results_dir, 150 mod_info=mod_info, 151 extra_args=extra_args, 152 ) 153 ret_code = test_runner.run_tests(tests, extra_args, reporter) 154 tests_ret_code |= ret_code 155 # pylint: disable=broad-except 156 except Exception: 157 stacktrace = traceback.format_exc() 158 reporter.runner_failure(test_runner.NAME, stacktrace) 159 tests_ret_code = ExitCode.TEST_FAILURE 160 is_success = False 161 run_time = metrics_utils.convert_duration(time.time() - test_start) 162 metrics.RunnerFinishEvent( 163 duration=run_time, 164 success=is_success, 165 runner_name=test_runner.NAME, 166 test=[{'name': test_name, 167 'result': ret_code, 168 'stacktrace': stacktrace}]) 169 # Tests that spends over 10 mins to finish will be stored in the 170 # shardable test file, and Atest will launch auto-sharding in the next 171 # runs. 172 for test in tests: 173 atest_utils.update_shardable_tests(test.test_name, 174 run_time.get('seconds', 0)) 175 if delay_print_summary: 176 return tests_ret_code, reporter 177 return reporter.print_summary() or tests_ret_code, reporter 178