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

Side by Side Diff: git_cl.py

Issue 23072039: Fix R= line rewriter to handle TBRs well. (Closed) Base URL: https://chromium.googlesource.com/chromium/tools/depot_tools.git@master
Patch Set: More comments. 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 | tests/git_cl_test.py » ('j') | tests/git_cl_test.py » ('J')
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.""" 8 """A git-command for integrating reviews on Rietveld."""
9 9
10 from distutils.version import LooseVersion 10 from distutils.version import LooseVersion
(...skipping 831 matching lines...) Expand 10 before | Expand all | Expand 10 after
842 'tree-status-url', False) 842 'tree-status-url', False)
843 SetProperty(settings.GetViewVCUrl(), 'ViewVC URL', 'viewvc-url', True) 843 SetProperty(settings.GetViewVCUrl(), 'ViewVC URL', 'viewvc-url', True)
844 844
845 # TODO: configure a default branch to diff against, rather than this 845 # TODO: configure a default branch to diff against, rather than this
846 # svn-based hackery. 846 # svn-based hackery.
847 847
848 848
849 class ChangeDescription(object): 849 class ChangeDescription(object):
850 """Contains a parsed form of the change description.""" 850 """Contains a parsed form of the change description."""
851 R_LINE = r'^[ \t]*(TBR|R)[ \t]*=[ \t]*(.*?)[ \t]*$' 851 R_LINE = r'^[ \t]*(TBR|R)[ \t]*=[ \t]*(.*?)[ \t]*$'
852 BUG_LINE = r'^[ \t]*(BUG)[ \t]*=[ \t]*(.*?)[ \t]*$'
852 853
853 def __init__(self, description): 854 def __init__(self, description):
854 self._description = (description or '').strip() 855 self._description_lines = (description or '').strip().splitlines()
855 856
856 @property 857 @property # www.logilab.org/ticket/89786
857 def description(self): 858 def description(self): # pylint: disable=E0202
858 return self._description 859 return '\n'.join(self._description_lines)
860
861 @description.setter
862 def description(self, desc): # pylint: disable=E0202
M-A Ruel 2013/08/30 15:52:36 I must say I'm not a fan of keeping a setter here,
agable 2013/09/11 21:08:05 Yeah, you're right that the setter is confusing. I
863 if isinstance(desc, basestring):
864 lines = desc.strip().splitlines()
M-A Ruel 2013/08/30 15:52:36 .strip() is not necessary because of lines 867-870
agable 2013/09/11 21:08:05 Done.
865 else:
866 lines = [line.strip() for line in desc]
M-A Ruel 2013/08/30 15:52:36 That's wrong, you are left-aligning everything.
agable 2013/09/11 21:08:05 Done.
867 while lines and not lines[0]:
868 lines = lines[1:]
M-A Ruel 2013/08/30 15:52:36 lines.pop(0) will just the same faster.
agable 2013/09/11 21:08:05 Done.
869 while lines and not lines[-1]:
870 lines = lines.pop()
M-A Ruel 2013/08/30 15:52:36 That's wrong since you are keeping the last line;
agable 2013/09/11 21:08:05 Done.
871 self._description_lines = lines
859 872
860 def update_reviewers(self, reviewers): 873 def update_reviewers(self, reviewers):
861 """Rewrites the R=/TBR= line(s) as a single line.""" 874 """Rewrites the R=/TBR= line(s) as a single line each."""
862 assert isinstance(reviewers, list), reviewers 875 assert isinstance(reviewers, list), reviewers
863 if not reviewers: 876 if not reviewers:
864 return 877 return
865 regexp = re.compile(self.R_LINE, re.MULTILINE)
866 matches = list(regexp.finditer(self._description))
867 is_tbr = any(m.group(1) == 'TBR' for m in matches)
868 if len(matches) > 1:
869 # Erase all except the first one.
870 for i in xrange(len(matches) - 1, 0, -1):
871 self._description = (
872 self._description[:matches[i].start()] +
873 self._description[matches[i].end():])
874 878
875 if is_tbr: 879 # Get the set of R= and TBR= lines and remove them from the desciption.
876 new_r_line = 'TBR=' + ', '.join(reviewers) 880 regexp = re.compile(self.R_LINE)
881 matches = [regexp.match(line) for line in self._description_lines]
882 new_desc = [self._description_lines[i] for i in range(len(matches))
883 if not matches[i]]
884 self.description = new_desc
885
886 # Construct new unified R= and TBR= lines.
887 r_names = []
888 tbr_names = []
889 for match in matches:
890 if match is None:
891 continue
892 is_tbr = match.group(1) == 'TBR'
893 people = cleanup_list([match.group(2).strip()])
894 if is_tbr:
895 tbr_names.extend(people)
896 else:
897 r_names.extend(people)
898 r_final_names = reviewers[:]
899 for name in r_names:
900 if name not in r_final_names:
901 r_final_names.append(name)
902 new_r_line = 'R=' + ', '.join(r_final_names) if r_final_names else None
903 new_tbr_line = 'TBR=' + ', '.join(tbr_names) if tbr_names else None
904
905 # Put the new lines in the desciption where the old first R= line was.
906 line_loc = next((i for i, match in enumerate(matches) if match), -1)
907 if line_loc != -1 and line_loc < len(self._description_lines):
908 if new_tbr_line:
909 self._description_lines.insert(line_loc, new_tbr_line)
910 if new_r_line:
911 self._description_lines.insert(line_loc, new_r_line)
877 else: 912 else:
878 new_r_line = 'R=' + ', '.join(reviewers) 913 if new_r_line:
879 914 self.append_footer(new_r_line)
880 if matches: 915 if new_tbr_line:
881 self._description = ( 916 self.append_footer(new_tbr_line)
882 self._description[:matches[0].start()] + new_r_line +
883 self._description[matches[0].end():]).strip()
884 else:
885 self.append_footer(new_r_line)
886 917
887 def prompt(self): 918 def prompt(self):
888 """Asks the user to update the description.""" 919 """Asks the user to update the description."""
889 self._description = ( 920 self.description = [
890 '# Enter a description of the change.\n' 921 '# Enter a description of the change.',
891 '# This will be displayed on the codereview site.\n' 922 '# This will be displayed on the codereview site.',
892 '# The first line will also be used as the subject of the review.\n' 923 '# The first line will also be used as the subject of the review.',
893 '#--------------------This line is 72 characters long' 924 '#--------------------This line is 72 characters long'
894 '--------------------\n' 925 '--------------------',
895 ) + self._description 926 ] + self._description_lines
896 927
897 if '\nBUG=' not in self._description: 928 regexp = re.compile(self.BUG_LINE)
929 if not any((regexp.match(line) for line in self._description_lines)):
898 self.append_footer('BUG=') 930 self.append_footer('BUG=')
899 content = gclient_utils.RunEditor(self._description, True, 931 content = gclient_utils.RunEditor(self.description, True,
900 git_editor=settings.GetGitEditor()) 932 git_editor=settings.GetGitEditor())
901 if not content: 933 if not content:
902 DieWithError('Running editor failed') 934 DieWithError('Running editor failed')
935 lines = content.splitlines()
903 936
904 # Strip off comments. 937 # Strip off comments.
905 content = re.compile(r'^#.*$', re.MULTILINE).sub('', content).strip() 938 clean_lines = [line.strip() for line in lines
906 if not content: 939 if not re.match(r'^#.*$', line)]
940 if not clean_lines:
907 DieWithError('No CL description, aborting') 941 DieWithError('No CL description, aborting')
908 self._description = content 942 self.description = clean_lines
909 943
910 def append_footer(self, line): 944 def append_footer(self, line):
911 # Adds a LF if the last line did not have 'FOO=BAR' or if the new one isn't. 945 if self._description_lines:
912 if self._description: 946 # Add an empty line if either the last line or the new line isn't a tag.
913 if '\n' not in self._description: 947 last_line = self._description_lines[-1]
914 self._description += '\n' 948 if (not presubmit_support.Change.TAG_LINE_RE.match(last_line) or
915 else: 949 not presubmit_support.Change.TAG_LINE_RE.match(line)):
916 last_line = self._description.rsplit('\n', 1)[1] 950 self._description_lines.append('')
917 if (not presubmit_support.Change.TAG_LINE_RE.match(last_line) or 951 self._description_lines.append(line)
918 not presubmit_support.Change.TAG_LINE_RE.match(line)):
919 self._description += '\n'
920 self._description += '\n' + line
921 952
922 def get_reviewers(self): 953 def get_reviewers(self):
923 """Retrieves the list of reviewers.""" 954 """Retrieves the list of reviewers."""
924 regexp = re.compile(self.R_LINE, re.MULTILINE) 955 matches = [re.match(self.R_LINE, line) for line in self._description_lines]
925 reviewers = [i.group(2).strip() for i in regexp.finditer(self._description)] 956 reviewers = [match.group(2).strip() for match in matches if match]
926 return cleanup_list(reviewers) 957 return cleanup_list(reviewers)
927 958
928 959
929 def get_approving_reviewers(props): 960 def get_approving_reviewers(props):
930 """Retrieves the reviewers that approved a CL from the issue properties with 961 """Retrieves the reviewers that approved a CL from the issue properties with
931 messages. 962 messages.
932 963
933 Note that the list may contain reviewers that are not committer, thus are not 964 Note that the list may contain reviewers that are not committer, thus are not
934 considered by the CQ. 965 considered by the CQ.
935 """ 966 """
(...skipping 1261 matching lines...) Expand 10 before | Expand all | Expand 10 after
2197 ('AppEngine is misbehaving and returned HTTP %d, again. Keep faith ' 2228 ('AppEngine is misbehaving and returned HTTP %d, again. Keep faith '
2198 'and retry or visit go/isgaeup.\n%s') % (e.code, str(e))) 2229 'and retry or visit go/isgaeup.\n%s') % (e.code, str(e)))
2199 2230
2200 2231
2201 if __name__ == '__main__': 2232 if __name__ == '__main__':
2202 # These affect sys.stdout so do it outside of main() to simplify mocks in 2233 # These affect sys.stdout so do it outside of main() to simplify mocks in
2203 # unit testing. 2234 # unit testing.
2204 fix_encoding.fix_encoding() 2235 fix_encoding.fix_encoding()
2205 colorama.init() 2236 colorama.init()
2206 sys.exit(main(sys.argv[1:])) 2237 sys.exit(main(sys.argv[1:]))
OLDNEW
« no previous file with comments | « no previous file | tests/git_cl_test.py » ('j') | tests/git_cl_test.py » ('J')

Powered by Google App Engine
This is Rietveld 408576698