blob: df7eb3219473a49fcebc5eb5d12048417ed80d9d [file] [log] [blame]
Xixuan Wu0c76d5b2017-08-30 16:40:17 -07001# Copyright 2017 The Chromium OS Authors. All rights reserved.
2# Use of this source code is governed by a BSD-style license that can be
3# found in the LICENSE file.
4
5"""Module for tasks triggered by suite scheduler."""
6# pylint: disable=g-bad-import-order
Xixuan Wu244e0ec2018-05-23 14:49:55 -07007# pylint: disable=dangerous-default-value
Xixuan Wu0c76d5b2017-08-30 16:40:17 -07008
9from distutils import version
10import MySQLdb
11import logging
12
13import build_lib
Xixuan Wued878ea2019-03-18 15:32:16 -070014import skylab
Xixuan Wu0c76d5b2017-08-30 16:40:17 -070015import task_executor
Xixuan Wu0c76d5b2017-08-30 16:40:17 -070016import tot_manager
17
18# The max lifetime of a suite scheduled by suite scheduler
19_JOB_MAX_RUNTIME_MINS_DEFAULT = 72 * 60
20
21# One kind of formats for RO firmware spec.
22_RELEASE_RO_FIRMWARE_SPEC = 'released_ro_'
23
24
25class SchedulingError(Exception):
26 """Raised to indicate a failure in scheduling a task."""
27
28
29class Task(object):
30 """Represents an entry from the suite_scheduler config file.
31
32 Each entry from the suite_scheduler config file maps one-to-one to a
33 Task. Each instance has enough information to schedule itself.
34 """
35
Po-Hsien Wangdd833072018-08-16 18:09:20 -070036 def __init__(self, task_info, board_family_config={}, tot=None):
Xixuan Wu0c76d5b2017-08-30 16:40:17 -070037 """Initialize a task instance.
38
39 Args:
40 task_info: a config_reader.TaskInfo object, which includes:
41 name, name of this task, e.g. 'NightlyPower'
42 suite, the name of the suite to run, e.g. 'graphics_per-day'
43 branch_specs, a pre-vetted iterable of branch specifiers,
44 e.g. ['>=R18', 'factory']
45 pool, the pool of machines to schedule tasks. Default is None.
46 num, the number of devices to shard the test suite. It could
47 be an Integer or None. By default it's None.
Po-Hsien Wangdd833072018-08-16 18:09:20 -070048 board_families, a common separated list of board family to run this
49 task on. Boards belong to one of the board family in this list
50 would be added to task_info.boards.
Xixuan Wu0c76d5b2017-08-30 16:40:17 -070051 boards, a comma separated list of boards to run this task on. Default
Po-Hsien Wangdd833072018-08-16 18:09:20 -070052 is None, which allows this task to run on all boards. If same board
53 is specified in 'boards' and 'exclude_boards', we exclude this
54 board.
55 exclude_board_families, a common separated list of board family not to
56 run task on. Boards belong to one of the board family in this list
57 would be added to task_info.exclude_boards.
Po-Hsien Wang6d589732018-05-15 17:19:34 -070058 exclude_boards, a comma separated list of boards not to run this task
59 on. Default is None, which allows this task to run on all boards.
Po-Hsien Wangdd833072018-08-16 18:09:20 -070060 If same board is specified in 'boards' and 'exclude_boards', we
61 exclude this board.
Xixuan Wu89897182019-01-03 15:28:01 -080062 models, a comma separated list of models to run this task on. Default
63 is None, which allows this task to run on all models. If same model
64 is specified in 'models' and 'exclude_models', we exclude this
65 model.
66 exclude_models, a comma separated list of models not to run this task
67 on. Default is None, which allows this task to run on all models.
Xixuan Wu0c76d5b2017-08-30 16:40:17 -070068 priority, the string name of a priority from constants.Priorities.
69 timeout, the max lifetime of the suite in hours.
70 cros_build_spec, spec used to determine the ChromeOS build to test
71 with a firmware build, e.g., tot, R41 etc.
72 firmware_rw_build_spec, spec used to determine the firmware RW build
73 test with a ChromeOS build.
74 firmware_ro_build_spec, spec used to determine the firmware RO build
75 test with a ChromeOS build.
76 test_source, the source of test code when firmware will be updated in
77 the test. The value can be 'firmware_rw', 'firmware_ro' or 'cros'.
78 job_retry, set to True to enable job-level retry. Default is False.
Xixuan Wu80531932017-10-12 17:26:51 -070079 no_delay, set to True to raise the priority of this task in task.
80 force, set to True to schedule this suite no matter whether there's
81 duplicate jobs before.
Xixuan Wu0c76d5b2017-08-30 16:40:17 -070082 queue, so the suite jobs can start running tests with no waiting.
83 hour, an integer specifying the hour that a nightly run should be
84 triggered, default is set to 21.
85 day, an integer specifying the day of a week that a weekly run should
86 be triggered, default is set to 5 (Saturday).
87 os_type, type of OS, e.g., cros, brillo, android. Default is cros.
88 The argument is required for android/brillo builds.
89 launch_control_branches, comma separated string of launch control
90 branches. The argument is required and only applicable for
91 android/brillo builds.
92 launch_control_targets, comma separated string of build targets for
93 launch control builds. The argument is required and only
94 applicable for android/brillo builds.
95 testbed_dut_count, number of duts to test when using a testbed.
Xixuan Wu83118dd2018-08-27 12:11:35 -070096 board_family_config: A board family dictionary mapping board_family name
97 to its corresponding boards.
Xixuan Wu0c76d5b2017-08-30 16:40:17 -070098 tot: The tot manager for checking ToT. If it's None, a new tot_manager
99 instance will be initialized.
100 """
Xixuan Wu5451a662017-10-17 10:57:40 -0700101 # Indicate whether there're suites get pushed into taskqueue for this task.
102 self.is_pushed = False
103
Xixuan Wu0c76d5b2017-08-30 16:40:17 -0700104 self.name = task_info.name
105 self.suite = task_info.suite
106 self.branch_specs = task_info.branch_specs
107 self.pool = task_info.pool
108 self.num = task_info.num
109 self.priority = task_info.priority
110 self.timeout = task_info.timeout
111 self.cros_build_spec = task_info.cros_build_spec
112 self.firmware_rw_build_spec = task_info.firmware_rw_build_spec
113 self.firmware_ro_build_spec = task_info.firmware_ro_build_spec
114 self.test_source = task_info.test_source
115 self.job_retry = task_info.job_retry
116 self.no_delay = task_info.no_delay
Xixuan Wu80531932017-10-12 17:26:51 -0700117 self.force = task_info.force
Xixuan Wu0c76d5b2017-08-30 16:40:17 -0700118 self.os_type = task_info.os_type
119 self.testbed_dut_count = task_info.testbed_dut_count
Xixuan Wu008ee832017-10-12 16:59:34 -0700120 self.hour = task_info.hour
121 self.day = task_info.day
Craig Bergstrom58263d32018-04-26 14:11:35 -0600122 self.only_hwtest_sanity_required = task_info.only_hwtest_sanity_required
Xixuan Wu0c76d5b2017-08-30 16:40:17 -0700123
124 if task_info.lc_branches:
125 self.launch_control_branches = [
126 t.strip() for t in task_info.lc_branches.split(',')]
127 else:
128 self.launch_control_branches = []
129
130 if task_info.lc_targets:
131 self.launch_control_targets = [
132 t.strip() for t in task_info.lc_targets.split(',')]
133 else:
134 self.launch_control_targets = []
135
136 if task_info.boards:
137 self.boards = [t.strip() for t in task_info.boards.split(',')]
138 else:
139 self.boards = []
140
Po-Hsien Wang6d589732018-05-15 17:19:34 -0700141 if task_info.exclude_boards:
142 self.exclude_boards = [
143 t.strip() for t in task_info.exclude_boards.split(',')]
144 else:
145 self.exclude_boards = []
146
Xixuan Wu89897182019-01-03 15:28:01 -0800147 if task_info.models:
148 self.models = [t.strip() for t in task_info.models.split(',')]
149 else:
150 self.models = []
151
152 if task_info.exclude_models:
153 self.exclude_models = [
154 t.strip() for t in task_info.exclude_models.split(',')]
155 else:
156 self.exclude_models = []
157
Po-Hsien Wangdd833072018-08-16 18:09:20 -0700158 if task_info.board_families:
159 # Finetune the allowed boards list with board_families & boards.
160 families = [family.strip()
161 for family in task_info.board_families.split(',')]
162 for family in families:
163 self.boards += board_family_config.get(family, [])
Xixuan Wu89897182019-01-03 15:28:01 -0800164
Po-Hsien Wangdd833072018-08-16 18:09:20 -0700165 if task_info.exclude_board_families:
166 # Finetune the disallowed boards list with exclude_board_families
167 # & exclude_boards.
168 families = [family.strip()
169 for family in task_info.exclude_board_families.split(',')]
170 for family in families:
171 self.exclude_boards += board_family_config.get(family, [])
172
Xixuan Wu0c76d5b2017-08-30 16:40:17 -0700173 if tot is None:
174 self.tot_manager = tot_manager.TotMilestoneManager()
175 else:
176 self.tot_manager = tot
177
178 self._set_spec_compare_info()
179
Xixuan Wuf4a4c882019-03-15 14:48:26 -0700180 def schedule(self, launch_control_builds, cros_builds_tuple, configs,
Xixuan Wuc6819012019-05-23 11:34:59 -0700181 build_client):
Xixuan Wu0c76d5b2017-08-30 16:40:17 -0700182 """Schedule the task by its settings.
183
184 Args:
185 launch_control_builds: the build dict for Android boards, see
186 return value of |get_launch_control_builds|.
Craig Bergstrom58263d32018-04-26 14:11:35 -0600187 cros_builds_tuple: the two-tuple of build dicts for ChromeOS boards,
188 see return value of |get_cros_builds|.
Xixuan Wuf4a4c882019-03-15 14:48:26 -0700189 configs: a config_reader.Configs object.
Xixuan Wuc6819012019-05-23 11:34:59 -0700190 build_client: a rest_client.BuildBucketBigqueryClient object, to
191 connect Buildbucket Bigquery.
Xixuan Wu0c76d5b2017-08-30 16:40:17 -0700192
193 Raises:
194 SchedulingError: if tasks that should be scheduled fail to schedule.
Xixuan Wu5451a662017-10-17 10:57:40 -0700195
196 Returns:
Jacob Kopczynski79d00102018-07-13 15:37:03 -0700197 A boolean indicator; true if there were any suites related to this
198 task which got pushed into the suites queue.
Xixuan Wu0c76d5b2017-08-30 16:40:17 -0700199 """
Xixuan Wuf4a4c882019-03-15 14:48:26 -0700200 assert configs.lab_config is not None
Xixuan Wu5451a662017-10-17 10:57:40 -0700201 self.is_pushed = False
202
Craig Bergstrom58263d32018-04-26 14:11:35 -0600203 branch_builds, relaxed_builds = cros_builds_tuple
204 builds_dict = branch_builds
205 if self.only_hwtest_sanity_required:
206 builds_dict = relaxed_builds
207
Xixuan Wu0c76d5b2017-08-30 16:40:17 -0700208 logging.info('######## Scheduling task %s ########', self.name)
209 if self.os_type == build_lib.OS_TYPE_CROS:
Craig Bergstrom58263d32018-04-26 14:11:35 -0600210 if not builds_dict:
Xixuan Wu0c76d5b2017-08-30 16:40:17 -0700211 logging.info('No CrOS build to run, skip running.')
212 else:
Xixuan Wuc6819012019-05-23 11:34:59 -0700213 self._schedule_cros_builds(builds_dict, configs, build_client)
Xixuan Wu0c76d5b2017-08-30 16:40:17 -0700214 else:
215 if not launch_control_builds:
216 logging.info('No Android build to run, skip running.')
217 else:
218 self._schedule_launch_control_builds(launch_control_builds)
219
Xixuan Wu5451a662017-10-17 10:57:40 -0700220 return self.is_pushed
221
Xixuan Wu0c76d5b2017-08-30 16:40:17 -0700222 def _set_spec_compare_info(self):
223 """Set branch spec compare info for task for further check."""
224 self._bare_branches = []
225 self._version_equal_constraint = False
226 self._version_gte_constraint = False
227 self._version_lte_constraint = False
228
229 if not self.branch_specs:
230 # Any milestone is OK.
231 self._numeric_constraint = version.LooseVersion('0')
232 else:
233 self._numeric_constraint = None
234 for spec in self.branch_specs:
235 if 'tot' in spec.lower():
236 # Convert spec >=tot-1 to >=RXX.
237 tot_str = spec[spec.index('tot'):]
238 spec = spec.replace(
239 tot_str, self.tot_manager.convert_tot_spec(tot_str))
240
241 if spec.startswith('>='):
242 self._numeric_constraint = version.LooseVersion(
243 spec.lstrip('>=R'))
244 self._version_gte_constraint = True
245 elif spec.startswith('<='):
246 self._numeric_constraint = version.LooseVersion(
247 spec.lstrip('<=R'))
248 self._version_lte_constraint = True
249 elif spec.startswith('=='):
250 self._version_equal_constraint = True
251 self._numeric_constraint = version.LooseVersion(
252 spec.lstrip('==R'))
253 else:
254 self._bare_branches.append(spec)
255
256 def _fits_spec(self, branch):
257 """Check if a branch is deemed OK by this task's branch specs.
258
259 Will return whether a branch 'fits' the specifications stored in this task.
260
261 Examples:
262 Assuming tot=R40
263 t = Task('Name', 'suite', ['factory', '>=tot-1'])
264 t._fits_spec('factory') # True
265 t._fits_spec('40') # True
266 t._fits_spec('38') # False
267 t._fits_spec('firmware') # False
268
269 Args:
270 branch: the branch to check.
271
272 Returns:
273 True if branch 'fits' with stored specs, False otherwise.
274 """
275 if branch in build_lib.BARE_BRANCHES:
276 return branch in self._bare_branches
277
278 if self._numeric_constraint:
279 if self._version_equal_constraint:
280 return version.LooseVersion(branch) == self._numeric_constraint
281 elif self._version_gte_constraint:
282 return version.LooseVersion(branch) >= self._numeric_constraint
283 elif self._version_lte_constraint:
284 return version.LooseVersion(branch) <= self._numeric_constraint
285 else:
286 return version.LooseVersion(branch) >= self._numeric_constraint
287 else:
288 return False
289
Xixuan Wuc6819012019-05-23 11:34:59 -0700290 def _get_latest_firmware_build_from_db(self, spec, board, build_client):
Xixuan Wu0c76d5b2017-08-30 16:40:17 -0700291 """Get the latest firmware build from CIDB database.
292
293 Args:
294 spec: a string build spec for RO or RW firmware, eg. firmware,
295 cros. For RO firmware, the value can also be released_ro_X.
296 board: the board against which this task will run suite job.
Xixuan Wuc6819012019-05-23 11:34:59 -0700297 build_client: a rest_client.BuildBucketBigqueryClient object, to
298 connect Buildbucket Bigquery.
Xixuan Wu0c76d5b2017-08-30 16:40:17 -0700299
300 Returns:
301 the latest firmware build.
302
303 Raises:
304 MySQLdb.OperationalError: if connection operations are not valid.
305 """
306 build_type = 'release' if spec == 'cros' else spec
307 # TODO(xixuan): not all firmware are saved in cidb.
Xixuan Wuc6819012019-05-23 11:34:59 -0700308 build = build_client.get_latest_passed_builds('%s-%s' % (board, build_type))
Xixuan Wu0c76d5b2017-08-30 16:40:17 -0700309
310 if build is None:
311 return None
312 else:
313 latest_build = str(build_lib.CrOSBuild(
314 board, build_type, build.milestone, build.platform))
315 logging.debug('latest firmware build for %s-%s: %s', board,
316 build_type, latest_build)
317 return latest_build
318
319 def _get_latest_firmware_build_from_lab_config(self, spec, board,
320 lab_config):
321 """Get latest firmware from lab config file.
322
323 Args:
324 spec: a string build spec for RO or RW firmware, eg. firmware,
325 cros. For RO firmware, the value can also be released_ro_X.
326 board: a string board against which this task will run suite job.
327 lab_config: a config.LabConfig object, to read lab config file.
328
329 Returns:
330 A string latest firmware build.
331
332 Raises:
333 ValueError: if no firmware build list is found in lab config file.
334 """
335 index = int(spec[len(_RELEASE_RO_FIRMWARE_SPEC):])
336 released_ro_builds = lab_config.get_firmware_ro_build_list(
337 'RELEASED_RO_BUILDS_%s' % board).split(',')
338 if len(released_ro_builds) < index:
339 raise ValueError('No %dth ro_builds in the lab_config firmware '
340 'list %r' % (index, released_ro_builds))
341
342 logging.debug('Get ro_build: %s', released_ro_builds[index - 1])
343 return released_ro_builds[index - 1]
344
Xixuan Wuc6819012019-05-23 11:34:59 -0700345 def _get_firmware_build(self, spec, board, lab_config, build_client):
Xixuan Wu0c76d5b2017-08-30 16:40:17 -0700346 """Get the firmware build name to test with ChromeOS build.
347
348 Args:
349 spec: a string build spec for RO or RW firmware, eg. firmware,
350 cros. For RO firmware, the value can also be released_ro_X.
351 board: a string board against which this task will run suite job.
352 lab_config: a config.LabConfig object, to read lab config file.
Xixuan Wuc6819012019-05-23 11:34:59 -0700353 build_client: a rest_client.BuildBucketBigqueryClient object, to
354 connect Buildbucket Bigquery.
Xixuan Wu0c76d5b2017-08-30 16:40:17 -0700355
356 Returns:
357 A string firmware build name.
358
359 Raises:
360 ValueError: if failing to get firmware from lab config file;
361 MySQLdb.OperationalError: if DB connection is not available.
362 """
363 if not spec or spec == 'stable':
364 # TODO(crbug.com/577316): Query stable RO firmware.
365 logging.debug('%s RO firmware build is not supported.', spec)
366 return None
367
368 try:
369 if spec.startswith(_RELEASE_RO_FIRMWARE_SPEC):
370 # For RO firmware whose value is released_ro_X, where X is the index of
371 # the list defined in lab config file:
372 # CROS/RELEASED_RO_BUILDS_[board].
373 # For example, for spec 'released_ro_2', and lab config file
374 # CROS/RELEASED_RO_BUILDS_veyron_jerry: build1,build2,
Craig Bergstrom58263d32018-04-26 14:11:35 -0600375 # return firmware RO build should be 'build2'.
Xixuan Wu0c76d5b2017-08-30 16:40:17 -0700376 return self._get_latest_firmware_build_from_lab_config(
377 spec, board, lab_config)
378 else:
Xixuan Wuc6819012019-05-23 11:34:59 -0700379 return self._get_latest_firmware_build_from_db(
380 spec, board, build_client)
Xixuan Wu0c76d5b2017-08-30 16:40:17 -0700381 except ValueError as e:
382 logging.warning('Failed to get firmware from lab config file'
383 'for spec %s, board %s: %s', spec, board, str(e))
384 return None
385 except MySQLdb.OperationalError as e:
386 logging.warning('Failed to get firmware from cidb for spec %s, '
387 'board %s: %s', spec, board, str(e))
388 return None
389
C Shapiro7f24a002017-12-05 14:25:09 -0700390 def _push_suite(
391 self,
392 board=None,
393 model=None,
394 cros_build=None,
395 firmware_rw_build=None,
396 firmware_ro_build=None,
397 test_source_build=None,
398 launch_control_build=None,
Xixuan Wubb74a372018-08-21 17:37:08 -0700399 run_prod_code=False,
Xixuan Wu028f6732019-04-11 14:47:42 -0700400 is_skylab=False,
401 override_pool='',
402 override_qs_account=''):
Xixuan Wu0c76d5b2017-08-30 16:40:17 -0700403 """Schedule suite job for the task by pushing suites to SuiteQueue.
404
405 Args:
406 board: the board against which this suite job run.
C Shapiro7f24a002017-12-05 14:25:09 -0700407 model: the model name for unibuild.
Xixuan Wu0c76d5b2017-08-30 16:40:17 -0700408 cros_build: the CrOS build of this suite job.
409 firmware_rw_build: Firmware RW build to run this suite job with.
410 firmware_ro_build: Firmware RO build to run this suite job with.
411 test_source_build: Test source build, used for server-side
412 packaging of this suite job.
413 launch_control_build: the launch control build of this suite job.
414 run_prod_code: If True, the suite will run the test code that lives
415 in prod aka the test code currently on the lab servers. If
416 False, the control files and test code for this suite run will
417 be retrieved from the build artifacts. Default is False.
Xixuan Wubb74a372018-08-21 17:37:08 -0700418 is_skylab: If True, schedule this suite to skylab, otherwise schedule
419 it to AFE. Default is False.
Xixuan Wu028f6732019-04-11 14:47:42 -0700420 override_pool: A string to indicate pool of quota scheduler.
421 override_qs_account: A string of quota scheduler account.
Xixuan Wu0c76d5b2017-08-30 16:40:17 -0700422 """
423 android_build = None
424 testbed_build = None
425
426 if self.testbed_dut_count:
427 launch_control_build = '%s#%d' % (launch_control_build,
428 self.testbed_dut_count)
429 test_source_build = launch_control_build
430 board = '%s-%d' % (board, self.testbed_dut_count)
431
432 if launch_control_build:
433 if not self.testbed_dut_count:
434 android_build = launch_control_build
435 else:
436 testbed_build = launch_control_build
437
438 suite_job_parameters = {
439 'suite': self.suite,
440 'board': board,
C Shapiro7f24a002017-12-05 14:25:09 -0700441 'model': model,
Xixuan Wu0c76d5b2017-08-30 16:40:17 -0700442 build_lib.BuildVersionKey.CROS_VERSION: cros_build,
443 build_lib.BuildVersionKey.FW_RW_VERSION: firmware_rw_build,
444 build_lib.BuildVersionKey.FW_RO_VERSION: firmware_ro_build,
445 build_lib.BuildVersionKey.ANDROID_BUILD_VERSION: android_build,
446 build_lib.BuildVersionKey.TESTBED_BUILD_VERSION: testbed_build,
447 'num': self.num,
448 'pool': self.pool,
449 'priority': self.priority,
450 'timeout': self.timeout,
451 'timeout_mins': _JOB_MAX_RUNTIME_MINS_DEFAULT,
452 'max_runtime_mins': _JOB_MAX_RUNTIME_MINS_DEFAULT,
Xixuan Wu2ba72652017-09-15 15:49:42 -0700453 'no_wait_for_results': not self.job_retry,
Xixuan Wu0c76d5b2017-08-30 16:40:17 -0700454 'test_source_build': test_source_build,
455 'job_retry': self.job_retry,
456 'no_delay': self.no_delay,
Xixuan Wu80531932017-10-12 17:26:51 -0700457 'force': self.force,
Xixuan Wu0c76d5b2017-08-30 16:40:17 -0700458 'run_prod_code': run_prod_code,
Xixuan Wubb74a372018-08-21 17:37:08 -0700459 'is_skylab': is_skylab,
Xixuan Wu028f6732019-04-11 14:47:42 -0700460 'override_pool': override_pool,
461 'override_qs_account': override_qs_account,
Xixuan Wu0c76d5b2017-08-30 16:40:17 -0700462 }
463
464 task_executor.push(task_executor.SUITES_QUEUE, **suite_job_parameters)
465 logging.info('Pushing task %r into taskqueue', suite_job_parameters)
Xixuan Wu5451a662017-10-17 10:57:40 -0700466 self.is_pushed = True
Xixuan Wu0c76d5b2017-08-30 16:40:17 -0700467
Xixuan Wuc6819012019-05-23 11:34:59 -0700468 def _schedule_cros_builds(self, build_dict, configs, build_client):
Xixuan Wu0c76d5b2017-08-30 16:40:17 -0700469 """Schedule tasks with branch builds.
470
471 Args:
Craig Bergstrom58263d32018-04-26 14:11:35 -0600472 build_dict: the build dict for ChromeOS boards, see return
Xixuan Wu0c76d5b2017-08-30 16:40:17 -0700473 value of |build_lib.get_cros_builds_since_date_from_db|.
Xixuan Wuf4a4c882019-03-15 14:48:26 -0700474 configs: A config_reader.Configs object.
Xixuan Wuc6819012019-05-23 11:34:59 -0700475 build_client: a rest_client.BuildBucketBigqueryClient object, to
476 connect Buildbucket Bigquery.
Xixuan Wu0c76d5b2017-08-30 16:40:17 -0700477 """
Xixuan Wuf4a4c882019-03-15 14:48:26 -0700478 lab_config = configs.lab_config
C Shapiro7f24a002017-12-05 14:25:09 -0700479 models_by_board = lab_config.get_cros_model_map() if lab_config else {}
Xixuan Wu8d2f2862018-08-28 16:48:04 -0700480 for (board, passed_model, build_type,
Craig Bergstrom58263d32018-04-26 14:11:35 -0600481 milestone), manifest in build_dict.iteritems():
Xixuan Wu0c76d5b2017-08-30 16:40:17 -0700482 cros_build = str(build_lib.CrOSBuild(board, build_type, milestone,
483 manifest))
484 logging.info('Running %s on %s', self.name, cros_build)
Po-Hsien Wang6d589732018-05-15 17:19:34 -0700485 if self.exclude_boards and board in self.exclude_boards:
486 logging.debug('Board %s is in excluded board list: %s',
487 board, self.exclude_boards)
488 continue
Xixuan Wu8d2f2862018-08-28 16:48:04 -0700489
Xixuan Wu0c76d5b2017-08-30 16:40:17 -0700490 if self.boards and board not in self.boards:
491 logging.debug('Board %s is not in supported board list: %s',
492 board, self.boards)
493 continue
494
495 # Check the fitness of the build's branch for task
496 branch_build_spec = _pick_branch(build_type, milestone)
497 if not self._fits_spec(branch_build_spec):
498 logging.debug("branch_build spec %s doesn't fit this task's "
499 "requirement: %s", branch_build_spec, self.branch_specs)
500 continue
501
502 firmware_rw_build = None
503 firmware_ro_build = None
504 if self.firmware_rw_build_spec or self.firmware_ro_build_spec:
505 firmware_rw_build = self._get_firmware_build(
Xixuan Wuc6819012019-05-23 11:34:59 -0700506 self.firmware_rw_build_spec, board, lab_config, build_client)
Xixuan Wu0c76d5b2017-08-30 16:40:17 -0700507 firmware_ro_build = self._get_firmware_build(
Xixuan Wuc6819012019-05-23 11:34:59 -0700508 self.firmware_ro_build_spec, board, lab_config, build_client)
Xixuan Wu0c76d5b2017-08-30 16:40:17 -0700509
510 if not firmware_ro_build and self.firmware_ro_build_spec:
511 logging.debug('No firmware ro build to run, skip running')
512 continue
513
514 if self.test_source == build_lib.BuildType.FIRMWARE_RW:
515 test_source_build = firmware_rw_build
516 else:
Jacob Kopczynski79d00102018-07-13 15:37:03 -0700517 # Default test source build to CrOS build if it's not specified.
518 # Past versions chose based on run_prod_code, but we no longer respect
519 # that option and scheduler settings should always set it to False.
Xixuan Wu0c76d5b2017-08-30 16:40:17 -0700520 test_source_build = cros_build
521
Xixuan Wub4b2f412019-05-03 11:22:31 -0700522 hwtest_board = build_lib.reshape_board(board)
Xixuan Wu1f0be4c2019-03-18 16:53:23 -0700523 is_skylab = skylab.should_run_in_skylab(configs.migration_config,
Xixuan Wub4b2f412019-05-03 11:22:31 -0700524 hwtest_board,
Xixuan Wued878ea2019-03-18 15:32:16 -0700525 passed_model,
Xixuan Wu1e42c752019-03-21 13:41:49 -0700526 self.suite,
527 self.pool)
Xixuan Wu028f6732019-04-11 14:47:42 -0700528 override_pool, override_qs_account = skylab.get_override_info(
529 configs.migration_config,
Xixuan Wub4b2f412019-05-03 11:22:31 -0700530 hwtest_board,
Xixuan Wu028f6732019-04-11 14:47:42 -0700531 passed_model,
532 self.suite,
533 self.pool)
Harpreet Grewalbbbb7de2019-02-05 19:35:03 +0000534 models = models_by_board.get(board, [None])
C Shapiro7f24a002017-12-05 14:25:09 -0700535
Harpreet Grewalbbbb7de2019-02-05 19:35:03 +0000536 for model in models:
537 if ((passed_model is not None and model == passed_model) or
538 passed_model is None):
Xixuan Wua41efa22019-05-17 14:28:04 -0700539 full_model_name = '%s_%s' % (board, model)
540 # Respect exclude first.
541 if self.exclude_models and full_model_name in self.exclude_models:
542 logging.debug("Skip model %s as it's in exclude model list %s",
543 model, self.exclude_models)
544 continue
545
546 if self.models and full_model_name not in self.models:
547 logging.debug("Skip model %s as it's not in support model list %s",
548 model, self.models)
549 continue
550
Harpreet Grewalbbbb7de2019-02-05 19:35:03 +0000551 self._push_suite(
Xixuan Wub4b2f412019-05-03 11:22:31 -0700552 board=hwtest_board,
Harpreet Grewalbbbb7de2019-02-05 19:35:03 +0000553 model=model,
554 cros_build=cros_build,
555 firmware_rw_build=firmware_rw_build,
556 firmware_ro_build=firmware_ro_build,
557 test_source_build=test_source_build,
Xixuan Wu028f6732019-04-11 14:47:42 -0700558 is_skylab=is_skylab,
559 override_pool=override_pool,
560 override_qs_account=override_qs_account)
Xixuan Wu0c76d5b2017-08-30 16:40:17 -0700561
562 def _schedule_launch_control_builds(self, launch_control_builds):
563 """Schedule tasks with launch control builds.
564
565 Args:
566 launch_control_builds: the build dict for Android boards.
567 """
568 for board, launch_control_build in launch_control_builds.iteritems():
569 logging.debug('Running %s on %s', self.name, board)
Po-Hsien Wang6d589732018-05-15 17:19:34 -0700570 if self.exclude_boards and board in self.exclude_boards:
571 logging.debug('Board %s is in excluded board list: %s',
572 board, self.exclude_boards)
573 continue
Xixuan Wu0c76d5b2017-08-30 16:40:17 -0700574 if self.boards and board not in self.boards:
575 logging.debug('Board %s is not in supported board list: %s',
576 board, self.boards)
577 continue
578
579 for android_build in launch_control_build:
580 if not any([branch in android_build
581 for branch in self.launch_control_branches]):
582 logging.debug('Branch %s is not required to run for task '
583 '%s', android_build, self.name)
584 continue
585
586 self._push_suite(board=board,
587 test_source_build=android_build,
588 launch_control_build=android_build)
589
590
591def _pick_branch(build_type, milestone):
592 """Select branch based on build type.
593
594 If the build_type is a bare branch, return build_type as the build spec.
595 If the build_type is a normal CrOS branch, return milestone as the build
596 spec.
597
598 Args:
599 build_type: a string builder name, like 'release'.
600 milestone: a string milestone, like '55'.
601
602 Returns:
603 A string milestone if build_type represents CrOS build, otherwise
604 return build_type.
605 """
606 return build_type if build_type in build_lib.BARE_BRANCHES else milestone