Chromium Code Reviews

Side by Side Diff: git_cl/git_cl.py

Issue 6685023: Make git_cl pass pylint cleanly (Closed) Base URL: svn://svn.chromium.org/chrome/trunk/tools/depot_tools
Patch Set: whoops Created 9 years, 9 months ago
Use n/p to move between diff chunks; N/P to move between comments.
Jump to:
View unified diff | | Annotate | Revision Log
OLDNEW
1 #!/usr/bin/python 1 #!/usr/bin/python
2 # git-cl -- a git-command for integrating reviews on Rietveld 2 # git-cl -- a git-command for integrating reviews on Rietveld
3 # Copyright (C) 2008 Evan Martin <martine@danga.com> 3 # Copyright (C) 2008 Evan Martin <martine@danga.com>
4 4
5 import errno 5 import errno
6 import logging 6 import logging
7 import optparse 7 import optparse
8 import os 8 import os
9 import re 9 import re
10 import StringIO 10 import StringIO
11 import subprocess 11 import subprocess
12 import sys 12 import sys
13 import tempfile 13 import tempfile
14 import textwrap 14 import textwrap
15 import upload
16 import urlparse 15 import urlparse
17 import urllib2 16 import urllib2
18 17
19 try: 18 try:
20 import readline 19 import readline # pylint: disable=W0611
21 except ImportError: 20 except ImportError:
22 pass 21 pass
23 22
23 # TODO(dpranke): don't use relative import.
24 import upload # pylint: disable=W0403
24 try: 25 try:
25 # TODO(dpranke): We wrap this in a try block for a limited form of 26 # TODO(dpranke): We wrap this in a try block for a limited form of
26 # backwards-compatibility with older versions of git-cl that weren't 27 # backwards-compatibility with older versions of git-cl that weren't
27 # dependent on depot_tools. This version should still work outside of 28 # dependent on depot_tools. This version should still work outside of
28 # depot_tools as long as --bypass-hooks is used. We should remove this 29 # depot_tools as long as --bypass-hooks is used. We should remove this
29 # once this has baked for a while and things seem safe. 30 # once this has baked for a while and things seem safe.
30 depot_tools_path = os.path.dirname(os.path.dirname(os.path.abspath(__file__))) 31 depot_tools_path = os.path.dirname(os.path.dirname(os.path.abspath(__file__)))
31 sys.path.append(depot_tools_path) 32 sys.path.append(depot_tools_path)
32 import breakpad 33 import breakpad # pylint: disable=W0611
33 except ImportError: 34 except ImportError:
34 pass 35 pass
35 36
36 DEFAULT_SERVER = 'http://codereview.appspot.com' 37 DEFAULT_SERVER = 'http://codereview.appspot.com'
37 POSTUPSTREAM_HOOK_PATTERN = '.git/hooks/post-cl-%s' 38 POSTUPSTREAM_HOOK_PATTERN = '.git/hooks/post-cl-%s'
38 DESCRIPTION_BACKUP_FILE = '~/.git_cl_description_backup' 39 DESCRIPTION_BACKUP_FILE = '~/.git_cl_description_backup'
39 40
40 def DieWithError(message): 41 def DieWithError(message):
41 print >> sys.stderr, message 42 print >> sys.stderr, message
42 sys.exit(1) 43 sys.exit(1)
(...skipping 557 matching lines...)
600 #PUSH_URL_CONFIG: url.ssh://gitrw.chromium.org.pushinsteadof 601 #PUSH_URL_CONFIG: url.ssh://gitrw.chromium.org.pushinsteadof
601 #ORIGIN_URL_CONFIG: http://src.chromium.org/git 602 #ORIGIN_URL_CONFIG: http://src.chromium.org/git
602 RunGit(['config', keyvals['PUSH_URL_CONFIG'], 603 RunGit(['config', keyvals['PUSH_URL_CONFIG'],
603 keyvals['ORIGIN_URL_CONFIG']]) 604 keyvals['ORIGIN_URL_CONFIG']])
604 605
605 606
606 @usage('[repo root containing codereview.settings]') 607 @usage('[repo root containing codereview.settings]')
607 def CMDconfig(parser, args): 608 def CMDconfig(parser, args):
608 """edit configuration for this tree""" 609 """edit configuration for this tree"""
609 610
610 (options, args) = parser.parse_args(args) 611 _, args = parser.parse_args(args)
611 if len(args) == 0: 612 if len(args) == 0:
612 GetCodereviewSettingsInteractively() 613 GetCodereviewSettingsInteractively()
613 return 0 614 return 0
614 615
615 url = args[0] 616 url = args[0]
616 if not url.endswith('codereview.settings'): 617 if not url.endswith('codereview.settings'):
617 url = os.path.join(url, 'codereview.settings') 618 url = os.path.join(url, 'codereview.settings')
618 619
619 # Load code review settings and download hooks (if available). 620 # Load code review settings and download hooks (if available).
620 LoadCodereviewSettingsFromFile(urllib2.urlopen(url)) 621 LoadCodereviewSettingsFromFile(urllib2.urlopen(url))
(...skipping 45 matching lines...)
666 print cl.GetDescription(pretty=True) 667 print cl.GetDescription(pretty=True)
667 return 0 668 return 0
668 669
669 670
670 @usage('[issue_number]') 671 @usage('[issue_number]')
671 def CMDissue(parser, args): 672 def CMDissue(parser, args):
672 """Set or display the current code review issue number. 673 """Set or display the current code review issue number.
673 674
674 Pass issue number 0 to clear the current issue. 675 Pass issue number 0 to clear the current issue.
675 """ 676 """
676 (options, args) = parser.parse_args(args) 677 _, args = parser.parse_args(args)
677 678
678 cl = Changelist() 679 cl = Changelist()
679 if len(args) > 0: 680 if len(args) > 0:
680 try: 681 try:
681 issue = int(args[0]) 682 issue = int(args[0])
682 except ValueError: 683 except ValueError:
683 DieWithError('Pass a number to set the issue or none to list it.\n' 684 DieWithError('Pass a number to set the issue or none to list it.\n'
684 'Maybe you want to run git cl status?') 685 'Maybe you want to run git cl status?')
685 cl.SetIssue(issue) 686 cl.SetIssue(issue)
686 print 'Issue number:', cl.GetIssue(), '(%s)' % cl.GetIssueURL() 687 print 'Issue number:', cl.GetIssue(), '(%s)' % cl.GetIssueURL()
(...skipping 497 matching lines...)
1184 parser.add_option('-f', action='store_true', dest='force', 1185 parser.add_option('-f', action='store_true', dest='force',
1185 help='with -b, clobber any existing branch') 1186 help='with -b, clobber any existing branch')
1186 parser.add_option('--reject', action='store_true', dest='reject', 1187 parser.add_option('--reject', action='store_true', dest='reject',
1187 help='allow failed patches and spew .rej files') 1188 help='allow failed patches and spew .rej files')
1188 parser.add_option('-n', '--no-commit', action='store_true', dest='nocommit', 1189 parser.add_option('-n', '--no-commit', action='store_true', dest='nocommit',
1189 help="don't commit after patch applies") 1190 help="don't commit after patch applies")
1190 (options, args) = parser.parse_args(args) 1191 (options, args) = parser.parse_args(args)
1191 if len(args) != 1: 1192 if len(args) != 1:
1192 parser.print_help() 1193 parser.print_help()
1193 return 1 1194 return 1
1194 input = args[0] 1195 issue_arg = args[0]
1195 1196
1196 if re.match(r'\d+', input): 1197 if re.match(r'\d+', input):
1197 # Input is an issue id. Figure out the URL. 1198 # Input is an issue id. Figure out the URL.
1198 issue = input 1199 issue = issue_arg
1199 server = settings.GetDefaultServerUrl() 1200 server = settings.GetDefaultServerUrl()
1200 fetch = urllib2.urlopen('%s/%s' % (server, issue)).read() 1201 fetch = urllib2.urlopen('%s/%s' % (server, issue)).read()
1201 m = re.search(r'/download/issue[0-9]+_[0-9]+.diff', fetch) 1202 m = re.search(r'/download/issue[0-9]+_[0-9]+.diff', fetch)
1202 if not m: 1203 if not m:
1203 DieWithError('Must pass an issue ID or full URL for ' 1204 DieWithError('Must pass an issue ID or full URL for '
1204 '\'Download raw patch set\'') 1205 '\'Download raw patch set\'')
1205 url = '%s%s' % (server, m.group(0).strip()) 1206 url = '%s%s' % (server, m.group(0).strip())
1206 else: 1207 else:
1207 # Assume it's a URL to the patch. Default to http. 1208 # Assume it's a URL to the patch. Default to http.
1208 input = FixUrl(input) 1209 issue_url = FixUrl(issue_arg)
1209 match = re.match(r'.*?/issue(\d+)_\d+.diff', input) 1210 match = re.match(r'.*?/issue(\d+)_\d+.diff', issue_url)
1210 if match: 1211 if match:
1211 issue = match.group(1) 1212 issue = match.group(1)
1212 url = input 1213 url = input
1213 else: 1214 else:
1214 DieWithError('Must pass an issue ID or full URL for ' 1215 DieWithError('Must pass an issue ID or full URL for '
1215 '\'Download raw patch set\'') 1216 '\'Download raw patch set\'')
1216 1217
1217 if options.newbranch: 1218 if options.newbranch:
1218 if options.force: 1219 if options.force:
1219 RunGit(['branch', '-D', options.newbranch], 1220 RunGit(['branch', '-D', options.newbranch],
(...skipping 62 matching lines...)
1282 return 'open' 1283 return 'open'
1283 return 'unknown' 1284 return 'unknown'
1284 return 'unset' 1285 return 'unset'
1285 1286
1286 1287
1287 def GetTreeStatusReason(): 1288 def GetTreeStatusReason():
1288 """Fetches the tree status from a json url and returns the message 1289 """Fetches the tree status from a json url and returns the message
1289 with the reason for the tree to be opened or closed.""" 1290 with the reason for the tree to be opened or closed."""
1290 # Don't import it at file level since simplejson is not installed by default 1291 # Don't import it at file level since simplejson is not installed by default
1291 # on python 2.5 and it is only used for git-cl tree which isn't often used, 1292 # on python 2.5 and it is only used for git-cl tree which isn't often used,
1292 # forcing everyone to install simplejson isn't efficient. 1293 # forcing everyone to install simplejson isn't efficient.
M-A Ruel 2011/03/13 20:45:48 We need to change that once git-cl is moved one di
1293 try: 1294 try:
1294 import simplejson as json 1295 import simplejson as json # pylint: disable=F0401
1295 except ImportError: 1296 except ImportError:
1296 try: 1297 try:
1297 import json 1298 import json
1298 # Some versions of python2.5 have an incomplete json module. Check to make 1299 # Some versions of python2.5 have an incomplete json module. Check to make
1299 # sure loads exists. 1300 # sure loads exists.
1300 json.loads 1301 if not json.loads:
M-A Ruel 2011/03/13 20:45:48 We don't need this check anymore.
1301 except (ImportError, AttributeError): 1302 raise ImportError
1303 except (ImportError):
1302 print >> sys.stderr, 'Please install simplejson' 1304 print >> sys.stderr, 'Please install simplejson'
1303 sys.exit(1) 1305 sys.exit(1)
1304 1306
1305 url = settings.GetTreeStatusUrl() 1307 url = settings.GetTreeStatusUrl()
1306 json_url = urlparse.urljoin(url, '/current?format=json') 1308 json_url = urlparse.urljoin(url, '/current?format=json')
1307 connection = urllib2.urlopen(json_url) 1309 connection = urllib2.urlopen(json_url)
1308 status = json.loads(connection.read()) 1310 status = json.loads(connection.read())
1309 connection.close() 1311 connection.close()
1310 return status['message'] 1312 return status['message']
1311 1313
1312 1314
1313 def CMDtree(parser, args): 1315 def CMDtree(parser, args):
1314 """show the status of the tree""" 1316 """show the status of the tree"""
1315 (options, args) = parser.parse_args(args) 1317 _, args = parser.parse_args(args)
1316 status = GetTreeStatus() 1318 status = GetTreeStatus()
1317 if 'unset' == status: 1319 if 'unset' == status:
1318 print 'You must configure your tree status URL by running "git cl config".' 1320 print 'You must configure your tree status URL by running "git cl config".'
1319 return 2 1321 return 2
1320 1322
1321 print "The tree is %s" % status 1323 print "The tree is %s" % status
1322 print 1324 print
1323 print GetTreeStatusReason() 1325 print GetTreeStatusReason()
1324 if status != 'open': 1326 if status != 'open':
1325 return 1 1327 return 1
1326 return 0 1328 return 0
1327 1329
1328 1330
1329 def CMDupstream(parser, args): 1331 def CMDupstream(parser, args):
1330 """print the name of the upstream branch, if any""" 1332 """print the name of the upstream branch, if any"""
1331 (options, args) = parser.parse_args(args) 1333 _, args = parser.parse_args(args)
1332 cl = Changelist() 1334 cl = Changelist()
1333 print cl.GetUpstreamBranch() 1335 print cl.GetUpstreamBranch()
1334 return 0 1336 return 0
1335 1337
1336 1338
1337 def Command(name): 1339 def Command(name):
1338 return getattr(sys.modules[__name__], 'CMD' + name, None) 1340 return getattr(sys.modules[__name__], 'CMD' + name, None)
1339 1341
1340 1342
1341 def CMDhelp(parser, args): 1343 def CMDhelp(parser, args):
1342 """print list of commands or help for a specific command""" 1344 """print list of commands or help for a specific command"""
1343 (options, args) = parser.parse_args(args) 1345 _, args = parser.parse_args(args)
1344 if len(args) == 1: 1346 if len(args) == 1:
1345 return main(args + ['--help']) 1347 return main(args + ['--help'])
1346 parser.print_help() 1348 parser.print_help()
1347 return 0 1349 return 0
1348 1350
1349 1351
1350 def GenUsage(parser, command): 1352 def GenUsage(parser, command):
1351 """Modify an OptParse object with the function's documentation.""" 1353 """Modify an OptParse object with the function's documentation."""
1352 obj = Command(command) 1354 obj = Command(command)
1353 more = getattr(obj, 'usage_more', '') 1355 more = getattr(obj, 'usage_more', '')
(...skipping 40 matching lines...)
1394 ('AppEngine is misbehaving and returned HTTP %d, again. Keep faith ' 1396 ('AppEngine is misbehaving and returned HTTP %d, again. Keep faith '
1395 'and retry or visit go/isgaeup.\n%s') % (e.code, str(e))) 1397 'and retry or visit go/isgaeup.\n%s') % (e.code, str(e)))
1396 1398
1397 # Not a known command. Default to help. 1399 # Not a known command. Default to help.
1398 GenUsage(parser, 'help') 1400 GenUsage(parser, 'help')
1399 return CMDhelp(parser, argv) 1401 return CMDhelp(parser, argv)
1400 1402
1401 1403
1402 if __name__ == '__main__': 1404 if __name__ == '__main__':
1403 sys.exit(main(sys.argv[1:])) 1405 sys.exit(main(sys.argv[1:]))
OLDNEW

Powered by Google App Engine