2017-10-16 18:22:47 +00:00
|
|
|
#!/usr/bin/env python
|
|
|
|
|
|
|
|
# Copyright 2017 Google Inc.
|
|
|
|
#
|
|
|
|
# Use of this source code is governed by a BSD-style license that can be
|
|
|
|
# found in the LICENSE file.
|
|
|
|
|
|
|
|
|
|
|
|
"""Submit one or more try jobs."""
|
|
|
|
|
|
|
|
|
2021-03-25 13:04:43 +00:00
|
|
|
from __future__ import print_function
|
2017-10-16 18:22:47 +00:00
|
|
|
import argparse
|
|
|
|
import json
|
|
|
|
import os
|
|
|
|
import re
|
|
|
|
import subprocess
|
|
|
|
import sys
|
2017-10-17 18:50:26 +00:00
|
|
|
import tempfile
|
2020-06-25 12:24:08 +00:00
|
|
|
import urllib2
|
2017-10-16 18:22:47 +00:00
|
|
|
|
|
|
|
|
2019-11-15 18:58:17 +00:00
|
|
|
BUCKET_SKIA_PRIMARY = 'skia/skia.primary'
|
|
|
|
BUCKET_SKIA_INTERNAL = 'skia-internal/skia.internal'
|
2020-01-14 11:23:58 +00:00
|
|
|
INFRA_BOTS = os.path.join('infra', 'bots')
|
|
|
|
TASKS_JSON = os.path.join(INFRA_BOTS, 'tasks.json')
|
2017-10-17 18:50:26 +00:00
|
|
|
REPO_INTERNAL = 'https://skia.googlesource.com/internal_test.git'
|
|
|
|
TMP_DIR = os.path.join(tempfile.gettempdir(), 'sktry')
|
2017-10-17 13:18:18 +00:00
|
|
|
|
2020-01-14 11:23:58 +00:00
|
|
|
SKIA_ROOT = os.path.realpath(os.path.join(
|
|
|
|
os.path.dirname(os.path.abspath(__file__)), os.pardir))
|
|
|
|
SKIA_INFRA_BOTS = os.path.join(SKIA_ROOT, INFRA_BOTS)
|
|
|
|
sys.path.insert(0, SKIA_INFRA_BOTS)
|
2017-10-17 13:18:18 +00:00
|
|
|
|
2017-10-17 18:50:26 +00:00
|
|
|
import utils
|
|
|
|
|
|
|
|
|
2020-01-14 11:23:58 +00:00
|
|
|
def find_repo_root():
|
|
|
|
"""Find the root directory of the current repository."""
|
|
|
|
cwd = os.getcwd()
|
|
|
|
while True:
|
|
|
|
if os.path.isdir(os.path.join(cwd, '.git')):
|
|
|
|
return cwd
|
|
|
|
next_cwd = os.path.dirname(cwd)
|
|
|
|
if next_cwd == cwd:
|
|
|
|
raise Exception('Failed to find repo root!')
|
2020-01-14 12:00:03 +00:00
|
|
|
cwd = next_cwd
|
2020-01-14 11:23:58 +00:00
|
|
|
|
|
|
|
|
2017-10-17 18:50:26 +00:00
|
|
|
def get_jobs(repo):
|
|
|
|
"""Obtain the list of jobs from the given repo."""
|
|
|
|
# Maintain a copy of the repo in the temp dir.
|
|
|
|
if not os.path.isdir(TMP_DIR):
|
|
|
|
os.mkdir(TMP_DIR)
|
|
|
|
with utils.chdir(TMP_DIR):
|
|
|
|
dirname = repo.split('/')[-1]
|
|
|
|
if not os.path.isdir(dirname):
|
|
|
|
subprocess.check_call([
|
|
|
|
utils.GIT, 'clone', '--mirror', repo, dirname])
|
|
|
|
with utils.chdir(dirname):
|
|
|
|
subprocess.check_call([utils.GIT, 'remote', 'update'])
|
|
|
|
jobs = json.loads(subprocess.check_output([
|
2020-01-14 11:23:58 +00:00
|
|
|
utils.GIT, 'show', 'master:%s' % JOBS_JSON]))
|
2017-10-17 18:50:26 +00:00
|
|
|
return (BUCKET_SKIA_INTERNAL, jobs)
|
2017-10-16 18:22:47 +00:00
|
|
|
|
|
|
|
|
|
|
|
def main():
|
|
|
|
# Parse arguments.
|
2020-01-14 11:23:58 +00:00
|
|
|
d = 'Helper script for triggering try jobs.'
|
2017-10-16 18:22:47 +00:00
|
|
|
parser = argparse.ArgumentParser(description=d)
|
|
|
|
parser.add_argument('--list', action='store_true', default=False,
|
|
|
|
help='Just list the jobs; do not trigger anything.')
|
2017-10-17 18:50:26 +00:00
|
|
|
parser.add_argument('--internal', action='store_true', default=False,
|
|
|
|
help=('If set, include internal jobs. You must have '
|
|
|
|
'permission to view internal repos.'))
|
2017-10-16 18:22:47 +00:00
|
|
|
parser.add_argument('job', nargs='?', default=None,
|
|
|
|
help='Job name or regular expression to match job names.')
|
|
|
|
args = parser.parse_args()
|
|
|
|
|
2020-06-25 12:24:08 +00:00
|
|
|
# First, find the Gerrit issue number. If the change was uploaded using Depot
|
|
|
|
# Tools, this configuration will be present in the git config.
|
|
|
|
branch = subprocess.check_output(['git', 'branch', '--show-current']).rstrip()
|
|
|
|
if not branch:
|
2021-03-25 13:04:43 +00:00
|
|
|
print('Not on any branch; cannot trigger try jobs.')
|
2020-06-25 12:24:08 +00:00
|
|
|
sys.exit(1)
|
|
|
|
branch_issue_config = 'branch.%s.gerritissue' % branch
|
|
|
|
try:
|
|
|
|
issue = subprocess.check_output([
|
|
|
|
'git', 'config', '--local', branch_issue_config])
|
|
|
|
except subprocess.CalledProcessError:
|
|
|
|
# Not using Depot Tools. Find the Change-Id line in the most recent commit
|
|
|
|
# and obtain the issue number using that.
|
2021-03-25 13:04:43 +00:00
|
|
|
print('"git cl issue" not set; searching for Change-Id footer.')
|
2020-06-25 12:24:08 +00:00
|
|
|
msg = subprocess.check_output(['git', 'log', '-n1', branch])
|
|
|
|
m = re.search('Change-Id: (I[a-f0-9]+)', msg)
|
|
|
|
if not m:
|
2021-03-25 13:04:43 +00:00
|
|
|
print('No gerrit issue found in `git config --local %s` and no Change-Id'
|
|
|
|
' found in most recent commit message.')
|
2020-06-25 12:24:08 +00:00
|
|
|
sys.exit(1)
|
|
|
|
url = 'https://skia-review.googlesource.com/changes/%s' % m.groups()[0]
|
|
|
|
resp = urllib2.urlopen(url).read()
|
|
|
|
issue = str(json.loads('\n'.join(resp.splitlines()[1:]))['_number'])
|
2021-03-25 13:04:43 +00:00
|
|
|
print('Setting "git cl issue %s"' % issue)
|
2020-06-25 12:24:08 +00:00
|
|
|
subprocess.check_call(['git', 'cl', 'issue', issue])
|
2017-10-17 18:50:26 +00:00
|
|
|
# Load and filter the list of jobs.
|
2017-10-17 13:18:18 +00:00
|
|
|
jobs = []
|
2020-01-14 11:23:58 +00:00
|
|
|
tasks_json = os.path.join(find_repo_root(), TASKS_JSON)
|
|
|
|
with open(tasks_json) as f:
|
|
|
|
tasks_cfg = json.load(f)
|
|
|
|
skia_primary_jobs = []
|
|
|
|
for k, v in tasks_cfg['jobs'].iteritems():
|
|
|
|
skia_primary_jobs.append(k)
|
|
|
|
skia_primary_jobs.sort()
|
|
|
|
|
|
|
|
# TODO(borenet): This assumes that the current repo is associated with the
|
|
|
|
# skia.primary bucket. This will work for most repos but it would be better to
|
|
|
|
# look up the correct bucket to use.
|
|
|
|
jobs.append((BUCKET_SKIA_PRIMARY, skia_primary_jobs))
|
2017-10-17 18:50:26 +00:00
|
|
|
if args.internal:
|
|
|
|
jobs.append(get_jobs(REPO_INTERNAL))
|
2017-10-16 18:22:47 +00:00
|
|
|
if args.job:
|
2017-10-18 16:53:49 +00:00
|
|
|
filtered_jobs = []
|
2017-10-17 13:18:18 +00:00
|
|
|
for bucket, job_list in jobs:
|
|
|
|
filtered = [j for j in job_list if re.search(args.job, j)]
|
|
|
|
if len(filtered) > 0:
|
2017-10-18 16:53:49 +00:00
|
|
|
filtered_jobs.append((bucket, filtered))
|
|
|
|
jobs = filtered_jobs
|
2017-10-16 18:22:47 +00:00
|
|
|
|
|
|
|
# Display the list of jobs.
|
|
|
|
if len(jobs) == 0:
|
2021-03-25 13:04:43 +00:00
|
|
|
print('Found no jobs matching "%s"' % repr(args.job))
|
2017-10-16 18:22:47 +00:00
|
|
|
sys.exit(1)
|
2017-10-17 13:18:18 +00:00
|
|
|
count = 0
|
|
|
|
for bucket, job_list in jobs:
|
|
|
|
count += len(job_list)
|
2021-03-25 13:04:43 +00:00
|
|
|
print('Found %d jobs:' % count)
|
2017-10-17 13:18:18 +00:00
|
|
|
for bucket, job_list in jobs:
|
2021-03-25 13:04:43 +00:00
|
|
|
print(' %s:' % bucket)
|
2017-10-17 13:18:18 +00:00
|
|
|
for j in job_list:
|
2021-03-25 13:04:43 +00:00
|
|
|
print(' %s' % j)
|
2017-10-16 18:22:47 +00:00
|
|
|
if args.list:
|
|
|
|
return
|
|
|
|
|
2017-12-11 18:18:52 +00:00
|
|
|
if count > 1:
|
|
|
|
# Prompt before triggering jobs.
|
|
|
|
resp = raw_input('\nDo you want to trigger these jobs? (y/n or i for '
|
|
|
|
'interactive): ')
|
2021-03-25 13:04:43 +00:00
|
|
|
print('')
|
2017-12-11 18:18:52 +00:00
|
|
|
if resp != 'y' and resp != 'i':
|
|
|
|
sys.exit(1)
|
|
|
|
if resp == 'i':
|
|
|
|
filtered_jobs = []
|
|
|
|
for bucket, job_list in jobs:
|
|
|
|
new_job_list = []
|
|
|
|
for j in job_list:
|
|
|
|
incl = raw_input(('Trigger %s? (y/n): ' % j))
|
|
|
|
if incl == 'y':
|
|
|
|
new_job_list.append(j)
|
|
|
|
if len(new_job_list) > 0:
|
|
|
|
filtered_jobs.append((bucket, new_job_list))
|
|
|
|
jobs = filtered_jobs
|
2017-10-16 18:22:47 +00:00
|
|
|
|
|
|
|
# Trigger the try jobs.
|
2017-10-17 13:18:18 +00:00
|
|
|
for bucket, job_list in jobs:
|
|
|
|
cmd = ['git', 'cl', 'try', '-B', bucket]
|
|
|
|
for j in job_list:
|
|
|
|
cmd.extend(['-b', j])
|
|
|
|
try:
|
|
|
|
subprocess.check_call(cmd)
|
|
|
|
except subprocess.CalledProcessError:
|
|
|
|
# Output from the command will fall through, so just exit here rather than
|
|
|
|
# printing a stack trace.
|
|
|
|
sys.exit(1)
|
2017-10-16 18:22:47 +00:00
|
|
|
|
|
|
|
|
|
|
|
if __name__ == '__main__':
|
|
|
|
main()
|