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

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: Added note about https for crosreview.com 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 },
122 # TODO: chrome-internal-review requires login credentials. Enable once login
123 # support is added to this client. See crbug.com/281695.
124 #{
125 # 'url': 'chrome-internal-review.googlesource.com',
126 # 'shorturl': 'crosreview.com/i',
127 #},
123 { 128 {
124 'url': 'gerrit-int.chromium.org', 129 'host': 'gerrit.chromium.org',
130 'port': 29418,
131 },
132 {
133 'host': 'gerrit-int.chromium.org',
125 'port': 29419, 134 'port': 29419,
126 'shorturl': 'crosreview.com/i',
127 }, 135 },
128 ] 136 ]
129 137
130 google_code_projects = [ 138 google_code_projects = [
131 { 139 {
132 'name': 'chromium', 140 'name': 'chromium',
133 'shorturl': 'crbug.com', 141 'shorturl': 'crbug.com',
134 }, 142 },
135 { 143 {
136 'name': 'chromium-os', 144 'name': 'chromium-os',
(...skipping 95 matching lines...) Expand 10 before | Expand all | Expand 10 after
232 240
233 def get_yes_or_no(msg): 241 def get_yes_or_no(msg):
234 while True: 242 while True:
235 response = raw_input(msg + ' yes/no [no] ') 243 response = raw_input(msg + ' yes/no [no] ')
236 if response == 'y' or response == 'yes': 244 if response == 'y' or response == 'yes':
237 return True 245 return True
238 elif not response or response == 'n' or response == 'no': 246 elif not response or response == 'n' or response == 'no':
239 return False 247 return False
240 248
241 249
250 def datetime_from_gerrit(date_string):
251 return datetime.strptime(date_string, '%Y-%m-%d %H:%M:%S.%f000')
252
253
242 def datetime_from_rietveld(date_string): 254 def datetime_from_rietveld(date_string):
243 return datetime.strptime(date_string, '%Y-%m-%d %H:%M:%S.%f') 255 return datetime.strptime(date_string, '%Y-%m-%d %H:%M:%S.%f')
244 256
245 257
246 def datetime_from_google_code(date_string): 258 def datetime_from_google_code(date_string):
247 return datetime.strptime(date_string, '%Y-%m-%dT%H:%M:%S.%fZ') 259 return datetime.strptime(date_string, '%Y-%m-%dT%H:%M:%S.%fZ')
248 260
249 261
250 class MyActivity(object): 262 class MyActivity(object):
251 def __init__(self, options): 263 def __init__(self, options):
(...skipping 117 matching lines...) Expand 10 before | Expand all | Expand 10 after
369 def process_rietveld_replies(replies): 381 def process_rietveld_replies(replies):
370 ret = [] 382 ret = []
371 for reply in replies: 383 for reply in replies:
372 r = {} 384 r = {}
373 r['author'] = reply['sender'] 385 r['author'] = reply['sender']
374 r['created'] = datetime_from_rietveld(reply['date']) 386 r['created'] = datetime_from_rietveld(reply['date'])
375 r['content'] = '' 387 r['content'] = ''
376 ret.append(r) 388 ret.append(r)
377 return ret 389 return ret
378 390
391 @staticmethod
392 def gerrit_changes_over_ssh(instance, filters):
393 # See https://review.openstack.org/Documentation/cmd-query.html
394 # Gerrit doesn't allow filtering by created time, only modified time.
395 gquery_cmd = ['ssh', '-p', str(instance['port']), instance['host'],
396 'gerrit', 'query',
397 '--format', 'JSON',
398 '--comments',
399 '--'] + filters
400 [stdout, _] = subprocess.Popen(gquery_cmd, stdout=subprocess.PIPE,
M-A Ruel 2013/09/11 13:47:26 (stdout, _)
deymo 2013/09/11 21:14:35 Done.
401 stderr=subprocess.PIPE).communicate()
M-A Ruel 2013/09/11 13:47:26 Failure will be hard to handle since you discard b
deymo 2013/09/11 21:14:35 This part of the code is using the old gerrit inte
402 issues = str(stdout).split('\n')[:-2]
M-A Ruel 2013/09/11 13:47:26 splitlines() stdout is already a str(), not need
deymo 2013/09/11 21:14:35 Done.
403 return map(json.loads, issues)
404
405 @staticmethod
406 def gerrit_changes_over_rest(instance, filters):
407 # See https://gerrit-review.googlesource.com/Documentation/rest-api.html
408 # Gerrit doesn't allow filtering by created time, only modified time.
409 args = urllib.urlencode([
410 ('q', ' '.join(filters)),
411 ('o', 'MESSAGES'),
412 ('o', 'LABELS')])
413 rest_url = 'https://%s/changes/?%s' % (instance['url'], args)
414
415 req = urllib2.Request(rest_url, headers={'Accept': 'text/plain'})
416 try:
417 response = urllib2.urlopen(req)
418 stdout = response.read()
419 except urllib2.HTTPError, e:
420 print 'ERROR: Looking up %r: %s' % (rest_url, e)
421 return []
422
423 # Check that the returned JSON starts with the right marker.
M-A Ruel 2013/09/11 13:47:26 s/starts/ends/
deymo 2013/09/11 21:14:35 hmmm... I think that "starts" is the right option
424 if stdout[:5] != ")]}'\n":
425 print 'ERROR: Marker not found on REST API response: %r' % stdout[:5]
426 return []
427 return json.loads(stdout[5:])
M-A Ruel 2013/09/11 13:47:26 What are you skipping exactly?
deymo 2013/09/11 21:14:35 I'm skipping the string ")]}'\n" from the beginnin
428
379 def gerrit_search(self, instance, owner=None, reviewer=None): 429 def gerrit_search(self, instance, owner=None, reviewer=None):
380 max_age = datetime.today() - self.modified_after 430 max_age = datetime.today() - self.modified_after
381 max_age = max_age.days * 24 * 3600 + max_age.seconds 431 max_age = max_age.days * 24 * 3600 + max_age.seconds
432 user_filter = 'owner:%s' % owner if owner else 'reviewer:%s' % reviewer
433 filters = ['-age:%ss' % max_age, user_filter]
382 434
383 # See https://review.openstack.org/Documentation/cmd-query.html 435 # Determine the gerrit interface to use: SSH or REST API:
384 # Gerrit doesn't allow filtering by created time, only modified time. 436 if 'host' in instance:
385 user_filter = 'owner:%s' % owner if owner else 'reviewer:%s' % reviewer 437 issues = self.gerrit_changes_over_ssh(instance, filters)
386 gquery_cmd = ['ssh', '-p', str(instance['port']), instance['url'], 438 issues = [self.process_gerrit_ssh_issue(instance, issue)
387 'gerrit', 'query', 439 for issue in issues]
388 '--format', 'JSON', 440 elif 'url' in instance:
389 '--comments', 441 issues = self.gerrit_changes_over_rest(instance, filters)
390 '--', 442 issues = [self.process_gerrit_rest_issue(instance, issue)
391 '-age:%ss' % str(max_age), 443 for issue in issues]
392 user_filter] 444 else:
393 [stdout, _] = subprocess.Popen(gquery_cmd, stdout=subprocess.PIPE, 445 raise Exception("Invalid gerrit_instances configuration.")
M-A Ruel 2013/09/11 13:47:26 resto f the file use single quotes, stay consisten
deymo 2013/09/11 21:14:35 Done.
394 stderr=subprocess.PIPE).communicate()
395 issues = str(stdout).split('\n')[:-2]
396 issues = map(json.loads, issues)
397 446
398 # TODO(cjhopman): should we filter abandoned changes? 447 # TODO(cjhopman): should we filter abandoned changes?
399 issues = [self.process_gerrit_issue(instance, issue) for issue in issues]
400 issues = filter(self.filter_issue, issues) 448 issues = filter(self.filter_issue, issues)
401 issues = sorted(issues, key=lambda i: i['modified'], reverse=True) 449 issues = sorted(issues, key=lambda i: i['modified'], reverse=True)
402 450
403 return issues 451 return issues
404 452
405 def process_gerrit_issue(self, instance, issue): 453 def process_gerrit_ssh_issue(self, instance, issue):
406 ret = {} 454 ret = {}
407 ret['review_url'] = issue['url'] 455 ret['review_url'] = issue['url']
408 if 'shorturl' in instance: 456 if 'shorturl' in instance:
409 ret['review_url'] = 'http://%s/%s' % (instance['shorturl'], 457 ret['review_url'] = 'http://%s/%s' % (instance['shorturl'],
410 issue['number']) 458 issue['number'])
411 ret['header'] = issue['subject'] 459 ret['header'] = issue['subject']
412 ret['owner'] = issue['owner']['email'] 460 ret['owner'] = issue['owner']['email']
413 ret['author'] = ret['owner'] 461 ret['author'] = ret['owner']
414 ret['created'] = datetime.fromtimestamp(issue['createdOn']) 462 ret['created'] = datetime.fromtimestamp(issue['createdOn'])
415 ret['modified'] = datetime.fromtimestamp(issue['lastUpdated']) 463 ret['modified'] = datetime.fromtimestamp(issue['lastUpdated'])
416 if 'comments' in issue: 464 if 'comments' in issue:
417 ret['replies'] = self.process_gerrit_issue_replies(issue['comments']) 465 ret['replies'] = self.process_gerrit_ssh_issue_replies(issue['comments'])
418 else: 466 else:
419 ret['replies'] = [] 467 ret['replies'] = []
420 ret['reviewers'] = set() 468 ret['reviewers'] = set()
421 for reply in ret['replies']: 469 for reply in ret['replies']:
422 if reply['author'] != ret['author']: 470 if reply['author'] != ret['author']:
423 ret['reviewers'].add(reply['author']) 471 ret['reviewers'].add(reply['author'])
424 return ret 472 return ret
425 473
426 @staticmethod 474 @staticmethod
427 def process_gerrit_issue_replies(replies): 475 def process_gerrit_ssh_issue_replies(replies):
428 ret = [] 476 ret = []
429 replies = filter(lambda r: 'email' in r['reviewer'], replies) 477 replies = filter(lambda r: 'email' in r['reviewer'], replies)
430 for reply in replies: 478 for reply in replies:
431 r = {} 479 r = {}
432 r['author'] = reply['reviewer']['email'] 480 r['author'] = reply['reviewer']['email']
433 r['created'] = datetime.fromtimestamp(reply['timestamp']) 481 r['created'] = datetime.fromtimestamp(reply['timestamp'])
434 r['content'] = '' 482 r['content'] = ''
435 ret.append(r) 483 ret.append(r)
436 return ret 484 return ret
437 485
486 def process_gerrit_rest_issue(self, instance, issue):
487 ret = {}
488 ret['review_url'] = 'https://%s/%s' % (instance['url'], issue['_number'])
489 if 'shorturl' in instance:
490 # TODO: Move this short link to https once crosreview.com supports it.
M-A Ruel 2013/09/11 13:47:26 TODO(deymo)
deymo 2013/09/11 21:14:35 Done.
491 ret['review_url'] = 'http://%s/%s' % (instance['shorturl'],
492 issue['_number'])
493 ret['header'] = issue['subject']
494 ret['owner'] = issue['owner']['email']
495 ret['author'] = ret['owner']
496 ret['created'] = datetime_from_gerrit(issue['created'])
497 ret['modified'] = datetime_from_gerrit(issue['updated'])
498 if 'messages' in issue:
499 ret['replies'] = self.process_gerrit_rest_issue_replies(issue['messages'])
500 else:
501 ret['replies'] = []
502 ret['reviewers'] = set()
M-A Ruel 2013/09/11 13:47:26 (optional) Could be done as a one liner: ret['rev
deymo 2013/09/11 21:14:35 Changed in both functions. On 2013/09/11 13:47:26
503 for reply in ret['replies']:
504 if reply['author'] != ret['author']:
505 ret['reviewers'].add(reply['author'])
506 return ret
507
508 @staticmethod
509 def process_gerrit_rest_issue_replies(replies):
510 ret = []
511 replies = filter(lambda r: 'email' in r['author'], replies)
512 for reply in replies:
513 r = {}
M-A Ruel 2013/09/11 13:47:26 ret.append( { 'author': reply['author'][
deymo 2013/09/11 21:14:35 This code is essentially the same as the previousl
514 r['author'] = reply['author']['email']
515 r['created'] = datetime_from_gerrit(reply['date'])
516 r['content'] = reply['message']
517 ret.append(r)
518 return ret
519
438 def google_code_issue_search(self, instance): 520 def google_code_issue_search(self, instance):
439 time_format = '%Y-%m-%dT%T' 521 time_format = '%Y-%m-%dT%T'
440 # See http://code.google.com/p/support/wiki/IssueTrackerAPI 522 # See http://code.google.com/p/support/wiki/IssueTrackerAPI
441 # q=<owner>@chromium.org does a full text search for <owner>@chromium.org. 523 # 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 524 # This will accept the issue if owner is the owner or in the cc list. Might
443 # have some false positives, though. 525 # have some false positives, though.
444 526
445 # Don't filter normally on modified_before because it can filter out things 527 # Don't filter normally on modified_before because it can filter out things
446 # that were modified in the time period and then modified again after it. 528 # that were modified in the time period and then modified again after it.
447 gcode_url = ('https://code.google.com/feeds/issues/p/%s/issues/full' % 529 gcode_url = ('https://code.google.com/feeds/issues/p/%s/issues/full' %
(...skipping 577 matching lines...) Expand 10 before | Expand all | Expand 10 after
1025 print '\n\n\n' 1107 print '\n\n\n'
1026 1108
1027 my_activity.print_changes() 1109 my_activity.print_changes()
1028 my_activity.print_reviews() 1110 my_activity.print_reviews()
1029 my_activity.print_issues() 1111 my_activity.print_issues()
1030 return 0 1112 return 0
1031 1113
1032 1114
1033 if __name__ == '__main__': 1115 if __name__ == '__main__':
1034 sys.exit(main()) 1116 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