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

Side by Side Diff: git_cl.py

Issue 2318903002: Prompt to delete pending edits before changing the Gerrit CL description. (Closed)
Patch Set: Created 4 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 | « gerrit_util.py ('k') | tests/git_cl_test.py » ('j') | 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 # 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
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
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
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
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
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
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
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
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)
OLDNEW
« no previous file with comments | « gerrit_util.py ('k') | tests/git_cl_test.py » ('j') | no next file with comments »

Powered by Google App Engine
This is Rietveld 408576698