Chromium Code Reviews
chromiumcodereview-hr@appspot.gserviceaccount.com (chromiumcodereview-hr) | Please choose your nickname with Settings | Help | Chromium Project | Gerrit Changes | Sign out
(646)

Side by Side Diff: build/android/pylib/host_driven/setup.py

Issue 19537004: [Android] Converts host driven tests to common test_dispatcher (Closed) Base URL: https://chromium.googlesource.com/chromium/src.git@sharding_refactoring
Patch Set: Renames InstrumentationPythonTestBase, renames python to host-driven globally Created 7 years, 5 months ago
Use n/p to move between diff chunks; N/P to move between comments. Draft comments are only viewable by you.
Jump to:
View unified diff | Download patch
OLDNEW
1 # Copyright (c) 2012 The Chromium Authors. All rights reserved. 1 # Copyright (c) 2012 The Chromium Authors. All rights reserved.
2 # Use of this source code is governed by a BSD-style license that can be 2 # Use of this source code is governed by a BSD-style license that can be
3 # found in the LICENSE file. 3 # found in the LICENSE file.
4 4
5 """Runs the Python tests (relies on using the Java test runner).""" 5 """Runs the host-driven tests (relies on using the Java test runner)."""
6 6
7 import functools
7 import logging 8 import logging
8 import os 9 import os
9 import sys 10 import sys
10 import types 11 import types
11 12
12 from pylib import android_commands 13 from pylib import android_commands
13 from pylib.base import base_test_result 14 from pylib.base import base_test_result
14 from pylib.instrumentation import test_package 15 from pylib.base import shard
frankf 2013/07/24 01:04:55 you mean test_dispatcher?
gkanwar1 2013/07/24 17:38:04 Actually, this should just get removed, since it's
15 from pylib.instrumentation import test_runner
16 from pylib.utils import report_results 16 from pylib.utils import report_results
17 17
18 import python_test_base 18 import instrumentation_test
19 from python_test_sharder import PythonTestSharder 19 import test_runner
20 from test_info_collection import TestInfoCollection 20 import test_info_collection
21 21
22 22
23 def _GetPythonFiles(root, files): 23 def _GetPythonFiles(root, files):
24 """Returns all files from |files| that end in 'Test.py'. 24 """Returns all files from |files| that end in 'Test.py'.
25 25
26 Args: 26 Args:
27 root: A directory name with python files. 27 root: A directory name with python files.
28 files: A list of file names. 28 files: A list of file names.
29 29
30 Returns: 30 Returns:
31 A list with all Python driven test file paths. 31 A list with all host-driven test file paths.
32 """ 32 """
33 return [os.path.join(root, f) for f in files if f.endswith('Test.py')] 33 return [os.path.join(root, f) for f in files if f.endswith('Test.py')]
34 34
35 35
36 def _InferImportNameFromFile(python_file): 36 def _InferImportNameFromFile(python_file):
37 """Given a file, infer the import name for that file. 37 """Given a file, infer the import name for that file.
38 38
39 Example: /usr/foo/bar/baz.py -> baz. 39 Example: /usr/foo/bar/baz.py -> baz.
40 40
41 Args: 41 Args:
42 python_file: path to the Python file, ostensibly to import later. 42 python_file: Path to the Python file, ostensibly to import later.
43 43
44 Returns: 44 Returns:
45 The module name for the given file. 45 The module name for the given file.
46 """ 46 """
47 return os.path.splitext(os.path.basename(python_file))[0] 47 return os.path.splitext(os.path.basename(python_file))[0]
48 48
49 49
50 def DispatchPythonTests(options): 50 def _GetTestModules(host_driven_test_root, is_official_build):
51 """Dispatches the Python tests. If there are multiple devices, use sharding. 51 """Retrieve a sorted list of host-driven tests.
52 52
53 Args: 53 Walks the location of host-driven tests, imports them, and provides the list
54 options: command line options.
55
56 Returns:
57 A tuple of (base_test_result.TestRunResults object, exit code)
58
59 Raises:
60 Exception: If there are no attached devices.
61 """
62
63 attached_devices = android_commands.GetAttachedDevices()
64 if not attached_devices:
65 raise Exception('You have no devices attached or visible!')
66 if options.test_device:
67 attached_devices = [options.test_device]
68
69 test_collection = TestInfoCollection()
70 all_tests = _GetAllTests(options.python_test_root, options.official_build)
71 test_collection.AddTests(all_tests)
72 test_names = [t.qualified_name for t in all_tests]
73 logging.debug('All available tests: ' + str(test_names))
74
75 available_tests = test_collection.GetAvailableTests(
76 options.annotations, options.exclude_annotations, options.test_filter)
77
78 if not available_tests:
79 logging.warning('No Python tests to run with current args.')
80 return (base_test_result.TestRunResults(), 0)
81
82 test_names = [t.qualified_name for t in available_tests]
83 logging.debug('Final list of tests to run: ' + str(test_names))
84
85 # Copy files to each device before running any tests.
86 for device_id in attached_devices:
87 logging.debug('Pushing files to device %s', device_id)
88 test_pkg = test_package.TestPackage(options.test_apk_path,
89 options.test_apk_jar_path)
90 test_files_copier = test_runner.TestRunner(
91 options.build_type, options.test_data, options.install_apk,
92 options.save_perf_json, options.screenshot_failures, options.tool,
93 options.wait_for_debugger, options.disable_assertions,
94 options.push_deps, options.cleanup_test_files, device_id, 0, test_pkg,
95 [])
96 test_files_copier.InstallTestPackage()
97 if options.push_deps:
98 logging.info('Pushing data deps to device.')
99 test_files_copier.PushDataDeps()
100 else:
101 logging.warning('Skipping pushing data deps to device.')
102
103 # Actually run the tests.
104 if len(attached_devices) > 1 and options.wait_for_debugger:
105 logging.warning('Debugger can not be sharded, '
106 'using first available device')
107 attached_devices = attached_devices[:1]
108 logging.debug('Running Python tests')
109 sharder = PythonTestSharder(attached_devices, available_tests, options)
110 test_results = sharder.RunShardedTests()
111
112 if not test_results.DidRunPass():
113 return (test_results, 1)
114
115 return (test_results, 0)
116
117
118 def _GetTestModules(python_test_root, is_official_build):
119 """Retrieve a sorted list of pythonDrivenTests.
120
121 Walks the location of pythonDrivenTests, imports them, and provides the list
122 of imported modules to the caller. 54 of imported modules to the caller.
123 55
124 Args: 56 Args:
125 python_test_root: the path to walk, looking for pythonDrivenTests 57 host_driven_test_root: The path to walk, looking for the
126 is_official_build: whether to run only those tests marked 'official' 58 pythonDrivenTests or host_driven_tests directory
59 is_official_build: Whether to run only those tests marked 'official'
127 60
128 Returns: 61 Returns:
129 A list of Python modules which may have zero or more tests. 62 A list of Python modules which may have zero or more tests.
130 """ 63 """
131 # By default run all python tests under pythonDrivenTests. 64 # By default run all host-driven tests under pythonDrivenTests or
132 python_test_file_list = [] 65 # host_driven_tests.
133 for root, _, files in os.walk(python_test_root): 66 host_driven_test_file_list = []
67 for root, _, files in os.walk(host_driven_test_root):
134 if (root.endswith('host_driven_tests') or 68 if (root.endswith('host_driven_tests') or
135 root.endswith('pythonDrivenTests') or 69 root.endswith('pythonDrivenTests') or
136 (is_official_build and root.endswith('pythonDrivenTests/official'))): 70 (is_official_build and root.endswith('pythonDrivenTests/official'))):
137 python_test_file_list += _GetPythonFiles(root, files) 71 host_driven_test_file_list += _GetPythonFiles(root, files)
138 python_test_file_list.sort() 72 host_driven_test_file_list.sort()
139 73
140 test_module_list = [_GetModuleFromFile(test_file) 74 test_module_list = [_GetModuleFromFile(test_file)
141 for test_file in python_test_file_list] 75 for test_file in host_driven_test_file_list]
142 return test_module_list 76 return test_module_list
143 77
144 78
145 def _GetModuleFromFile(python_file): 79 def _GetModuleFromFile(python_file):
146 """Gets the module associated with a file by importing it. 80 """Gets the module associated with a file by importing it.
147 81
148 Args: 82 Args:
149 python_file: file to import 83 python_file: file to import
150 84
151 Returns: 85 Returns:
152 The module object. 86 The module object.
153 """ 87 """
154 sys.path.append(os.path.dirname(python_file)) 88 sys.path.append(os.path.dirname(python_file))
155 import_name = _InferImportNameFromFile(python_file) 89 import_name = _InferImportNameFromFile(python_file)
156 return __import__(import_name) 90 return __import__(import_name)
157 91
158 92
159 def _GetTestsFromClass(test_class): 93 def _GetTestsFromClass(test_class, extra_args):
160 """Create a list of test objects for each test method on this class. 94 """Create a list of test objects for each test method on this class.
161 95
162 Test methods are methods on the class which begin with 'test'. 96 Test methods are methods on the class which begin with 'test'.
163 97
164 Args: 98 Args:
165 test_class: class object which contains zero or more test methods. 99 test_class: Class object which contains zero or more test methods.
100 extra_args: List of extra args to pass into the constructor of the
101 test cases.
166 102
167 Returns: 103 Returns:
168 A list of test objects, each of which is bound to one test. 104 A list of test case objects, each initialized for particular test method.
169 """ 105 """
170 test_names = [m for m in dir(test_class) 106 test_names = [m for m in dir(test_class)
171 if _IsTestMethod(m, test_class)] 107 if _IsTestMethod(m, test_class)]
172 return map(test_class, test_names) 108 return [test_class(name, *extra_args) for name in test_names]
173 109
174 110
175 def _GetTestClassesFromModule(test_module): 111 def _GetTestClassesFromModule(test_module, extra_args):
112 """Get all test classes from |test_module|."""
113
176 tests = [] 114 tests = []
177 for name in dir(test_module): 115 for name in dir(test_module):
178 attr = getattr(test_module, name) 116 attr = getattr(test_module, name)
179 if _IsTestClass(attr): 117 if _IsTestClass(attr):
180 tests.extend(_GetTestsFromClass(attr)) 118 tests.extend(_GetTestsFromClass(attr, extra_args))
181 return tests 119 return tests
182 120
183 121
184 def _IsTestClass(test_class): 122 def _IsTestClass(test_class,
123 base_class=instrumentation_test.HostDrivenInstrumentationTest):
185 return (type(test_class) is types.TypeType and 124 return (type(test_class) is types.TypeType and
186 issubclass(test_class, python_test_base.PythonTestBase) and 125 issubclass(test_class, base_class) and
187 test_class is not python_test_base.PythonTestBase) 126 test_class is not base_class)
188 127
189 128
190 def _IsTestMethod(attrname, test_case_class): 129 def _IsTestMethod(attrname, test_case_class):
191 """Checks whether this is a valid test method. 130 """Checks whether this is a valid test method.
192 131
193 Args: 132 Args:
194 attrname: the method name. 133 attrname: the method name.
195 test_case_class: the test case class. 134 test_case_class: the test case class.
196 135
197 Returns: 136 Returns:
198 True if test_case_class.'attrname' is callable and it starts with 'test'; 137 True if test_case_class.'attrname' is callable and it starts with 'test';
199 False otherwise. 138 False otherwise.
200 """ 139 """
201 attr = getattr(test_case_class, attrname) 140 attr = getattr(test_case_class, attrname)
202 return callable(attr) and attrname.startswith('test') 141 return callable(attr) and attrname.startswith('test')
203 142
204 143
205 def _GetAllTests(test_root, is_official_build): 144 def _GetAllTests(test_root, is_official_build, extra_args):
206 """Retrieve a list of Python test modules and their respective methods. 145 """Retrieve a list of host-driven test modules and their respective methods.
207 146
208 Args: 147 Args:
209 test_root: path which contains Python-driven test files 148 test_root: Path which contains host-driven test files
210 is_official_build: whether this is an official build 149 is_official_build: Whether this is an official build
150 extra_args: List of extra args to pass to the constructor of the test case.
211 151
212 Returns: 152 Returns:
213 List of test case objects for all available test methods. 153 List of test case objects for all available test methods.
214 """ 154 """
215 if not test_root: 155 if not test_root:
216 return [] 156 return []
217 all_tests = [] 157 all_tests = []
218 test_module_list = _GetTestModules(test_root, is_official_build) 158 test_module_list = _GetTestModules(test_root, is_official_build)
219 for module in test_module_list: 159 for module in test_module_list:
220 all_tests.extend(_GetTestClassesFromModule(module)) 160 all_tests.extend(_GetTestClassesFromModule(module, extra_args))
221 return all_tests 161 return all_tests
162
163
164 def InstrumentationSetup(host_driven_test_root, official_build, annotations,
165 exclude_annotations, test_filter, tool, build_type,
166 push_deps, cleanup_test_files, test_apk_path,
167 test_apk_jar_path, test_data, install_apk,
168 save_perf_json, screenshot_failures,
169 wait_for_debugger, disable_assertions):
170 """Creates test set of host-driven instrumentation tests and runner factory.
171
172 Args:
173 host_driven_test_root: Directory where the host-driven tests are.
174 official_build: True if this is an official build.
175 annotations: Annotations for the tests.
176 exclude_annotations: Any annotations to exclude from running.
177 test_filter: Filter string for tests.
178 tool: Name of the Valgrind tool.
179 build_type: 'Release' or 'Debug'.
180 push_deps: If True, push all dependencies to the device.
181 cleanup_test_files: If True, cleanup test files on device.
182 test_apk_path: Path to the test apk file.
183 test_apk_jar_path: Path to the accompanying jar file.
184 test_data: Location of the test data.
185 install_apk: Re-installs the apk if opted.
186 save_perf_json: Whether or not to save the JSON file from UI perf tests.
187 screenshot_failures: Take a screenshot for a test failure
188 wait_for_debugger: Blocks until the debugger is connected.
189 disable_assertions: Whether to disable java assertions on the device.
190
191 Returns:
192 A tuple of (TestRunnerFactory, tests).
193 """
194
195 test_collection = test_info_collection.TestInfoCollection()
196 all_tests = _GetAllTests(
197 host_driven_test_root, official_build, [
198 test_apk_path, test_apk_jar_path, test_data, install_apk,
199 save_perf_json, screenshot_failures, tool, wait_for_debugger,
200 disable_assertions])
201 test_collection.AddTests(all_tests)
202 test_names = [t.qualified_name for t in all_tests]
203 logging.debug('All available tests: ' + str(test_names))
204
205 available_tests = test_collection.GetAvailableTests(
206 annotations, exclude_annotations, test_filter)
207
208 def TestRunnerFactory(device, shard_index):
209 return test_runner.HostDrivenTestRunner(
210 device, shard_index, tool, build_type, push_deps, cleanup_test_files)
211
212 return (TestRunnerFactory, available_tests)
OLDNEW

Powered by Google App Engine
This is Rietveld 408576698