OLD | NEW |
---|---|
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 Loading... | |
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 Loading... | |
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 Loading... | |
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 Loading... | |
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()) |
OLD | NEW |