blob: 4df87d6f51c0f20eec618688829e3d969b9d7e16 [file] [log] [blame]
# Copyright 2014 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.
import datetime
import json
import re
import string
class Test(object):
"""
Base class for tests that can be retried after deapplying a previously
applied patch.
"""
def __init__(self):
super(Test, self).__init__()
self._test_runs = {}
@property
def abort_on_failure(self):
"""If True, abort build when test fails."""
return False
@property
def name(self): # pragma: no cover
"""Name of the test."""
raise NotImplementedError()
def isolate_target(self, _api):
"""Returns isolate target name. Defaults to name.
The _api is here in case classes want to use api information to alter the
isolation target.
"""
return self.name # pragma: no cover
@staticmethod
def compile_targets(api):
"""List of compile targets needed by this test."""
raise NotImplementedError() # pragma: no cover
def pre_run(self, api, suffix, test_filter=None): # pragma: no cover
"""Steps to execute before running the test."""
return []
def run(self, api, suffix, test_filter=None): # pragma: no cover
"""Run the test. suffix is 'with patch' or 'without patch'."""
raise NotImplementedError()
def post_run(self, api, suffix, test_filter=None): # pragma: no cover
"""Steps to execute after running the test."""
return []
def has_valid_results(self, api, suffix): # pragma: no cover
"""
Returns True if results (failures) are valid.
This makes it possible to distinguish between the case of no failures
and the test failing to even report its results in machine-readable
format.
"""
raise NotImplementedError()
def failures(self, api, suffix): # pragma: no cover
"""Return list of failures (list of strings)."""
raise NotImplementedError()
@property
def uses_swarming(self):
"""Returns true if the test uses swarming."""
return False
@property
def uses_local_devices(self):
return False # pragma: no cover
def _step_name(self, suffix):
"""Helper to uniformly combine tests's name with a suffix."""
if not suffix:
return self.name
return '%s (%s)' % (self.name, suffix)
class ArchiveBuildStep(Test):
def __init__(self, gs_bucket, gs_acl=None):
self.gs_bucket = gs_bucket
self.gs_acl = gs_acl
def run(self, api, suffix, test_filter=None):
return api.chromium.archive_build(
'archive build',
self.gs_bucket,
gs_acl=self.gs_acl,
)
@staticmethod
def compile_targets(_):
return []
class SizesStep(Test):
def __init__(self, results_url, perf_id):
self.results_url = results_url
self.perf_id = perf_id
def run(self, api, suffix, test_filter=None):
return api.chromium.sizes(self.results_url, self.perf_id)
@staticmethod
def compile_targets(_):
return ['chrome']
@property
def name(self):
return 'sizes' # pragma: no cover
def has_valid_results(self, api, suffix):
# TODO(sebmarchand): implement this function as well as the
# |failures| one.
return True
def failures(self, api, suffix):
return []
class ScriptTest(Test): # pylint: disable=W0232
"""
Test which uses logic from script inside chromium repo.
This makes it possible to keep the logic src-side as opposed
to the build repo most Chromium developers are unfamiliar with.
Another advantage is being to test changes to these scripts
on trybots.
All new tests are strongly encouraged to use this infrastructure.
"""
def __init__(self, name, script, all_compile_targets, script_args=None,
override_compile_targets=None):
super(ScriptTest, self).__init__()
self._name = name
self._script = script
self._all_compile_targets = all_compile_targets
self._script_args = script_args
self._override_compile_targets = override_compile_targets
@property
def name(self):
return self._name
def compile_targets(self, api):
if self._override_compile_targets:
return self._override_compile_targets
try:
substitutions = {'name': self._name}
return [string.Template(s).safe_substitute(substitutions)
for s in self._all_compile_targets[self._script]]
except KeyError: # pragma: no cover
# There are internal recipes that appear to configure
# test script steps, but ones that don't have data.
# We get around this by returning a default value for that case.
# But the recipes should be updated to not do this.
# We mark this as pragma: no cover since the public recipes
# will not exercise this block.
#
# TODO(phajdan.jr): Revisit this when all script tests
# lists move src-side. We should be able to provide
# test data then.
if api.chromium._test_data.enabled:
return []
raise
def run(self, api, suffix, test_filter=None):
name = self.name
if suffix:
name += ' (%s)' % suffix
run_args = []
if suffix == 'without patch':
run_args.extend([
'--filter-file', api.json.input(self.failures(api, 'with patch'))
]) # pragma: no cover
try:
script_args = []
if self._script_args:
script_args = ['--args', api.json.input(self._script_args)]
api.python(
name,
# Enforce that all scripts are in the specified directory
# for consistency.
api.path['checkout'].join(
'testing', 'scripts', api.path.basename(self._script)),
args=(api.chromium_tests.get_common_args_for_scripts() +
script_args +
['run', '--output', api.json.output()] +
run_args),
step_test_data=lambda: api.json.test_api.output(
{'valid': True, 'failures': []}))
finally:
self._test_runs[suffix] = api.step.active_result
if self.has_valid_results(api, suffix):
self._test_runs[suffix].presentation.step_text += (
api.test_utils.format_step_text([
['failures:', self.failures(api, suffix)]
]))
return self._test_runs[suffix]
def has_valid_results(self, api, suffix):
try:
# Make sure the JSON includes all necessary data.
self.failures(api, suffix)
return self._test_runs[suffix].json.output['valid']
except Exception: # pragma: no cover
return False
def failures(self, api, suffix):
return self._test_runs[suffix].json.output['failures']
class LocalGTestTest(Test):
def __init__(self, name, args=None, target_name=None, use_isolate=False,
revision=None, webkit_revision=None,
android_shard_timeout=None, android_tool=None,
override_compile_targets=None, override_isolate_target=None,
use_xvfb=True, **runtest_kwargs):
"""Constructs an instance of LocalGTestTest.
Args:
name: Displayed name of the test. May be modified by suffixes.
args: Arguments to be passed to the test.
target_name: Actual name of the test. Defaults to name.
use_isolate: When set, uses api.isolate.runtest to invoke the test.
Calling recipe should have isolate in their DEPS.
revision: Revision of the Chrome checkout.
webkit_revision: Revision of the WebKit checkout.
override_compile_targets: List of compile targets for this test
(for tests that don't follow target naming conventions).
override_isolate_target: List of isolate targets for this test
(for tests that don't follow target naming conventions).
use_xvfb: whether to use the X virtual frame buffer. Only has an
effect on Linux. Defaults to True. Mostly harmless to
specify this, except on GPU bots.
runtest_kwargs: Additional keyword args forwarded to the runtest.
"""
super(LocalGTestTest, self).__init__()
self._name = name
self._args = args or []
self._target_name = target_name
self._use_isolate = use_isolate
self._revision = revision
self._webkit_revision = webkit_revision
self._android_shard_timeout = android_shard_timeout
self._android_tool = android_tool
self._override_compile_targets = override_compile_targets
self._override_isolate_target = override_isolate_target
self._use_xvfb = use_xvfb
self._runtest_kwargs = runtest_kwargs
@property
def name(self):
return self._name
@property
def target_name(self):
return self._target_name or self._name
@property
def uses_local_devices(self):
return True # pragma: no cover
def isolate_target(self, _api): # pragma: no cover
if self._override_isolate_target:
return self._override_isolate_target
return self.target_name
def compile_targets(self, api):
# TODO(phajdan.jr): clean up override_compile_targets (remove or cover).
if self._override_compile_targets: # pragma: no cover
return self._override_compile_targets
if api.chromium.c.TARGET_PLATFORM == 'android':
# TODO(agrieve): Remove _apk suffix in favour of bin/run_${target} once
# GYP is gone. http://crbug.com/599919
return [self.target_name + '_apk']
return [self.target_name]
def run(self, api, suffix, test_filter=None):
# Copy the list because run can be invoked multiple times and we modify
# the local copy.
args = self._args[:]
is_android = api.chromium.c.TARGET_PLATFORM == 'android'
if suffix == 'without patch':
test_filter = self.failures(api, 'with patch')
kwargs = {}
if test_filter:
if is_android:
kwargs['gtest_filter'] = ':'.join(test_filter) # pragma: no cover
else:
args.append(api.chromium.test_launcher_filter(test_filter))
gtest_results_file = api.test_utils.gtest_results(add_json_log=False)
step_test_data = lambda: api.test_utils.test_api.canned_gtest_output(True)
kwargs['name'] = self._step_name(suffix)
kwargs['args'] = args
kwargs['step_test_data'] = step_test_data
if is_android:
kwargs['json_results_file'] = gtest_results_file
kwargs['shard_timeout'] = self._android_shard_timeout
kwargs['tool'] = self._android_tool
else:
kwargs['xvfb'] = self._use_xvfb
kwargs['test_type'] = self.name
kwargs['annotate'] = 'gtest'
kwargs['test_launcher_summary_output'] = gtest_results_file
kwargs.update(self._runtest_kwargs)
try:
if is_android:
api.chromium_android.run_test_suite(self.target_name, **kwargs)
else:
api.chromium.runtest(self.target_name, revision=self._revision,
webkit_revision=self._webkit_revision, **kwargs)
# TODO(kbr): add functionality to generate_gtest to be able to
# force running these local gtests via isolate from the src-side
# JSON files. crbug.com/584469
finally:
step_result = api.step.active_result
self._test_runs[suffix] = step_result
if hasattr(step_result, 'test_utils'):
r = step_result.test_utils.gtest_results
p = step_result.presentation
if r.valid:
p.step_text += api.test_utils.format_step_text([
['failures:', r.failures]
])
if api.test_results.c.test_results_server:
api.test_results.upload(
api.json.input(r.raw),
test_type=self.name,
chrome_revision=api.bot_update.last_returned_properties.get(
'got_revision_cp', 'x@{#0}'))
return step_result
def has_valid_results(self, api, suffix):
if suffix not in self._test_runs:
return False # pragma: no cover
if not hasattr(self._test_runs[suffix], 'test_utils'):
return False # pragma: no cover
gtest_results = self._test_runs[suffix].test_utils.gtest_results
if not gtest_results.valid: # pragma: no cover
return False
global_tags = gtest_results.raw.get('global_tags', [])
return 'UNRELIABLE_RESULTS' not in global_tags
def failures(self, api, suffix):
return self._test_runs[suffix].test_utils.gtest_results.failures
def get_args_for_test(api, chromium_tests_api, test_spec, bot_update_step):
"""Gets the argument list for a dynamically generated test, as
provided by the JSON files in src/testing/buildbot/ in the Chromium
workspace. This function provides the following build properties in
the form of variable substitutions in the tests' argument lists:
buildername
got_revision
so, for example, a test can declare the argument:
--test-machine-name=\"${buildername}\"
and ${buildername} will be replaced with the associated build
property. In this example, it will also be double-quoted, to handle
the case where the machine name contains contains spaces.
This function also supports trybot-only and waterfall-only
arguments, so that a test can pass a different argument lists on the
continuous builders compared to tryjobs. This is useful when the
waterfall bots generate some reference data that is tested against
during tryjobs.
"""
args = test_spec.get('args', [])
if chromium_tests_api.is_precommit_mode():
args = args + test_spec.get('precommit_args', [])
else:
args = args + test_spec.get('non_precommit_args', [])
# Perform substitution of known variables.
substitutions = {
'buildername': api.properties.get('buildername'),
'got_revision': bot_update_step.presentation.properties['got_revision']
}
return [string.Template(arg).safe_substitute(substitutions) for arg in args]
def generate_gtest(api, chromium_tests_api, mastername, buildername, test_spec,
bot_update_step, enable_swarming=False,
swarming_dimensions=None, scripts_compile_targets=None):
def canonicalize_test(test):
if isinstance(test, basestring):
canonical_test = {'test': test}
else:
canonical_test = test.copy()
canonical_test.setdefault('shard_index', 0)
canonical_test.setdefault('total_shards', 1)
return canonical_test
def get_tests(api):
return [canonicalize_test(t) for t in
test_spec.get(buildername, {}).get('gtest_tests', [])]
for test in get_tests(api):
args = test.get('args', [])
if test['shard_index'] != 0 or test['total_shards'] != 1:
args.extend(['--test-launcher-shard-index=%d' % test['shard_index'],
'--test-launcher-total-shards=%d' % test['total_shards']])
use_swarming = False
swarming_shards = 1
swarming_dimension_sets = None
swarming_priority = None
swarming_expiration = None
swarming_hard_timeout = None
cipd_packages = None
if enable_swarming:
swarming_spec = test.get('swarming', {})
if swarming_spec.get('can_use_on_swarming_builders'):
use_swarming = True
swarming_shards = swarming_spec.get('shards', 1)
swarming_dimension_sets = swarming_spec.get('dimension_sets')
swarming_priority = swarming_spec.get('priority_adjustment')
swarming_expiration = swarming_spec.get('expiration')
swarming_hard_timeout = swarming_spec.get('hard_timeout')
packages = swarming_spec.get('cipd_packages')
if packages:
cipd_packages = [(p['location'],
p['cipd_package'],
p['revision'])
for p in packages]
override_compile_targets = test.get('override_compile_targets', None)
override_isolate_target = test.get('override_isolate_target', None)
target_name = str(test['test'])
name = str(test.get('name', target_name))
swarming_dimensions = swarming_dimensions or {}
use_xvfb = test.get('use_xvfb', True)
if use_swarming and swarming_dimension_sets:
for dimensions in swarming_dimension_sets:
# Yield potentially multiple invocations of the same test, on
# different machine configurations.
new_dimensions = dict(swarming_dimensions)
new_dimensions.update(dimensions)
yield GTestTest(name, args=args, target_name=target_name,
flakiness_dash=True,
enable_swarming=True,
swarming_shards=swarming_shards,
swarming_dimensions=new_dimensions,
swarming_priority=swarming_priority,
swarming_expiration=swarming_expiration,
swarming_hard_timeout=swarming_hard_timeout,
override_compile_targets=override_compile_targets,
override_isolate_target=override_isolate_target,
use_xvfb=use_xvfb, cipd_packages=cipd_packages)
else:
yield GTestTest(name, args=args, target_name=target_name,
flakiness_dash=True,
enable_swarming=use_swarming,
swarming_dimensions=swarming_dimensions,
swarming_shards=swarming_shards,
swarming_priority=swarming_priority,
swarming_expiration=swarming_expiration,
swarming_hard_timeout=swarming_hard_timeout,
override_compile_targets=override_compile_targets,
override_isolate_target=override_isolate_target,
use_xvfb=use_xvfb, cipd_packages=cipd_packages)
def generate_instrumentation_test(api, chromium_tests_api, mastername,
buildername, test_spec, bot_update_step,
enable_swarming=False,
swarming_dimensions=None,
scripts_compile_targets=None):
for test in test_spec.get(buildername, {}).get('instrumentation_tests', []):
test_name = str(test.get('test'))
use_swarming = False
if enable_swarming:
swarming_spec = test.get('swarming', {})
if swarming_spec.get('can_use_on_swarming_builders'):
use_swarming = True
swarming_shards = swarming_spec.get('shards', 1)
swarming_dimension_sets = swarming_spec.get('dimension_sets')
swarming_priority = swarming_spec.get('priority_adjustment')
swarming_expiration = swarming_spec.get('expiration')
args = get_args_for_test(api, chromium_tests_api, test,
bot_update_step)
if use_swarming and swarming_dimension_sets:
for dimensions in swarming_dimension_sets:
# TODO(stip): Swarmify instrumentation tests
pass
else:
yield AndroidInstrumentationTest(
test_name,
compile_targets=test.get('override_compile_targets'),
timeout_scale=test.get('timeout_scale'),
result_details=True,
store_tombstones=True,
args=args)
def generate_junit_test(api, chromium_tests_api, mastername, buildername,
test_spec, bot_update_step, enable_swarming=False,
swarming_dimensions=None,
scripts_compile_targets=None):
for test in test_spec.get(buildername, {}).get('junit_tests', []):
yield AndroidJunitTest(str(test['test']))
def generate_script(api, chromium_tests_api, mastername, buildername, test_spec,
bot_update_step, enable_swarming=False,
swarming_dimensions=None, scripts_compile_targets=None):
for script_spec in test_spec.get(buildername, {}).get('scripts', []):
yield ScriptTest(
str(script_spec['name']),
script_spec['script'],
scripts_compile_targets,
script_spec.get('args', []),
script_spec.get('override_compile_targets', []))
class DynamicPerfTests(Test):
def __init__(self, perf_id, platform, target_bits, max_battery_temp=350,
num_device_shards=1, num_host_shards=1, shard_index=0,
override_browser_name=None, enable_platform_mode=False,
pass_adb_path=True, num_retries=0):
self._perf_id = perf_id
self._platform = platform
self._target_bits = target_bits
self._enable_platform_mode = enable_platform_mode
self._max_battery_temp = max_battery_temp
self._num_host_shards = num_host_shards
self._num_device_shards = num_device_shards
self._num_retries = num_retries
self._pass_adb_path = pass_adb_path
self._shard_index = shard_index
if override_browser_name:
# TODO(phajdan.jr): restore coverage after moving to chromium/src .
self._browser_name = override_browser_name # pragma: no cover
else:
if platform == 'android':
self._browser_name = 'android-chromium'
elif platform == 'win' and target_bits == 64:
self._browser_name = 'release_x64'
else:
self._browser_name ='release'
@property
def name(self):
return 'dynamic_perf_tests' # pragma: no cover
@property
def uses_local_devices(self):
return True
def run(self, api, suffix, test_filter=None):
tests = self._test_list(api)
if self._num_device_shards == 1:
self._run_serially(api, tests)
else:
self._run_sharded(api, tests)
def _test_list(self, api):
if self._platform == 'android':
# Must have already called device_status_check().
device = api.chromium_android.devices[0]
else:
device = None
tests = api.chromium.list_perf_tests(
browser=self._browser_name,
num_shards=self._num_host_shards * self._num_device_shards,
device=device).json.output
tests['steps'] = {k: v for k, v in tests['steps'].iteritems()
if v['device_affinity'] / self._num_device_shards == self._shard_index}
for test_info in tests['steps'].itervalues():
test_info['device_affinity'] %= self._num_device_shards
return tests
def _run_sharded(self, api, tests):
api.chromium_android.run_sharded_perf_tests(
config=api.json.input(data=tests),
perf_id=self._perf_id,
chartjson_file=True,
max_battery_temp=self._max_battery_temp,
known_devices_file=api.chromium_android.known_devices_file,
enable_platform_mode=self._enable_platform_mode,
pass_adb_path=self._pass_adb_path,
num_retries=self._num_retries)
def _run_serially(self, api, tests):
failure = None
for test_name, test in sorted(tests['steps'].iteritems()):
test_name = str(test_name)
annotate = api.chromium.get_annotate_by_test_name(test_name)
cmd = test['cmd'].split()
try:
api.chromium.runtest(
cmd[1] if len(cmd) > 1 else cmd[0],
args=cmd[2:],
name=test_name,
annotate=annotate,
python_mode=True,
results_url='https://chromeperf.appspot.com',
perf_dashboard_id=test.get('perf_dashboard_id', test_name),
perf_id=self._perf_id,
test_type=test.get('perf_dashboard_id', test_name),
xvfb=True,
chartjson_file=True)
except api.step.StepFailure as f: # pragma: no cover
failure = f
if failure: # pragma: no cover
raise failure
@staticmethod
def compile_targets(_):
return []
class SwarmingTest(Test):
PRIORITY_ADJUSTMENTS = {
'higher': -10,
'normal': 0,
'lower': +10,
}
def __init__(self, name, dimensions=None, tags=None, target_name=None,
extra_suffix=None, priority=None, expiration=None,
hard_timeout=None):
self._name = name
self._tasks = {}
self._results = {}
self._target_name = target_name
self._dimensions = dimensions
self._tags = tags
self._extra_suffix = extra_suffix
self._priority = priority
self._expiration = expiration
self._hard_timeout = hard_timeout
if dimensions and not extra_suffix:
self._extra_suffix = self._get_gpu_suffix(dimensions)
@staticmethod
def _get_gpu_suffix(dimensions):
if not dimensions.get('gpu'):
return None
gpu_vendor_id = dimensions.get('gpu', '').split(':')[0].lower()
vendor_ids = {
'8086': 'Intel',
'10de': 'NVIDIA',
'1002': 'ATI',
}
gpu_vendor = vendor_ids.get(gpu_vendor_id) or '(%s)' % gpu_vendor_id
os = dimensions.get('os', '')
if os.startswith('Mac'):
if dimensions.get('hidpi', '') == '1':
os_name = 'Mac Retina'
else:
os_name = 'Mac'
elif os.startswith('Windows'):
os_name = 'Windows'
else:
os_name = 'Linux'
return 'on %s GPU on %s' % (gpu_vendor, os_name)
@property
def name(self):
if self._extra_suffix:
return '%s %s' % (self._name, self._extra_suffix)
else:
return self._name
@property
def target_name(self):
return self._target_name or self._name
def isolate_target(self, _api):
return self.target_name
def create_task(self, api, suffix, isolated_hash, test_filter=None):
"""Creates a swarming task. Must be overridden in subclasses.
Args:
api: Caller's API.
suffix: Suffix added to the test name.
isolated_hash: Hash of the isolated test to be run.
Returns:
A SwarmingTask object.
"""
raise NotImplementedError() # pragma: no cover
def pre_run(self, api, suffix, test_filter=None):
"""Launches the test on Swarming."""
assert suffix not in self._tasks, (
'Test %s was already triggered' % self._step_name(suffix))
# *.isolated may be missing if *_run target is misconfigured. It's a error
# in gyp, not a recipe failure. So carry on with recipe execution.
isolated_hash = api.isolate.isolated_tests.get(self.isolate_target(api))
if not isolated_hash:
return api.python.inline(
'[error] %s' % self._step_name(suffix),
r"""
import sys
print '*.isolated file for target %s is missing' % sys.argv[1]
sys.exit(1)
""",
args=[self.isolate_target(api)])
# Create task.
self._tasks[suffix] = self.create_task(
api, suffix, isolated_hash, test_filter=test_filter)
if self._priority in self.PRIORITY_ADJUSTMENTS:
self._tasks[suffix].priority += self.PRIORITY_ADJUSTMENTS[self._priority]
if self._expiration:
self._tasks[suffix].expiration = self._expiration
if self._hard_timeout:
self._tasks[suffix].hard_timeout = self._hard_timeout
# Add custom dimensions.
if self._dimensions: # pragma: no cover
#TODO(stip): concoct a test case that will trigger this codepath
for k, v in self._dimensions.iteritems():
if v is None:
# Remove key if it exists. This allows one to use None to remove
# default dimensions.
self._tasks[suffix].dimensions.pop(k, None)
else:
self._tasks[suffix].dimensions[k] = v
# Add config-specific tags.
self._tasks[suffix].tags.update(api.chromium.c.runtests.swarming_tags)
# Add custom tags.
if self._tags:
# TODO(kbr): figure out how to cover this line of code with
# tests after the removal of the GPU recipe. crbug.com/584469
self._tasks[suffix].tags.update(self._tags) # pragma: no cover
# Set default value.
if 'os' not in self._tasks[suffix].dimensions:
self._tasks[suffix].dimensions['os'] = api.swarming.prefered_os_dimension(
api.platform.name)
return api.swarming.trigger_task(self._tasks[suffix])
def run(self, api, suffix, test_filter=None): # pylint: disable=R0201
"""Not used. All logic in pre_run, post_run."""
return []
def validate_task_results(self, api, step_result):
"""Interprets output of a task (provided as StepResult object).
Called for successful and failed tasks.
Args:
api: Caller's API.
step_result: StepResult object to examine.
Returns:
A tuple (valid, failures), where valid is True if valid results are
available and failures is a list of names of failed tests (ignored if
valid is False).
"""
raise NotImplementedError() # pragma: no cover
def post_run(self, api, suffix, test_filter=None):
"""Waits for launched test to finish and collects the results."""
assert suffix not in self._results, (
'Results of %s were already collected' % self._step_name(suffix))
# Emit error if test wasn't triggered. This happens if *.isolated is not
# found. (The build is already red by this moment anyway).
if suffix not in self._tasks:
return api.python.inline(
'[collect error] %s' % self._step_name(suffix),
r"""
import sys
print '%s wasn\'t triggered' % sys.argv[1]
sys.exit(1)
""",
args=[self.target_name])
try:
api.swarming.collect_task(self._tasks[suffix])
finally:
valid, failures = self.validate_task_results(api, api.step.active_result)
self._results[suffix] = {'valid': valid, 'failures': failures}
def has_valid_results(self, api, suffix):
# Test wasn't triggered or wasn't collected.
if suffix not in self._tasks or not suffix in self._results:
# TODO(kbr): figure out how to cover this line of code with
# tests after the removal of the GPU recipe. crbug.com/584469
return False # pragma: no cover
return self._results[suffix]['valid']
def failures(self, api, suffix):
assert self.has_valid_results(api, suffix)
return self._results[suffix]['failures']
@property
def uses_swarming(self):
return True
class SwarmingGTestTest(SwarmingTest):
def __init__(self, name, args=None, target_name=None, shards=1,
dimensions=None, tags=None, extra_suffix=None, priority=None,
expiration=None, hard_timeout=None, upload_test_results=True,
override_compile_targets=None, override_isolate_target=None,
cipd_packages=None):
super(SwarmingGTestTest, self).__init__(name, dimensions, tags, target_name,
extra_suffix, priority, expiration,
hard_timeout)
self._args = args or []
self._shards = shards
self._upload_test_results = upload_test_results
self._override_compile_targets = override_compile_targets
self._override_isolate_target = override_isolate_target
self._cipd_packages = cipd_packages
def compile_targets(self, api):
# <X>_run target depends on <X>, and then isolates it invoking isolate.py.
# It is a convention, not a hard coded rule.
# Also include name without the _run suffix to help recipes correctly
# interpret results returned by "analyze".
if self._override_compile_targets:
return self._override_compile_targets
if api.chromium.c.TARGET_PLATFORM == 'android':
# Not all _apk_runs have a corresponding _apk, so we only return the
# _apk_run here.
return [self.target_name + '_apk_run']
return [self.target_name, self.target_name + '_run']
def isolate_target(self, api):
# TODO(agrieve): Remove override_isolate_target and _apk suffix in
# favour of bin/run_${target} once GYP is gone. http://crbug.com/599919
if self._override_isolate_target:
return self._override_isolate_target
if api.chromium.c.TARGET_PLATFORM == 'android':
return self.target_name + '_apk'
return self.target_name
def create_task(self, api, suffix, isolated_hash, test_filter=None):
# For local tests test_args are added inside api.chromium.runtest.
args = self._args[:]
args.extend(api.chromium.c.runtests.test_args)
if suffix == 'without patch':
# If rerunning without a patch, run only tests that failed.
test_filter = sorted(self.failures(api, 'with patch'))
# Make sure there's no existing --test-launcher-filter-file.
# http://crbug.com/587527.
# TODO(phajdan.jr): write test.
for arg in args:
if arg.startswith("--test-launcher-filter-file"): # pragma: no cover
args.remove(arg)
if test_filter:
args.append('--gtest_filter=%s' % ':'.join(test_filter))
args.extend(api.chromium.c.runtests.swarming_extra_args)
return api.swarming.gtest_task(
title=self._step_name(suffix),
isolated_hash=isolated_hash,
shards=self._shards,
test_launcher_summary_output=api.test_utils.gtest_results(add_json_log=False),
cipd_packages=self._cipd_packages, extra_args=args)
def validate_task_results(self, api, step_result):
if not hasattr(step_result, 'test_utils'):
return False, None # pragma: no cover
gtest_results = step_result.test_utils.gtest_results
if not gtest_results:
return False, None # pragma: no cover
global_tags = gtest_results.raw.get('global_tags', [])
if 'UNRELIABLE_RESULTS' in global_tags:
return False, None # pragma: no cover
return True, gtest_results.failures
def post_run(self, api, suffix, test_filter=None):
"""Waits for launched test to finish and collects the results."""
try:
super(SwarmingGTestTest, self).post_run(
api, suffix,test_filter=test_filter)
finally:
step_result = api.step.active_result
# Only upload test results if we have gtest results.
if (self._upload_test_results and
hasattr(step_result, 'test_utils') and
hasattr(step_result.test_utils, 'gtest_results')):
gtest_results = getattr(step_result.test_utils, 'gtest_results', None)
if gtest_results and gtest_results.raw:
parsed_gtest_data = gtest_results.raw
chrome_revision_cp = api.bot_update.last_returned_properties.get(
'got_revision_cp', 'x@{#0}')
chrome_revision = str(api.commit_position.parse_revision(
chrome_revision_cp))
api.test_results.upload(
api.json.input(parsed_gtest_data),
chrome_revision=chrome_revision,
test_type=step_result.step['name'],
test_results_server='test-results.appspot.com')
class LocalIsolatedScriptTest(Test):
def __init__(self, name, args=None, target_name=None,
override_compile_targets=None, **runtest_kwargs):
"""Constructs an instance of LocalIsolatedScriptTest.
An LocalIsolatedScriptTest knows how to invoke an isolate which obeys a
certain contract. The isolate's main target must be a wrapper script which
must interpret certain command line arguments as follows:
--isolated-script-test-output [FILENAME]
The wrapper script must write the simplified json output that the recipes
consume (similar to GTestTest and ScriptTest) into |FILENAME|.
The contract may be expanded later to support functionality like sharding
and retries of specific failed tests. Currently the examples of such wrapper
scripts live in src/testing/scripts/ in the Chromium workspace.
Args:
name: Displayed name of the test. May be modified by suffixes.
args: Arguments to be passed to the test.
target_name: Actual name of the test. Defaults to name.
runtest_kwargs: Additional keyword args forwarded to the runtest.
override_compile_targets: The list of compile targets to use. If not
specified this is the same as target_name.
"""
super(LocalIsolatedScriptTest, self).__init__()
self._name = name
self._args = args or []
self._target_name = target_name
self._runtest_kwargs = runtest_kwargs
self._override_compile_targets = override_compile_targets
@property
def name(self):
return self._name
@property
def target_name(self):
return self._target_name or self._name
def isolate_target(self, _api):
return self.target_name
@property
def uses_swarming(self):
return True
def compile_targets(self, _):
if self._override_compile_targets:
return self._override_compile_targets
return [self.target_name]
# TODO(nednguyen, kbr): figure out what to do with Android.
# (crbug.com/533480)
def run(self, api, suffix, test_filter=None):
# Copy the list because run can be invoked multiple times and we modify
# the local copy.
args = self._args[:]
# TODO(nednguyen, kbr): define contract with the wrapper script to rerun
# a subset of the tests. (crbug.com/533481)
json_results_file = api.json.output()
args.extend(
['--isolated-script-test-output', json_results_file])
step_test_data = lambda: api.json.test_api.output(
{'valid': True, 'failures': []})
try:
api.isolate.run_isolated(
self.name,
api.isolate.isolated_tests[self.target_name],
args,
step_test_data=step_test_data)
finally:
self._test_runs[suffix] = api.step.active_result
if self.has_valid_results(api, suffix):
self._test_runs[suffix].presentation.step_text += (
api.test_utils.format_step_text([
['failures:', self.failures(api, suffix)]
]))
elif api.step.active_result.retcode == 0:
# This failure won't be caught automatically. Need to manually
# raise it as a step failure.
api.step.active_result.presentation.status = api.step.FAILURE
raise api.step.StepFailure('Test results were invalid')
return self._test_runs[suffix]
def has_valid_results(self, api, suffix):
try:
# Make sure the JSON includes all necessary data.
self.failures(api, suffix)
return self._test_runs[suffix].json.output['valid']
except Exception: # pragma: no cover
return False
def failures(self, api, suffix):
return self._test_runs[suffix].json.output['failures']
def is_json_results_format(results):
return results.get('version', 0) == 3
class SwarmingIsolatedScriptTest(SwarmingTest):
def __init__(self, name, args=None, target_name=None, shards=1,
dimensions=None, tags=None, extra_suffix=None, priority=None,
expiration=None, hard_timeout=None, upload_test_results=True,
override_compile_targets=None, perf_id=None,
results_url=None, perf_dashboard_id=None):
super(SwarmingIsolatedScriptTest, self).__init__(
name, dimensions, tags, target_name, extra_suffix, priority, expiration, hard_timeout)
self._args = args or []
self._shards = shards
self._upload_test_results = upload_test_results
self._override_compile_targets = override_compile_targets
self._perf_id=perf_id
self._results_url = results_url
self._perf_dashboard_id = perf_dashboard_id
self._isolated_script_results = {}
@property
def target_name(self):
return self._target_name or self._name
def compile_targets(self, _):
if self._override_compile_targets:
return self._override_compile_targets
return [self.target_name]
@property
def uses_swarming(self):
return True
def create_task(self, api, suffix, isolated_hash, test_filter=None):
browser_config = api.chromium.c.build_config_fs.lower()
args = self._args[:]
# TODO(nednguyen): only rerun the tests that failed for the "without patch"
# suffix.
# For the time being, we assume all isolated_script_test are not idempotent
# TODO(nednguyen): make this configurable in isolated_scripts's spec.
return api.swarming.isolated_script_task(
title=self._step_name(suffix), isolated_hash=isolated_hash,
shards=self._shards, idempotent=False, extra_args=args)
def validate_simplified_results(self, results):
return results['valid'], results['failures']
def validate_json_test_results(self, api, results):
test_results = api.test_utils.create_results_from_json(results)
tests = test_results.tests
failures = list(
t for t in tests
if all(res not in tests[t]['expected'].split()
for res in tests[t]['actual'].split()))
return True, failures
def upload_json_format_results(self, api, results):
chrome_revision_cp = api.bot_update.last_returned_properties.get(
'got_revision_cp', 'x@{#0}')
chrome_revision = str(api.commit_position.parse_revision(
chrome_revision_cp))
api.test_results.upload(
api.json.input(results), chrome_revision=chrome_revision,
test_type=self.name,
test_results_server='test-results.appspot.com')
def validate_task_results(self, api, step_result):
results = getattr(step_result, 'isolated_script_results', None) or {}
valid = True
failures = []
try:
if is_json_results_format(results):
valid, failures = self.validate_json_test_results(api, results)
else:
valid, failures = self.validate_simplified_results(results)
except (ValueError, KeyError) as e:
step_result.presentation.logs['invalid_results_exc'] = [repr(e)]
valid = False
failures = None
if not failures and step_result.retcode != 0:
failures = ['%s (entire test suite)' % self.name]
valid = False
if valid:
self._isolated_script_results = results
step_result.presentation.step_text += api.test_utils.format_step_text([
['failures:', failures]
])
# Check for chartjson results and upload to results dashboard if present.
self._output_chartjson_results_if_present(api, step_result)
return valid, failures
def post_run(self, api, suffix, test_filter=None):
try:
super(SwarmingIsolatedScriptTest, self).post_run(
api, suffix, test_filter=test_filter)
finally:
results = self._isolated_script_results
if self._upload_test_results and is_json_results_format(results):
self.upload_json_format_results(api, results)
def _output_chartjson_results_if_present(self, api, step_result):
results = \
getattr(step_result, 'isolated_script_chartjson_results', None) or {}
try:
if not results.get('enabled', True):
step_result.presentation.logs['DISABLED_BENCHMARK'] = \
['Info: Benchmark disabled, not sending results to dashboard']
return
# TODO(eyaich): Remove logging once we debug uploading chartjson
# to perf dashboard
step_result.presentation.logs['chartjson_info'] = \
['Info: Setting up arguments for perf dashboard']
results_file = api.raw_io.input(data=json.dumps(results))
"""Produces a step that uploads results to dashboard"""
args = [
'--results-file', results_file,
# We are passing this in solely to have the output show up as a link
# in the step log, it will not be used after the upload is complete.
'--output-json-file', api.json.output(),
'--perf-id', self._perf_id,
'--results-url', self._results_url,
'--name', self._perf_dashboard_id,
'--buildername', api.properties['buildername'],
'--buildnumber', api.properties['buildnumber'],
]
if api.chromium.c.build_dir:
args.extend(['--build-dir', api.chromium.c.build_dir])
if 'got_revision_cp' in api.properties:
args.extend(['--got-revision-cp', api.properties['got_revision_cp']])
if 'version' in api.properties:
args.extend(['--version', api.properties['version']])
if 'git_revision' in api.properties:
args.extend(['--git-revision', api.properties['git_revision']])
if 'got_webrtc_revision' in api.properties:
args.extend(['--got-webrtc-revision',
api.properties['got_webrtc_revision']])
if 'got_v8_revision' in api.properties:
args.extend(['--got-v8-revision', api.properties['got_v8_revision']])
api.python(
'Upload Perf Dashboard Results',
api.chromium.package_repo_resource(
'scripts', 'slave', 'upload_perf_dashboard_results.py'),
args)
except Exception as e:
step_result.presentation.logs['chartjson_info'].append(
'Error: Unable to upload chartjson results to perf dashboard')
step_result.presentation.logs['chartjson_info'].append('%r' % e)
def generate_isolated_script(api, chromium_tests_api, mastername, buildername,
test_spec, bot_update_step, enable_swarming=False,
swarming_dimensions=None,
scripts_compile_targets=None):
# Get the perf id and results url if present.
bot_config = (chromium_tests_api.builders.get(mastername, {})
.get('builders', {}).get(buildername, {}))
perf_id = bot_config.get('perf-id')
results_url = bot_config.get('results-url')
for spec in test_spec.get(buildername, {}).get('isolated_scripts', []):
perf_dashboard_id = spec.get('name', '')
use_swarming = False
swarming_shards = 1
swarming_dimension_sets = None
swarming_priority = None
swarming_expiration = None
swarming_hard_timeout = None
if enable_swarming:
swarming_spec = spec.get('swarming', {})
if swarming_spec.get('can_use_on_swarming_builders', False):
use_swarming = True
swarming_shards = swarming_spec.get('shards', 1)
swarming_dimension_sets = swarming_spec.get('dimension_sets')
swarming_priority = swarming_spec.get('priority_adjustment')
swarming_expiration = swarming_spec.get('expiration')
swarming_hard_timeout = swarming_spec.get('hard_timeout')
name = str(spec['name'])
# The variable substitution and precommit/non-precommit arguments
# could be supported for the other test types too, but that wasn't
# desired at the time of this writing.
args = get_args_for_test(api, chromium_tests_api, spec, bot_update_step)
target_name = spec['isolate_name']
# This features is only needed for the cases in which the *_run compile
# target is needed to generate isolate files that contains dynamically libs.
# TODO(nednguyen, kbr): Remove this once all the GYP builds are converted
# to GN.
override_compile_targets = spec.get('override_compile_targets', None)
swarming_dimensions = swarming_dimensions or {}
if use_swarming:
if swarming_dimension_sets:
for dimensions in swarming_dimension_sets:
# Yield potentially multiple invocations of the same test,
# on different machine configurations.
new_dimensions = dict(swarming_dimensions)
new_dimensions.update(dimensions)
yield SwarmingIsolatedScriptTest(
name=name, args=args, target_name=target_name,
shards=swarming_shards, dimensions=new_dimensions,
override_compile_targets=override_compile_targets,
priority=swarming_priority, expiration=swarming_expiration,
hard_timeout=swarming_hard_timeout, perf_id=perf_id,
results_url=results_url, perf_dashboard_id=perf_dashboard_id)
else:
yield SwarmingIsolatedScriptTest(
name=name, args=args, target_name=target_name,
shards=swarming_shards, dimensions=swarming_dimensions,
override_compile_targets=override_compile_targets,
priority=swarming_priority, expiration=swarming_expiration,
hard_timeout=swarming_hard_timeout, perf_id=perf_id,
results_url=results_url, perf_dashboard_id=perf_dashboard_id)
else:
yield LocalIsolatedScriptTest(
name=name, args=args, target_name=target_name,
override_compile_targets=override_compile_targets)
class GTestTest(Test):
def __init__(self, name, args=None, target_name=None, enable_swarming=False,
swarming_shards=1, swarming_dimensions=None, swarming_tags=None,
swarming_extra_suffix=None, swarming_priority=None,
swarming_expiration=None, swarming_hard_timeout=None,
cipd_packages=None, **runtest_kwargs):
super(GTestTest, self).__init__()
if enable_swarming:
self._test = SwarmingGTestTest(
name, args, target_name, swarming_shards, swarming_dimensions,
swarming_tags, swarming_extra_suffix, swarming_priority,
swarming_expiration, swarming_hard_timeout,
cipd_packages=cipd_packages,
override_compile_targets=runtest_kwargs.get(
'override_compile_targets'),
override_isolate_target=runtest_kwargs.get(
'override_isolate_target'))
else:
self._test = LocalGTestTest(name, args, target_name, **runtest_kwargs)
self.enable_swarming = enable_swarming
@property
def name(self):
return self._test.name
@property
def uses_local_devices(self):
# Return True unless this test has swarming enabled.
return not self.enable_swarming
def isolate_target(self, api):
return self._test.isolate_target(api)
def compile_targets(self, api):
return self._test.compile_targets(api)
def pre_run(self, api, suffix, test_filter=None):
return self._test.pre_run(api, suffix, test_filter=test_filter)
def run(self, api, suffix, test_filter=None):
return self._test.run(api, suffix, test_filter=test_filter)
def post_run(self, api, suffix, test_filter=None):
return self._test.post_run(api, suffix, test_filter=test_filter)
def has_valid_results(self, api, suffix):
return self._test.has_valid_results(api, suffix)
def failures(self, api, suffix):
return self._test.failures(api, suffix)
@property
def uses_swarming(self):
return self._test.uses_swarming
class PythonBasedTest(Test):
@staticmethod
def compile_targets(_):
return [] # pragma: no cover
def run_step(self, api, suffix, cmd_args, **kwargs):
raise NotImplementedError() # pragma: no cover
def run(self, api, suffix, test_filter=None):
cmd_args = ['--write-full-results-to',
api.test_utils.test_results(add_json_log=False)]
if suffix == 'without patch':
cmd_args.extend(self.failures(api, 'with patch')) # pragma: no cover
try:
self.run_step(
api,
suffix,
cmd_args,
step_test_data=lambda: api.test_utils.test_api.canned_test_output(True))
finally:
step_result = api.step.active_result
self._test_runs[suffix] = step_result
if hasattr(step_result, 'test_utils'):
r = step_result.test_utils.test_results
p = step_result.presentation
p.step_text += api.test_utils.format_step_text([
['unexpected_failures:', r.unexpected_failures.keys()],
])
return step_result
def has_valid_results(self, api, suffix):
# TODO(dpranke): we should just return zero/nonzero for success/fail.
# crbug.com/357866
step = self._test_runs[suffix]
if not hasattr(step, 'test_utils'):
return False # pragma: no cover
return (step.test_utils.test_results.valid and
step.retcode <= step.test_utils.test_results.MAX_FAILURES_EXIT_STATUS and
(step.retcode == 0) or self.failures(api, suffix))
def failures(self, api, suffix):
return self._test_runs[suffix].test_utils.test_results.unexpected_failures
class PrintPreviewTests(PythonBasedTest): # pylint: disable=W032
name = 'print_preview_tests'
def run_step(self, api, suffix, cmd_args, **kwargs):
platform_arg = '.'.join(['browser_test',
api.platform.normalize_platform_name(api.platform.name)])
args = cmd_args
path = api.path['checkout'].join(
'third_party', 'WebKit', 'Tools', 'Scripts', 'run-webkit-tests')
args.extend(['--platform', platform_arg])
env = {}
if api.platform.is_linux:
env['CHROME_DEVEL_SANDBOX'] = api.path.join(
'/opt', 'chromium', 'chrome_sandbox')
return api.chromium.runtest(
test=path,
args=args,
xvfb=True,
name=self._step_name(suffix),
python_mode=True,
env=env,
**kwargs)
@staticmethod
def compile_targets(api):
return ['browser_tests', 'blink_tests']
class BisectTest(Test): # pylint: disable=W0232
name = 'bisect_test'
def __init__(self, test_parameters={}, **kwargs):
super(BisectTest, self).__init__()
self._test_parameters = test_parameters
self.run_results = {}
self.kwargs = kwargs
@property
def abort_on_failure(self):
return True # pragma: no cover
@property
def uses_local_devices(self):
return False
@staticmethod
def compile_targets(_): # pragma: no cover
return ['chrome'] # Bisect always uses a separate bot for building.
def pre_run(self, api, _, test_filter=None):
self.test_config = api.bisect_tester.load_config_from_dict(
self._test_parameters.get('bisect_config',
api.properties.get('bisect_config')))
def run(self, api, _, test_filter=None):
self.run_results = api.bisect_tester.run_test(
self.test_config, **self.kwargs)
def has_valid_results(self, *_):
return bool(self.run_results.get('retcodes')) # pragma: no cover
def failures(self, *_):
return self._failures # pragma: no cover
class BisectTestStaging(Test): # pylint: disable=W0232
name = 'bisect_test_staging'
def __init__(self, test_parameters={}, **kwargs):
super(BisectTestStaging, self).__init__()
self._test_parameters = test_parameters
self.run_results = {}
self.kwargs = kwargs
@property
def abort_on_failure(self):
return True # pragma: no cover
@property
def uses_local_devices(self):
return False
@staticmethod
def compile_targets(_): # pragma: no cover
return ['chrome'] # Bisect always uses a separate bot for building.
def pre_run(self, api, _, test_filter=None):
self.test_config = api.bisect_tester_staging.load_config_from_dict(
self._test_parameters.get('bisect_config',
api.properties.get('bisect_config')))
def run(self, api, _, test_filter=None):
self.run_results = api.bisect_tester_staging.run_test(
self.test_config, **self.kwargs)
def has_valid_results(self, *_):
return bool(self.run_results.get('retcodes')) # pragma: no cover
def failures(self, *_):
return self._failures # pragma: no cover
class AndroidTest(Test):
def __init__(self, name, compile_targets, isolate_file_path=None):
super(AndroidTest, self).__init__()
self._name = name
self._compile_targets = compile_targets
self.isolate_file_path = isolate_file_path
@property
def name(self):
return self._name
def run_tests(self, api, suffix, json_results_file):
"""Runs the Android test suite and outputs the json results to a file.
Args:
api: Caller's API.
suffix: Suffix added to the test name.
json_results_file: File to output the test results.
"""
raise NotImplementedError() # pragma: no cover
def run(self, api, suffix, test_filter=None):
assert api.chromium.c.TARGET_PLATFORM == 'android'
json_results_file = api.test_utils.gtest_results(add_json_log=False)
try:
self.run_tests(api, suffix, json_results_file)
finally:
step_result = api.step.active_result
self._test_runs[suffix] = {'valid': False}
if (hasattr(step_result, 'test_utils') and
hasattr(step_result.test_utils, 'gtest_results')):
gtest_results = step_result.test_utils.gtest_results
failures = gtest_results.failures
self._test_runs[suffix] = {'valid': True, 'failures': failures}
step_result.presentation.step_text += (
api.test_utils.format_step_text([['failures:', failures]]))
api.test_results.upload(
api.json.input(gtest_results.raw),
test_type=self.name,
chrome_revision=api.bot_update.last_returned_properties.get(
'got_revision_cp', 'x@{#0}'),
test_results_server='test-results.appspot.com')
def compile_targets(self, _):
return self._compile_targets
def has_valid_results(self, api, suffix):
if suffix not in self._test_runs:
return False # pragma: no cover
return self._test_runs[suffix]['valid']
def failures(self, api, suffix):
assert self.has_valid_results(api, suffix)
return self._test_runs[suffix]['failures']
class AndroidJunitTest(AndroidTest):
def __init__(self, name):
super(AndroidJunitTest, self).__init__(name, compile_targets=[name],
isolate_file_path=None)
@property
def uses_local_devices(self):
return False
#override
def run_tests(self, api, suffix, json_results_file):
api.chromium_android.run_java_unit_test_suite(
self.name, verbose=True, suffix=suffix,
json_results_file=json_results_file,
step_test_data=lambda: api.test_utils.test_api.canned_gtest_output(False))
class AndroidInstrumentationTest(AndroidTest):
_DEFAULT_SUITES = {
'AndroidWebViewTest': {
'compile_target': 'android_webview_test_apk',
},
'ChromePublicTest': {
'compile_target': 'chrome_public_test_apk',
},
'ChromeSyncShellTest': {
'compile_target': 'chrome_sync_shell_test_apk',
},
'ChromotingTest': {
'compile_target': 'remoting_test_apk',
},
'ContentShellTest': {
'compile_target': 'content_shell_test_apk',
},
'MojoTest': {
'compile_target': 'mojo_test_apk',
},
'SystemWebViewShellLayoutTest': {
'compile_target': 'system_webview_shell_layout_test_apk',
# TODO(agrieve): These should be listed as deps for
# system_webview_shell_layout_test_apk.
'additional_compile_targets': [
'system_webview_apk',
'android_tools'
],
# TODO(jbudorick): Remove this once it's handled by the generated script.
'additional_apks': [
'SystemWebView.apk',
],
},
'WebViewUiTest': {
'compile_target': 'webview_ui_test_app_test_apk',
# TODO(yolandyan): These should be removed once crbug/643660 is resolved
'additional_compile_targets': [
'system_webview_apk',
],
'additional_apks': [
'SystemWebView.apk',
],
}
}
_DEFAULT_SUITES_BY_TARGET = {
'android_webview_test_apk': _DEFAULT_SUITES['AndroidWebViewTest'],
'chrome_public_test_apk': _DEFAULT_SUITES['ChromePublicTest'],
'chrome_sync_shell_test_apk': _DEFAULT_SUITES['ChromeSyncShellTest'],
'content_shell_test_apk': _DEFAULT_SUITES['ContentShellTest'],
'mojo_test_apk': _DEFAULT_SUITES['MojoTest'],
'remoting_test_apk': _DEFAULT_SUITES['ChromotingTest'],
'system_webview_shell_layout_test_apk':
_DEFAULT_SUITES['SystemWebViewShellLayoutTest'],
'webview_ui_test_app_test_apk': _DEFAULT_SUITES['WebViewUiTest'],
}
def __init__(self, name, compile_targets=None, apk_under_test=None,
test_apk=None, isolate_file_path=None, timeout_scale=None,
annotation=None, except_annotation=None, screenshot=False,
verbose=True, tool=None, additional_apks=None,
store_tombstones=False, result_details=False, args=None):
suite_defaults = (
AndroidInstrumentationTest._DEFAULT_SUITES.get(name)
or AndroidInstrumentationTest._DEFAULT_SUITES_BY_TARGET.get(name)
or {})
if not compile_targets:
compile_targets = [suite_defaults.get('compile_target', name)]
compile_targets.extend(
suite_defaults.get('additional_compile_targets', []))
super(AndroidInstrumentationTest, self).__init__(
name,
compile_targets,
isolate_file_path or suite_defaults.get('isolate_file_path'))
self._additional_apks = (
additional_apks or suite_defaults.get('additional_apks'))
self._annotation = annotation
self._apk_under_test = (
apk_under_test or suite_defaults.get('apk_under_test'))
self._except_annotation = except_annotation
self._screenshot = screenshot
self._test_apk = test_apk or suite_defaults.get('test_apk')
self._timeout_scale = timeout_scale
self._tool = tool
self._verbose = verbose
self._wrapper_script_suite_name = compile_targets[0]
self._store_tombstones = store_tombstones
self._result_details = result_details
self._args = args
@property
def uses_local_devices(self):
return True
#override
def run_tests(self, api, suffix, json_results_file):
api.chromium_android.run_instrumentation_suite(
self.name,
test_apk=api.chromium_android.apk_path(self._test_apk),
apk_under_test=api.chromium_android.apk_path(self._apk_under_test),
additional_apks=[
api.chromium_android.apk_path(a)
for a in self._additional_apks or []],
suffix=suffix,
isolate_file_path=self.isolate_file_path,
annotation=self._annotation, except_annotation=self._except_annotation,
screenshot=self._screenshot, verbose=self._verbose, tool=self._tool,
json_results_file=json_results_file,
timeout_scale=self._timeout_scale,
result_details=self._result_details,
store_tombstones=self._store_tombstones,
wrapper_script_suite_name=self._wrapper_script_suite_name,
step_test_data=lambda: api.test_utils.test_api.canned_gtest_output(False),
args=self._args)
class BlinkTest(Test):
# TODO(dpranke): This should be converted to a PythonBasedTest, although it
# will need custom behavior because we archive the results as well.
def __init__(self, extra_args=None):
super(BlinkTest, self).__init__()
self._extra_args = extra_args
name = 'webkit_tests'
@staticmethod
def compile_targets(api):
return ['blink_tests']
@property
def uses_local_devices(self):
return True
def run(self, api, suffix, test_filter=None):
results_dir = api.path['start_dir'].join('layout-test-results')
step_name = self._step_name(suffix)
args = [
'--target', api.chromium.c.BUILD_CONFIG,
'-o', results_dir,
'--build-dir', api.chromium.c.build_dir,
'--json-test-results', api.test_utils.test_results(add_json_log=False),
'--test-results-server', 'test-results.appspot.com',
'--build-number', str(api.properties['buildnumber']),
'--builder-name', api.properties['buildername'],
'--step-name', step_name,
]
if api.chromium.c.TARGET_PLATFORM == 'android':
args.extend(['--platform', 'android'])
if self._extra_args:
args.extend(self._extra_args)
if suffix == 'without patch':
test_list = "\n".join(self.failures(api, 'with patch'))
args.extend(['--test-list', api.raw_io.input(test_list),
'--skipped', 'always'])
try:
if api.platform.is_win:
args[2] = '--results-directory'
args += [
'--master-name', api.properties['mastername'],
'--debug-rwt-logging',
]
step_result = api.python(
step_name,
api.path['checkout'].join('third_party', 'WebKit', 'Tools',
'Scripts', 'run-webkit-tests'),
args,
step_test_data=lambda: api.test_utils.test_api.canned_test_output(
passing=True, minimal=True))
else:
step_result = api.chromium.runtest(
api.chromium.package_repo_resource(
'scripts', 'slave', 'chromium', 'layout_test_wrapper.py'),
args, name=step_name,
# TODO(phajdan.jr): Clean up the runtest.py mess.
disable_src_side_runtest_py=True,
step_test_data=lambda: api.test_utils.test_api.canned_test_output(
passing=True, minimal=True))
# Mark steps with unexpected flakes as warnings. Do this here instead of
# "finally" blocks because we only want to do this if step was successful.
# We don't want to possibly change failing steps to warnings.
if step_result and step_result.test_utils.test_results.unexpected_flakes:
step_result.presentation.status = api.step.WARNING
finally:
step_result = api.step.active_result
self._test_runs[suffix] = step_result
if step_result:
r = step_result.test_utils.test_results
p = step_result.presentation
if r.valid:
p.step_text += api.test_utils.format_step_text([
['unexpected_flakes:', r.unexpected_flakes.keys()],
['unexpected_failures:', r.unexpected_failures.keys()],
['Total executed: %s' % r.num_passes],
])
if suffix in ('', 'with patch'):
buildername = api.properties['buildername']
buildnumber = api.properties['buildnumber']
archive_layout_test_results = api.chromium.package_repo_resource(
'scripts', 'slave', 'chromium', 'archive_layout_test_results.py')
archive_layout_test_args = [
'--results-dir', results_dir,
'--build-dir', api.chromium.c.build_dir,
'--build-number', buildnumber,
'--builder-name', buildername,
'--gs-bucket', 'gs://chromium-layout-test-archives',
'--staging-dir', api.path['cache'].join('chrome_staging'),
]
archive_layout_test_args += api.build.slave_utils_args
# TODO(phajdan.jr): Pass gs_acl as a parameter, not build property.
if api.properties.get('gs_acl'):
archive_layout_test_args.extend(['--gs-acl', api.properties['gs_acl']])
archive_result = api.python(
'archive_webkit_tests_results',
archive_layout_test_results,
archive_layout_test_args)
# TODO(infra): http://crbug.com/418946 .
sanitized_buildername = re.sub('[ .()]', '_', buildername)
base = (
"https://storage.googleapis.com/chromium-layout-test-archives/%s/%s"
% (sanitized_buildername, buildnumber))
archive_result.presentation.links['layout_test_results'] = (
base + '/layout-test-results/results.html')
archive_result.presentation.links['(zip)'] = (
base + '/layout-test-results.zip')
def has_valid_results(self, api, suffix):
if suffix not in self._test_runs:
return False
step = self._test_runs[suffix]
# TODO(dpranke): crbug.com/357866 - note that all comparing against
# MAX_FAILURES_EXIT_STATUS tells us is that we did not exit early
# or abnormally; it does not tell us how many failures there actually
# were, which might be much higher (up to 5000 diffs, where we
# would bail out early with --exit-after-n-failures) or lower
# if we bailed out after 100 crashes w/ -exit-after-n-crashes, in
# which case the retcode is actually 130
return (step.test_utils.test_results.valid and
step.retcode <= step.test_utils.test_results.MAX_FAILURES_EXIT_STATUS)
def failures(self, api, suffix):
return self._test_runs[suffix].test_utils.test_results.unexpected_failures
class MiniInstallerTest(PythonBasedTest): # pylint: disable=W0232
name = 'test_installer'
@staticmethod
def compile_targets(_):
return ['mini_installer', 'next_version_mini_installer']
def run_step(self, api, suffix, cmd_args, **kwargs):
test_path = api.path['checkout'].join('chrome', 'test', 'mini_installer')
args = [
'--build-dir', api.chromium.c.build_dir,
'--target', api.chromium.c.build_config_fs,
'--force-clean',
'--config', test_path.join('config', 'config.config'),
]
args.extend(cmd_args)
return api.python(
self._step_name(suffix),
test_path.join('test_installer.py'),
args,
**kwargs)
class WebViewCTSTest(Test):
@property
def name(self):
return 'WebView CTS'
@property
def uses_local_devices(self):
return True
@staticmethod
def compile_targets(api):
return ['system_webview_apk']
def run(self, api, suffix, test_filter=None):
api.chromium_android.adb_install_apk(
api.chromium_android.apk_path('SystemWebView.apk'))
api.chromium_android.run_webview_cts()
class IncrementalCoverageTest(Test):
name = 'incremental_coverage'
@property
def uses_local_devices(self):
return True
def has_valid_results(self, api, suffix):
return True
def failures(self, api, suffix):
return []
@property
def name(self): # pragma: no cover
"""Name of the test."""
return IncrementalCoverageTest.name
@staticmethod
def compile_targets(api):
"""List of compile targets needed by this test."""
return []
def run(self, api, suffix, test_filter=None):
api.chromium_android.coverage_report(upload=False)
api.chromium_android.get_changed_lines_for_revision()
api.chromium_android.incremental_coverage_report()
class FindAnnotatedTest(Test):
_TEST_APKS = {
'android_webview_test_apk': 'AndroidWebViewTest',
'blimp_test_apk': 'BlimpTest',
'chrome_public_test_apk': 'ChromePublicTest',
'chrome_sync_shell_test_apk': 'ChromeSyncShellTest',
'content_shell_test_apk': 'ContentShellTest',
'system_webview_shell_layout_test_apk': 'SystemWebViewShellLayoutTest',
}
@staticmethod
def compile_targets(api):
return FindAnnotatedTest._TEST_APKS.keys()
def run(self, api, suffix, test_filter=None):
with api.tempfile.temp_dir('annotated_tests_json') as temp_output_dir:
timestamp_string = datetime.datetime.utcnow().strftime('%Y%m%dT%H%M%S')
if api.properties.get('buildername') is not None:
timestamp_string = api.properties.get('current_time', timestamp_string)
args = [
'--apk-output-dir', api.chromium.output_dir,
'--json-output-dir', temp_output_dir,
'--timestamp-string', timestamp_string,
'-v']
args.extend(
['--test-apks'] + [i for i in FindAnnotatedTest._TEST_APKS.values()])
api.python(
'run find_annotated_tests.py',
api.path['checkout'].join(
'tools', 'android', 'find_annotated_tests.py'),
cwd=api.path['checkout'],
args=args)
api.gsutil.upload(
temp_output_dir.join(
'%s-android-chrome.json' % timestamp_string),
'chromium-annotated-tests', 'android')
GOMA_TESTS = [
GTestTest('base_unittests'),
GTestTest('content_unittests'),
]