274 lines
9.2 KiB
Python
274 lines
9.2 KiB
Python
|
#!/usr/bin/env python
|
||
|
#
|
||
|
# Copyright (c) 2013 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 collections
|
||
|
import copy
|
||
|
import json
|
||
|
import os
|
||
|
import pipes
|
||
|
import re
|
||
|
import subprocess
|
||
|
import sys
|
||
|
|
||
|
import bb_utils
|
||
|
|
||
|
sys.path.append(os.path.join(os.path.dirname(__file__), '..'))
|
||
|
from pylib import constants
|
||
|
|
||
|
|
||
|
_BotConfig = collections.namedtuple(
|
||
|
'BotConfig', ['bot_id', 'host_obj', 'test_obj'])
|
||
|
|
||
|
HostConfig = collections.namedtuple(
|
||
|
'HostConfig',
|
||
|
['script', 'host_steps', 'extra_args', 'extra_gyp_defines', 'target_arch'])
|
||
|
|
||
|
TestConfig = collections.namedtuple('Tests', ['script', 'tests', 'extra_args'])
|
||
|
|
||
|
|
||
|
def BotConfig(bot_id, host_object, test_object=None):
|
||
|
return _BotConfig(bot_id, host_object, test_object)
|
||
|
|
||
|
|
||
|
def DictDiff(d1, d2):
|
||
|
diff = []
|
||
|
for key in sorted(set(d1.keys() + d2.keys())):
|
||
|
if key in d1 and d1[key] != d2.get(key):
|
||
|
diff.append('- %s=%s' % (key, pipes.quote(d1[key])))
|
||
|
if key in d2 and d2[key] != d1.get(key):
|
||
|
diff.append('+ %s=%s' % (key, pipes.quote(d2[key])))
|
||
|
return '\n'.join(diff)
|
||
|
|
||
|
|
||
|
def GetEnvironment(host_obj, testing):
|
||
|
init_env = dict(os.environ)
|
||
|
init_env['GYP_GENERATORS'] = 'ninja'
|
||
|
init_env['GOMA_DIR'] = bb_utils.GOMA_DIR
|
||
|
envsetup_cmd = '. build/android/envsetup.sh'
|
||
|
if host_obj.target_arch:
|
||
|
envsetup_cmd += ' --target-arch=%s' % host_obj.target_arch
|
||
|
if testing:
|
||
|
# Skip envsetup to avoid presubmit dependence on android deps.
|
||
|
print 'Testing mode - skipping "%s"' % envsetup_cmd
|
||
|
envsetup_cmd = ':'
|
||
|
else:
|
||
|
print 'Running %s' % envsetup_cmd
|
||
|
proc = subprocess.Popen(['bash', '-exc',
|
||
|
envsetup_cmd + ' >&2; python build/android/buildbot/env_to_json.py'],
|
||
|
stdout=subprocess.PIPE, stderr=subprocess.PIPE,
|
||
|
cwd=bb_utils.CHROME_SRC, env=init_env)
|
||
|
json_env, envsetup_output = proc.communicate()
|
||
|
if proc.returncode != 0:
|
||
|
print >> sys.stderr, 'FATAL Failure in envsetup.'
|
||
|
print >> sys.stderr, envsetup_output
|
||
|
sys.exit(1)
|
||
|
env = json.loads(json_env)
|
||
|
env['GYP_DEFINES'] = env.get('GYP_DEFINES', '') + ' fastbuild=1'
|
||
|
extra_gyp = host_obj.extra_gyp_defines
|
||
|
if extra_gyp:
|
||
|
env['GYP_DEFINES'] += ' %s' % extra_gyp
|
||
|
if re.search('(asan|clang)=1', extra_gyp):
|
||
|
env.pop('CXX_target', None)
|
||
|
|
||
|
# Bots checkout chrome in /b/build/slave/<name>/build/src
|
||
|
build_internal_android = os.path.abspath(os.path.join(
|
||
|
bb_utils.CHROME_SRC, '..', '..', '..', '..', '..', 'build_internal',
|
||
|
'scripts', 'slave', 'android'))
|
||
|
if os.path.exists(build_internal_android):
|
||
|
env['PATH'] = os.pathsep.join([build_internal_android, env['PATH']])
|
||
|
return env
|
||
|
|
||
|
|
||
|
def GetCommands(options, bot_config):
|
||
|
"""Get a formatted list of commands.
|
||
|
|
||
|
Args:
|
||
|
options: Options object.
|
||
|
bot_config: A BotConfig named tuple.
|
||
|
host_step_script: Host step script.
|
||
|
device_step_script: Device step script.
|
||
|
Returns:
|
||
|
list of Command objects.
|
||
|
"""
|
||
|
property_args = bb_utils.EncodeProperties(options)
|
||
|
commands = [[bot_config.host_obj.script,
|
||
|
'--steps=%s' % ','.join(bot_config.host_obj.host_steps)] +
|
||
|
property_args + (bot_config.host_obj.extra_args or [])]
|
||
|
|
||
|
test_obj = bot_config.test_obj
|
||
|
if test_obj:
|
||
|
run_test_cmd = [test_obj.script, '--reboot'] + property_args
|
||
|
for test in test_obj.tests:
|
||
|
run_test_cmd.extend(['-f', test])
|
||
|
if test_obj.extra_args:
|
||
|
run_test_cmd.extend(test_obj.extra_args)
|
||
|
commands.append(run_test_cmd)
|
||
|
return commands
|
||
|
|
||
|
|
||
|
def GetBotStepMap():
|
||
|
compile_step = ['compile']
|
||
|
std_host_tests = ['check_webview_licenses', 'findbugs']
|
||
|
std_build_steps = ['compile', 'zip_build']
|
||
|
std_test_steps = ['extract_build']
|
||
|
std_tests = ['ui', 'unit']
|
||
|
flakiness_server = (
|
||
|
'--flakiness-server=%s' % constants.UPSTREAM_FLAKINESS_SERVER)
|
||
|
experimental = ['--experimental']
|
||
|
|
||
|
B = BotConfig
|
||
|
H = (lambda steps, extra_args=None, extra_gyp=None, target_arch=None :
|
||
|
HostConfig('build/android/buildbot/bb_host_steps.py', steps, extra_args,
|
||
|
extra_gyp, target_arch))
|
||
|
T = (lambda tests, extra_args=None :
|
||
|
TestConfig('build/android/buildbot/bb_device_steps.py', tests,
|
||
|
extra_args))
|
||
|
|
||
|
bot_configs = [
|
||
|
# Main builders
|
||
|
B('main-builder-dbg', H(std_build_steps + std_host_tests)),
|
||
|
B('main-builder-rel', H(std_build_steps)),
|
||
|
B('main-clang-builder',
|
||
|
H(compile_step, extra_gyp='clang=1 component=shared_library')),
|
||
|
B('main-clobber', H(compile_step)),
|
||
|
B('main-tests', H(std_test_steps), T(std_tests, [flakiness_server])),
|
||
|
|
||
|
# Other waterfalls
|
||
|
B('asan-builder-tests', H(compile_step, extra_gyp='asan=1'),
|
||
|
T(std_tests, ['--asan'])),
|
||
|
B('chromedriver-fyi-tests-dbg', H(std_test_steps),
|
||
|
T(['chromedriver'], ['--install=ChromiumTestShell'])),
|
||
|
B('fyi-x86-builder-dbg',
|
||
|
H(compile_step + std_host_tests, experimental, target_arch='x86')),
|
||
|
B('fyi-builder-dbg',
|
||
|
H(std_build_steps + std_host_tests, experimental)),
|
||
|
B('x86-builder-dbg',
|
||
|
H(compile_step + std_host_tests, target_arch='x86')),
|
||
|
B('fyi-builder-rel', H(std_build_steps, experimental)),
|
||
|
B('fyi-tests', H(std_test_steps),
|
||
|
T(std_tests, ['--experimental', flakiness_server])),
|
||
|
B('fyi-component-builder-tests-dbg',
|
||
|
H(compile_step, extra_gyp='component=shared_library'),
|
||
|
T(std_tests, ['--experimental', flakiness_server])),
|
||
|
B('perf-bisect-builder-tests-dbg', H(['bisect_perf_regression'])),
|
||
|
B('perf-tests-rel', H(std_test_steps),
|
||
|
T([], ['--install=ChromiumTestShell'])),
|
||
|
B('webkit-latest-webkit-tests', H(std_test_steps),
|
||
|
T(['webkit_layout', 'webkit'], ['--auto-reconnect'])),
|
||
|
B('webkit-latest-contentshell', H(compile_step),
|
||
|
T(['webkit_layout'], ['--auto-reconnect'])),
|
||
|
B('builder-unit-tests', H(compile_step), T(['unit'])),
|
||
|
B('webrtc-tests', H(std_test_steps), T(['webrtc'], [flakiness_server])),
|
||
|
|
||
|
# Generic builder config (for substring match).
|
||
|
B('builder', H(std_build_steps)),
|
||
|
]
|
||
|
|
||
|
bot_map = dict((config.bot_id, config) for config in bot_configs)
|
||
|
|
||
|
# These bots have identical configuration to ones defined earlier.
|
||
|
copy_map = [
|
||
|
('lkgr-clobber', 'main-clobber'),
|
||
|
('try-builder-dbg', 'main-builder-dbg'),
|
||
|
('try-builder-rel', 'main-builder-rel'),
|
||
|
('try-clang-builder', 'main-clang-builder'),
|
||
|
('try-fyi-builder-dbg', 'fyi-builder-dbg'),
|
||
|
('try-x86-builder-dbg', 'x86-builder-dbg'),
|
||
|
('try-tests', 'main-tests'),
|
||
|
('try-fyi-tests', 'fyi-tests'),
|
||
|
('webkit-latest-tests', 'main-tests'),
|
||
|
('webrtc-builder', 'main-builder-rel'),
|
||
|
]
|
||
|
for to_id, from_id in copy_map:
|
||
|
assert to_id not in bot_map
|
||
|
# pylint: disable=W0212
|
||
|
bot_map[to_id] = copy.deepcopy(bot_map[from_id])._replace(bot_id=to_id)
|
||
|
|
||
|
# Trybots do not upload to flakiness dashboard. They should be otherwise
|
||
|
# identical in configuration to their trunk building counterparts.
|
||
|
test_obj = bot_map[to_id].test_obj
|
||
|
if to_id.startswith('try') and test_obj:
|
||
|
extra_args = test_obj.extra_args
|
||
|
if extra_args and flakiness_server in extra_args:
|
||
|
extra_args.remove(flakiness_server)
|
||
|
return bot_map
|
||
|
|
||
|
|
||
|
# Return an object from the map, looking first for an exact id match.
|
||
|
# If this fails, look for an id which is a substring of the specified id.
|
||
|
# Choose the longest of all substring matches.
|
||
|
# pylint: disable=W0622
|
||
|
def GetBestMatch(id_map, id):
|
||
|
config = id_map.get(id)
|
||
|
if not config:
|
||
|
substring_matches = filter(lambda x: x in id, id_map.iterkeys())
|
||
|
if substring_matches:
|
||
|
max_id = max(substring_matches, key=len)
|
||
|
print 'Using config from id="%s" (substring match).' % max_id
|
||
|
config = id_map[max_id]
|
||
|
return config
|
||
|
|
||
|
|
||
|
def GetRunBotOptParser():
|
||
|
parser = bb_utils.GetParser()
|
||
|
parser.add_option('--bot-id', help='Specify bot id directly.')
|
||
|
parser.add_option('--testing', action='store_true',
|
||
|
help='For testing: print, but do not run commands')
|
||
|
|
||
|
return parser
|
||
|
|
||
|
|
||
|
def GetBotConfig(options, bot_step_map):
|
||
|
bot_id = options.bot_id or options.factory_properties.get('android_bot_id')
|
||
|
if not bot_id:
|
||
|
print (sys.stderr,
|
||
|
'A bot id must be specified through option or factory_props.')
|
||
|
return
|
||
|
|
||
|
bot_config = GetBestMatch(bot_step_map, bot_id)
|
||
|
if not bot_config:
|
||
|
print 'Error: config for id="%s" cannot be inferred.' % bot_id
|
||
|
return bot_config
|
||
|
|
||
|
|
||
|
def RunBotCommands(options, commands, env):
|
||
|
print 'Environment changes:'
|
||
|
print DictDiff(dict(os.environ), env)
|
||
|
|
||
|
for command in commands:
|
||
|
print bb_utils.CommandToString(command)
|
||
|
sys.stdout.flush()
|
||
|
if options.testing:
|
||
|
env['BUILDBOT_TESTING'] = '1'
|
||
|
return_code = subprocess.call(command, cwd=bb_utils.CHROME_SRC, env=env)
|
||
|
if return_code != 0:
|
||
|
return return_code
|
||
|
|
||
|
|
||
|
def main(argv):
|
||
|
parser = GetRunBotOptParser()
|
||
|
options, args = parser.parse_args(argv[1:])
|
||
|
if args:
|
||
|
parser.error('Unused args: %s' % args)
|
||
|
|
||
|
bot_config = GetBotConfig(options, GetBotStepMap())
|
||
|
if not bot_config:
|
||
|
sys.exit(1)
|
||
|
|
||
|
print 'Using config:', bot_config
|
||
|
|
||
|
commands = GetCommands(options, bot_config)
|
||
|
for command in commands:
|
||
|
print 'Will run: ', bb_utils.CommandToString(command)
|
||
|
print
|
||
|
|
||
|
env = GetEnvironment(bot_config.host_obj, options.testing)
|
||
|
return RunBotCommands(options, commands, env)
|
||
|
|
||
|
|
||
|
if __name__ == '__main__':
|
||
|
sys.exit(main(sys.argv))
|