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

Side by Side Diff: my_activity.py

Issue 24047003: my_activity.py: Use gerrit new REST API. (Closed) Base URL: https://chromium.googlesource.com/chromium/tools/depot_tools.git@master
Patch Set: Created 7 years, 3 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
« 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 """Get stats about your activity. 6 """Get stats about your activity.
7 7
8 Example: 8 Example:
9 - my_activity.py for stats for the current week (last week on mondays). 9 - my_activity.py for stats for the current week (last week on mondays).
10 - my_activity.py -Q for stats for last quarter. 10 - my_activity.py -Q for stats for last quarter.
(...skipping 98 matching lines...) Expand 10 before | Expand all | Expand 10 after
109 { 109 {
110 'url': 'breakpad.appspot.com', 110 'url': 'breakpad.appspot.com',
111 'supports_owner_modified_query': False, 111 'supports_owner_modified_query': False,
112 'requires_auth': False, 112 'requires_auth': False,
113 'email_domain': 'chromium.org', 113 'email_domain': 'chromium.org',
114 }, 114 },
115 ] 115 ]
116 116
117 gerrit_instances = [ 117 gerrit_instances = [
118 { 118 {
119 'url': 'gerrit.chromium.org', 119 'url': 'chromium-review.googlesource.com',
120 'port': 29418,
121 'shorturl': 'crosreview.com', 120 'shorturl': 'crosreview.com',
122 }, 121 },
123 { 122 {
124 'url': 'gerrit-int.chromium.org', 123 'url': 'chrome-internal-review.googlesource.com',
125 'port': 29419,
126 'shorturl': 'crosreview.com/i', 124 'shorturl': 'crosreview.com/i',
127 }, 125 },
128 ] 126 ]
129 127
130 google_code_projects = [ 128 google_code_projects = [
131 { 129 {
132 'name': 'chromium', 130 'name': 'chromium',
133 'shorturl': 'crbug.com', 131 'shorturl': 'crbug.com',
134 }, 132 },
135 { 133 {
(...skipping 96 matching lines...) Expand 10 before | Expand all | Expand 10 after
232 230
233 def get_yes_or_no(msg): 231 def get_yes_or_no(msg):
234 while True: 232 while True:
235 response = raw_input(msg + ' yes/no [no] ') 233 response = raw_input(msg + ' yes/no [no] ')
236 if response == 'y' or response == 'yes': 234 if response == 'y' or response == 'yes':
237 return True 235 return True
238 elif not response or response == 'n' or response == 'no': 236 elif not response or response == 'n' or response == 'no':
239 return False 237 return False
240 238
241 239
240 def datetime_from_gerrit(date_string):
241 return datetime.strptime(date_string, '%Y-%m-%d %H:%M:%S.%f000')
242
243
242 def datetime_from_rietveld(date_string): 244 def datetime_from_rietveld(date_string):
243 return datetime.strptime(date_string, '%Y-%m-%d %H:%M:%S.%f') 245 return datetime.strptime(date_string, '%Y-%m-%d %H:%M:%S.%f')
244 246
245 247
246 def datetime_from_google_code(date_string): 248 def datetime_from_google_code(date_string):
247 return datetime.strptime(date_string, '%Y-%m-%dT%H:%M:%S.%fZ') 249 return datetime.strptime(date_string, '%Y-%m-%dT%H:%M:%S.%fZ')
248 250
249 251
250 class MyActivity(object): 252 class MyActivity(object):
251 def __init__(self, options): 253 def __init__(self, options):
(...skipping 121 matching lines...) Expand 10 before | Expand all | Expand 10 after
373 r['author'] = reply['sender'] 375 r['author'] = reply['sender']
374 r['created'] = datetime_from_rietveld(reply['date']) 376 r['created'] = datetime_from_rietveld(reply['date'])
375 r['content'] = '' 377 r['content'] = ''
376 ret.append(r) 378 ret.append(r)
377 return ret 379 return ret
378 380
379 def gerrit_search(self, instance, owner=None, reviewer=None): 381 def gerrit_search(self, instance, owner=None, reviewer=None):
380 max_age = datetime.today() - self.modified_after 382 max_age = datetime.today() - self.modified_after
381 max_age = max_age.days * 24 * 3600 + max_age.seconds 383 max_age = max_age.days * 24 * 3600 + max_age.seconds
382 384
383 # See https://review.openstack.org/Documentation/cmd-query.html 385 # See https://gerrit-review.googlesource.com/Documentation/rest-api.html
384 # Gerrit doesn't allow filtering by created time, only modified time. 386 # Gerrit doesn't allow filtering by created time, only modified time.
385 user_filter = 'owner:%s' % owner if owner else 'reviewer:%s' % reviewer 387 user_filter = 'owner:%s' % owner if owner else 'reviewer:%s' % reviewer
386 gquery_cmd = ['ssh', '-p', str(instance['port']), instance['url'], 388 args = 'q=-age:%ss+%s&o=LABELS&o=MESSAGES' % (str(max_age), user_filter)
cjhopman 2013/09/09 21:50:05 use urllib.urlencode to encode parameters if possi
deymo 2013/09/09 22:09:41 Done.
387 'gerrit', 'query', 389 rest_url = 'https://%s/changes/?%s' % (instance['url'], args)
388 '--format', 'JSON', 390
389 '--comments', 391 req = urllib2.Request(rest_url, headers={'Accept': 'text/plain'})
390 '--', 392 try:
391 '-age:%ss' % str(max_age), 393 response = urllib2.urlopen(req)
392 user_filter] 394 stdout = response.read()
393 [stdout, _] = subprocess.Popen(gquery_cmd, stdout=subprocess.PIPE, 395 except urllib2.HTTPError, e:
394 stderr=subprocess.PIPE).communicate() 396 print "Looking up %r: %s" % (rest_url, e)
395 issues = str(stdout).split('\n')[:-2] 397 return []
396 issues = map(json.loads, issues) 398 open('zaraza', 'w').write(stdout)
cjhopman 2013/09/09 21:50:05 ?
deymo 2013/09/09 22:09:41 aaagr... I forgot to remove that debug line. Sorry
399 # Check that the returned JSON starts with the right marker.
400 if stdout[0:5] != ")]}'\n":
401 return []
402 issues = json.loads(stdout[5:])
397 403
398 # TODO(cjhopman): should we filter abandoned changes? 404 # TODO(cjhopman): should we filter abandoned changes?
399 issues = [self.process_gerrit_issue(instance, issue) for issue in issues] 405 issues = [self.process_gerrit_issue(instance, issue) for issue in issues]
400 issues = filter(self.filter_issue, issues) 406 issues = filter(self.filter_issue, issues)
401 issues = sorted(issues, key=lambda i: i['modified'], reverse=True) 407 issues = sorted(issues, key=lambda i: i['modified'], reverse=True)
402 408
403 return issues 409 return issues
404 410
405 def process_gerrit_issue(self, instance, issue): 411 def process_gerrit_issue(self, instance, issue):
406 ret = {} 412 ret = {}
407 ret['review_url'] = issue['url'] 413 ret['review_url'] = 'http://%s/%s' % (instance['url'], issue['_number'])
408 if 'shorturl' in instance: 414 if 'shorturl' in instance:
409 ret['review_url'] = 'http://%s/%s' % (instance['shorturl'], 415 ret['review_url'] = 'http://%s/%s' % (instance['shorturl'],
410 issue['number']) 416 issue['_number'])
411 ret['header'] = issue['subject'] 417 ret['header'] = issue['subject']
412 ret['owner'] = issue['owner']['email'] 418 ret['owner'] = issue['owner']['email']
413 ret['author'] = ret['owner'] 419 ret['author'] = ret['owner']
414 ret['created'] = datetime.fromtimestamp(issue['createdOn']) 420 ret['created'] = datetime_from_gerrit(issue['created'])
415 ret['modified'] = datetime.fromtimestamp(issue['lastUpdated']) 421 ret['modified'] = datetime_from_gerrit(issue['updated'])
416 if 'comments' in issue: 422 if 'messages' in issue:
417 ret['replies'] = self.process_gerrit_issue_replies(issue['comments']) 423 ret['replies'] = self.process_gerrit_issue_replies(issue['messages'])
418 else: 424 else:
419 ret['replies'] = [] 425 ret['replies'] = []
420 ret['reviewers'] = set() 426 ret['reviewers'] = set()
421 for reply in ret['replies']: 427 for reply in ret['replies']:
422 if reply['author'] != ret['author']: 428 if reply['author'] != ret['author']:
423 ret['reviewers'].add(reply['author']) 429 ret['reviewers'].add(reply['author'])
424 return ret 430 return ret
425 431
426 @staticmethod 432 @staticmethod
427 def process_gerrit_issue_replies(replies): 433 def process_gerrit_issue_replies(replies):
428 ret = [] 434 ret = []
429 replies = filter(lambda r: 'email' in r['reviewer'], replies) 435 replies = filter(lambda r: 'email' in r['author'], replies)
430 for reply in replies: 436 for reply in replies:
431 r = {} 437 r = {}
432 r['author'] = reply['reviewer']['email'] 438 r['author'] = reply['author']['email']
433 r['created'] = datetime.fromtimestamp(reply['timestamp']) 439 r['created'] = datetime_from_gerrit(reply['date'])
434 r['content'] = '' 440 r['content'] = reply['message']
435 ret.append(r) 441 ret.append(r)
436 return ret 442 return ret
437 443
438 def google_code_issue_search(self, instance): 444 def google_code_issue_search(self, instance):
439 time_format = '%Y-%m-%dT%T' 445 time_format = '%Y-%m-%dT%T'
440 # See http://code.google.com/p/support/wiki/IssueTrackerAPI 446 # See http://code.google.com/p/support/wiki/IssueTrackerAPI
441 # q=<owner>@chromium.org does a full text search for <owner>@chromium.org. 447 # q=<owner>@chromium.org does a full text search for <owner>@chromium.org.
442 # This will accept the issue if owner is the owner or in the cc list. Might 448 # This will accept the issue if owner is the owner or in the cc list. Might
443 # have some false positives, though. 449 # have some false positives, though.
444 450
(...skipping 580 matching lines...) Expand 10 before | Expand all | Expand 10 after
1025 print '\n\n\n' 1031 print '\n\n\n'
1026 1032
1027 my_activity.print_changes() 1033 my_activity.print_changes()
1028 my_activity.print_reviews() 1034 my_activity.print_reviews()
1029 my_activity.print_issues() 1035 my_activity.print_issues()
1030 return 0 1036 return 0
1031 1037
1032 1038
1033 if __name__ == '__main__': 1039 if __name__ == '__main__':
1034 sys.exit(main()) 1040 sys.exit(main())
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