• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
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