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 # 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 and Gerrit.""" | 8 """A git-command for integrating reviews on Rietveld and Gerrit.""" |
9 | 9 |
10 from distutils.version import LooseVersion | 10 from distutils.version import LooseVersion |
(...skipping 1564 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... | |
1575 print( | 1575 print( |
1576 'You aren\'t using Rietveld at the moment, but Gerrit.\n' | 1576 'You aren\'t using Rietveld at the moment, but Gerrit.\n' |
1577 'Using Rietveld in your PRESUBMIT scripts won\'t work.\n' | 1577 'Using Rietveld in your PRESUBMIT scripts won\'t work.\n' |
1578 'Please, either change your PRESUBIT to not use rietveld_obj.%s,\n' | 1578 'Please, either change your PRESUBIT to not use rietveld_obj.%s,\n' |
1579 'or use Rietveld for codereview.\n' | 1579 'or use Rietveld for codereview.\n' |
1580 'See also http://crbug.com/579160.' % attr) | 1580 'See also http://crbug.com/579160.' % attr) |
1581 raise NotImplementedError() | 1581 raise NotImplementedError() |
1582 return ThisIsNotRietveldIssue() | 1582 return ThisIsNotRietveldIssue() |
1583 | 1583 |
1584 def GetStatus(self): | 1584 def GetStatus(self): |
1585 # TODO(tandrii) | 1585 """Apply a rough heuristic to give a simple summary of an issue's review |
1586 raise NotImplementedError() | 1586 or CQ status, assuming adherence to a common workflow. |
1587 | |
1588 Returns None if no issue for this branch, or one of the following keywords: | |
1589 * 'error' - error from review tool (including deleted issues) | |
1590 * 'unsent' - no reviewers added | |
1591 * 'waiting' - waiting for review | |
1592 * 'reply' - waiting for owner to reply to review | |
1593 * 'lgtm' - Code-Review +2 from at least one approved reviewer | |
1594 * 'commit' - in the commit queue | |
1595 * 'closed' - abandoned | |
1596 """ | |
1597 if not self.GetIssue(): | |
1598 return None | |
1599 | |
1600 try: | |
1601 data = self._GetChangeDetail(['DETAILED_LABELS', 'CURRENT_REVISION']) | |
1602 except httplib.HTTPException: | |
1603 return 'error' | |
1604 | |
1605 if data['status'] == 'ABANDONED': | |
1606 return 'abandoned' | |
ukai
2016/03/28 02:19:51
'closed' ?
(or add 'abondoned' in docstring above?
tandrii(chromium)
2016/03/28 18:57:21
Done.
| |
1607 | |
1608 cq_label = data['labels'].get('Commit-Queue', {}) | |
1609 if cq_label: | |
1610 # Vote value is a stringified integer, which we expect from 0 to 2. | |
1611 vote_value = cq_label.get('value', '0') | |
1612 vote_text = cq_label.get('values', {}).get(vote_value, '') | |
1613 if vote_text.lower() == 'Commit'.lower(): | |
ukai
2016/03/28 02:19:51
if vote.text.lower() == 'commit':
?
tandrii(chromium)
2016/03/28 18:57:21
Done.
| |
1614 return 'commit' | |
1615 | |
1616 lgtm_label = data['labels'].get('Code-Review', {}) | |
1617 if lgtm_label: | |
1618 if 'rejected' in lgtm_label: | |
1619 return 'not lgtm' | |
ukai
2016/03/28 02:19:51
'waiting' ?
(or add 'not lgtm' in docstring above?
tandrii(chromium)
2016/03/28 18:57:21
Done.
| |
1620 if 'approved' in lgtm_label: | |
1621 return 'lgtm' | |
1622 | |
1623 if not data.get('reviewers', {}).get('REVIEWER', []): | |
1624 return 'unsent' | |
1625 | |
1626 messages = data.get('messages', []) | |
1627 if messages: | |
1628 owner = data['owner'].get('_account_id') | |
1629 last_message_author = messages[-1].get('author', {}).get('_account_id') | |
1630 if owner != last_message_author: | |
1631 # Some reply from non-owner. | |
1632 return 'reply' | |
1633 | |
1634 return 'waiting' | |
1587 | 1635 |
1588 def GetMostRecentPatchset(self): | 1636 def GetMostRecentPatchset(self): |
1589 data = gerrit_util.GetChangeDetail(self._GetGerritHost(), self.GetIssue(), | 1637 data = self._GetChangeDetail(['CURRENT_REVISION']) |
1590 ['CURRENT_REVISION']) | |
1591 return data['revisions'][data['current_revision']]['_number'] | 1638 return data['revisions'][data['current_revision']]['_number'] |
1592 | 1639 |
1593 def FetchDescription(self): | 1640 def FetchDescription(self): |
1594 data = gerrit_util.GetChangeDetail(self._GetGerritHost(), self.GetIssue(), | 1641 data = self._GetChangeDetail(['COMMIT_FOOTERS', 'CURRENT_REVISION']) |
1595 ['COMMIT_FOOTERS', 'CURRENT_REVISION']) | |
1596 return data['revisions'][data['current_revision']]['commit_with_footers'] | 1642 return data['revisions'][data['current_revision']]['commit_with_footers'] |
1597 | 1643 |
1598 def UpdateDescriptionRemote(self, description): | 1644 def UpdateDescriptionRemote(self, description): |
1599 # TODO(tandrii) | 1645 # TODO(tandrii) |
1600 raise NotImplementedError() | 1646 raise NotImplementedError() |
1601 | 1647 |
1602 def CloseIssue(self): | 1648 def CloseIssue(self): |
1603 gerrit_util.AbandonChange(self._GetGerritHost(), self.GetIssue(), msg='') | 1649 gerrit_util.AbandonChange(self._GetGerritHost(), self.GetIssue(), msg='') |
1604 | 1650 |
1605 | 1651 |
1652 def _GetChangeDetail(self, options): | |
1653 return gerrit_util.GetChangeDetail(self._GetGerritHost(), self.GetIssue(), | |
1654 options) | |
1655 | |
1656 | |
1606 class ChangeDescription(object): | 1657 class ChangeDescription(object): |
1607 """Contains a parsed form of the change description.""" | 1658 """Contains a parsed form of the change description.""" |
1608 R_LINE = r'^[ \t]*(TBR|R)[ \t]*=[ \t]*(.*?)[ \t]*$' | 1659 R_LINE = r'^[ \t]*(TBR|R)[ \t]*=[ \t]*(.*?)[ \t]*$' |
1609 BUG_LINE = r'^[ \t]*(BUG)[ \t]*=[ \t]*(.*?)[ \t]*$' | 1660 BUG_LINE = r'^[ \t]*(BUG)[ \t]*=[ \t]*(.*?)[ \t]*$' |
1610 | 1661 |
1611 def __init__(self, description): | 1662 def __init__(self, description): |
1612 self._description_lines = (description or '').strip().splitlines() | 1663 self._description_lines = (description or '').strip().splitlines() |
1613 | 1664 |
1614 @property # www.logilab.org/ticket/89786 | 1665 @property # www.logilab.org/ticket/89786 |
1615 def description(self): # pylint: disable=E0202 | 1666 def description(self): # pylint: disable=E0202 |
(...skipping 2653 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... | |
4269 if __name__ == '__main__': | 4320 if __name__ == '__main__': |
4270 # These affect sys.stdout so do it outside of main() to simplify mocks in | 4321 # These affect sys.stdout so do it outside of main() to simplify mocks in |
4271 # unit testing. | 4322 # unit testing. |
4272 fix_encoding.fix_encoding() | 4323 fix_encoding.fix_encoding() |
4273 colorama.init() | 4324 colorama.init() |
4274 try: | 4325 try: |
4275 sys.exit(main(sys.argv[1:])) | 4326 sys.exit(main(sys.argv[1:])) |
4276 except KeyboardInterrupt: | 4327 except KeyboardInterrupt: |
4277 sys.stderr.write('interrupted\n') | 4328 sys.stderr.write('interrupted\n') |
4278 sys.exit(1) | 4329 sys.exit(1) |
OLD | NEW |