Add support to hide and unhide patches
[stgit] / stgit / main.py
CommitLineData
41a6d859
CM
1"""Basic quilt-like functionality
2"""
3
4__copyright__ = """
5Copyright (C) 2005, Catalin Marinas <catalin.marinas@gmail.com>
6
7This program is free software; you can redistribute it and/or modify
8it under the terms of the GNU General Public License version 2 as
9published by the Free Software Foundation.
10
11This program is distributed in the hope that it will be useful,
12but WITHOUT ANY WARRANTY; without even the implied warranty of
13MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
14GNU General Public License for more details.
15
16You should have received a copy of the GNU General Public License
17along with this program; if not, write to the Free Software
18Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
19"""
20
21import sys, os
c333afcf 22from optparse import OptionParser
41a6d859 23
c333afcf 24import stgit.commands
41a6d859 25
41a6d859
CM
26#
27# The commands map
28#
c333afcf
CM
29class Commands(dict):
30 """Commands class. It performs on-demand module loading
31 """
32 def __getitem__(self, key):
514dd4f2
CM
33 """Return the command python module name based.
34 """
35 global prog
36
c333afcf 37 cmd_mod = self.get(key)
514dd4f2
CM
38 if not cmd_mod:
39 candidates = [cmd for cmd in self.keys() if cmd.startswith(key)]
40
41 if not candidates:
42 print >> sys.stderr, 'Unknown command: %s' % key
43 print >> sys.stderr, ' Try "%s help" for a list of ' \
44 'supported commands' % prog
45 sys.exit(1)
46 elif len(candidates) > 1:
47 print >> sys.stderr, 'Ambiguous command: %s' % key
48 print >> sys.stderr, ' Candidates are: %s' \
49 % ', '.join(candidates)
50 sys.exit(1)
51
52 cmd_mod = self.get(candidates[0])
53
c333afcf
CM
54 __import__('stgit.commands.' + cmd_mod)
55 return getattr(stgit.commands, cmd_mod)
56
57commands = Commands({
58 'add': 'add',
59 'applied': 'applied',
60 'assimilate': 'assimilate',
61 'branch': 'branch',
62 'delete': 'delete',
63 'diff': 'diff',
64 'clean': 'clean',
65 'clone': 'clone',
66 'commit': 'commit',
67 'export': 'export',
68 'files': 'files',
69 'float': 'float',
70 'fold': 'fold',
71 'goto': 'goto',
841c7b2a 72 'hide': 'hide',
c333afcf
CM
73 'id': 'id',
74 'import': 'imprt',
75 'init': 'init',
76 'log': 'log',
77 'mail': 'mail',
78 'new': 'new',
79 'patches': 'patches',
80 'pick': 'pick',
81 'pop': 'pop',
82 'pull': 'pull',
83 'push': 'push',
22037590 84 'rebase': 'rebase',
c333afcf
CM
85 'refresh': 'refresh',
86 'rename': 'rename',
87 'resolved': 'resolved',
88 'rm': 'rm',
89 'series': 'series',
90 'show': 'show',
91 'status': 'status',
06848fab 92 'sync': 'sync',
c333afcf
CM
93 'top': 'top',
94 'unapplied': 'unapplied',
841c7b2a
CM
95 'uncommit': 'uncommit',
96 'unhide': 'unhide'
c333afcf 97 })
41a6d859 98
5dc51fea
YD
99# classification: repository, stack, patch, working copy
100repocommands = (
101 'branch',
102 'clone',
103 'id',
104 'pull'
105 )
106stackcommands = (
107 'applied',
4d0ba818 108 'assimilate',
5dc51fea
YD
109 'clean',
110 'commit',
d98a499c 111 'float',
5dc51fea 112 'goto',
841c7b2a 113 'hide',
5dc51fea
YD
114 'init',
115 'pop',
116 'push',
22037590 117 'rebase',
5dc51fea
YD
118 'series',
119 'top',
120 'unapplied',
841c7b2a
CM
121 'uncommit',
122 'unhide'
5dc51fea
YD
123 )
124patchcommands = (
125 'delete',
126 'export',
127 'files',
128 'fold',
129 'import',
64354a2d 130 'log',
5dc51fea
YD
131 'mail',
132 'new',
133 'pick',
134 'refresh',
135 'rename',
06848fab
CM
136 'show',
137 'sync'
5dc51fea
YD
138 )
139wccommands = (
140 'add',
141 'diff',
142 'patches',
143 'resolved',
144 'rm',
145 'status'
146 )
147
148def _print_helpstring(cmd):
149 print ' ' + cmd + ' ' * (12 - len(cmd)) + commands[cmd].help
150
41a6d859
CM
151def print_help():
152 print 'usage: %s <command> [options]' % os.path.basename(sys.argv[0])
153 print
5dc51fea 154 print 'Generic commands:'
f0699cc7 155 print ' help print the detailed command usage'
1e0bdf2a 156 print ' version display version information'
22ea9102 157 print ' copyright display copyright information'
5dc51fea 158 # unclassified commands if any
41a6d859
CM
159 cmds = commands.keys()
160 cmds.sort()
161 for cmd in cmds:
5dc51fea
YD
162 if not cmd in repocommands and not cmd in stackcommands \
163 and not cmd in patchcommands and not cmd in wccommands:
164 _print_helpstring(cmd)
165 print
166
167 print 'Repository commands:'
168 for cmd in repocommands:
169 _print_helpstring(cmd)
170 print
171
172 print 'Stack commands:'
173 for cmd in stackcommands:
174 _print_helpstring(cmd)
175 print
176
177 print 'Patch commands:'
178 for cmd in patchcommands:
179 _print_helpstring(cmd)
180 print
181
182 print 'Working-copy commands:'
183 for cmd in wccommands:
184 _print_helpstring(cmd)
41a6d859
CM
185
186#
187# The main function (command dispatcher)
188#
189def main():
190 """The main function
191 """
514dd4f2
CM
192 global prog
193
41a6d859
CM
194 prog = os.path.basename(sys.argv[0])
195
196 if len(sys.argv) < 2:
d00e708a 197 print >> sys.stderr, 'usage: %s <command>' % prog
41a6d859 198 print >> sys.stderr, \
f0699cc7 199 ' Try "%s --help" for a list of supported commands' % prog
41a6d859
CM
200 sys.exit(1)
201
202 cmd = sys.argv[1]
203
f0699cc7 204 if cmd in ['-h', '--help']:
a9d82c36 205 if len(sys.argv) >= 3 and sys.argv[2] in commands:
31c5abf2 206 cmd = sys.argv[2]
f0699cc7 207 sys.argv[2] = '--help'
31c5abf2
PR
208 else:
209 print_help()
210 sys.exit(0)
f0699cc7
CM
211 if cmd == 'help':
212 if len(sys.argv) == 3 and not sys.argv[2] in ['-h', '--help']:
213 cmd = sys.argv[2]
214 if not cmd in commands:
215 print >> sys.stderr, '%s help: "%s" command unknown' \
216 % (prog, cmd)
217 sys.exit(1)
218
219 sys.argv[0] += ' %s' % cmd
220 command = commands[cmd]
221 parser = OptionParser(usage = command.usage,
222 option_list = command.options)
0d4cd7ce
CM
223 from pydoc import pager
224 pager(parser.format_help())
f0699cc7 225 else:
4fe42e63 226 print_help()
f0699cc7 227 sys.exit(0)
1e0bdf2a 228 if cmd in ['-v', '--version', 'version']:
c333afcf 229 from stgit.version import version
4df2f866 230 print 'Stacked GIT %s' % version
50725547 231 os.system('git --version')
1e0bdf2a 232 print 'Python version %s' % sys.version
41a6d859 233 sys.exit(0)
22ea9102
CM
234 if cmd in ['copyright']:
235 print __copyright__
236 sys.exit(0)
41a6d859
CM
237
238 # re-build the command line arguments
239 sys.argv[0] += ' %s' % cmd
240 del(sys.argv[1])
241
242 command = commands[cmd]
f0699cc7
CM
243 usage = command.usage.split('\n')[0].strip()
244 parser = OptionParser(usage = usage, option_list = command.options)
41a6d859 245 options, args = parser.parse_args()
22d87516
CM
246
247 # These modules are only used from this point onwards and do not
248 # need to be imported earlier
eee7283e 249 from stgit.config import config_setup
9e3f506f 250 from ConfigParser import ParsingError, NoSectionError
22d87516
CM
251 from stgit.stack import Series, StackException
252 from stgit.git import GitException
253 from stgit.commands.common import CmdException
254 from stgit.gitmergeonefile import GitMergeException
255
41a6d859 256 try:
eee7283e
CM
257 config_setup()
258
98290387
CM
259 # 'clone' doesn't expect an already initialised GIT tree. A Series
260 # object will be created after the GIT tree is cloned
261 if cmd != 'clone':
262 if hasattr(options, 'branch') and options.branch:
c333afcf 263 command.crt_series = Series(options.branch)
98290387 264 else:
c333afcf 265 command.crt_series = Series()
98290387 266 stgit.commands.common.crt_series = command.crt_series
9ac09909 267
41a6d859 268 command.func(parser, options, args)
9e3f506f
KH
269 except (IOError, ParsingError, NoSectionError, CmdException,
270 StackException, GitException, GitMergeException), err:
41a6d859
CM
271 print >> sys.stderr, '%s %s: %s' % (prog, cmd, err)
272 sys.exit(2)
4d9fc826
CM
273 except KeyboardInterrupt:
274 sys.exit(1)
41a6d859
CM
275
276 sys.exit(0)