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

Unified 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 side-by-side diff with in-line comments
Download patch
« no previous file with comments | « no previous file | no next file » | no next file with comments »
Expand Comments ('e') | Collapse Comments ('c') | Show Comments Hide Comments ('s')
Index: git_cl.py
diff --git a/git_cl.py b/git_cl.py
index db9823042e62a91a1e8258e007cc5a90380c162b..69538499b035fbe76c2f02031f7a6ec3ced94828 100755
--- a/git_cl.py
+++ b/git_cl.py
@@ -13,10 +13,12 @@ import json
import logging
import optparse
import os
+import Queue
import re
import stat
import sys
import textwrap
+import threading
import urllib2
import urlparse
@@ -436,6 +438,7 @@ class Changelist(object):
self.cc = None
self.watchers = ()
self._remote = None
+ self._props = None
def GetCCList(self):
"""Return the users cc'd on this CL.
@@ -675,17 +678,28 @@ or verify this branch is set up to track another (via the --track argument to
stderr=subprocess2.PIPE, error_ok=True)
self.has_patchset = False
- def GetMostRecentPatchset(self, issue):
- return self.RpcServer().get_issue_properties(
- int(issue), False)['patchsets'][-1]
+ def GetMostRecentPatchset(self, issue=None):
+ return self.GetIssueProperties(issue)['patchsets'][-1]
def GetPatchSetDiff(self, issue, patchset):
return self.RpcServer().get(
'/download/issue%s_%s.diff' % (issue, patchset))
+ def GetIssueProperties(self, issue=None):
+ 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.
+ # Ignore self._props if issue is specified.
+ return self.RpcServer().get_issue_properties(issue, True)
+
+ if self._props is None:
+ issue = self.GetIssue()
+ if not issue:
+ self._props = {}
+ else:
+ 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.
+ return self._props
+
def GetApprovingReviewers(self):
- return get_approving_reviewers(
- self.RpcServer().get_issue_properties(self.GetIssue(), True))
+ return get_approving_reviewers(self.GetIssueProperties())
def SetIssue(self, issue):
"""Set this branch's issue. If issue=0, clears the issue."""
@@ -1064,6 +1078,8 @@ def CMDstatus(parser, args):
"""show status of changelists"""
parser.add_option('--field',
help='print only specific field (desc|id|patch|url)')
+ parser.add_option('-f', '--fast', action='store_true',
+ help='Do not retrieve review status')
(options, args) = parser.parse_args(args)
if options.field:
@@ -1093,8 +1109,57 @@ def CMDstatus(parser, args):
branches = dict((c.GetBranch(), c.GetIssueURL()) for c in changes)
alignment = max(5, max(len(b) for b in branches))
print 'Branches associated with reviews:'
+ # Adhoc thread pool to request data concurrently.
+ output = Queue.Queue()
+
+ # Silence upload.py otherwise it becomes unweldly.
+ upload.verbosity = 0
+
+ if not options.fast:
+ def fetch(b):
+ c = Changelist(branchref=b)
+ i = c.GetIssueURL()
+ try:
+ props = c.GetIssueProperties()
+ r = c.GetApprovingReviewers() if i else None
+ if not props.get('messages'):
+ r = None
+ except urllib2.HTTPError:
+ # The issue probably doesn't exist anymore.
+ i += ' (broken)'
+ r = None
+ output.put((b, i, r))
+
+ threads = [threading.Thread(target=fetch, args=(b,)) for b in branches]
+ for t in threads:
+ t.daemon = True
+ t.start()
+ else:
+ # Do not use GetApprovingReviewers(), since it requires an HTTP request.
+ for b in branches:
+ c = Changelist(branchref=b)
+ output.put((b, c.GetIssue(), None))
+
+ tmp = {}
+ alignment = max(5, max(len(ShortBranchName(b)) for b in branches))
for branch in sorted(branches):
- print " %*s: %s" % (alignment, branch, branches[branch])
+ while branch not in tmp:
+ b, i, r = output.get()
+ tmp[b] = (i, r)
+ issue, reviewers = tmp.pop(branch)
+ if not issue:
+ color = Fore.WHITE
+ elif reviewers:
+ # Was approved.
+ color = Fore.GREEN
+ elif reviewers is None:
+ # No message was sent.
+ color = Fore.RED
+ else:
+ color = Fore.BLUE
+ print ' %*s: %s%s%s' % (
+ alignment, ShortBranchName(branch), color, issue, Fore.RESET)
+
cl = Changelist()
print
print 'Current branch:',
@@ -1136,7 +1201,7 @@ def CMDcomments(parser, args):
cl = Changelist()
if cl.GetIssue():
- data = cl.RpcServer().get_issue_properties(cl.GetIssue(), True)
+ data = cl.GetIssueProperties()
for message in sorted(data['messages'], key=lambda x: x['date']):
if message['disapproval']:
color = Fore.RED
@@ -1434,7 +1499,7 @@ def CMDupload(parser, args):
options.reviewers = hook_results.reviewers.split(',')
if cl.GetIssue():
- latest_patchset = cl.GetMostRecentPatchset(cl.GetIssue())
+ latest_patchset = cl.GetMostRecentPatchset()
local_patchset = cl.GetPatchset()
if latest_patchset and local_patchset and local_patchset != latest_patchset:
print ('The last upload made from this repository was patchset #%d but '
@@ -1673,7 +1738,7 @@ def SendUpstream(parser, args, cmd):
'(you may be prompted for your codereview password)...')
cl.UpdateDescription(change_desc.description)
cl.CloseIssue()
- props = cl.RpcServer().get_issue_properties(cl.GetIssue(), False)
+ props = cl.GetIssueProperties()
patch_num = len(props['patchsets'])
comment = "Committed patchset #%d manually as r%s" % (patch_num, revision)
comment += ' (presubmit successful).' if not options.bypass_hooks else '.'
@@ -1946,7 +2011,7 @@ def CMDtry(parser, args):
patchset = cl.GetPatchset()
if not cl.GetPatchset():
- patchset = cl.GetMostRecentPatchset(cl.GetIssue())
+ patchset = cl.GetMostRecentPatchset()
cl.RpcServer().trigger_try_jobs(
cl.GetIssue(), patchset, options.name, options.clobber, options.revision,
« 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