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

Side by Side Diff: subcommand.py

Issue 22824018: Convert gclient to use subcommand.py (Closed) Base URL: svn://svn.chromium.org/chrome/trunk/tools/depot_tools
Patch Set: Created 7 years, 4 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 | Annotate | Revision Log
« gclient.py ('K') | « git_cl.py ('k') | no next file » | no next file with comments »
Toggle Intra-line Diffs ('i') | Expand Comments ('e') | Collapse Comments ('c') | Show Comments Hide Comments ('s')
OLDNEW
1 # Copyright 2013 The Chromium Authors. All rights reserved. 1 # Copyright 2013 The Chromium Authors. All rights reserved.
2 # Use of this source code is governed by a BSD-style license that can be 2 # Use of this source code is governed by a BSD-style license that can be
3 # found in the LICENSE file. 3 # found in the LICENSE file.
4 4
5 """Manages subcommands in a script. 5 """Manages subcommands in a script.
6 6
7 Each subcommand should look like this: 7 Each subcommand should look like this:
8 @usage('[pet name]') 8 @usage('[pet name]')
9 def CMDpet(parser, args): 9 def CMDpet(parser, args):
10 '''Prints a pet. 10 '''Prints a pet.
(...skipping 33 matching lines...) Expand 10 before | Expand all | Expand 10 after
44 44
45 45
46 def usage(more): 46 def usage(more):
47 """Adds a 'usage_more' property to a CMD function.""" 47 """Adds a 'usage_more' property to a CMD function."""
48 def hook(fn): 48 def hook(fn):
49 fn.usage_more = more 49 fn.usage_more = more
50 return fn 50 return fn
51 return hook 51 return hook
52 52
53 53
54 def example(more):
55 """Adds a 'example' property to a CMD function.
56
57 It will be shown in the epilog.
58 """
59 def hook(fn):
60 fn.example = more
61 return fn
62 return hook
63
64
54 def CMDhelp(parser, args): 65 def CMDhelp(parser, args):
55 """Prints list of commands or help for a specific command.""" 66 """Prints list of commands or help for a specific command."""
56 # This is the default help implementation. It can be disabled or overriden if 67 # This is the default help implementation. It can be disabled or overriden if
57 # wanted. 68 # wanted.
58 if not any(i in ('-h', '--help') for i in args): 69 if not any(i in ('-h', '--help') for i in args):
59 args = args + ['--help'] 70 args = args + ['--help']
60 _, args = parser.parse_args(args) 71 _, args = parser.parse_args(args)
61 # Never gets there. 72 # Never gets there.
62 assert False 73 assert False
63 74
64 75
76 def _is_color_enabled():
iannucci 2013/08/16 21:32:05 _get_color_module() -> colorama or None
M-A Ruel 2013/08/17 00:19:10 Done.
77 """Looks if a module named colorama was imported.
78
79 If so, assumes colors are supported and return the module handle.
80 """
81 return sys.modules.get('colorama') or sys.modules.get('third_party.colorama')
82
83
65 class CommandDispatcher(object): 84 class CommandDispatcher(object):
66 def __init__(self, module): 85 def __init__(self, module):
67 """module is the name of the main python module where to look for commands. 86 """module is the name of the main python module where to look for commands.
68 87
69 The python builtin variable __name__ MUST be used for |module|. If the 88 The python builtin variable __name__ MUST be used for |module|. If the
70 script is executed in the form 'python script.py', __name__ == '__main__' 89 script is executed in the form 'python script.py', __name__ == '__main__'
71 and sys.modules['script'] doesn't exist. On the other hand if it is unit 90 and sys.modules['script'] doesn't exist. On the other hand if it is unit
72 tested, __main__ will be the unit test's module so it has to reference to 91 tested, __main__ will be the unit test's module so it has to reference to
73 itself with 'script'. __name__ always match the right value. 92 itself with 'script'. __name__ always match the right value.
74 """ 93 """
(...skipping 44 matching lines...) Expand 10 before | Expand all | Expand 10 after
119 if (hamming_commands[0][0] - hamming_commands[1][0]) < 0.3: 138 if (hamming_commands[0][0] - hamming_commands[1][0]) < 0.3:
120 # Too ambiguous. 139 # Too ambiguous.
121 return 140 return
122 141
123 if hamming_commands[0][0] < 0.8: 142 if hamming_commands[0][0] < 0.8:
124 # Not similar enough. Don't be a fool and run a random command. 143 # Not similar enough. Don't be a fool and run a random command.
125 return 144 return
126 145
127 return commands[hamming_commands[0][1]] 146 return commands[hamming_commands[0][1]]
128 147
148 def _gen_commands_list(self):
149 """Generates the short list of supported commands."""
150 commands = self.enumerate_commands()
151 docs = sorted(
152 (name, self._create_command_summary(name, handler))
153 for name, handler in commands.iteritems())
154 # Skip commands without a docstring.
155 docs = [i for i in docs if i[1]]
156 # Then calculate maximum length for alignment:
157 length = max(len(c) for c in commands)
158
159 # Look if color is supported.
160 colors = _is_color_enabled()
161 green = reset = ''
162 if colors:
163 green = colors.Fore.GREEN
164 reset = colors.Fore.RESET
165 return (
166 'Commands are:\n' +
167 ''.join(
168 ' %s%-*s%s %s\n' % (green, length, name, reset, doc)
169 for name, doc in docs))
170
129 def _add_command_usage(self, parser, command): 171 def _add_command_usage(self, parser, command):
130 """Modifies an OptionParser object with the function's documentation.""" 172 """Modifies an OptionParser object with the function's documentation."""
131 name = command.__name__[3:] 173 name = command.__name__[3:]
132 more = getattr(command, 'usage_more', '')
133 if name == 'help': 174 if name == 'help':
134 name = '<command>' 175 name = '<command>'
135 # Use the module's docstring as the description for the 'help' command if 176 # Use the module's docstring as the description for the 'help' command if
136 # available. 177 # available.
137 parser.description = self.module.__doc__ 178 parser.description = (self.module.__doc__ or '').rstrip()
M-A Ruel 2013/08/16 21:08:05 The following is formatting fine tuning so the out
179 if parser.description:
180 parser.description += '\n\n'
181 parser.description += self._gen_commands_list()
182 # Do not touch epilog.
138 else: 183 else:
139 # Use the command's docstring if available. 184 # Use the command's docstring if available. For commands, unlike module
140 parser.description = command.__doc__ 185 # docstring, realign.
141 parser.description = (parser.description or '').strip() 186 lines = (command.__doc__ or '').rstrip().splitlines()
iannucci 2013/08/16 21:32:05 you may want to just use textwrap.dedent()
M-A Ruel 2013/08/17 00:19:10 Done.
142 if parser.description: 187 # Determine alignment automatically.
143 parser.description += '\n' 188 alignment = 0
189 for i in lines[1:]:
190 if not i:
191 continue
192 if i:
193 # Cheezy way to count the leading number of whitespaces.
194 alignment = len(i) - len(i.lstrip(' '))
195 break
196 lines_fixed = [lines[0]] + [
197 l[alignment:] if len(l) >= alignment else l for l in lines[1:]]
198 parser.description = '\n'.join(lines_fixed)
199 if parser.description:
200 parser.description += '\n'
201 parser.epilog = getattr(command, 'example', None)
iannucci 2013/08/16 21:32:05 let's call it .example instead, and then add 'Exam
M-A Ruel 2013/08/17 00:19:10 I've decided to switch to naming it epilog, so the
202 if parser.epilog:
203 parser.epilog = '\n' + parser.epilog.strip() + '\n'
204
205 more = getattr(command, 'usage_more', '')
144 parser.set_usage( 206 parser.set_usage(
145 'usage: %%prog %s [options]%s' % (name, '' if not more else ' ' + more)) 207 'usage: %%prog %s [options]%s' % (name, '' if not more else ' ' + more))
146 208
147 @staticmethod 209 @staticmethod
148 def _create_command_summary(name, command): 210 def _create_command_summary(name, command):
149 """Creates a oneline summary from the command's docstring.""" 211 """Creates a oneline summary from the command's docstring."""
150 if name != command.__name__[3:]: 212 if name != command.__name__[3:]:
151 # Skip aliases. 213 # Skip aliases.
152 return '' 214 return ''
153 doc = command.__doc__ or '' 215 doc = command.__doc__ or ''
154 line = doc.split('\n', 1)[0].rstrip('.') 216 line = doc.split('\n', 1)[0].rstrip('.')
155 if not line: 217 if not line:
156 return line 218 return line
157 return (line[0].lower() + line[1:]).strip() 219 return (line[0].lower() + line[1:]).strip()
158 220
159 def execute(self, parser, args): 221 def execute(self, parser, args):
160 """Dispatches execution to the right command. 222 """Dispatches execution to the right command.
161 223
162 Fallbacks to 'help' if not disabled. 224 Fallbacks to 'help' if not disabled.
163 """ 225 """
164 commands = self.enumerate_commands() 226 # Unconditionally disable format_description() and format_epilog().
165 length = max(len(c) for c in commands) 227 # Technically, a formatter should be used but it's not worth (yet) the
166 228 # trouble.
167 # Lists all the commands in 'help'. 229 parser.format_description = lambda _: parser.description or ''
168 if commands['help']: 230 parser.format_epilog = lambda _: parser.epilog or ''
169 docs = sorted(
170 (name, self._create_command_summary(name, handler))
171 for name, handler in commands.iteritems())
172 # Skip commands without a docstring.
173 commands['help'].usage_more = (
174 '\n\nCommands are:\n' + '\n'.join(
175 ' %-*s %s' % (length, name, doc) for name, doc in docs if doc))
176 231
177 if args: 232 if args:
178 if args[0] in ('-h', '--help') and len(args) > 1: 233 if args[0] in ('-h', '--help') and len(args) > 1:
179 # Inverse the argument order so 'tool --help cmd' is rewritten to 234 # Inverse the argument order so 'tool --help cmd' is rewritten to
180 # 'tool cmd --help'. 235 # 'tool cmd --help'.
181 args = [args[1], args[0]] + args[2:] 236 args = [args[1], args[0]] + args[2:]
182 command = self.find_nearest_command(args[0]) 237 command = self.find_nearest_command(args[0])
183 if command: 238 if command:
184 if command.__name__ == 'CMDhelp' and len(args) > 1: 239 if command.__name__ == 'CMDhelp' and len(args) > 1:
185 # Inverse the arguments order so 'tool help cmd' is rewritten to 240 # Inverse the arguments order so 'tool help cmd' is rewritten to
186 # 'tool cmd --help'. Do it here since we want 'tool hel cmd' to work 241 # 'tool cmd --help'. Do it here since we want 'tool hel cmd' to work
187 # too. 242 # too.
188 args = [args[1], '--help'] + args[2:] 243 args = [args[1], '--help'] + args[2:]
189 command = self.find_nearest_command(args[0]) or command 244 command = self.find_nearest_command(args[0]) or command
190 245
191 # "fix" the usage and the description now that we know the subcommand. 246 # "fix" the usage and the description now that we know the subcommand.
192 self._add_command_usage(parser, command) 247 self._add_command_usage(parser, command)
193 return command(parser, args[1:]) 248 return command(parser, args[1:])
194 249
195 if commands['help']: 250 cmdhelp = self.enumerate_commands().get('help')
251 if cmdhelp:
196 # Not a known command. Default to help. 252 # Not a known command. Default to help.
197 self._add_command_usage(parser, commands['help']) 253 self._add_command_usage(parser, cmdhelp)
198 return commands['help'](parser, args) 254 return cmdhelp(parser, args)
199 255
200 # Nothing can be done. 256 # Nothing can be done.
201 return 2 257 return 2
OLDNEW
« gclient.py ('K') | « git_cl.py ('k') | no next file » | no next file with comments »

Powered by Google App Engine
This is Rietveld 408576698