Update To 11.40.268.0
[platform/framework/web/crosswalk.git] / src / build / android / buildbot / bb_run_bot.py
1 #!/usr/bin/env python
2 #
3 # Copyright (c) 2013 The Chromium Authors. All rights reserved.
4 # Use of this source code is governed by a BSD-style license that can be
5 # found in the LICENSE file.
6
7 import collections
8 import copy
9 import json
10 import os
11 import pipes
12 import re
13 import subprocess
14 import sys
15
16 import bb_utils
17
18 sys.path.append(os.path.join(os.path.dirname(__file__), '..'))
19 from pylib import constants
20
21
22 CHROMIUM_COVERAGE_BUCKET = 'chromium-code-coverage'
23
24 _BotConfig = collections.namedtuple(
25     'BotConfig', ['bot_id', 'host_obj', 'test_obj'])
26
27 HostConfig = collections.namedtuple(
28     'HostConfig',
29     ['script', 'host_steps', 'extra_args', 'extra_gyp_defines', 'target_arch'])
30
31 TestConfig = collections.namedtuple('Tests', ['script', 'tests', 'extra_args'])
32
33
34 def BotConfig(bot_id, host_object, test_object=None):
35   return _BotConfig(bot_id, host_object, test_object)
36
37
38 def DictDiff(d1, d2):
39   diff = []
40   for key in sorted(set(d1.keys() + d2.keys())):
41     if key in d1 and d1[key] != d2.get(key):
42       diff.append('- %s=%s' % (key, pipes.quote(d1[key])))
43     if key in d2 and d2[key] != d1.get(key):
44       diff.append('+ %s=%s' % (key, pipes.quote(d2[key])))
45   return '\n'.join(diff)
46
47
48 def GetEnvironment(host_obj, testing, extra_env_vars=None):
49   init_env = dict(os.environ)
50   init_env['GYP_GENERATORS'] = 'ninja'
51   if extra_env_vars:
52     init_env.update(extra_env_vars)
53   envsetup_cmd = '. build/android/envsetup.sh'
54   if testing:
55     # Skip envsetup to avoid presubmit dependence on android deps.
56     print 'Testing mode - skipping "%s"' % envsetup_cmd
57     envsetup_cmd = ':'
58   else:
59     print 'Running %s' % envsetup_cmd
60   proc = subprocess.Popen(['bash', '-exc',
61     envsetup_cmd + ' >&2; python build/android/buildbot/env_to_json.py'],
62     stdout=subprocess.PIPE, stderr=subprocess.PIPE,
63     cwd=bb_utils.CHROME_SRC, env=init_env)
64   json_env, envsetup_output = proc.communicate()
65   if proc.returncode != 0:
66     print >> sys.stderr, 'FATAL Failure in envsetup.'
67     print >> sys.stderr, envsetup_output
68     sys.exit(1)
69   env = json.loads(json_env)
70   env['GYP_DEFINES'] = env.get('GYP_DEFINES', '') + \
71       ' OS=android fastbuild=1 use_goma=1 gomadir=%s' % bb_utils.GOMA_DIR
72   if host_obj.target_arch:
73     env['GYP_DEFINES'] += ' target_arch=%s' % host_obj.target_arch
74   extra_gyp = host_obj.extra_gyp_defines
75   if extra_gyp:
76     env['GYP_DEFINES'] += ' %s' % extra_gyp
77     if re.search('(asan|clang)=1', extra_gyp):
78       env.pop('CXX_target', None)
79
80   # Bots checkout chrome in /b/build/slave/<name>/build/src
81   build_internal_android = os.path.abspath(os.path.join(
82       bb_utils.CHROME_SRC, '..', '..', '..', '..', '..', 'build_internal',
83       'scripts', 'slave', 'android'))
84   if os.path.exists(build_internal_android):
85     env['PATH'] = os.pathsep.join([build_internal_android, env['PATH']])
86   return env
87
88
89 def GetCommands(options, bot_config):
90   """Get a formatted list of commands.
91
92   Args:
93     options: Options object.
94     bot_config: A BotConfig named tuple.
95     host_step_script: Host step script.
96     device_step_script: Device step script.
97   Returns:
98     list of Command objects.
99   """
100   property_args = bb_utils.EncodeProperties(options)
101   commands = [[bot_config.host_obj.script,
102                '--steps=%s' % ','.join(bot_config.host_obj.host_steps)] +
103               property_args + (bot_config.host_obj.extra_args or [])]
104
105   test_obj = bot_config.test_obj
106   if test_obj:
107     run_test_cmd = [test_obj.script] + property_args
108     for test in test_obj.tests:
109       run_test_cmd.extend(['-f', test])
110     if test_obj.extra_args:
111       run_test_cmd.extend(test_obj.extra_args)
112     commands.append(run_test_cmd)
113   return commands
114
115
116 def GetBotStepMap():
117   compile_step = ['compile']
118   chrome_proxy_tests = ['chrome_proxy']
119   python_unittests = ['python_unittests']
120   std_host_tests = ['check_webview_licenses', 'findbugs']
121   emma_coverage_tests = [x for x in std_host_tests if x is not 'findbugs']
122   std_build_steps = ['compile', 'zip_build']
123   std_test_steps = ['extract_build']
124   std_tests = ['ui', 'unit']
125   telemetry_tests = ['telemetry_perf_unittests']
126   flakiness_server = (
127       '--flakiness-server=%s' % constants.UPSTREAM_FLAKINESS_SERVER)
128   experimental = ['--experimental']
129   bisect_chrome_output_dir = os.path.abspath(
130       os.path.join(os.path.dirname(__file__), os.pardir, os.pardir, os.pardir,
131                    os.pardir, 'bisect', 'src', 'out'))
132   B = BotConfig
133   H = (lambda steps, extra_args=None, extra_gyp=None, target_arch=None :
134        HostConfig('build/android/buildbot/bb_host_steps.py', steps, extra_args,
135                   extra_gyp, target_arch))
136   T = (lambda tests, extra_args=None :
137        TestConfig('build/android/buildbot/bb_device_steps.py', tests,
138                   extra_args))
139
140   bot_configs = [
141       # Main builders
142       B('main-builder-dbg', H(std_build_steps + std_host_tests)),
143       B('main-builder-rel', H(std_build_steps)),
144       B('main-clang-builder',
145         H(compile_step, extra_gyp='clang=1 component=shared_library')),
146       B('main-clobber', H(compile_step)),
147       B('main-tests-rel', H(std_test_steps),
148         T(std_tests + telemetry_tests + chrome_proxy_tests,
149           ['--cleanup', flakiness_server])),
150       B('main-tests', H(std_test_steps),
151         T(std_tests,['--cleanup', flakiness_server])),
152
153       # Other waterfalls
154       B('asan-builder-tests', H(compile_step,
155                                 extra_gyp='asan=1 component=shared_library'),
156         T(std_tests, ['--asan', '--asan-symbolize'])),
157       B('blink-try-builder', H(compile_step)),
158       B('chromedriver-fyi-tests-dbg', H(std_test_steps),
159         T(['chromedriver'], ['--install=ChromeShell', '--skip-wipe',
160           '--cleanup'])),
161       B('fyi-x86-builder-dbg',
162         H(compile_step + std_host_tests, experimental, target_arch='ia32')),
163       B('fyi-builder-dbg',
164         H(std_build_steps + emma_coverage_tests, experimental,
165           extra_gyp='emma_coverage=1')),
166       B('x86-builder-dbg',
167         H(compile_step + std_host_tests, target_arch='ia32')),
168       B('fyi-builder-rel', H(std_build_steps,  experimental)),
169       B('fyi-tests', H(std_test_steps),
170         T(std_tests + python_unittests,
171                       ['--experimental', flakiness_server,
172                       '--coverage-bucket', CHROMIUM_COVERAGE_BUCKET,
173                       '--cleanup'])),
174       B('fyi-component-builder-tests-dbg',
175         H(compile_step, extra_gyp='component=shared_library'),
176         T(std_tests, ['--experimental', flakiness_server])),
177       B('gpu-builder-tests-dbg',
178         H(compile_step),
179         T(['gpu'], ['--install=ContentShell'])),
180       # Pass empty T([]) so that logcat monitor and device status check are run.
181       B('perf-bisect-builder-tests-dbg',
182         H(['bisect_perf_regression']),
183         T([], ['--chrome-output-dir', bisect_chrome_output_dir])),
184       B('perf-tests-rel', H(std_test_steps),
185         T([], ['--install=ChromeShell', '--cleanup'])),
186       B('webkit-latest-webkit-tests', H(std_test_steps),
187         T(['webkit_layout', 'webkit'], ['--cleanup', '--auto-reconnect'])),
188       B('webkit-latest-contentshell', H(compile_step),
189         T(['webkit_layout'], ['--auto-reconnect'])),
190       B('builder-unit-tests', H(compile_step), T(['unit'])),
191
192       # Generic builder config (for substring match).
193       B('builder', H(std_build_steps)),
194   ]
195
196   bot_map = dict((config.bot_id, config) for config in bot_configs)
197
198   # These bots have identical configuration to ones defined earlier.
199   copy_map = [
200       ('lkgr-clobber', 'main-clobber'),
201       ('try-builder-dbg', 'main-builder-dbg'),
202       ('try-builder-rel', 'main-builder-rel'),
203       ('try-clang-builder', 'main-clang-builder'),
204       ('try-fyi-builder-dbg', 'fyi-builder-dbg'),
205       ('try-x86-builder-dbg', 'x86-builder-dbg'),
206       ('try-tests-rel', 'main-tests-rel'),
207       ('try-tests', 'main-tests'),
208       ('try-fyi-tests', 'fyi-tests'),
209       ('webkit-latest-tests', 'main-tests'),
210   ]
211   for to_id, from_id in copy_map:
212     assert to_id not in bot_map
213     # pylint: disable=W0212
214     bot_map[to_id] = copy.deepcopy(bot_map[from_id])._replace(bot_id=to_id)
215
216     # Trybots do not upload to flakiness dashboard. They should be otherwise
217     # identical in configuration to their trunk building counterparts.
218     test_obj = bot_map[to_id].test_obj
219     if to_id.startswith('try') and test_obj:
220       extra_args = test_obj.extra_args
221       if extra_args and flakiness_server in extra_args:
222         extra_args.remove(flakiness_server)
223   return bot_map
224
225
226 # Return an object from the map, looking first for an exact id match.
227 # If this fails, look for an id which is a substring of the specified id.
228 # Choose the longest of all substring matches.
229 # pylint: disable=W0622
230 def GetBestMatch(id_map, id):
231   config = id_map.get(id)
232   if not config:
233     substring_matches = filter(lambda x: x in id, id_map.iterkeys())
234     if substring_matches:
235       max_id = max(substring_matches, key=len)
236       print 'Using config from id="%s" (substring match).' % max_id
237       config = id_map[max_id]
238   return config
239
240
241 def GetRunBotOptParser():
242   parser = bb_utils.GetParser()
243   parser.add_option('--bot-id', help='Specify bot id directly.')
244   parser.add_option('--testing', action='store_true',
245                     help='For testing: print, but do not run commands')
246
247   return parser
248
249
250 def GetBotConfig(options, bot_step_map):
251   bot_id = options.bot_id or options.factory_properties.get('android_bot_id')
252   if not bot_id:
253     print (sys.stderr,
254            'A bot id must be specified through option or factory_props.')
255     return
256
257   bot_config = GetBestMatch(bot_step_map, bot_id)
258   if not bot_config:
259     print 'Error: config for id="%s" cannot be inferred.' % bot_id
260   return bot_config
261
262
263 def RunBotCommands(options, commands, env):
264   print 'Environment changes:'
265   print DictDiff(dict(os.environ), env)
266
267   for command in commands:
268     print bb_utils.CommandToString(command)
269     sys.stdout.flush()
270     if options.testing:
271       env['BUILDBOT_TESTING'] = '1'
272     return_code = subprocess.call(command, cwd=bb_utils.CHROME_SRC, env=env)
273     if return_code != 0:
274       return return_code
275
276
277 def main(argv):
278   proc = subprocess.Popen(
279       ['/bin/hostname', '-f'], stdout=subprocess.PIPE, stderr=subprocess.PIPE)
280   hostname_stdout, hostname_stderr = proc.communicate()
281   if proc.returncode == 0:
282     print 'Running on: ' + hostname_stdout
283   else:
284     print >> sys.stderr, 'WARNING: failed to run hostname'
285     print >> sys.stderr, hostname_stdout
286     print >> sys.stderr, hostname_stderr
287     sys.exit(1)
288
289   parser = GetRunBotOptParser()
290   options, args = parser.parse_args(argv[1:])
291   if args:
292     parser.error('Unused args: %s' % args)
293
294   bot_config = GetBotConfig(options, GetBotStepMap())
295   if not bot_config:
296     sys.exit(1)
297
298   print 'Using config:', bot_config
299
300   commands = GetCommands(options, bot_config)
301   for command in commands:
302     print 'Will run: ', bb_utils.CommandToString(command)
303   print
304
305   env = GetEnvironment(bot_config.host_obj, options.testing)
306   return RunBotCommands(options, commands, env)
307
308
309 if __name__ == '__main__':
310   sys.exit(main(sys.argv))