OLD | NEW |
---|---|
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 Loading... | |
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 Loading... | |
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 |
OLD | NEW |