blob: 58b0ff8256805fc62558b2ee56bfc0c185d08478 [file] [log] [blame]
#!/usr/bin/env python3
# Copyright 2018 The ChromiumOS Authors
# Use of this source code is governed by a BSD-style license that can be
# found in the LICENSE file.
"""Chrome bisector to bisect a range of chrome commits.
This bisector bisects commits between branched chrome and releases.
"""
import logging
import os
from bisect_kit import bisector_cli
from bisect_kit import cli
from bisect_kit import codechange
from bisect_kit import configure
from bisect_kit import core
from bisect_kit import cr_util
from bisect_kit import cros_lab_util
from bisect_kit import cros_util
from bisect_kit import errors
from bisect_kit import gclient_util
logger = logging.getLogger(__name__)
def guess_chrome_version(opts, rev):
if cros_util.is_cros_version(rev):
assert opts.board, 'need to specify BOARD for cros version'
chrome_version = cros_util.query_chrome_version(opts.board, rev)
assert cr_util.is_chrome_version(chrome_version)
logger.info(
'Converted given CrOS version %s to Chrome version %s',
rev,
chrome_version,
)
rev = chrome_version
return rev
def generate_action_link(action):
if action['action_type'] == 'commit':
repo_url = action['repo_url']
# normalize
if repo_url == 'https://chromium.googlesource.com/a/chromium/src.git':
repo_url = 'https://chromium.googlesource.com/chromium/src.git'
action['link'] = repo_url + '/+/' + action['rev']
class ChromeSrcDomain(core.BisectDomain):
"""BisectDomain for Chrome branched tree"""
revtype = staticmethod(
cli.argtype_multiplexer(
cr_util.argtype_chrome_version, cros_util.argtype_cros_version
)
)
intra_revtype = staticmethod(
codechange.argtype_intra_rev(cr_util.argtype_chrome_version)
)
help = globals()['__doc__']
@staticmethod
def add_init_arguments(parser):
parser.add_argument(
'--chrome_root',
required=True,
metavar='CHROME_ROOT',
type=cli.argtype_dir_path,
default=configure.get('CHROME_ROOT'),
help='Root of chrome source tree, like ~/chromium',
)
parser.add_argument(
'--chrome_mirror',
required=True,
metavar='CHROME_MIRROR',
type=cli.argtype_dir_path,
default=configure.get('CHROME_MIRROR'),
help='gclient cache dir',
)
# Only used for Chrome on ChromeOS.
parser.add_argument(
'--dut',
type=cli.argtype_notempty,
metavar='DUT',
default=configure.get('DUT'),
help='For ChromeOS, address of DUT (Device Under Test)',
)
parser.add_argument(
'--board',
metavar='BOARD',
default=configure.get('BOARD'),
help='For ChromeOS, board name',
)
@staticmethod
def init(opts):
chrome_src = os.path.join(opts.chrome_root, 'src')
if not os.path.exists(chrome_src):
raise errors.ArgumentError(
'--chrome_root', "chrome src directory doesn't exist"
)
if opts.dut:
if cros_lab_util.is_satlab_dut(opts.dut):
cros_lab_util.write_satlab_ssh_config(opts.dut)
if not cros_util.is_dut(opts.dut):
raise errors.ArgumentError(
'--dut', 'invalid DUT or unable to connect'
)
if not opts.board:
opts.board = cros_util.query_dut_board(opts.dut)
old = guess_chrome_version(opts, opts.old)
new = guess_chrome_version(opts, opts.new)
if old == new:
raise errors.ArgumentError(
'--old and --new',
'start and end of chrome versions are identical',
)
if not cr_util.is_version_lesseq(old, new):
raise errors.ArgumentError(
'--old and --new',
'start chrome version (%s) is newer than end version (%s)'
% (old, new),
)
config = dict(
chrome_root=opts.chrome_root,
old=old,
new=new,
board=opts.board,
dut=opts.dut,
chrome_mirror=opts.chrome_mirror,
)
spec_manager = cr_util.ChromeSpecManager(config)
cache = gclient_util.GclientCache(opts.chrome_mirror)
# b/227415522: Sync Chrome source tree to old and new to fetch unreachable
# commits.
gclient_util.sync(opts.chrome_root, revision=opts.new)
gclient_util.sync(opts.chrome_root, revision=opts.old)
if not cr_util.is_direct_relative_version(old, new):
logger.warning('old=%s is not parent of new=%s', old, new)
lowest_common_ancestor = spec_manager.get_lca_chrome_localbuild(
old, new
)
logger.warning(
'Assume their lowest common ancestor, %s,'
'still have expected old behavior as %s',
lowest_common_ancestor,
old,
)
config['old'] = old = lowest_common_ancestor
code_manager = codechange.CodeManager(
opts.chrome_root, spec_manager, cache
)
revlist, details = code_manager.build_revlist(old, new)
for detail in details.values():
for action in detail.get('actions', []):
generate_action_link(action)
return config, {'revlist': revlist, 'details': details}
def __init__(self, config):
self.config = config
def setenv(self, env, rev):
env['CHROME_ROOT'] = self.config['chrome_root']
env['CHROME_MIRROR'] = self.config['chrome_mirror']
env['REV'] = rev
if self.config['board']:
env['BOARD'] = self.config['board']
if self.config['dut']:
env['DUT'] = self.config['dut']
def fill_candidate_summary(self, summary):
if 'current_range' in summary:
old, new = summary['current_range']
old_base, _, _ = codechange.parse_intra_rev(old)
_, new_next, _ = codechange.parse_intra_rev(new)
log_url = (
'https://chromium.googlesource.com/chromium/src/+log/%s..%s?n=10000'
% (old_base, new_next)
)
summary['links'] = [
{
'name': 'change_list',
'url': log_url,
'note': 'The link of change list only lists chrome src/ commits. For '
'example, commits inside v8 and third party repos are not '
'listed.',
},
{
'name': 'fuller',
'url': log_url + '&pretty=fuller',
},
]
if __name__ == '__main__':
bisector_cli.BisectorCommandLine(ChromeSrcDomain).main()