| Index: build/android/pylib/python_test_base.py
|
| diff --git a/build/android/pylib/python_test_base.py b/build/android/pylib/python_test_base.py
|
| new file mode 100644
|
| index 0000000000000000000000000000000000000000..69e5bbec587fe59949377dee65830202bb5088fe
|
| --- /dev/null
|
| +++ b/build/android/pylib/python_test_base.py
|
| @@ -0,0 +1,177 @@
|
| +# Copyright (c) 2012 The Chromium Authors. All rights reserved.
|
| +# Use of this source code is governed by a BSD-style license that can be
|
| +# found in the LICENSE file.
|
| +
|
| +"""Base class for Android Python-driven tests.
|
| +
|
| +This test case is intended to serve as the base class for any Python-driven
|
| +tests. It is similar to the Python unitttest module in that the user's tests
|
| +inherit from this case and add their tests in that case.
|
| +
|
| +When a PythonTestBase object is instantiated, its purpose is to run only one of
|
| +its tests. The test runner gives it the name of the test the instance will
|
| +run. The test runner calls SetUp with the Android device ID which the test will
|
| +run against. The runner runs the test method itself, collecting the result,
|
| +and calls TearDown.
|
| +
|
| +Tests can basically do whatever they want in the test methods, such as call
|
| +Java tests using _RunJavaTests. Those methods have the advantage of massaging
|
| +the Java test results into Python test results.
|
| +"""
|
| +
|
| +import logging
|
| +import os
|
| +import time
|
| +
|
| +import android_commands
|
| +import apk_info
|
| +from run_java_tests import TestRunner
|
| +import test_options_parser
|
| +from test_result import SingleTestResult, TestResults, PYTHON
|
| +
|
| +
|
| +# aka the parent of com.google.android
|
| +BASE_ROOT = 'src' + os.sep
|
| +
|
| +
|
| +class PythonTestBase(object):
|
| + """Base class for Python-driven tests."""
|
| +
|
| + def __init__(self, test_name):
|
| + # test_name must match one of the test methods defined on a subclass which
|
| + # inherits from this class.
|
| + # It's stored so we can do the attr lookup on demand, allowing this class
|
| + # to be pickled, a requirement for the multiprocessing module.
|
| + self.test_name = test_name
|
| + class_name = self.__class__.__name__
|
| + self.qualified_name = class_name + '.' + self.test_name
|
| + self.ports_to_forward = []
|
| +
|
| + def SetUp(self, device_id, shard_index):
|
| + self.shard_index = shard_index
|
| + self.device_id = device_id
|
| + self.adb = android_commands.AndroidCommands(self.device_id)
|
| +
|
| + def TearDown(self):
|
| + pass
|
| +
|
| + def Run(self):
|
| + logging.warning('Running Python-driven test: %s', self.test_name)
|
| + return getattr(self, self.test_name)()
|
| +
|
| + def _RunJavaTest(self, fname, suite, test):
|
| + """Runs a single Java test with a Java TestRunner.
|
| +
|
| + Args:
|
| + fname: filename for the test (e.g. foo/bar/baz/tests/FooTest.py)
|
| + suite: name of the Java test suite (e.g. FooTest)
|
| + test: name of the test method to run (e.g. testFooBar)
|
| +
|
| + Returns:
|
| + TestResults object with a single test result.
|
| + """
|
| + test = self._ComposeFullTestName(fname, suite, test)
|
| + # Get a set of default options
|
| + options = test_options_parser.ParseInstrumentationArgs([''])
|
| + apks = [apk_info.ApkInfo(options.test_apk_path, options.test_apk_jar_path)]
|
| + java_test_runner = TestRunner(options, self.device_id, [test], False,
|
| + self.shard_index,
|
| + apks,
|
| + self.ports_to_forward)
|
| + return java_test_runner.Run()
|
| +
|
| + def _RunJavaTests(self, fname, tests):
|
| + """Calls a list of tests and stops at the first test failure.
|
| +
|
| + This method iterates until either it encounters a non-passing test or it
|
| + exhausts the list of tests. Then it returns the appropriate Python result.
|
| +
|
| + Args:
|
| + fname: filename for the Python test
|
| + tests: a list of Java test names which will be run
|
| +
|
| + Returns:
|
| + A TestResults object containing a result for this Python test.
|
| + """
|
| + start_ms = int(time.time()) * 1000
|
| +
|
| + result = None
|
| + for test in tests:
|
| + # We're only running one test at a time, so this TestResults object will
|
| + # hold only one result.
|
| + suite, test_name = test.split('.')
|
| + result = self._RunJavaTest(fname, suite, test_name)
|
| + # A non-empty list means the test did not pass.
|
| + if result.GetAllBroken():
|
| + break
|
| +
|
| + duration_ms = int(time.time()) * 1000 - start_ms
|
| +
|
| + # Do something with result.
|
| + return self._ProcessResults(result, start_ms, duration_ms)
|
| +
|
| + def _ProcessResults(self, result, start_ms, duration_ms):
|
| + """Translates a Java test result into a Python result for this test.
|
| +
|
| + The TestRunner class that we use under the covers will return a test result
|
| + for that specific Java test. However, to make reporting clearer, we have
|
| + this method to abstract that detail and instead report that as a failure of
|
| + this particular test case while still including the Java stack trace.
|
| +
|
| + Args:
|
| + result: TestResults with a single Java test result
|
| + start_ms: the time the test started
|
| + duration_ms: the length of the test
|
| +
|
| + Returns:
|
| + A TestResults object containing a result for this Python test.
|
| + """
|
| + test_results = TestResults()
|
| +
|
| + # If our test is in broken, then it crashed/failed.
|
| + broken = result.GetAllBroken()
|
| + if broken:
|
| + # Since we have run only one test, take the first and only item.
|
| + single_result = broken[0]
|
| +
|
| + log = single_result.log
|
| + if not log:
|
| + log = 'No logging information.'
|
| +
|
| + short_error_msg = single_result.log.split('\n')[0]
|
| + # err_info is ostensibly for Sponge to consume; it's a short error
|
| + # message and a longer one.
|
| + err_info = (short_error_msg, log)
|
| +
|
| + python_result = SingleTestResult(self.qualified_name, start_ms,
|
| + duration_ms,
|
| + PYTHON,
|
| + log,
|
| + err_info)
|
| +
|
| + # Figure out where the test belonged. There's probably a cleaner way of
|
| + # doing this.
|
| + if single_result in result.crashed:
|
| + test_results.crashed = [python_result]
|
| + elif single_result in result.failed:
|
| + test_results.failed = [python_result]
|
| + elif single_result in result.unknown:
|
| + test_results.unknown = [python_result]
|
| +
|
| + else:
|
| + python_result = SingleTestResult(self.qualified_name, start_ms,
|
| + duration_ms,
|
| + PYTHON)
|
| + test_results.ok = [python_result]
|
| +
|
| + return test_results
|
| +
|
| + def _ComposeFullTestName(self, fname, suite, test):
|
| + package_name = self._GetPackageName(fname)
|
| + return package_name + '.' + suite + '#' + test
|
| +
|
| + def _GetPackageName(self, fname):
|
| + """Extracts the package name from the test file path."""
|
| + dirname = os.path.dirname(fname)
|
| + package = dirname[dirname.rfind(BASE_ROOT) + len(BASE_ROOT):]
|
| + return package.replace(os.sep, '.')
|
|
|