Chromium Code Reviews
chromiumcodereview-hr@appspot.gserviceaccount.com (chromiumcodereview-hr) | Please choose your nickname with Settings | Help | Chromium Project | Gerrit Changes | Sign out
(27)

Side by Side Diff: git_cl.py

Issue 19967004: Add color support to git cl and fetch issue properties in parallel. (Closed) Base URL: svn://svn.chromium.org/chrome/trunk/tools/depot_tools
Patch Set: rietveld ate the previous patchset Created 7 years, 5 months ago
Use n/p to move between diff chunks; N/P to move between comments. Draft comments are only viewable by you.
Jump to:
View unified diff | Download patch | Annotate | Revision Log
« no previous file with comments | « no previous file | no next file » | no next file with comments »
Toggle Intra-line Diffs ('i') | Expand Comments ('e') | Collapse Comments ('c') | Show Comments Hide Comments ('s')
OLDNEW
1 #!/usr/bin/env python 1 #!/usr/bin/env python
2 # Copyright (c) 2012 The Chromium Authors. All rights reserved. 2 # Copyright (c) 2012 The Chromium Authors. All rights reserved.
3 # Use of this source code is governed by a BSD-style license that can be 3 # Use of this source code is governed by a BSD-style license that can be
4 # found in the LICENSE file. 4 # found in the LICENSE file.
5 5
6 # Copyright (C) 2008 Evan Martin <martine@danga.com> 6 # Copyright (C) 2008 Evan Martin <martine@danga.com>
7 7
8 """A git-command for integrating reviews on Rietveld.""" 8 """A git-command for integrating reviews on Rietveld."""
9 9
10 import difflib 10 import difflib
11 from distutils.version import LooseVersion 11 from distutils.version import LooseVersion
12 import json 12 import json
13 import logging 13 import logging
14 import optparse 14 import optparse
15 import os 15 import os
16 import Queue
16 import re 17 import re
17 import stat 18 import stat
18 import sys 19 import sys
19 import textwrap 20 import textwrap
21 import threading
20 import urllib2 22 import urllib2
21 import urlparse 23 import urlparse
22 24
23 try: 25 try:
24 import readline # pylint: disable=F0401,W0611 26 import readline # pylint: disable=F0401,W0611
25 except ImportError: 27 except ImportError:
26 pass 28 pass
27 29
28 30
29 from third_party import colorama 31 from third_party import colorama
(...skipping 399 matching lines...) Expand 10 before | Expand all | Expand 10 after
429 self.has_issue = False 431 self.has_issue = False
430 self.issue = None 432 self.issue = None
431 self.has_description = False 433 self.has_description = False
432 self.description = None 434 self.description = None
433 self.has_patchset = False 435 self.has_patchset = False
434 self.patchset = None 436 self.patchset = None
435 self._rpc_server = None 437 self._rpc_server = None
436 self.cc = None 438 self.cc = None
437 self.watchers = () 439 self.watchers = ()
438 self._remote = None 440 self._remote = None
441 self._props = None
439 442
440 def GetCCList(self): 443 def GetCCList(self):
441 """Return the users cc'd on this CL. 444 """Return the users cc'd on this CL.
442 445
443 Return is a string suitable for passing to gcl with the --cc flag. 446 Return is a string suitable for passing to gcl with the --cc flag.
444 """ 447 """
445 if self.cc is None: 448 if self.cc is None:
446 base_cc = settings .GetDefaultCCList() 449 base_cc = settings .GetDefaultCCList()
447 more_cc = ','.join(self.watchers) 450 more_cc = ','.join(self.watchers)
448 self.cc = ','.join(filter(None, (base_cc, more_cc))) or '' 451 self.cc = ','.join(filter(None, (base_cc, more_cc))) or ''
(...skipping 219 matching lines...) Expand 10 before | Expand all | Expand 10 after
668 671
669 def SetPatchset(self, patchset): 672 def SetPatchset(self, patchset):
670 """Set this branch's patchset. If patchset=0, clears the patchset.""" 673 """Set this branch's patchset. If patchset=0, clears the patchset."""
671 if patchset: 674 if patchset:
672 RunGit(['config', self._PatchsetSetting(), str(patchset)]) 675 RunGit(['config', self._PatchsetSetting(), str(patchset)])
673 else: 676 else:
674 RunGit(['config', '--unset', self._PatchsetSetting()], 677 RunGit(['config', '--unset', self._PatchsetSetting()],
675 stderr=subprocess2.PIPE, error_ok=True) 678 stderr=subprocess2.PIPE, error_ok=True)
676 self.has_patchset = False 679 self.has_patchset = False
677 680
678 def GetMostRecentPatchset(self, issue): 681 def GetMostRecentPatchset(self, issue=None):
679 return self.RpcServer().get_issue_properties( 682 return self.GetIssueProperties(issue)['patchsets'][-1]
680 int(issue), False)['patchsets'][-1]
681 683
682 def GetPatchSetDiff(self, issue, patchset): 684 def GetPatchSetDiff(self, issue, patchset):
683 return self.RpcServer().get( 685 return self.RpcServer().get(
684 '/download/issue%s_%s.diff' % (issue, patchset)) 686 '/download/issue%s_%s.diff' % (issue, patchset))
685 687
688 def GetIssueProperties(self, issue=None):
689 if issue:
iannucci 2013/07/23 18:34:16 probably should do issue is None... technically 0
M-A Ruel 2013/07/23 18:49:43 Deleted this code.
690 # Ignore self._props if issue is specified.
691 return self.RpcServer().get_issue_properties(issue, True)
692
693 if self._props is None:
694 issue = self.GetIssue()
695 if not issue:
696 self._props = {}
697 else:
698 self._props = self.RpcServer().get_issue_properties(issue, True)
iannucci 2013/07/23 18:34:16 Hm, this function is a quite weird... Can't we jus
M-A Ruel 2013/07/23 18:49:43 Fixed.
699 return self._props
700
686 def GetApprovingReviewers(self): 701 def GetApprovingReviewers(self):
687 return get_approving_reviewers( 702 return get_approving_reviewers(self.GetIssueProperties())
688 self.RpcServer().get_issue_properties(self.GetIssue(), True))
689 703
690 def SetIssue(self, issue): 704 def SetIssue(self, issue):
691 """Set this branch's issue. If issue=0, clears the issue.""" 705 """Set this branch's issue. If issue=0, clears the issue."""
692 if issue: 706 if issue:
693 RunGit(['config', self._IssueSetting(), str(issue)]) 707 RunGit(['config', self._IssueSetting(), str(issue)])
694 if self.rietveld_server: 708 if self.rietveld_server:
695 RunGit(['config', self._RietveldServer(), self.rietveld_server]) 709 RunGit(['config', self._RietveldServer(), self.rietveld_server])
696 else: 710 else:
697 RunGit(['config', '--unset', self._IssueSetting()]) 711 RunGit(['config', '--unset', self._IssueSetting()])
698 self.SetPatchset(0) 712 self.SetPatchset(0)
(...skipping 358 matching lines...) Expand 10 before | Expand all | Expand 10 after
1057 else: 1071 else:
1058 print("Setting base-url to %s" % args[0]) 1072 print("Setting base-url to %s" % args[0])
1059 return RunGit(['config', 'branch.%s.base-url' % branch, args[0]], 1073 return RunGit(['config', 'branch.%s.base-url' % branch, args[0]],
1060 error_ok=False).strip() 1074 error_ok=False).strip()
1061 1075
1062 1076
1063 def CMDstatus(parser, args): 1077 def CMDstatus(parser, args):
1064 """show status of changelists""" 1078 """show status of changelists"""
1065 parser.add_option('--field', 1079 parser.add_option('--field',
1066 help='print only specific field (desc|id|patch|url)') 1080 help='print only specific field (desc|id|patch|url)')
1081 parser.add_option('-f', '--fast', action='store_true',
1082 help='Do not retrieve review status')
1067 (options, args) = parser.parse_args(args) 1083 (options, args) = parser.parse_args(args)
1068 1084
1069 if options.field: 1085 if options.field:
1070 cl = Changelist() 1086 cl = Changelist()
1071 if options.field.startswith('desc'): 1087 if options.field.startswith('desc'):
1072 print cl.GetDescription() 1088 print cl.GetDescription()
1073 elif options.field == 'id': 1089 elif options.field == 'id':
1074 issueid = cl.GetIssue() 1090 issueid = cl.GetIssue()
1075 if issueid: 1091 if issueid:
1076 print issueid 1092 print issueid
1077 elif options.field == 'patch': 1093 elif options.field == 'patch':
1078 patchset = cl.GetPatchset() 1094 patchset = cl.GetPatchset()
1079 if patchset: 1095 if patchset:
1080 print patchset 1096 print patchset
1081 elif options.field == 'url': 1097 elif options.field == 'url':
1082 url = cl.GetIssueURL() 1098 url = cl.GetIssueURL()
1083 if url: 1099 if url:
1084 print url 1100 print url
1085 return 0 1101 return 0
1086 1102
1087 branches = RunGit(['for-each-ref', '--format=%(refname)', 'refs/heads']) 1103 branches = RunGit(['for-each-ref', '--format=%(refname)', 'refs/heads'])
1088 if not branches: 1104 if not branches:
1089 print('No local branch found.') 1105 print('No local branch found.')
1090 return 0 1106 return 0
1091 1107
1092 changes = (Changelist(branchref=b) for b in branches.splitlines()) 1108 changes = (Changelist(branchref=b) for b in branches.splitlines())
1093 branches = dict((c.GetBranch(), c.GetIssueURL()) for c in changes) 1109 branches = dict((c.GetBranch(), c.GetIssueURL()) for c in changes)
1094 alignment = max(5, max(len(b) for b in branches)) 1110 alignment = max(5, max(len(b) for b in branches))
1095 print 'Branches associated with reviews:' 1111 print 'Branches associated with reviews:'
1112 # Adhoc thread pool to request data concurrently.
1113 output = Queue.Queue()
1114
1115 # Silence upload.py otherwise it becomes unweldly.
1116 upload.verbosity = 0
1117
1118 if not options.fast:
1119 def fetch(b):
1120 c = Changelist(branchref=b)
1121 i = c.GetIssueURL()
1122 try:
1123 props = c.GetIssueProperties()
1124 r = c.GetApprovingReviewers() if i else None
1125 if not props.get('messages'):
1126 r = None
1127 except urllib2.HTTPError:
1128 # The issue probably doesn't exist anymore.
1129 i += ' (broken)'
1130 r = None
1131 output.put((b, i, r))
1132
1133 threads = [threading.Thread(target=fetch, args=(b,)) for b in branches]
1134 for t in threads:
1135 t.daemon = True
1136 t.start()
1137 else:
1138 # Do not use GetApprovingReviewers(), since it requires an HTTP request.
1139 for b in branches:
1140 c = Changelist(branchref=b)
1141 output.put((b, c.GetIssue(), None))
1142
1143 tmp = {}
1144 alignment = max(5, max(len(ShortBranchName(b)) for b in branches))
1096 for branch in sorted(branches): 1145 for branch in sorted(branches):
1097 print " %*s: %s" % (alignment, branch, branches[branch]) 1146 while branch not in tmp:
1147 b, i, r = output.get()
1148 tmp[b] = (i, r)
1149 issue, reviewers = tmp.pop(branch)
1150 if not issue:
1151 color = Fore.WHITE
1152 elif reviewers:
1153 # Was approved.
1154 color = Fore.GREEN
1155 elif reviewers is None:
1156 # No message was sent.
1157 color = Fore.RED
1158 else:
1159 color = Fore.BLUE
1160 print ' %*s: %s%s%s' % (
1161 alignment, ShortBranchName(branch), color, issue, Fore.RESET)
1162
1098 cl = Changelist() 1163 cl = Changelist()
1099 print 1164 print
1100 print 'Current branch:', 1165 print 'Current branch:',
1101 if not cl.GetIssue(): 1166 if not cl.GetIssue():
1102 print 'no issue assigned.' 1167 print 'no issue assigned.'
1103 return 0 1168 return 0
1104 print cl.GetBranch() 1169 print cl.GetBranch()
1105 print 'Issue number: %s (%s)' % (cl.GetIssue(), cl.GetIssueURL()) 1170 print 'Issue number: %s (%s)' % (cl.GetIssue(), cl.GetIssueURL())
1106 print 'Issue description:' 1171 print 'Issue description:'
1107 print cl.GetDescription(pretty=True) 1172 print cl.GetDescription(pretty=True)
(...skipping 21 matching lines...) Expand all
1129 1194
1130 1195
1131 def CMDcomments(parser, args): 1196 def CMDcomments(parser, args):
1132 """show review comments of the current changelist""" 1197 """show review comments of the current changelist"""
1133 (_, args) = parser.parse_args(args) 1198 (_, args) = parser.parse_args(args)
1134 if args: 1199 if args:
1135 parser.error('Unsupported argument: %s' % args) 1200 parser.error('Unsupported argument: %s' % args)
1136 1201
1137 cl = Changelist() 1202 cl = Changelist()
1138 if cl.GetIssue(): 1203 if cl.GetIssue():
1139 data = cl.RpcServer().get_issue_properties(cl.GetIssue(), True) 1204 data = cl.GetIssueProperties()
1140 for message in sorted(data['messages'], key=lambda x: x['date']): 1205 for message in sorted(data['messages'], key=lambda x: x['date']):
1141 if message['disapproval']: 1206 if message['disapproval']:
1142 color = Fore.RED 1207 color = Fore.RED
1143 elif message['approval']: 1208 elif message['approval']:
1144 color = Fore.GREEN 1209 color = Fore.GREEN
1145 elif message['sender'] == data['owner_email']: 1210 elif message['sender'] == data['owner_email']:
1146 color = Fore.MAGENTA 1211 color = Fore.MAGENTA
1147 else: 1212 else:
1148 color = Fore.BLUE 1213 color = Fore.BLUE
1149 print '\n%s%s %s%s' % ( 1214 print '\n%s%s %s%s' % (
(...skipping 277 matching lines...) Expand 10 before | Expand all | Expand 10 after
1427 hook_results = cl.RunHook(committing=False, 1492 hook_results = cl.RunHook(committing=False,
1428 may_prompt=not options.force, 1493 may_prompt=not options.force,
1429 verbose=options.verbose, 1494 verbose=options.verbose,
1430 change=change) 1495 change=change)
1431 if not hook_results.should_continue(): 1496 if not hook_results.should_continue():
1432 return 1 1497 return 1
1433 if not options.reviewers and hook_results.reviewers: 1498 if not options.reviewers and hook_results.reviewers:
1434 options.reviewers = hook_results.reviewers.split(',') 1499 options.reviewers = hook_results.reviewers.split(',')
1435 1500
1436 if cl.GetIssue(): 1501 if cl.GetIssue():
1437 latest_patchset = cl.GetMostRecentPatchset(cl.GetIssue()) 1502 latest_patchset = cl.GetMostRecentPatchset()
1438 local_patchset = cl.GetPatchset() 1503 local_patchset = cl.GetPatchset()
1439 if latest_patchset and local_patchset and local_patchset != latest_patchset: 1504 if latest_patchset and local_patchset and local_patchset != latest_patchset:
1440 print ('The last upload made from this repository was patchset #%d but ' 1505 print ('The last upload made from this repository was patchset #%d but '
1441 'the most recent patchset on the server is #%d.' 1506 'the most recent patchset on the server is #%d.'
1442 % (local_patchset, latest_patchset)) 1507 % (local_patchset, latest_patchset))
1443 print ('Uploading will still work, but if you\'ve uploaded to this issue ' 1508 print ('Uploading will still work, but if you\'ve uploaded to this issue '
1444 'from another machine or branch the patch you\'re uploading now ' 1509 'from another machine or branch the patch you\'re uploading now '
1445 'might not include those changes.') 1510 'might not include those changes.')
1446 ask_for_data('About to upload; enter to confirm.') 1511 ask_for_data('About to upload; enter to confirm.')
1447 1512
(...skipping 218 matching lines...) Expand 10 before | Expand all | Expand 10 after
1666 return 1 1731 return 1
1667 viewvc_url = settings.GetViewVCUrl() 1732 viewvc_url = settings.GetViewVCUrl()
1668 if viewvc_url and revision: 1733 if viewvc_url and revision:
1669 change_desc.append_footer('Committed: ' + viewvc_url + revision) 1734 change_desc.append_footer('Committed: ' + viewvc_url + revision)
1670 elif revision: 1735 elif revision:
1671 change_desc.append_footer('Committed: ' + revision) 1736 change_desc.append_footer('Committed: ' + revision)
1672 print ('Closing issue ' 1737 print ('Closing issue '
1673 '(you may be prompted for your codereview password)...') 1738 '(you may be prompted for your codereview password)...')
1674 cl.UpdateDescription(change_desc.description) 1739 cl.UpdateDescription(change_desc.description)
1675 cl.CloseIssue() 1740 cl.CloseIssue()
1676 props = cl.RpcServer().get_issue_properties(cl.GetIssue(), False) 1741 props = cl.GetIssueProperties()
1677 patch_num = len(props['patchsets']) 1742 patch_num = len(props['patchsets'])
1678 comment = "Committed patchset #%d manually as r%s" % (patch_num, revision) 1743 comment = "Committed patchset #%d manually as r%s" % (patch_num, revision)
1679 comment += ' (presubmit successful).' if not options.bypass_hooks else '.' 1744 comment += ' (presubmit successful).' if not options.bypass_hooks else '.'
1680 cl.RpcServer().add_comment(cl.GetIssue(), comment) 1745 cl.RpcServer().add_comment(cl.GetIssue(), comment)
1681 cl.SetIssue(0) 1746 cl.SetIssue(0)
1682 1747
1683 if retcode == 0: 1748 if retcode == 0:
1684 hook = POSTUPSTREAM_HOOK_PATTERN % cmd 1749 hook = POSTUPSTREAM_HOOK_PATTERN % cmd
1685 if os.path.isfile(hook): 1750 if os.path.isfile(hook):
1686 RunCommand([hook, base_branch], error_ok=True) 1751 RunCommand([hook, base_branch], error_ok=True)
(...skipping 252 matching lines...) Expand 10 before | Expand all | Expand 10 after
1939 if any('triggered' in b for b in builders_and_tests): 2004 if any('triggered' in b for b in builders_and_tests):
1940 print >> sys.stderr, ( 2005 print >> sys.stderr, (
1941 'ERROR You are trying to send a job to a triggered bot. This type of' 2006 'ERROR You are trying to send a job to a triggered bot. This type of'
1942 ' bot requires an\ninitial job from a parent (usually a builder). ' 2007 ' bot requires an\ninitial job from a parent (usually a builder). '
1943 'Instead send your job to the parent.\n' 2008 'Instead send your job to the parent.\n'
1944 'Bot list: %s' % builders_and_tests) 2009 'Bot list: %s' % builders_and_tests)
1945 return 1 2010 return 1
1946 2011
1947 patchset = cl.GetPatchset() 2012 patchset = cl.GetPatchset()
1948 if not cl.GetPatchset(): 2013 if not cl.GetPatchset():
1949 patchset = cl.GetMostRecentPatchset(cl.GetIssue()) 2014 patchset = cl.GetMostRecentPatchset()
1950 2015
1951 cl.RpcServer().trigger_try_jobs( 2016 cl.RpcServer().trigger_try_jobs(
1952 cl.GetIssue(), patchset, options.name, options.clobber, options.revision, 2017 cl.GetIssue(), patchset, options.name, options.clobber, options.revision,
1953 builders_and_tests) 2018 builders_and_tests)
1954 print('Tried jobs on:') 2019 print('Tried jobs on:')
1955 length = max(len(builder) for builder in builders_and_tests) 2020 length = max(len(builder) for builder in builders_and_tests)
1956 for builder in sorted(builders_and_tests): 2021 for builder in sorted(builders_and_tests):
1957 print ' %*s: %s' % (length, builder, ','.join(builders_and_tests[builder])) 2022 print ' %*s: %s' % (length, builder, ','.join(builders_and_tests[builder]))
1958 return 0 2023 return 0
1959 2024
(...skipping 217 matching lines...) Expand 10 before | Expand all | Expand 10 after
2177 GenUsage(parser, 'help') 2242 GenUsage(parser, 'help')
2178 return CMDhelp(parser, argv) 2243 return CMDhelp(parser, argv)
2179 2244
2180 2245
2181 if __name__ == '__main__': 2246 if __name__ == '__main__':
2182 # These affect sys.stdout so do it outside of main() to simplify mocks in 2247 # These affect sys.stdout so do it outside of main() to simplify mocks in
2183 # unit testing. 2248 # unit testing.
2184 fix_encoding.fix_encoding() 2249 fix_encoding.fix_encoding()
2185 colorama.init() 2250 colorama.init()
2186 sys.exit(main(sys.argv[1:])) 2251 sys.exit(main(sys.argv[1:]))
OLDNEW
« no previous file with comments | « no previous file | no next file » | no next file with comments »

Powered by Google App Engine
This is Rietveld 408576698