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

Side by Side Diff: subcommand.py

Issue 23250002: Split generic subcommand code off its own module. (Closed) Base URL: svn://svn.chromium.org/chrome/trunk/tools/depot_tools
Patch Set: typo 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
« no previous file with comments | « git_cl.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
(Empty)
1 # Copyright (c) 2013 The Chromium Authors. All rights reserved.
2 # Use of this source code is governed by a BSD-style license that can be
3 # found in the LICENSE file.
4
5 """Manages subcommands in a script.
6
7 Each subcommand should look like this:
8 @usage('[pet name]')
9 def CMDpet(parser, args):
10 '''Prints a pet.
11
12 Many people likes pet. This command prints a pet for your pleasure.
13 '''
14 parser.add_option('--color', help='color of your pet')
15 options, args = parser.parse_args(args)
16 if len(args) != 1:
17 parser.error('A pet name is required')
18 pet = args[0]
19 if options.color:
20 print('Nice %s %d' % (options.color, pet))
21 else:
22 print('Nice %s' % pet)
23 return 0
24
25 Explanation:
26 - usage decorator alter the usage: line in the command's help.
27 - docstring is used to both short help line and long help line.
28 - parser can be augmented with arguments.
29 - return the exit code.
30 - Every function in the main module with a name starting with 'CMD' will be a
31 subcommand.
32 - The module's docstring will be used in the default help page.
33 """
34
35 import difflib
36 import sys
37
38 # This is the main module where to search for commands. It should never be
39 # modified except for unit tests, where the main module is the unit test so this
40 # value needs to be updated manually.
41 MAIN_MODULE = sys.modules['__main__']
iannucci 2013/08/15 20:46:04 Hm.. I don't like this (esp. modifying it in the t
M-A Ruel 2013/08/15 20:56:55 I'll prototype this and will reply.
42
43
44 def usage(more):
45 """Adds a 'usage_more' property to a CMD function."""
46 def hook(fn):
47 fn.usage_more = more
48 return fn
49 return hook
iannucci 2013/08/15 20:46:04 I've actually always wondered why this doesn't jus
M-A Ruel 2013/08/15 20:56:55 wraps() is not applicable because it's not a hook,
iannucci 2013/08/15 21:23:22 Ah right.
50
51
52 def get_commands():
53 """Returns a dict of command and their handling function.
54
55 The commands must be in the '__main__' modules. To import a command from a
iannucci 2013/08/15 20:46:04 I think some of these docstrings need to be update
56 submodule, use:
57 from mysubcommand import CMDfoo
58
59 Automatically adds 'help' if not already defined.
60
61 A command can be effectively disabled by defining a global variable to None,
62 e.g.:
63 CMDhelp = None
64 """
65 cmds = dict(
66 (fn[3:], getattr(MAIN_MODULE, fn))
67 for fn in dir(MAIN_MODULE) if fn.startswith('CMD'))
68 cmds.setdefault('help', CMDhelp)
69 return cmds
70
71
72 def get_command(name):
iannucci 2013/08/15 20:46:04 Hm... I would expect this to be equivalent to get_
M-A Ruel 2013/08/15 20:56:55 That was ripped directly from git_cl.py The docstr
iannucci 2013/08/15 21:23:22 maybe find_closest_command?
M-A Ruel 2013/08/16 13:47:20 Done.
73 """Retrieves the function to handle a command.
74
75 It automatically tries to guess the intended command by handling typos or
76 incomplete names.
77 """
78 commands = get_commands()
79 if name in commands:
80 return commands[name]
81
82 # An exact match was not found. Try to be smart and look if there's something
83 # similar.
84 commands_with_prefix = [c for c in commands if c.startswith(name)]
85 if len(commands_with_prefix) == 1:
86 return commands[commands_with_prefix[0]]
87
88 # A #closeenough approximation of levenshtein distance.
89 def close_enough(a, b):
90 return difflib.SequenceMatcher(a=a, b=b).ratio()
91
92 hamming_commands = sorted(
93 ((close_enough(c, name), c) for c in commands),
94 reverse=True)
95 if (hamming_commands[0][0] - hamming_commands[1][0]) < 0.3:
96 # Too ambiguous.
97 return
98
99 if hamming_commands[0][0] < 0.8:
100 # Not similar enough. Don't be a fool and run a random command.
101 return
102
103 return commands[hamming_commands[0][1]]
104
105
106 def CMDhelp(parser, args):
107 """Prints list of commands or help for a specific command."""
108 # This is the default help implementation. It can be disabled or overriden if
109 # wanted.
110 if not any(i in ('-h', '--help') for i in args):
111 args = args + ['--help']
112 _, args = parser.parse_args(args)
113 # Never gets there.
114 assert False
115
116
117 def add_command_usage(parser, command):
118 """Modifies an OptionParser object with the function's documentation."""
119 name = command.__name__[3:]
120 more = getattr(command, 'usage_more', '')
121 if name == 'help':
122 name = '<command>'
123 # Use the __main__ module docstring as the help.
124 parser.description = sys.modules['__main__'].__doc__.strip() + '\n'
iannucci 2013/08/15 20:46:04 This shouldn't be sys.modules['__main__'], I don't
M-A Ruel 2013/08/15 20:56:55 Exact, fixing.
125 else:
126 parser.description = command.__doc__.strip() or ''
127 if parser.description:
128 parser.description += '\n'
129 parser.set_usage('usage: %%prog %s [options] %s' % (name, more))
130
131
132 def dispatch_command(parser, args):
133 """Dispatches execution to the right command.
134
135 Fallbacks to 'help'.
136 """
137 commands = get_commands()
138 length = max(len(c) for c in commands)
139
140 def gen_summary(x):
141 """Creates a oneline summary from the docstring."""
142 line = x.__doc__.split('\n', 1)[0].rstrip('.')
143 return (line[0].lower() + line[1:]).strip()
144
145 docs = sorted(
146 (name, gen_summary(handler)) for name, handler in commands.iteritems())
147
148 # Lists all the commands in 'help'.
149 if commands['help']:
150 commands['help'].usage_more = (
151 '\n\nCommands are:\n' + '\n'.join(
152 ' %-*s %s' % (length, name, doc) for name, doc in docs))
153
154 if args:
155 if args[0] in ('-h', '--help') and len(args) > 1:
156 # Inverse the argument order so 'tool --help cmd' is rewritten to
157 # 'tool cmd --help'.
158 args = [args[1], args[0]] + args[2:]
159 command = get_command(args[0])
160 if command:
161 if command.__name__ == 'CMDhelp' and len(args) > 1:
162 # Inverse the arguments order so 'tool help cmd' is rewritten to
163 # 'tool cmd --help'. Do it here since we want 'tool hel cmd' to work
164 # too.
165 args = [args[1], '--help'] + args[2:]
166 command = get_command(args[0])
167
168 # "fix" the usage and the description now that we know the subcommand.
169 add_command_usage(parser, command)
170 return command(parser, args[1:])
171
172 if commands['help']:
173 # Not a known command. Default to help.
174 add_command_usage(parser, commands['help'])
175 return commands['help'](parser, args)
176
177 # Nothing can be done.
178 return 2
OLDNEW
« no previous file with comments | « git_cl.py ('k') | tests/git_cl_test.py » ('j') | no next file with comments »

Powered by Google App Engine
This is Rietveld 408576698