Chromium Code Reviews| OLD | NEW |
|---|---|
| 1 #!/usr/bin/python | 1 #!/usr/bin/python |
| 2 # Copyright (c) 2011 The Chromium Authors. All rights reserved. | 2 # Copyright (c) 2011 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 """Given a filename as an argument, sort the #include/#imports in that file. | 6 """Given a filename as an argument, sort the #include/#imports in that file. |
| 7 | 7 |
| 8 Shows a diff and prompts for confirmation before doing the deed. | 8 Shows a diff and prompts for confirmation before doing the deed. |
| 9 """ | 9 """ |
| 10 | 10 |
| 11 import optparse | 11 import optparse |
| 12 import os | 12 import os |
| 13 import subprocess | |
| 13 import sys | 14 import sys |
| 14 import termios | 15 import termios |
| 15 import tty | 16 import tty |
| 16 | 17 |
| 18 # List of extensions whose headers this script knows how to sort. | |
| 19 _SOURCE_EXTENSIONS = ('h', 'hh', 'hpp', 'c', 'cc', 'cpp', 'cxx', 'mm',) | |
| 20 | |
| 17 def YesNo(prompt): | 21 def YesNo(prompt): |
| 18 """Prompts with a yes/no question, returns True if yes.""" | 22 """Prompts with a yes/no question, returns True if yes.""" |
| 19 print prompt, | 23 print prompt, |
| 20 sys.stdout.flush() | 24 sys.stdout.flush() |
| 21 # http://code.activestate.com/recipes/134892/ | 25 # http://code.activestate.com/recipes/134892/ |
| 22 fd = sys.stdin.fileno() | 26 fd = sys.stdin.fileno() |
| 23 old_settings = termios.tcgetattr(fd) | 27 old_settings = termios.tcgetattr(fd) |
| 24 ch = 'n' | 28 ch = 'n' |
| 25 try: | 29 try: |
| 26 tty.setraw(sys.stdin.fileno()) | 30 tty.setraw(sys.stdin.fileno()) |
| (...skipping 37 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... | |
| 64 while IsInclude(line): | 68 while IsInclude(line): |
| 65 headerblock.append(line) | 69 headerblock.append(line) |
| 66 line = infile.next() | 70 line = infile.next() |
| 67 for header in sorted(headerblock, key=IncludeCompareKey): | 71 for header in sorted(headerblock, key=IncludeCompareKey): |
| 68 outfile.write(header) | 72 outfile.write(header) |
| 69 # Intentionally fall through, to write the line that caused | 73 # Intentionally fall through, to write the line that caused |
| 70 # the above while loop to exit. | 74 # the above while loop to exit. |
| 71 outfile.write(line) | 75 outfile.write(line) |
| 72 | 76 |
| 73 | 77 |
| 78 def DiffAndConfirm(filename, should_confirm): | |
| 79 fixfilename = filename + '.new' | |
| 80 infile = open(filename, 'r') | |
| 81 outfile = open(fixfilename, 'w') | |
| 82 SortHeader(infile, outfile) | |
| 83 infile.close() | |
| 84 outfile.close() # Important so the below diff gets the updated contents. | |
| 85 | |
| 86 try: | |
| 87 diff = os.system('diff -u %s %s' % (filename, fixfilename)) | |
| 88 if diff >> 8 == 0: # Check exit code. | |
| 89 print '%s: no change' % filename | |
| 90 return | |
| 91 | |
| 92 if not should_confirm or YesNo('Use new file (y/N)?'): | |
| 93 os.rename(fixfilename, filename) | |
| 94 finally: | |
| 95 try: | |
| 96 os.remove(fixfilename) | |
| 97 except OSError: | |
| 98 # If the file isn't there, we don't care. | |
| 99 pass | |
| 100 | |
| 101 | |
| 102 def GitShell(args, ignore_return=False): | |
| 103 """A shell invocation suitable for communicating with git. Returns | |
| 104 output as list of lines, raises exception on error. | |
| 105 """ | |
| 106 job = subprocess.Popen(args, | |
| 107 shell=True, | |
| 108 stdout=subprocess.PIPE, | |
| 109 stderr=subprocess.STDOUT) | |
| 110 (out, err) = job.communicate() | |
| 111 if job.returncode != 0 and not ignore_return: | |
| 112 print out | |
| 113 raise Exception("Error %d running command %s" % ( | |
| 114 job.returncode, args)) | |
| 115 return out.split('\n') | |
| 116 | |
| 117 | |
| 118 def FilenamesFromGit(branch_name): | |
|
Nico
2011/10/26 17:53:38
Needs docstring
| |
| 119 lines = GitShell('git diff --stat=600,500 %s' % branch_name) | |
| 120 filenames = [] | |
| 121 for line in lines: | |
| 122 line = line.lstrip() | |
| 123 # Avoid summary line, and files that have been deleted (no plus). | |
| 124 if line.find('|') != -1 and line.find('+') != -1: | |
| 125 filename = line.split()[0] | |
| 126 if filename: | |
| 127 filename = filename.rstrip() | |
| 128 ext = filename.rsplit('.')[-1] | |
| 129 if ext in _SOURCE_EXTENSIONS: | |
| 130 filenames.append(filename) | |
| 131 return filenames | |
| 132 | |
| 133 | |
| 74 def main(): | 134 def main(): |
| 75 parser = optparse.OptionParser(usage='%prog filename1 filename2 ...') | 135 parser = optparse.OptionParser(usage='%prog filename1 filename2 ...') |
| 76 opts, args = parser.parse_args() | 136 parser.add_option('-g', '--gitbranch', dest='git_branch', |
| 137 help='Get list of files as diff from git branch.') | |
| 138 parser.add_option('-q', '--quiet', action='store_false', default=True, | |
|
Nico
2011/10/26 17:53:38
Confirm prompt suppression flags are usually calle
| |
| 139 dest='should_confirm', | |
| 140 help='Turn off confirmation prompt.') | |
| 141 opts, filenames = parser.parse_args() | |
| 77 | 142 |
| 78 if len(args) < 1: | 143 if len(filenames) < 1 and not opts.git_branch: |
| 79 parser.print_help() | 144 parser.print_help() |
| 80 sys.exit(1) | 145 sys.exit(1) |
| 81 | 146 |
| 82 for filename in args: | 147 if opts.git_branch: |
| 83 fixfilename = filename + '.new' | 148 filenames = FilenamesFromGit(opts.git_branch) |
| 84 infile = open(filename, 'r') | |
| 85 outfile = open(fixfilename, 'w') | |
| 86 SortHeader(infile, outfile) | |
| 87 infile.close() | |
| 88 outfile.close() # Important so the below diff gets the updated contents. | |
| 89 | 149 |
| 90 try: | 150 for filename in filenames: |
| 91 diff = os.system('diff -u %s %s' % (filename, fixfilename)) | 151 DiffAndConfirm(filename, opts.should_confirm) |
| 92 if diff >> 8 == 0: # Check exit code. | |
| 93 print '%s: no change' % filename | |
| 94 continue | |
| 95 | |
| 96 if YesNo('Use new file (y/N)?'): | |
| 97 os.rename(fixfilename, filename) | |
| 98 finally: | |
| 99 try: | |
| 100 os.remove(fixfilename) | |
| 101 except OSError: | |
| 102 # If the file isn't there, we don't care. | |
| 103 pass | |
| 104 | 152 |
| 105 | 153 |
| 106 if __name__ == '__main__': | 154 if __name__ == '__main__': |
| 107 main() | 155 main() |
| OLD | NEW |