|
|
|
#!/usr/bin/env python
|
|
|
|
# Copyright (c) 2012 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.
|
|
|
|
|
|
|
|
"""Applies an issue from Rietveld.
|
|
|
|
"""
|
|
|
|
|
|
|
|
import getpass
|
|
|
|
import logging
|
|
|
|
import optparse
|
|
|
|
import os
|
|
|
|
import subprocess
|
|
|
|
import sys
|
|
|
|
import urllib2
|
|
|
|
|
|
|
|
import breakpad # pylint: disable=W0611
|
|
|
|
|
|
|
|
import checkout
|
|
|
|
import fix_encoding
|
|
|
|
import gclient_utils
|
|
|
|
import rietveld
|
|
|
|
import scm
|
|
|
|
|
|
|
|
BASE_DIR = os.path.dirname(os.path.abspath(__file__))
|
|
|
|
|
|
|
|
|
|
|
|
def main():
|
|
|
|
parser = optparse.OptionParser(description=sys.modules[__name__].__doc__)
|
|
|
|
parser.add_option(
|
|
|
|
'-v', '--verbose', action='count', default=0,
|
|
|
|
help='Prints debugging infos')
|
|
|
|
parser.add_option(
|
|
|
|
'-e', '--email',
|
|
|
|
help='Email address to access rietveld. If not specified, anonymous '
|
|
|
|
'access will be used.')
|
|
|
|
parser.add_option(
|
|
|
|
'-w', '--password', default=None,
|
|
|
|
help='Password for email addressed. Use - to read password from stdin.')
|
|
|
|
parser.add_option(
|
|
|
|
'-i', '--issue', type='int', help='Rietveld issue number')
|
|
|
|
parser.add_option(
|
|
|
|
'-p', '--patchset', type='int', help='Rietveld issue\'s patchset number')
|
|
|
|
parser.add_option(
|
|
|
|
'-r',
|
|
|
|
'--root_dir',
|
|
|
|
default=os.getcwd(),
|
|
|
|
help='Root directory to apply the patch')
|
|
|
|
parser.add_option(
|
|
|
|
'-s',
|
|
|
|
'--server',
|
|
|
|
default='http://codereview.chromium.org',
|
|
|
|
help='Rietveld server')
|
|
|
|
options, args = parser.parse_args()
|
|
|
|
logging.basicConfig(
|
|
|
|
format='%(levelname)5s %(module)11s(%(lineno)4d): %(message)s',
|
|
|
|
level=[logging.WARNING, logging.INFO, logging.DEBUG][
|
|
|
|
min(2, options.verbose)])
|
|
|
|
if args:
|
|
|
|
parser.error('Extra argument(s) "%s" not understood' % ' '.join(args))
|
|
|
|
if not options.issue:
|
|
|
|
parser.error('Require --issue')
|
|
|
|
options.server = options.server.rstrip('/')
|
|
|
|
if not options.server:
|
|
|
|
parser.error('Require a valid server')
|
|
|
|
|
|
|
|
if options.password == '-':
|
|
|
|
options.password = sys.stdin.readline().strip()
|
|
|
|
|
|
|
|
# Always try un-authenticated first.
|
|
|
|
# TODO(maruel): Use OAuth2 properly so we don't hit rate-limiting on login
|
|
|
|
# attempts.
|
|
|
|
# Bad except clauses order (HTTPError is an ancestor class of
|
|
|
|
# ClientLoginError)
|
|
|
|
# pylint: disable=E0701
|
|
|
|
obj = rietveld.Rietveld(options.server, '', None)
|
|
|
|
properties = None
|
|
|
|
try:
|
|
|
|
properties = obj.get_issue_properties(options.issue, False)
|
|
|
|
except urllib2.HTTPError, e:
|
|
|
|
if e.getcode() != 302:
|
|
|
|
raise
|
|
|
|
# TODO(maruel): A few 'Invalid username or password.' are printed first, we
|
|
|
|
# should get rid of those.
|
|
|
|
except rietveld.upload.ClientLoginError, e:
|
|
|
|
# Fine, we'll do proper authentication.
|
|
|
|
pass
|
|
|
|
if properties is None:
|
|
|
|
if options.email is not None:
|
|
|
|
obj = rietveld.Rietveld(options.server, options.email, options.password)
|
|
|
|
try:
|
|
|
|
properties = obj.get_issue_properties(options.issue, False)
|
|
|
|
except rietveld.upload.ClientLoginError, e:
|
|
|
|
if sys.stdout.closed:
|
|
|
|
print('Accessing the issue requires proper credentials.')
|
|
|
|
return 1
|
|
|
|
else:
|
|
|
|
print('Accessing the issue requires login.')
|
|
|
|
obj = rietveld.Rietveld(options.server, None, None)
|
|
|
|
try:
|
|
|
|
properties = obj.get_issue_properties(options.issue, False)
|
|
|
|
except rietveld.upload.ClientLoginError, e:
|
|
|
|
print('Accessing the issue requires proper credentials.')
|
|
|
|
return 1
|
|
|
|
|
|
|
|
if not options.patchset:
|
|
|
|
options.patchset = properties['patchsets'][-1]
|
|
|
|
print('No patchset specified. Using patchset %d' % options.patchset)
|
|
|
|
|
|
|
|
print('Downloading the patch.')
|
|
|
|
try:
|
|
|
|
patchset = obj.get_patch(options.issue, options.patchset)
|
|
|
|
except urllib2.HTTPError, e:
|
|
|
|
print(
|
|
|
|
'Failed to fetch the patch for issue %d, patchset %d.\n'
|
|
|
|
'Try visiting %s/%d') % (
|
|
|
|
options.issue, options.patchset,
|
|
|
|
options.server, options.issue)
|
|
|
|
return 1
|
|
|
|
for patch in patchset.patches:
|
|
|
|
print(patch)
|
|
|
|
full_dir = os.path.abspath(options.root_dir)
|
|
|
|
scm_type = scm.determine_scm(full_dir)
|
|
|
|
if scm_type == 'svn':
|
|
|
|
scm_obj = checkout.SvnCheckout(full_dir, None, None, None, None)
|
|
|
|
elif scm_type == 'git':
|
|
|
|
scm_obj = checkout.GitCheckoutBase(full_dir, None, None)
|
|
|
|
elif scm_type == None:
|
|
|
|
scm_obj = checkout.RawCheckout(full_dir, None, None)
|
|
|
|
else:
|
|
|
|
parser.error('Couldn\'t determine the scm')
|
|
|
|
|
|
|
|
# TODO(maruel): HACK, remove me.
|
|
|
|
# When run a build slave, make sure buildbot knows that the checkout was
|
|
|
|
# modified.
|
|
|
|
if options.root_dir == 'src' and getpass.getuser() == 'chrome-bot':
|
|
|
|
# See sourcedirIsPatched() in:
|
|
|
|
# http://src.chromium.org/viewvc/chrome/trunk/tools/build/scripts/slave/
|
|
|
|
# chromium_commands.py?view=markup
|
|
|
|
open('.buildbot-patched', 'w').close()
|
|
|
|
|
|
|
|
print('\nApplying the patch.')
|
|
|
|
try:
|
|
|
|
scm_obj.apply_patch(patchset, verbose=True)
|
|
|
|
except checkout.PatchApplicationFailed, e:
|
|
|
|
print(str(e))
|
|
|
|
print('CWD=%s' % os.getcwd())
|
|
|
|
print('Checkout path=%s' % scm_obj.project_path)
|
|
|
|
return 1
|
|
|
|
|
|
|
|
if 'DEPS' in map(os.path.basename, patchset.filenames):
|
|
|
|
gclient_root = gclient_utils.FindGclientRoot(full_dir)
|
|
|
|
if gclient_root and scm_type:
|
|
|
|
print(
|
|
|
|
'A DEPS file was updated inside a gclient checkout, running gclient '
|
|
|
|
'sync.')
|
|
|
|
base_rev = 'BASE' if scm_type == 'svn' else 'HEAD'
|
|
|
|
gclient_path = os.path.join(BASE_DIR, 'gclient')
|
|
|
|
if sys.platform == 'win32':
|
|
|
|
gclient_path += '.bat'
|
|
|
|
return subprocess.call(
|
|
|
|
[gclient_path, 'sync', '--revision', base_rev], cwd=gclient_root)
|
|
|
|
return 0
|
|
|
|
|
|
|
|
|
|
|
|
if __name__ == "__main__":
|
|
|
|
fix_encoding.fix_encoding()
|
|
|
|
sys.exit(main())
|