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
26import atest_error
27import constants
28import result_reporter
29
30from metrics import metrics
31from metrics import metrics_utils
32from test_runners import atest_tf_test_runner
33from test_runners import robolectric_test_runner
34from test_runners import suite_plan_test_runner
35from test_runners import vts_tf_test_runner
36
37_TEST_RUNNERS = {
38    atest_tf_test_runner.AtestTradefedTestRunner.NAME: atest_tf_test_runner.AtestTradefedTestRunner,
39    robolectric_test_runner.RobolectricTestRunner.NAME: robolectric_test_runner.RobolectricTestRunner,
40    suite_plan_test_runner.SuitePlanTestRunner.NAME: suite_plan_test_runner.SuitePlanTestRunner,
41    vts_tf_test_runner.VtsTradefedTestRunner.NAME: vts_tf_test_runner.VtsTradefedTestRunner,
42}
43
44
45def _get_test_runners():
46    """Returns the test runners.
47
48    If external test runners are defined outside atest, they can be try-except
49    imported into here.
50
51    Returns:
52        Dict of test runner name to test runner class.
53    """
54    test_runners_dict = _TEST_RUNNERS
55    # Example import of example test runner:
56    try:
57        from test_runners import example_test_runner
58        test_runners_dict[example_test_runner.ExampleTestRunner.NAME] = example_test_runner.ExampleTestRunner
59    except ImportError:
60        pass
61    return test_runners_dict
62
63
64def group_tests_by_test_runners(test_infos):
65    """Group the test_infos by test runners
66
67    Args:
68        test_infos: List of TestInfo.
69
70    Returns:
71        List of tuples (test runner, tests).
72    """
73    tests_by_test_runner = []
74    test_runner_dict = _get_test_runners()
75    key = lambda x: x.test_runner
76    sorted_test_infos = sorted(list(test_infos), key=key)
77    for test_runner, tests in itertools.groupby(sorted_test_infos, key):
78        # groupby returns a grouper object, we want to operate on a list.
79        tests = list(tests)
80        test_runner_class = test_runner_dict.get(test_runner)
81        if test_runner_class is None:
82            raise atest_error.UnknownTestRunnerError('Unknown Test Runner %s' %
83                                                     test_runner)
84        tests_by_test_runner.append((test_runner_class, tests))
85    return tests_by_test_runner
86
87
88def get_test_runner_reqs(module_info, test_infos):
89    """Returns the requirements for all test runners specified in the tests.
90
91    Args:
92        module_info: ModuleInfo object.
93        test_infos: List of TestInfo.
94
95    Returns:
96        Set of build targets required by the test runners.
97    """
98    unused_result_dir = ''
99    test_runner_build_req = set()
100    for test_runner, _ in group_tests_by_test_runners(test_infos):
101        test_runner_build_req |= test_runner(
102            unused_result_dir,
103            module_info=module_info).get_test_runner_build_reqs()
104    return test_runner_build_req
105
106
107def run_all_tests(results_dir, test_infos, extra_args, module_info,
108                  delay_print_summary=False):
109    """Run the given tests.
110
111    Args:
112        results_dir: String directory to store atest results.
113        test_infos: List of TestInfo.
114        extra_args: Dict of extra args for test runners to use.
115        module_info: ModuleInfo object.
116
117    Returns:
118        0 if tests succeed, non-zero otherwise.
119    """
120    reporter = result_reporter.ResultReporter(
121        collect_only=extra_args.get(constants.COLLECT_TESTS_ONLY),
122        flakes_info=extra_args.get(constants.FLAKES_INFO))
123    reporter.print_starting_text()
124    tests_ret_code = constants.EXIT_CODE_SUCCESS
125    for test_runner, tests in group_tests_by_test_runners(test_infos):
126        test_name = ' '.join([test.test_name for test in tests])
127        test_start = time.time()
128        is_success = True
129        ret_code = constants.EXIT_CODE_TEST_FAILURE
130        stacktrace = ''
131        try:
132            test_runner = test_runner(results_dir, module_info=module_info)
133            ret_code = test_runner.run_tests(tests, extra_args, reporter)
134            tests_ret_code |= ret_code
135        # pylint: disable=broad-except
136        except Exception:
137            stacktrace = traceback.format_exc()
138            reporter.runner_failure(test_runner.NAME, stacktrace)
139            tests_ret_code = constants.EXIT_CODE_TEST_FAILURE
140            is_success = False
141        metrics.RunnerFinishEvent(
142            duration=metrics_utils.convert_duration(time.time() - test_start),
143            success=is_success,
144            runner_name=test_runner.NAME,
145            test=[{'name': test_name,
146                   'result': ret_code,
147                   'stacktrace': stacktrace}])
148    if delay_print_summary:
149        return tests_ret_code, reporter
150    return reporter.print_summary() or tests_ret_code, reporter
151