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 __future__ import print_function | 10 from __future__ import print_function |
(...skipping 1375 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... | |
1386 return presubmit_support.GitChange( | 1386 return presubmit_support.GitChange( |
1387 name, | 1387 name, |
1388 description, | 1388 description, |
1389 absroot, | 1389 absroot, |
1390 files, | 1390 files, |
1391 issue, | 1391 issue, |
1392 patchset, | 1392 patchset, |
1393 author, | 1393 author, |
1394 upstream=upstream_branch) | 1394 upstream=upstream_branch) |
1395 | 1395 |
1396 def UpdateDescription(self, description): | 1396 def UpdateDescription(self, force, description): |
tandrii(chromium)
2016/09/07 13:33:39
light suggestion: i'd add force=False as second ar
dsansome
2016/09/08 07:07:17
Done.
| |
1397 self.description = description | 1397 self.description = description |
1398 return self._codereview_impl.UpdateDescriptionRemote(description) | 1398 return self._codereview_impl.UpdateDescriptionRemote(force, description) |
1399 | 1399 |
1400 def RunHook(self, committing, may_prompt, verbose, change): | 1400 def RunHook(self, committing, may_prompt, verbose, change): |
1401 """Calls sys.exit() if the hook fails; returns a HookResults otherwise.""" | 1401 """Calls sys.exit() if the hook fails; returns a HookResults otherwise.""" |
1402 try: | 1402 try: |
1403 return presubmit_support.DoPresubmitChecks(change, committing, | 1403 return presubmit_support.DoPresubmitChecks(change, committing, |
1404 verbose=verbose, output_stream=sys.stdout, input_stream=sys.stdin, | 1404 verbose=verbose, output_stream=sys.stdout, input_stream=sys.stdin, |
1405 default_presubmit=None, may_prompt=may_prompt, | 1405 default_presubmit=None, may_prompt=may_prompt, |
1406 rietveld_obj=self._codereview_impl.GetRieveldObjForPresubmit(), | 1406 rietveld_obj=self._codereview_impl.GetRieveldObjForPresubmit(), |
1407 gerrit_obj=self._codereview_impl.GetGerritObjForPresubmit()) | 1407 gerrit_obj=self._codereview_impl.GetGerritObjForPresubmit()) |
1408 except presubmit_support.PresubmitFailure as e: | 1408 except presubmit_support.PresubmitFailure as e: |
(...skipping 187 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... | |
1596 | 1596 |
1597 def GetRieveldObjForPresubmit(self): | 1597 def GetRieveldObjForPresubmit(self): |
1598 # This is an unfortunate Rietveld-embeddedness in presubmit. | 1598 # This is an unfortunate Rietveld-embeddedness in presubmit. |
1599 # For non-Rietveld codereviews, this probably should return a dummy object. | 1599 # For non-Rietveld codereviews, this probably should return a dummy object. |
1600 raise NotImplementedError() | 1600 raise NotImplementedError() |
1601 | 1601 |
1602 def GetGerritObjForPresubmit(self): | 1602 def GetGerritObjForPresubmit(self): |
1603 # None is valid return value, otherwise presubmit_support.GerritAccessor. | 1603 # None is valid return value, otherwise presubmit_support.GerritAccessor. |
1604 return None | 1604 return None |
1605 | 1605 |
1606 def UpdateDescriptionRemote(self, description): | 1606 def UpdateDescriptionRemote(self, force, description): |
1607 """Update the description on codereview site.""" | 1607 """Update the description on codereview site.""" |
1608 raise NotImplementedError() | 1608 raise NotImplementedError() |
1609 | 1609 |
1610 def CloseIssue(self): | 1610 def CloseIssue(self): |
1611 """Closes the issue.""" | 1611 """Closes the issue.""" |
1612 raise NotImplementedError() | 1612 raise NotImplementedError() |
1613 | 1613 |
1614 def GetApprovingReviewers(self): | 1614 def GetApprovingReviewers(self): |
1615 """Returns a list of reviewers approving the change. | 1615 """Returns a list of reviewers approving the change. |
1616 | 1616 |
(...skipping 174 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... | |
1791 break | 1791 break |
1792 | 1792 |
1793 if not messages: | 1793 if not messages: |
1794 # No message was sent. | 1794 # No message was sent. |
1795 return 'unsent' | 1795 return 'unsent' |
1796 if messages[-1]['sender'] != props.get('owner_email'): | 1796 if messages[-1]['sender'] != props.get('owner_email'): |
1797 # Non-LGTM reply from non-owner and not CQ bot. | 1797 # Non-LGTM reply from non-owner and not CQ bot. |
1798 return 'reply' | 1798 return 'reply' |
1799 return 'waiting' | 1799 return 'waiting' |
1800 | 1800 |
1801 def UpdateDescriptionRemote(self, description): | 1801 def UpdateDescriptionRemote(self, force, description): |
1802 return self.RpcServer().update_description( | 1802 return self.RpcServer().update_description( |
1803 self.GetIssue(), self.description) | 1803 self.GetIssue(), self.description) |
1804 | 1804 |
1805 def CloseIssue(self): | 1805 def CloseIssue(self): |
1806 return self.RpcServer().close_issue(self.GetIssue()) | 1806 return self.RpcServer().close_issue(self.GetIssue()) |
1807 | 1807 |
1808 def SetFlag(self, flag, value): | 1808 def SetFlag(self, flag, value): |
1809 return self.SetFlags({flag: value}) | 1809 return self.SetFlags({flag: value}) |
1810 | 1810 |
1811 def SetFlags(self, flags): | 1811 def SetFlags(self, flags): |
(...skipping 475 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... | |
2287 def GetMostRecentPatchset(self): | 2287 def GetMostRecentPatchset(self): |
2288 data = self._GetChangeDetail(['CURRENT_REVISION']) | 2288 data = self._GetChangeDetail(['CURRENT_REVISION']) |
2289 return data['revisions'][data['current_revision']]['_number'] | 2289 return data['revisions'][data['current_revision']]['_number'] |
2290 | 2290 |
2291 def FetchDescription(self): | 2291 def FetchDescription(self): |
2292 data = self._GetChangeDetail(['CURRENT_REVISION']) | 2292 data = self._GetChangeDetail(['CURRENT_REVISION']) |
2293 current_rev = data['current_revision'] | 2293 current_rev = data['current_revision'] |
2294 url = data['revisions'][current_rev]['fetch']['http']['url'] | 2294 url = data['revisions'][current_rev]['fetch']['http']['url'] |
2295 return gerrit_util.GetChangeDescriptionFromGitiles(url, current_rev) | 2295 return gerrit_util.GetChangeDescriptionFromGitiles(url, current_rev) |
2296 | 2296 |
2297 def UpdateDescriptionRemote(self, description): | 2297 def UpdateDescriptionRemote(self, force, description): |
2298 if gerrit_util.HasPendingChangeEdit(self._GetGerritHost(), self.GetIssue()): | |
2299 if not force: | |
2300 ask_for_data( | |
2301 'The description cannot be modified while the issue has a pending ' | |
2302 'unpublished\nedit. Either publish the edit in the Gerrit web UI ' | |
tandrii(chromium)
2016/09/07 13:33:39
nit: i prefer not avoid manual text wrapping (ie n
dsansome
2016/09/08 07:07:17
Done.
| |
2303 'or delete it.\n\n' | |
2304 'Press Enter to delete the unpublished edit, Ctrl+C to abort.') | |
2305 | |
2306 gerrit_util.DeletePendingChangeEdit(self._GetGerritHost(), | |
2307 self.GetIssue()) | |
2298 gerrit_util.SetCommitMessage(self._GetGerritHost(), self.GetIssue(), | 2308 gerrit_util.SetCommitMessage(self._GetGerritHost(), self.GetIssue(), |
2299 description) | 2309 description) |
2300 | 2310 |
2301 def CloseIssue(self): | 2311 def CloseIssue(self): |
2302 gerrit_util.AbandonChange(self._GetGerritHost(), self.GetIssue(), msg='') | 2312 gerrit_util.AbandonChange(self._GetGerritHost(), self.GetIssue(), msg='') |
2303 | 2313 |
2304 def GetApprovingReviewers(self): | 2314 def GetApprovingReviewers(self): |
2305 """Returns a list of reviewers approving the change. | 2315 """Returns a list of reviewers approving the change. |
2306 | 2316 |
2307 Note: not necessarily committers. | 2317 Note: not necessarily committers. |
(...skipping 1292 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... | |
3600 | 3610 |
3601 | 3611 |
3602 @subcommand.usage('[codereview url or issue id]') | 3612 @subcommand.usage('[codereview url or issue id]') |
3603 def CMDdescription(parser, args): | 3613 def CMDdescription(parser, args): |
3604 """Brings up the editor for the current CL's description.""" | 3614 """Brings up the editor for the current CL's description.""" |
3605 parser.add_option('-d', '--display', action='store_true', | 3615 parser.add_option('-d', '--display', action='store_true', |
3606 help='Display the description instead of opening an editor') | 3616 help='Display the description instead of opening an editor') |
3607 parser.add_option('-n', '--new-description', | 3617 parser.add_option('-n', '--new-description', |
3608 help='New description to set for this issue (- for stdin, ' | 3618 help='New description to set for this issue (- for stdin, ' |
3609 '+ to load from local commit HEAD)') | 3619 '+ to load from local commit HEAD)') |
3620 parser.add_option('-f', '--force', action='store_true', | |
3621 help='Delete any unpublished Gerrit edits for this issue ' | |
3622 'without prompting') | |
3610 | 3623 |
3611 _add_codereview_select_options(parser) | 3624 _add_codereview_select_options(parser) |
3612 auth.add_auth_options(parser) | 3625 auth.add_auth_options(parser) |
3613 options, args = parser.parse_args(args) | 3626 options, args = parser.parse_args(args) |
3614 _process_codereview_select_options(parser, options) | 3627 _process_codereview_select_options(parser, options) |
3615 | 3628 |
3616 target_issue = None | 3629 target_issue = None |
3617 if len(args) > 0: | 3630 if len(args) > 0: |
3618 target_issue = ParseIssueNumberArgument(args[0]) | 3631 target_issue = ParseIssueNumberArgument(args[0]) |
3619 if not target_issue.valid: | 3632 if not target_issue.valid: |
(...skipping 28 matching lines...) Expand all Loading... | |
3648 elif text == '+': | 3661 elif text == '+': |
3649 base_branch = cl.GetCommonAncestorWithUpstream() | 3662 base_branch = cl.GetCommonAncestorWithUpstream() |
3650 change = cl.GetChange(base_branch, None, local_description=True) | 3663 change = cl.GetChange(base_branch, None, local_description=True) |
3651 text = change.FullDescriptionText() | 3664 text = change.FullDescriptionText() |
3652 | 3665 |
3653 description.set_description(text) | 3666 description.set_description(text) |
3654 else: | 3667 else: |
3655 description.prompt() | 3668 description.prompt() |
3656 | 3669 |
3657 if cl.GetDescription() != description.description: | 3670 if cl.GetDescription() != description.description: |
3658 cl.UpdateDescription(description.description) | 3671 cl.UpdateDescription(options.force, description.description) |
3659 return 0 | 3672 return 0 |
3660 | 3673 |
3661 | 3674 |
3662 def CreateDescriptionFromLog(args): | 3675 def CreateDescriptionFromLog(args): |
3663 """Pulls out the commit log to use as a base for the CL description.""" | 3676 """Pulls out the commit log to use as a base for the CL description.""" |
3664 log_args = [] | 3677 log_args = [] |
3665 if len(args) == 1 and not args[0].endswith('.'): | 3678 if len(args) == 1 and not args[0].endswith('.'): |
3666 log_args = [args[0] + '..'] | 3679 log_args = [args[0] + '..'] |
3667 elif len(args) == 1 and args[0].endswith('...'): | 3680 elif len(args) == 1 and args[0].endswith('...'): |
3668 log_args = [args[0][:-1]] | 3681 log_args = [args[0][:-1]] |
(...skipping 562 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... | |
4231 to_pending = ' to pending queue' if pushed_to_pending else '' | 4244 to_pending = ' to pending queue' if pushed_to_pending else '' |
4232 viewvc_url = settings.GetViewVCUrl() | 4245 viewvc_url = settings.GetViewVCUrl() |
4233 if not to_pending: | 4246 if not to_pending: |
4234 if viewvc_url and revision: | 4247 if viewvc_url and revision: |
4235 change_desc.append_footer( | 4248 change_desc.append_footer( |
4236 'Committed: %s%s' % (viewvc_url, revision)) | 4249 'Committed: %s%s' % (viewvc_url, revision)) |
4237 elif revision: | 4250 elif revision: |
4238 change_desc.append_footer('Committed: %s' % (revision,)) | 4251 change_desc.append_footer('Committed: %s' % (revision,)) |
4239 print('Closing issue ' | 4252 print('Closing issue ' |
4240 '(you may be prompted for your codereview password)...') | 4253 '(you may be prompted for your codereview password)...') |
4241 cl.UpdateDescription(change_desc.description) | 4254 cl.UpdateDescription(False, change_desc.description) |
4242 cl.CloseIssue() | 4255 cl.CloseIssue() |
4243 props = cl.GetIssueProperties() | 4256 props = cl.GetIssueProperties() |
4244 patch_num = len(props['patchsets']) | 4257 patch_num = len(props['patchsets']) |
4245 comment = "Committed patchset #%d (id:%d)%s manually as %s" % ( | 4258 comment = "Committed patchset #%d (id:%d)%s manually as %s" % ( |
4246 patch_num, props['patchsets'][-1], to_pending, revision) | 4259 patch_num, props['patchsets'][-1], to_pending, revision) |
4247 if options.bypass_hooks: | 4260 if options.bypass_hooks: |
4248 comment += ' (tree was closed).' if GetTreeStatus() == 'closed' else '.' | 4261 comment += ' (tree was closed).' if GetTreeStatus() == 'closed' else '.' |
4249 else: | 4262 else: |
4250 comment += ' (presubmit successful).' | 4263 comment += ' (presubmit successful).' |
4251 cl.RpcServer().add_comment(cl.GetIssue(), comment) | 4264 cl.RpcServer().add_comment(cl.GetIssue(), comment) |
(...skipping 1008 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... | |
5260 if __name__ == '__main__': | 5273 if __name__ == '__main__': |
5261 # These affect sys.stdout so do it outside of main() to simplify mocks in | 5274 # These affect sys.stdout so do it outside of main() to simplify mocks in |
5262 # unit testing. | 5275 # unit testing. |
5263 fix_encoding.fix_encoding() | 5276 fix_encoding.fix_encoding() |
5264 setup_color.init() | 5277 setup_color.init() |
5265 try: | 5278 try: |
5266 sys.exit(main(sys.argv[1:])) | 5279 sys.exit(main(sys.argv[1:])) |
5267 except KeyboardInterrupt: | 5280 except KeyboardInterrupt: |
5268 sys.stderr.write('interrupted\n') | 5281 sys.stderr.write('interrupted\n') |
5269 sys.exit(1) | 5282 sys.exit(1) |
OLD | NEW |