~bzr-pqm/bzr/bzr.dev

« back to all changes in this revision

Viewing changes to bzrlib/commands.py

  • Committer: Martin Pool
  • Date: 2005-05-10 03:55:34 UTC
  • Revision ID: mbp@sourcefrog.net-20050510035534-643062e821052ac5
- Add fortune-cookie external plugin demonstration

Show diffs side-by-side

added added

removed removed

Lines of Context:
1
 
# Copyright (C) 2006 by Canonical Ltd
2
 
#
 
1
# Copyright (C) 2004, 2005 by Canonical Ltd
 
2
 
3
3
# This program is free software; you can redistribute it and/or modify
4
4
# it under the terms of the GNU General Public License as published by
5
5
# the Free Software Foundation; either version 2 of the License, or
6
6
# (at your option) any later version.
7
 
#
 
7
 
8
8
# This program is distributed in the hope that it will be useful,
9
9
# but WITHOUT ANY WARRANTY; without even the implied warranty of
10
10
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
11
11
# GNU General Public License for more details.
12
 
#
 
12
 
13
13
# You should have received a copy of the GNU General Public License
14
14
# along with this program; if not, write to the Free Software
15
15
# Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA  02111-1307  USA
16
16
 
17
17
 
18
 
# TODO: probably should say which arguments are candidates for glob
19
 
# expansion on windows and do that at the command level.
20
 
 
21
 
# TODO: Define arguments by objects, rather than just using names.
22
 
# Those objects can specify the expected type of the argument, which
23
 
# would help with validation and shell completion.  They could also provide
24
 
# help/explanation for that argument in a structured way.
25
 
 
26
 
# TODO: Specific "examples" property on commands for consistent formatting.
27
 
 
28
 
# TODO: "--profile=cum", to change sort order.  Is there any value in leaving
29
 
# the profile output behind so it can be interactively examined?
30
 
 
31
 
import sys
32
 
import os
33
 
from warnings import warn
34
 
import errno
35
 
import codecs
 
18
 
 
19
import sys, os, time, os.path
 
20
from sets import Set
36
21
 
37
22
import bzrlib
38
 
import bzrlib.errors as errors
39
 
from bzrlib.errors import (BzrError,
40
 
                           BzrCommandError,
41
 
                           BzrCheckError,
42
 
                           NotBranchError)
43
 
from bzrlib.option import Option
44
 
import bzrlib.osutils
45
 
from bzrlib.revisionspec import RevisionSpec
46
 
from bzrlib.symbol_versioning import (deprecated_method, zero_eight)
47
 
from bzrlib import trace
48
 
from bzrlib.trace import mutter, note, log_error, warning, be_quiet
49
 
 
50
 
plugin_cmds = {}
51
 
 
52
 
 
53
 
def register_command(cmd, decorate=False):
54
 
    """Utility function to help register a command
55
 
 
56
 
    :param cmd: Command subclass to register
57
 
    :param decorate: If true, allow overriding an existing command
58
 
        of the same name; the old command is returned by this function.
59
 
        Otherwise it is an error to try to override an existing command.
60
 
    """
61
 
    global plugin_cmds
62
 
    k = cmd.__name__
63
 
    if k.startswith("cmd_"):
64
 
        k_unsquished = _unsquish_command_name(k)
65
 
    else:
66
 
        k_unsquished = k
67
 
    if not plugin_cmds.has_key(k_unsquished):
68
 
        plugin_cmds[k_unsquished] = cmd
69
 
        mutter('registered plugin command %s', k_unsquished)
70
 
        if decorate and k_unsquished in builtin_command_names():
71
 
            return _builtin_commands()[k_unsquished]
72
 
    elif decorate:
73
 
        result = plugin_cmds[k_unsquished]
74
 
        plugin_cmds[k_unsquished] = cmd
75
 
        return result
76
 
    else:
77
 
        log_error('Two plugins defined the same command: %r' % k)
78
 
        log_error('Not loading the one in %r' % sys.modules[cmd.__module__])
 
23
from bzrlib.trace import mutter, note, log_error
 
24
from bzrlib.errors import bailout, BzrError, BzrCheckError, BzrCommandError
 
25
from bzrlib.osutils import quotefn, pumpfile, isdir, isfile
 
26
from bzrlib.tree import RevisionTree, EmptyTree, WorkingTree, Tree
 
27
from bzrlib.revision import Revision
 
28
from bzrlib import Branch, Inventory, InventoryEntry, ScratchBranch, BZRDIR, \
 
29
     format_date
79
30
 
80
31
 
81
32
def _squish_command_name(cmd):
86
37
    assert cmd.startswith("cmd_")
87
38
    return cmd[4:].replace('_','-')
88
39
 
89
 
 
90
 
def _builtin_commands():
91
 
    import bzrlib.builtins
92
 
    r = {}
93
 
    builtins = bzrlib.builtins.__dict__
94
 
    for name in builtins:
95
 
        if name.startswith("cmd_"):
96
 
            real_name = _unsquish_command_name(name)
97
 
            r[real_name] = builtins[name]
98
 
    return r
99
 
            
100
 
 
101
 
def builtin_command_names():
102
 
    """Return list of builtin command names."""
103
 
    return _builtin_commands().keys()
104
 
    
105
 
 
106
 
def plugin_command_names():
107
 
    return plugin_cmds.keys()
108
 
 
109
 
 
110
 
def _get_cmd_dict(plugins_override=True):
111
 
    """Return name->class mapping for all commands."""
112
 
    d = _builtin_commands()
113
 
    if plugins_override:
114
 
        d.update(plugin_cmds)
115
 
    return d
116
 
 
117
 
    
118
 
def get_all_cmds(plugins_override=True):
 
40
def get_all_cmds():
119
41
    """Return canonical name and class for all registered commands."""
120
 
    for k, v in _get_cmd_dict(plugins_override=plugins_override).iteritems():
121
 
        yield k,v
122
 
 
123
 
 
124
 
def get_cmd_object(cmd_name, plugins_override=True):
 
42
    for k, v in globals().iteritems():
 
43
        if k.startswith("cmd_"):
 
44
            yield _unsquish_command_name(k), v
 
45
 
 
46
def get_cmd_class(cmd):
125
47
    """Return the canonical name and command class for a command.
126
 
 
127
 
    plugins_override
128
 
        If true, plugin commands can override builtins.
129
48
    """
130
 
    from bzrlib.externalcommand import ExternalCommand
131
 
 
132
 
    cmd_name = str(cmd_name)            # not unicode
 
49
    cmd = str(cmd)                      # not unicode
133
50
 
134
51
    # first look up this command under the specified name
135
 
    cmds = _get_cmd_dict(plugins_override=plugins_override)
136
52
    try:
137
 
        return cmds[cmd_name]()
 
53
        return cmd, globals()[_squish_command_name(cmd)]
138
54
    except KeyError:
139
55
        pass
140
56
 
141
57
    # look for any command which claims this as an alias
142
 
    for real_cmd_name, cmd_class in cmds.iteritems():
143
 
        if cmd_name in cmd_class.aliases:
144
 
            return cmd_class()
145
 
 
146
 
    cmd_obj = ExternalCommand.find_command(cmd_name)
147
 
    if cmd_obj:
148
 
        return cmd_obj
149
 
 
150
 
    raise BzrCommandError('unknown command "%s"' % cmd_name)
151
 
 
152
 
 
153
 
class Command(object):
 
58
    for cmdname, cmdclass in get_all_cmds():
 
59
        if cmd in cmdclass.aliases:
 
60
            return cmdname, cmdclass
 
61
 
 
62
    cmdclass = ExternalCommand.find_command(cmd)
 
63
    if cmdclass:
 
64
        return cmd, cmdclass
 
65
 
 
66
    raise BzrCommandError("unknown command %r" % cmd)
 
67
 
 
68
 
 
69
class Command:
154
70
    """Base class for commands.
155
71
 
156
 
    Commands are the heart of the command-line bzr interface.
157
 
 
158
 
    The command object mostly handles the mapping of command-line
159
 
    parameters into one or more bzrlib operations, and of the results
160
 
    into textual output.
161
 
 
162
 
    Commands normally don't have any state.  All their arguments are
163
 
    passed in to the run method.  (Subclasses may take a different
164
 
    policy if the behaviour of the instance needs to depend on e.g. a
165
 
    shell plugin and not just its Python class.)
166
 
 
167
72
    The docstring for an actual command should give a single-line
168
73
    summary, then a complete description of the command.  A grammar
169
74
    description will be inserted.
170
75
 
171
 
    aliases
172
 
        Other accepted names for this command.
173
 
 
174
76
    takes_args
175
77
        List of argument forms, marked with whether they are optional,
176
78
        repeated, etc.
177
79
 
178
 
                Examples:
179
 
 
180
 
                ['to_location', 'from_branch?', 'file*']
181
 
 
182
 
                'to_location' is required
183
 
                'from_branch' is optional
184
 
                'file' can be specified 0 or more times
185
 
 
186
80
    takes_options
187
 
        List of options that may be given for this command.  These can
188
 
        be either strings, referring to globally-defined options,
189
 
        or option objects.  Retrieve through options().
 
81
        List of options that may be given for this command.
190
82
 
191
83
    hidden
192
 
        If true, this command isn't advertised.  This is typically
193
 
        for commands intended for expert users.
194
 
 
195
 
    encoding_type
196
 
        Command objects will get a 'outf' attribute, which has been
197
 
        setup to properly handle encoding of unicode strings.
198
 
        encoding_type determines what will happen when characters cannot
199
 
        be encoded
200
 
            strict - abort if we cannot decode
201
 
            replace - put in a bogus character (typically '?')
202
 
            exact - do not encode sys.stdout
203
 
 
 
84
        If true, this command isn't advertised.
204
85
    """
205
86
    aliases = []
 
87
    
206
88
    takes_args = []
207
89
    takes_options = []
208
 
    encoding_type = 'strict'
209
90
 
210
91
    hidden = False
211
92
    
212
 
    def __init__(self):
213
 
        """Construct an instance of this command."""
214
 
        if self.__doc__ == Command.__doc__:
215
 
            warn("No help message set for %r" % self)
216
 
 
217
 
    def options(self):
218
 
        """Return dict of valid options for this command.
219
 
 
220
 
        Maps from long option name to option object."""
221
 
        r = dict()
222
 
        r['help'] = Option.OPTIONS['help']
223
 
        for o in self.takes_options:
224
 
            if not isinstance(o, Option):
225
 
                o = Option.OPTIONS[o]
226
 
            r[o.name] = o
227
 
        return r
228
 
 
229
 
    def _setup_outf(self):
230
 
        """Return a file linked to stdout, which has proper encoding."""
231
 
        assert self.encoding_type in ['strict', 'exact', 'replace']
232
 
 
233
 
        # Originally I was using self.stdout, but that looks
234
 
        # *way* too much like sys.stdout
235
 
        if self.encoding_type == 'exact':
236
 
            self.outf = sys.stdout
237
 
            return
238
 
 
239
 
        output_encoding = bzrlib.osutils.get_terminal_encoding()
240
 
 
241
 
        # use 'replace' so that we don't abort if trying to write out
242
 
        # in e.g. the default C locale.
243
 
        self.outf = codecs.getwriter(output_encoding)(sys.stdout, errors=self.encoding_type)
244
 
        # For whatever reason codecs.getwriter() does not advertise its encoding
245
 
        # it just returns the encoding of the wrapped file, which is completely
246
 
        # bogus. So set the attribute, so we can find the correct encoding later.
247
 
        self.outf.encoding = output_encoding
248
 
 
249
 
    @deprecated_method(zero_eight)
250
 
    def run_argv(self, argv):
251
 
        """Parse command line and run.
252
 
        
253
 
        See run_argv_aliases for the 0.8 and beyond api.
254
 
        """
255
 
        return self.run_argv_aliases(argv)
256
 
 
257
 
    def run_argv_aliases(self, argv, alias_argv=None):
258
 
        """Parse the command line and run with extra aliases in alias_argv."""
259
 
        args, opts = parse_args(self, argv, alias_argv)
260
 
        if 'help' in opts:  # e.g. bzr add --help
261
 
            from bzrlib.help import help_on_command
262
 
            help_on_command(self.name())
263
 
            return 0
264
 
        # XXX: This should be handled by the parser
265
 
        allowed_names = self.options().keys()
266
 
        for oname in opts:
267
 
            if oname not in allowed_names:
268
 
                raise BzrOptionError("option '--%s' is not allowed for"
269
 
                                " command %r" % (oname, self.name()))
270
 
        # mix arguments and options into one dictionary
271
 
        cmdargs = _match_argform(self.name(), self.takes_args, args)
272
 
        cmdopts = {}
273
 
        for k, v in opts.items():
274
 
            cmdopts[k.replace('-', '_')] = v
275
 
 
276
 
        all_cmd_args = cmdargs.copy()
277
 
        all_cmd_args.update(cmdopts)
278
 
 
279
 
        self._setup_outf()
280
 
 
281
 
        return self.run(**all_cmd_args)
 
93
    def __init__(self, options, arguments):
 
94
        """Construct and run the command.
 
95
 
 
96
        Sets self.status to the return value of run()."""
 
97
        assert isinstance(options, dict)
 
98
        assert isinstance(arguments, dict)
 
99
        cmdargs = options.copy()
 
100
        cmdargs.update(arguments)
 
101
        assert self.__doc__ != Command.__doc__, \
 
102
               ("No help message set for %r" % self)
 
103
        self.status = self.run(**cmdargs)
 
104
 
282
105
    
283
106
    def run(self):
284
 
        """Actually run the command.
 
107
        """Override this in sub-classes.
285
108
 
286
109
        This is invoked with the options and arguments bound to
287
110
        keyword parameters.
288
111
 
289
 
        Return 0 or None if the command was successful, or a non-zero
290
 
        shell error code if not.  It's OK for this method to allow
291
 
        an exception to raise up.
292
 
        """
293
 
        raise NotImplementedError('no implementation of command %r' 
294
 
                                  % self.name())
295
 
 
296
 
    def help(self):
297
 
        """Return help message for this class."""
298
 
        from inspect import getdoc
299
 
        if self.__doc__ is Command.__doc__:
300
 
            return None
301
 
        return getdoc(self)
302
 
 
303
 
    def name(self):
304
 
        return _unsquish_command_name(self.__class__.__name__)
305
 
 
306
 
    def plugin_name(self):
307
 
        """Get the name of the plugin that provides this command.
308
 
 
309
 
        :return: The name of the plugin or None if the command is builtin.
310
 
        """
311
 
        mod_parts = self.__module__.split('.')
312
 
        if len(mod_parts) >= 3 and mod_parts[1] == 'plugins':
313
 
            return mod_parts[2]
314
 
        else:
315
 
            return None
316
 
 
317
 
 
318
 
def parse_spec(spec):
319
 
    """
320
 
    >>> parse_spec(None)
321
 
    [None, None]
322
 
    >>> parse_spec("./")
323
 
    ['./', None]
324
 
    >>> parse_spec("../@")
325
 
    ['..', -1]
326
 
    >>> parse_spec("../f/@35")
327
 
    ['../f', 35]
328
 
    >>> parse_spec('./@revid:john@arbash-meinel.com-20050711044610-3ca0327c6a222f67')
329
 
    ['.', 'revid:john@arbash-meinel.com-20050711044610-3ca0327c6a222f67']
330
 
    """
331
 
    if spec is None:
332
 
        return [None, None]
333
 
    if '/@' in spec:
334
 
        parsed = spec.split('/@')
335
 
        assert len(parsed) == 2
336
 
        if parsed[1] == "":
337
 
            parsed[1] = -1
338
 
        else:
339
 
            try:
340
 
                parsed[1] = int(parsed[1])
341
 
            except ValueError:
342
 
                pass # We can allow stuff like ./@revid:blahblahblah
343
 
            else:
344
 
                assert parsed[1] >=0
345
 
    else:
346
 
        parsed = [spec, None]
347
 
    return parsed
348
 
 
349
 
def parse_args(command, argv, alias_argv=None):
 
112
        Return 0 or None if the command was successful, or a shell
 
113
        error code if not.
 
114
        """
 
115
        return 0
 
116
 
 
117
 
 
118
class ExternalCommand(Command):
 
119
    """Class to wrap external commands.
 
120
 
 
121
    We cheat a little here, when get_cmd_class() calls us we actually give it back
 
122
    an object we construct that has the appropriate path, help, options etc for the
 
123
    specified command.
 
124
 
 
125
    When run_bzr() tries to instantiate that 'class' it gets caught by the __call__
 
126
    method, which we override to call the Command.__init__ method. That then calls
 
127
    our run method which is pretty straight forward.
 
128
 
 
129
    The only wrinkle is that we have to map bzr's dictionary of options and arguments
 
130
    back into command line options and arguments for the script.
 
131
    """
 
132
 
 
133
    def find_command(cls, cmd):
 
134
        bzrpath = os.environ.get('BZRPATH', '')
 
135
 
 
136
        for dir in bzrpath.split(':'):
 
137
            path = os.path.join(dir, cmd)
 
138
            if os.path.isfile(path):
 
139
                return ExternalCommand(path)
 
140
 
 
141
        return None
 
142
 
 
143
    find_command = classmethod(find_command)
 
144
 
 
145
    def __init__(self, path):
 
146
        self.path = path
 
147
 
 
148
        pipe = os.popen('%s --bzr-usage' % path, 'r')
 
149
        self.takes_options = pipe.readline().split()
 
150
        self.takes_args = pipe.readline().split()
 
151
        pipe.close()
 
152
 
 
153
        pipe = os.popen('%s --bzr-help' % path, 'r')
 
154
        self.__doc__ = pipe.read()
 
155
        pipe.close()
 
156
 
 
157
    def __call__(self, options, arguments):
 
158
        Command.__init__(self, options, arguments)
 
159
        return self
 
160
 
 
161
    def run(self, **kargs):
 
162
        opts = []
 
163
        args = []
 
164
 
 
165
        keys = kargs.keys()
 
166
        keys.sort()
 
167
        for name in keys:
 
168
            value = kargs[name]
 
169
            if OPTIONS.has_key(name):
 
170
                # it's an option
 
171
                opts.append('--%s' % name)
 
172
                if value is not None and value is not True:
 
173
                    opts.append(str(value))
 
174
            else:
 
175
                # it's an arg, or arg list
 
176
                if type(value) is not list:
 
177
                    value = [value]
 
178
                for v in value:
 
179
                    if v is not None:
 
180
                        args.append(str(v))
 
181
 
 
182
        self.status = os.spawnv(os.P_WAIT, self.path, [self.path] + opts + args)
 
183
        return self.status
 
184
 
 
185
 
 
186
class cmd_status(Command):
 
187
    """Display status summary.
 
188
 
 
189
    For each file there is a single line giving its file state and name.
 
190
    The name is that in the current revision unless it is deleted or
 
191
    missing, in which case the old name is shown.
 
192
    """
 
193
    takes_args = ['file*']
 
194
    takes_options = ['all']
 
195
    aliases = ['st', 'stat']
 
196
    
 
197
    def run(self, all=False, file_list=None):
 
198
        b = Branch('.', lock_mode='r')
 
199
        b.show_status(show_all=all, file_list=file_list)
 
200
 
 
201
 
 
202
class cmd_cat_revision(Command):
 
203
    """Write out metadata for a revision."""
 
204
 
 
205
    hidden = True
 
206
    takes_args = ['revision_id']
 
207
    
 
208
    def run(self, revision_id):
 
209
        Branch('.').get_revision(revision_id).write_xml(sys.stdout)
 
210
 
 
211
 
 
212
class cmd_revno(Command):
 
213
    """Show current revision number.
 
214
 
 
215
    This is equal to the number of revisions on this branch."""
 
216
    def run(self):
 
217
        print Branch('.').revno()
 
218
 
 
219
    
 
220
class cmd_add(Command):
 
221
    """Add specified files or directories.
 
222
 
 
223
    In non-recursive mode, all the named items are added, regardless
 
224
    of whether they were previously ignored.  A warning is given if
 
225
    any of the named files are already versioned.
 
226
 
 
227
    In recursive mode (the default), files are treated the same way
 
228
    but the behaviour for directories is different.  Directories that
 
229
    are already versioned do not give a warning.  All directories,
 
230
    whether already versioned or not, are searched for files or
 
231
    subdirectories that are neither versioned or ignored, and these
 
232
    are added.  This search proceeds recursively into versioned
 
233
    directories.
 
234
 
 
235
    Therefore simply saying 'bzr add .' will version all files that
 
236
    are currently unknown.
 
237
 
 
238
    TODO: Perhaps adding a file whose directly is not versioned should
 
239
    recursively add that parent, rather than giving an error?
 
240
    """
 
241
    takes_args = ['file+']
 
242
    takes_options = ['verbose']
 
243
    
 
244
    def run(self, file_list, verbose=False):
 
245
        bzrlib.add.smart_add(file_list, verbose)
 
246
 
 
247
 
 
248
class cmd_relpath(Command):
 
249
    """Show path of a file relative to root"""
 
250
    takes_args = ['filename']
 
251
    
 
252
    def run(self, filename):
 
253
        print Branch(filename).relpath(filename)
 
254
 
 
255
 
 
256
 
 
257
class cmd_inventory(Command):
 
258
    """Show inventory of the current working copy or a revision."""
 
259
    takes_options = ['revision']
 
260
    
 
261
    def run(self, revision=None):
 
262
        b = Branch('.')
 
263
        if revision == None:
 
264
            inv = b.read_working_inventory()
 
265
        else:
 
266
            inv = b.get_revision_inventory(b.lookup_revision(revision))
 
267
 
 
268
        for path, entry in inv.iter_entries():
 
269
            print '%-50s %s' % (entry.file_id, path)
 
270
 
 
271
 
 
272
class cmd_move(Command):
 
273
    """Move files to a different directory.
 
274
 
 
275
    examples:
 
276
        bzr move *.txt doc
 
277
 
 
278
    The destination must be a versioned directory in the same branch.
 
279
    """
 
280
    takes_args = ['source$', 'dest']
 
281
    def run(self, source_list, dest):
 
282
        b = Branch('.')
 
283
 
 
284
        b.move([b.relpath(s) for s in source_list], b.relpath(dest))
 
285
 
 
286
 
 
287
class cmd_rename(Command):
 
288
    """Change the name of an entry.
 
289
 
 
290
    examples:
 
291
      bzr rename frob.c frobber.c
 
292
      bzr rename src/frob.c lib/frob.c
 
293
 
 
294
    It is an error if the destination name exists.
 
295
 
 
296
    See also the 'move' command, which moves files into a different
 
297
    directory without changing their name.
 
298
 
 
299
    TODO: Some way to rename multiple files without invoking bzr for each
 
300
    one?"""
 
301
    takes_args = ['from_name', 'to_name']
 
302
    
 
303
    def run(self, from_name, to_name):
 
304
        b = Branch('.')
 
305
        b.rename_one(b.relpath(from_name), b.relpath(to_name))
 
306
 
 
307
 
 
308
 
 
309
class cmd_renames(Command):
 
310
    """Show list of renamed files.
 
311
 
 
312
    TODO: Option to show renames between two historical versions.
 
313
 
 
314
    TODO: Only show renames under dir, rather than in the whole branch.
 
315
    """
 
316
    takes_args = ['dir?']
 
317
 
 
318
    def run(self, dir='.'):
 
319
        b = Branch(dir)
 
320
        old_inv = b.basis_tree().inventory
 
321
        new_inv = b.read_working_inventory()
 
322
 
 
323
        renames = list(bzrlib.tree.find_renames(old_inv, new_inv))
 
324
        renames.sort()
 
325
        for old_name, new_name in renames:
 
326
            print "%s => %s" % (old_name, new_name)        
 
327
 
 
328
 
 
329
class cmd_info(Command):
 
330
    """Show statistical information for this branch"""
 
331
    def run(self):
 
332
        import info
 
333
        info.show_info(Branch('.'))        
 
334
 
 
335
 
 
336
class cmd_remove(Command):
 
337
    """Make a file unversioned.
 
338
 
 
339
    This makes bzr stop tracking changes to a versioned file.  It does
 
340
    not delete the working copy.
 
341
    """
 
342
    takes_args = ['file+']
 
343
    takes_options = ['verbose']
 
344
    
 
345
    def run(self, file_list, verbose=False):
 
346
        b = Branch(file_list[0])
 
347
        b.remove([b.relpath(f) for f in file_list], verbose=verbose)
 
348
 
 
349
 
 
350
class cmd_file_id(Command):
 
351
    """Print file_id of a particular file or directory.
 
352
 
 
353
    The file_id is assigned when the file is first added and remains the
 
354
    same through all revisions where the file exists, even when it is
 
355
    moved or renamed.
 
356
    """
 
357
    hidden = True
 
358
    takes_args = ['filename']
 
359
    def run(self, filename):
 
360
        b = Branch(filename)
 
361
        i = b.inventory.path2id(b.relpath(filename))
 
362
        if i == None:
 
363
            bailout("%r is not a versioned file" % filename)
 
364
        else:
 
365
            print i
 
366
 
 
367
 
 
368
class cmd_file_path(Command):
 
369
    """Print path of file_ids to a file or directory.
 
370
 
 
371
    This prints one line for each directory down to the target,
 
372
    starting at the branch root."""
 
373
    hidden = True
 
374
    takes_args = ['filename']
 
375
    def run(self, filename):
 
376
        b = Branch(filename)
 
377
        inv = b.inventory
 
378
        fid = inv.path2id(b.relpath(filename))
 
379
        if fid == None:
 
380
            bailout("%r is not a versioned file" % filename)
 
381
        for fip in inv.get_idpath(fid):
 
382
            print fip
 
383
 
 
384
 
 
385
class cmd_revision_history(Command):
 
386
    """Display list of revision ids on this branch."""
 
387
    def run(self):
 
388
        for patchid in Branch('.').revision_history():
 
389
            print patchid
 
390
 
 
391
 
 
392
class cmd_directories(Command):
 
393
    """Display list of versioned directories in this branch."""
 
394
    def run(self):
 
395
        for name, ie in Branch('.').read_working_inventory().directories():
 
396
            if name == '':
 
397
                print '.'
 
398
            else:
 
399
                print name
 
400
 
 
401
 
 
402
class cmd_init(Command):
 
403
    """Make a directory into a versioned branch.
 
404
 
 
405
    Use this to create an empty branch, or before importing an
 
406
    existing project.
 
407
 
 
408
    Recipe for importing a tree of files:
 
409
        cd ~/project
 
410
        bzr init
 
411
        bzr add -v .
 
412
        bzr status
 
413
        bzr commit -m 'imported project'
 
414
    """
 
415
    def run(self):
 
416
        Branch('.', init=True)
 
417
 
 
418
 
 
419
class cmd_diff(Command):
 
420
    """Show differences in working tree.
 
421
    
 
422
    If files are listed, only the changes in those files are listed.
 
423
    Otherwise, all changes for the tree are listed.
 
424
 
 
425
    TODO: Given two revision arguments, show the difference between them.
 
426
 
 
427
    TODO: Allow diff across branches.
 
428
 
 
429
    TODO: Option to use external diff command; could be GNU diff, wdiff,
 
430
          or a graphical diff.
 
431
 
 
432
    TODO: Python difflib is not exactly the same as unidiff; should
 
433
          either fix it up or prefer to use an external diff.
 
434
 
 
435
    TODO: If a directory is given, diff everything under that.
 
436
 
 
437
    TODO: Selected-file diff is inefficient and doesn't show you
 
438
          deleted files.
 
439
 
 
440
    TODO: This probably handles non-Unix newlines poorly.
 
441
    """
 
442
    
 
443
    takes_args = ['file*']
 
444
    takes_options = ['revision']
 
445
    aliases = ['di']
 
446
 
 
447
    def run(self, revision=None, file_list=None):
 
448
        from bzrlib.diff import show_diff
 
449
    
 
450
        show_diff(Branch('.'), revision, file_list)
 
451
 
 
452
 
 
453
class cmd_deleted(Command):
 
454
    """List files deleted in the working tree.
 
455
 
 
456
    TODO: Show files deleted since a previous revision, or between two revisions.
 
457
    """
 
458
    def run(self, show_ids=False):
 
459
        b = Branch('.')
 
460
        old = b.basis_tree()
 
461
        new = b.working_tree()
 
462
 
 
463
        ## TODO: Much more efficient way to do this: read in new
 
464
        ## directories with readdir, rather than stating each one.  Same
 
465
        ## level of effort but possibly much less IO.  (Or possibly not,
 
466
        ## if the directories are very large...)
 
467
 
 
468
        for path, ie in old.inventory.iter_entries():
 
469
            if not new.has_id(ie.file_id):
 
470
                if show_ids:
 
471
                    print '%-50s %s' % (path, ie.file_id)
 
472
                else:
 
473
                    print path
 
474
 
 
475
class cmd_root(Command):
 
476
    """Show the tree root directory.
 
477
 
 
478
    The root is the nearest enclosing directory with a .bzr control
 
479
    directory."""
 
480
    takes_args = ['filename?']
 
481
    def run(self, filename=None):
 
482
        """Print the branch root."""
 
483
        from branch import find_branch
 
484
        b = find_branch(filename)
 
485
        print getattr(b, 'base', None) or getattr(b, 'baseurl')
 
486
 
 
487
 
 
488
class cmd_log(Command):
 
489
    """Show log of this branch.
 
490
 
 
491
    TODO: Option to limit range.
 
492
 
 
493
    TODO: Perhaps show most-recent first with an option for last.
 
494
    """
 
495
    takes_args = ['filename?']
 
496
    takes_options = ['timezone', 'verbose', 'show-ids']
 
497
    def run(self, filename=None, timezone='original', verbose=False, show_ids=False):
 
498
        from branch import find_branch
 
499
        b = find_branch((filename or '.'), lock_mode='r')
 
500
        if filename:
 
501
            filename = b.relpath(filename)
 
502
        bzrlib.show_log(b, filename,
 
503
                        show_timezone=timezone,
 
504
                        verbose=verbose,
 
505
                        show_ids=show_ids)
 
506
 
 
507
 
 
508
 
 
509
class cmd_touching_revisions(Command):
 
510
    """Return revision-ids which affected a particular file."""
 
511
    hidden = True
 
512
    takes_args = ["filename"]
 
513
    def run(self, filename):
 
514
        b = Branch(filename, lock_mode='r')
 
515
        inv = b.read_working_inventory()
 
516
        file_id = inv.path2id(b.relpath(filename))
 
517
        for revno, revision_id, what in bzrlib.log.find_touching_revisions(b, file_id):
 
518
            print "%6d %s" % (revno, what)
 
519
 
 
520
 
 
521
class cmd_ls(Command):
 
522
    """List files in a tree.
 
523
 
 
524
    TODO: Take a revision or remote path and list that tree instead.
 
525
    """
 
526
    hidden = True
 
527
    def run(self, revision=None, verbose=False):
 
528
        b = Branch('.')
 
529
        if revision == None:
 
530
            tree = b.working_tree()
 
531
        else:
 
532
            tree = b.revision_tree(b.lookup_revision(revision))
 
533
 
 
534
        for fp, fc, kind, fid in tree.list_files():
 
535
            if verbose:
 
536
                if kind == 'directory':
 
537
                    kindch = '/'
 
538
                elif kind == 'file':
 
539
                    kindch = ''
 
540
                else:
 
541
                    kindch = '???'
 
542
 
 
543
                print '%-8s %s%s' % (fc, fp, kindch)
 
544
            else:
 
545
                print fp
 
546
 
 
547
 
 
548
 
 
549
class cmd_unknowns(Command):
 
550
    """List unknown files"""
 
551
    def run(self):
 
552
        for f in Branch('.').unknowns():
 
553
            print quotefn(f)
 
554
 
 
555
 
 
556
 
 
557
class cmd_ignore(Command):
 
558
    """Ignore a command or pattern
 
559
 
 
560
    To remove patterns from the ignore list, edit the .bzrignore file.
 
561
 
 
562
    If the pattern contains a slash, it is compared to the whole path
 
563
    from the branch root.  Otherwise, it is comapred to only the last
 
564
    component of the path.
 
565
 
 
566
    Ignore patterns are case-insensitive on case-insensitive systems.
 
567
 
 
568
    Note: wildcards must be quoted from the shell on Unix.
 
569
 
 
570
    examples:
 
571
        bzr ignore ./Makefile
 
572
        bzr ignore '*.class'
 
573
    """
 
574
    takes_args = ['name_pattern']
 
575
    
 
576
    def run(self, name_pattern):
 
577
        from bzrlib.atomicfile import AtomicFile
 
578
        import codecs
 
579
 
 
580
        b = Branch('.')
 
581
        ifn = b.abspath('.bzrignore')
 
582
 
 
583
        # FIXME: probably doesn't handle non-ascii patterns
 
584
 
 
585
        if os.path.exists(ifn):
 
586
            f = b.controlfile(ifn, 'rt')
 
587
            igns = f.read()
 
588
            f.close()
 
589
        else:
 
590
            igns = ''
 
591
 
 
592
        if igns and igns[-1] != '\n':
 
593
            igns += '\n'
 
594
        igns += name_pattern + '\n'
 
595
 
 
596
        f = AtomicFile(ifn, 'wt')
 
597
        f.write(igns)
 
598
        f.commit()
 
599
 
 
600
        inv = b.working_tree().inventory
 
601
        if inv.path2id('.bzrignore'):
 
602
            mutter('.bzrignore is already versioned')
 
603
        else:
 
604
            mutter('need to make new .bzrignore file versioned')
 
605
            b.add(['.bzrignore'])
 
606
 
 
607
 
 
608
 
 
609
class cmd_ignored(Command):
 
610
    """List ignored files and the patterns that matched them.
 
611
 
 
612
    See also: bzr ignore"""
 
613
    def run(self):
 
614
        tree = Branch('.').working_tree()
 
615
        for path, file_class, kind, file_id in tree.list_files():
 
616
            if file_class != 'I':
 
617
                continue
 
618
            ## XXX: Slightly inefficient since this was already calculated
 
619
            pat = tree.is_ignored(path)
 
620
            print '%-50s %s' % (path, pat)
 
621
 
 
622
 
 
623
class cmd_lookup_revision(Command):
 
624
    """Lookup the revision-id from a revision-number
 
625
 
 
626
    example:
 
627
        bzr lookup-revision 33
 
628
    """
 
629
    hidden = True
 
630
    takes_args = ['revno']
 
631
    
 
632
    def run(self, revno):
 
633
        try:
 
634
            revno = int(revno)
 
635
        except ValueError:
 
636
            raise BzrCommandError("not a valid revision-number: %r" % revno)
 
637
 
 
638
        print Branch('.').lookup_revision(revno)
 
639
 
 
640
 
 
641
class cmd_export(Command):
 
642
    """Export past revision to destination directory.
 
643
 
 
644
    If no revision is specified this exports the last committed revision."""
 
645
    takes_args = ['dest']
 
646
    takes_options = ['revision']
 
647
    def run(self, dest, revision=None):
 
648
        b = Branch('.')
 
649
        if revision == None:
 
650
            rh = b.revision_history()[-1]
 
651
        else:
 
652
            rh = b.lookup_revision(int(revision))
 
653
        t = b.revision_tree(rh)
 
654
        t.export(dest)
 
655
 
 
656
 
 
657
class cmd_cat(Command):
 
658
    """Write a file's text from a previous revision."""
 
659
 
 
660
    takes_options = ['revision']
 
661
    takes_args = ['filename']
 
662
 
 
663
    def run(self, filename, revision=None):
 
664
        if revision == None:
 
665
            raise BzrCommandError("bzr cat requires a revision number")
 
666
        b = Branch('.')
 
667
        b.print_file(b.relpath(filename), int(revision))
 
668
 
 
669
 
 
670
class cmd_local_time_offset(Command):
 
671
    """Show the offset in seconds from GMT to local time."""
 
672
    hidden = True    
 
673
    def run(self):
 
674
        print bzrlib.osutils.local_time_offset()
 
675
 
 
676
 
 
677
 
 
678
class cmd_commit(Command):
 
679
    """Commit changes into a new revision.
 
680
 
 
681
    TODO: Commit only selected files.
 
682
 
 
683
    TODO: Run hooks on tree to-be-committed, and after commit.
 
684
 
 
685
    TODO: Strict commit that fails if there are unknown or deleted files.
 
686
    """
 
687
    takes_options = ['message', 'file', 'verbose']
 
688
    aliases = ['ci', 'checkin']
 
689
 
 
690
    def run(self, message=None, file=None, verbose=False):
 
691
        ## Warning: shadows builtin file()
 
692
        if not message and not file:
 
693
            raise BzrCommandError("please specify a commit message",
 
694
                                  ["use either --message or --file"])
 
695
        elif message and file:
 
696
            raise BzrCommandError("please specify either --message or --file")
 
697
        
 
698
        if file:
 
699
            import codecs
 
700
            message = codecs.open(file, 'rt', bzrlib.user_encoding).read()
 
701
 
 
702
        Branch('.').commit(message, verbose=verbose)
 
703
 
 
704
 
 
705
class cmd_check(Command):
 
706
    """Validate consistency of branch history.
 
707
 
 
708
    This command checks various invariants about the branch storage to
 
709
    detect data corruption or bzr bugs.
 
710
    """
 
711
    takes_args = ['dir?']
 
712
    def run(self, dir='.'):
 
713
        import bzrlib.check
 
714
        bzrlib.check.check(Branch(dir, find_root=False))
 
715
 
 
716
 
 
717
 
 
718
class cmd_whoami(Command):
 
719
    """Show bzr user id."""
 
720
    takes_options = ['email']
 
721
    
 
722
    def run(self, email=False):
 
723
        if email:
 
724
            print bzrlib.osutils.user_email()
 
725
        else:
 
726
            print bzrlib.osutils.username()
 
727
 
 
728
 
 
729
class cmd_selftest(Command):
 
730
    """Run internal test suite"""
 
731
    hidden = True
 
732
    def run(self):
 
733
        failures, tests = 0, 0
 
734
 
 
735
        import doctest, bzrlib.store, bzrlib.tests
 
736
        bzrlib.trace.verbose = False
 
737
 
 
738
        for m in bzrlib.store, bzrlib.inventory, bzrlib.branch, bzrlib.osutils, \
 
739
            bzrlib.tree, bzrlib.tests, bzrlib.commands, bzrlib.add:
 
740
            mf, mt = doctest.testmod(m)
 
741
            failures += mf
 
742
            tests += mt
 
743
            print '%-40s %3d tests' % (m.__name__, mt),
 
744
            if mf:
 
745
                print '%3d FAILED!' % mf
 
746
            else:
 
747
                print
 
748
 
 
749
        print '%-40s %3d tests' % ('total', tests),
 
750
        if failures:
 
751
            print '%3d FAILED!' % failures
 
752
        else:
 
753
            print
 
754
 
 
755
 
 
756
 
 
757
class cmd_version(Command):
 
758
    """Show version of bzr"""
 
759
    def run(self):
 
760
        show_version()
 
761
 
 
762
def show_version():
 
763
    print "bzr (bazaar-ng) %s" % bzrlib.__version__
 
764
    print bzrlib.__copyright__
 
765
    print "http://bazaar-ng.org/"
 
766
    print
 
767
    print "bzr comes with ABSOLUTELY NO WARRANTY.  bzr is free software, and"
 
768
    print "you may use, modify and redistribute it under the terms of the GNU"
 
769
    print "General Public License version 2 or later."
 
770
 
 
771
 
 
772
class cmd_rocks(Command):
 
773
    """Statement of optimism."""
 
774
    hidden = True
 
775
    def run(self):
 
776
        print "it sure does!"
 
777
 
 
778
 
 
779
class cmd_assert_fail(Command):
 
780
    """Test reporting of assertion failures"""
 
781
    hidden = True
 
782
    def run(self):
 
783
        assert False, "always fails"
 
784
 
 
785
 
 
786
class cmd_help(Command):
 
787
    """Show help on a command or other topic.
 
788
 
 
789
    For a list of all available commands, say 'bzr help commands'."""
 
790
    takes_args = ['topic?']
 
791
    aliases = ['?']
 
792
    
 
793
    def run(self, topic=None):
 
794
        import help
 
795
        help.help(topic)
 
796
 
 
797
 
 
798
######################################################################
 
799
# main routine
 
800
 
 
801
 
 
802
# list of all available options; the rhs can be either None for an
 
803
# option that takes no argument, or a constructor function that checks
 
804
# the type.
 
805
OPTIONS = {
 
806
    'all':                    None,
 
807
    'help':                   None,
 
808
    'file':                   unicode,
 
809
    'message':                unicode,
 
810
    'profile':                None,
 
811
    'revision':               int,
 
812
    'show-ids':               None,
 
813
    'timezone':               str,
 
814
    'verbose':                None,
 
815
    'version':                None,
 
816
    'email':                  None,
 
817
    }
 
818
 
 
819
SHORT_OPTIONS = {
 
820
    'm':                      'message',
 
821
    'F':                      'file', 
 
822
    'r':                      'revision',
 
823
    'v':                      'verbose',
 
824
}
 
825
 
 
826
 
 
827
def parse_args(argv):
350
828
    """Parse command line.
351
829
    
352
830
    Arguments and options are parsed at this level before being passed
353
831
    down to specific command handlers.  This routine knows, from a
354
832
    lookup table, something about the available options, what optargs
355
833
    they take, and which commands will accept them.
 
834
 
 
835
    >>> parse_args('--help'.split())
 
836
    ([], {'help': True})
 
837
    >>> parse_args('--version'.split())
 
838
    ([], {'version': True})
 
839
    >>> parse_args('status --all'.split())
 
840
    (['status'], {'all': True})
 
841
    >>> parse_args('commit --message=biter'.split())
 
842
    (['commit'], {'message': u'biter'})
356
843
    """
357
 
    # TODO: chop up this beast; make it a method of the Command
358
844
    args = []
359
845
    opts = {}
360
 
    alias_opts = {}
361
 
 
362
 
    cmd_options = command.options()
363
 
    argsover = False
364
 
    proc_aliasarg = True # Are we processing alias_argv now?
365
 
    for proc_argv in alias_argv, argv:
366
 
        while proc_argv:
367
 
            a = proc_argv.pop(0)
368
 
            if argsover:
369
 
                args.append(a)
370
 
                continue
371
 
            elif a == '--':
372
 
                # We've received a standalone -- No more flags
373
 
                argsover = True
374
 
                continue
375
 
            if a[0] == '-':
376
 
                # option names must not be unicode
377
 
                a = str(a)
378
 
                optarg = None
379
 
                if a[1] == '-':
380
 
                    mutter("  got option %r", a)
381
 
                    if '=' in a:
382
 
                        optname, optarg = a[2:].split('=', 1)
383
 
                    else:
384
 
                        optname = a[2:]
385
 
                    if optname not in cmd_options:
386
 
                        raise BzrCommandError('unknown option "%s"' % a)
387
 
                else:
388
 
                    shortopt = a[1:]
389
 
                    if shortopt in Option.SHORT_OPTIONS:
390
 
                        # Multi-character options must have a space to delimit
391
 
                        # their value
392
 
                        # ^^^ what does this mean? mbp 20051014
393
 
                        optname = Option.SHORT_OPTIONS[shortopt].name
394
 
                    else:
395
 
                        # Single character short options, can be chained,
396
 
                        # and have their value appended to their name
397
 
                        shortopt = a[1:2]
398
 
                        if shortopt not in Option.SHORT_OPTIONS:
399
 
                            # We didn't find the multi-character name, and we
400
 
                            # didn't find the single char name
401
 
                            raise BzrCommandError('unknown option "%s"' % a)
402
 
                        optname = Option.SHORT_OPTIONS[shortopt].name
403
 
 
404
 
                        if a[2:]:
405
 
                            # There are extra things on this option
406
 
                            # see if it is the value, or if it is another
407
 
                            # short option
408
 
                            optargfn = Option.OPTIONS[optname].type
409
 
                            if optargfn is None:
410
 
                                # This option does not take an argument, so the
411
 
                                # next entry is another short option, pack it
412
 
                                # back into the list
413
 
                                proc_argv.insert(0, '-' + a[2:])
414
 
                            else:
415
 
                                # This option takes an argument, so pack it
416
 
                                # into the array
417
 
                                optarg = a[2:]
418
 
                    if optname not in cmd_options:
419
 
                        raise BzrCommandError('unknown option "%s"' % shortopt)
420
 
                if optname in opts:
421
 
                    # XXX: Do we ever want to support this, e.g. for -r?
422
 
                    if proc_aliasarg:
423
 
                        raise BzrCommandError('repeated option %r' % a)
424
 
                    elif optname in alias_opts:
425
 
                        # Replace what's in the alias with what's in the real
426
 
                        # argument
427
 
                        del alias_opts[optname]
428
 
                        del opts[optname]
429
 
                        proc_argv.insert(0, a)
430
 
                        continue
431
 
                    else:
432
 
                        raise BzrCommandError('repeated option %r' % a)
433
 
                    
434
 
                option_obj = cmd_options[optname]
435
 
                optargfn = option_obj.type
436
 
                if optargfn:
437
 
                    if optarg == None:
438
 
                        if not proc_argv:
439
 
                            raise BzrCommandError('option %r needs an argument' % a)
440
 
                        else:
441
 
                            optarg = proc_argv.pop(0)
442
 
                    opts[optname] = optargfn(optarg)
443
 
                    if proc_aliasarg:
444
 
                        alias_opts[optname] = optargfn(optarg)
445
 
                else:
446
 
                    if optarg != None:
447
 
                        raise BzrCommandError('option %r takes no argument' % optname)
448
 
                    opts[optname] = True
449
 
                    if proc_aliasarg:
450
 
                        alias_opts[optname] = True
451
 
            else:
452
 
                args.append(a)
453
 
        proc_aliasarg = False # Done with alias argv
 
846
 
 
847
    # TODO: Maybe handle '--' to end options?
 
848
 
 
849
    while argv:
 
850
        a = argv.pop(0)
 
851
        if a[0] == '-':
 
852
            # option names must not be unicode
 
853
            a = str(a)
 
854
            optarg = None
 
855
            if a[1] == '-':
 
856
                mutter("  got option %r" % a)
 
857
                if '=' in a:
 
858
                    optname, optarg = a[2:].split('=', 1)
 
859
                else:
 
860
                    optname = a[2:]
 
861
                if optname not in OPTIONS:
 
862
                    bailout('unknown long option %r' % a)
 
863
            else:
 
864
                shortopt = a[1:]
 
865
                if shortopt not in SHORT_OPTIONS:
 
866
                    bailout('unknown short option %r' % a)
 
867
                optname = SHORT_OPTIONS[shortopt]
 
868
            
 
869
            if optname in opts:
 
870
                # XXX: Do we ever want to support this, e.g. for -r?
 
871
                bailout('repeated option %r' % a)
 
872
                
 
873
            optargfn = OPTIONS[optname]
 
874
            if optargfn:
 
875
                if optarg == None:
 
876
                    if not argv:
 
877
                        bailout('option %r needs an argument' % a)
 
878
                    else:
 
879
                        optarg = argv.pop(0)
 
880
                opts[optname] = optargfn(optarg)
 
881
            else:
 
882
                if optarg != None:
 
883
                    bailout('option %r takes no argument' % optname)
 
884
                opts[optname] = True
 
885
        else:
 
886
            args.append(a)
 
887
 
454
888
    return args, opts
455
889
 
456
890
 
 
891
 
 
892
 
457
893
def _match_argform(cmd, takes_args, args):
458
894
    argdict = {}
459
895
 
481
917
                raise BzrCommandError("command %r needs one or more %s"
482
918
                        % (cmd, argname.upper()))
483
919
            argdict[argname + '_list'] = args[:-1]
484
 
            args[:-1] = []
 
920
            args[:-1] = []                
485
921
        else:
486
922
            # just a plain arg
487
923
            argname = ap
499
935
 
500
936
 
501
937
 
502
 
def apply_profiled(the_callable, *args, **kwargs):
503
 
    import hotshot
504
 
    import tempfile
505
 
    import hotshot.stats
506
 
    pffileno, pfname = tempfile.mkstemp()
507
 
    try:
508
 
        prof = hotshot.Profile(pfname)
509
 
        try:
510
 
            ret = prof.runcall(the_callable, *args, **kwargs) or 0
511
 
        finally:
512
 
            prof.close()
513
 
        stats = hotshot.stats.load(pfname)
514
 
        stats.strip_dirs()
515
 
        stats.sort_stats('cum')   # 'time'
516
 
        ## XXX: Might like to write to stderr or the trace file instead but
517
 
        ## print_stats seems hardcoded to stdout
518
 
        stats.print_stats(20)
519
 
        return ret
520
 
    finally:
521
 
        os.close(pffileno)
522
 
        os.remove(pfname)
523
 
 
524
 
 
525
 
def apply_lsprofiled(filename, the_callable, *args, **kwargs):
526
 
    from bzrlib.lsprof import profile
527
 
    import cPickle
528
 
    ret, stats = profile(the_callable, *args, **kwargs)
529
 
    stats.sort()
530
 
    if filename is None:
531
 
        stats.pprint()
532
 
    else:
533
 
        stats.freeze()
534
 
        cPickle.dump(stats, open(filename, 'w'), 2)
535
 
        print 'Profile data written to %r.' % filename
536
 
    return ret
537
 
 
538
 
 
539
 
def get_alias(cmd):
540
 
    """Return an expanded alias, or None if no alias exists"""
541
 
    import bzrlib.config
542
 
    alias = bzrlib.config.GlobalConfig().get_alias(cmd)
543
 
    if (alias):
544
 
        return alias.split(' ')
545
 
    return None
546
 
 
547
 
 
548
938
def run_bzr(argv):
549
939
    """Execute a command.
550
940
 
551
941
    This is similar to main(), but without all the trappings for
552
942
    logging and error handling.  
553
 
    
554
 
    argv
555
 
       The command-line arguments, without the program name from argv[0]
556
 
       These should already be decoded. All library/test code calling
557
 
       run_bzr should be passing valid strings (don't need decoding).
558
 
    
559
 
    Returns a command status or raises an exception.
560
 
 
561
 
    Special master options: these must come before the command because
562
 
    they control how the command is interpreted.
563
 
 
564
 
    --no-plugins
565
 
        Do not load plugin modules at all
566
 
 
567
 
    --no-aliases
568
 
        Do not allow aliases
569
 
 
570
 
    --builtin
571
 
        Only use builtin commands.  (Plugins are still allowed to change
572
 
        other behaviour.)
573
 
 
574
 
    --profile
575
 
        Run under the Python hotshot profiler.
576
 
 
577
 
    --lsprof
578
 
        Run under the Python lsprof profiler.
579
943
    """
580
 
    argv = list(argv)
581
 
 
582
 
    opt_lsprof = opt_profile = opt_no_plugins = opt_builtin =  \
583
 
                opt_no_aliases = False
584
 
    opt_lsprof_file = None
585
 
 
586
 
    # --no-plugins is handled specially at a very early stage. We need
587
 
    # to load plugins before doing other command parsing so that they
588
 
    # can override commands, but this needs to happen first.
589
 
 
590
 
    argv_copy = []
591
 
    i = 0
592
 
    while i < len(argv):
593
 
        a = argv[i]
594
 
        if a == '--profile':
595
 
            opt_profile = True
596
 
        elif a == '--lsprof':
597
 
            opt_lsprof = True
598
 
        elif a == '--lsprof-file':
599
 
            opt_lsprof_file = argv[i + 1]
600
 
            i += 1
601
 
        elif a == '--no-plugins':
602
 
            opt_no_plugins = True
603
 
        elif a == '--no-aliases':
604
 
            opt_no_aliases = True
605
 
        elif a == '--builtin':
606
 
            opt_builtin = True
607
 
        elif a in ('--quiet', '-q'):
608
 
            be_quiet()
609
 
        else:
610
 
            argv_copy.append(a)
611
 
        i += 1
612
 
 
613
 
    argv = argv_copy
614
 
    if (not argv):
615
 
        from bzrlib.builtins import cmd_help
616
 
        cmd_help().run_argv_aliases([])
617
 
        return 0
618
 
 
619
 
    if argv[0] == '--version':
620
 
        from bzrlib.builtins import show_version
621
 
        show_version()
622
 
        return 0
623
 
        
624
 
    if not opt_no_plugins:
625
 
        from bzrlib.plugin import load_plugins
626
 
        load_plugins()
627
 
    else:
628
 
        from bzrlib.plugin import disable_plugins
629
 
        disable_plugins()
630
 
 
631
 
    alias_argv = None
632
 
 
633
 
    if not opt_no_aliases:
634
 
        alias_argv = get_alias(argv[0])
635
 
        if alias_argv:
636
 
            alias_argv = [a.decode(bzrlib.user_encoding) for a in alias_argv]
637
 
            argv[0] = alias_argv.pop(0)
638
 
 
639
 
    cmd = str(argv.pop(0))
640
 
 
641
 
    cmd_obj = get_cmd_object(cmd, plugins_override=not opt_builtin)
642
 
    if not getattr(cmd_obj.run_argv, 'is_deprecated', False):
643
 
        run = cmd_obj.run_argv
644
 
        run_argv = [argv]
645
 
    else:
646
 
        run = cmd_obj.run_argv_aliases
647
 
        run_argv = [argv, alias_argv]
648
 
 
 
944
    argv = [a.decode(bzrlib.user_encoding) for a in argv]
 
945
    
649
946
    try:
650
 
        if opt_lsprof:
651
 
            ret = apply_lsprofiled(opt_lsprof_file, run, *run_argv)
652
 
        elif opt_profile:
653
 
            ret = apply_profiled(run, *run_argv)
654
 
        else:
655
 
            ret = run(*run_argv)
656
 
        return ret or 0
657
 
    finally:
658
 
        # reset, in case we may do other commands later within the same process
659
 
        be_quiet(False)
660
 
 
661
 
def display_command(func):
662
 
    """Decorator that suppresses pipe/interrupt errors."""
663
 
    def ignore_pipe(*args, **kwargs):
 
947
        args, opts = parse_args(argv[1:])
 
948
        if 'help' in opts:
 
949
            import help
 
950
            if args:
 
951
                help.help(args[0])
 
952
            else:
 
953
                help.help()
 
954
            return 0
 
955
        elif 'version' in opts:
 
956
            show_version()
 
957
            return 0
 
958
        cmd = str(args.pop(0))
 
959
    except IndexError:
 
960
        log_error('usage: bzr COMMAND')
 
961
        log_error('  try "bzr help"')
 
962
        return 1
 
963
 
 
964
    canonical_cmd, cmd_class = get_cmd_class(cmd)
 
965
 
 
966
    # global option
 
967
    if 'profile' in opts:
 
968
        profile = True
 
969
        del opts['profile']
 
970
    else:
 
971
        profile = False
 
972
 
 
973
    # check options are reasonable
 
974
    allowed = cmd_class.takes_options
 
975
    for oname in opts:
 
976
        if oname not in allowed:
 
977
            raise BzrCommandError("option '--%s' is not allowed for command %r"
 
978
                                  % (oname, cmd))
 
979
 
 
980
    # mix arguments and options into one dictionary
 
981
    cmdargs = _match_argform(cmd, cmd_class.takes_args, args)
 
982
    cmdopts = {}
 
983
    for k, v in opts.items():
 
984
        cmdopts[k.replace('-', '_')] = v
 
985
 
 
986
    if profile:
 
987
        import hotshot, tempfile
 
988
        pffileno, pfname = tempfile.mkstemp()
664
989
        try:
665
 
            result = func(*args, **kwargs)
666
 
            sys.stdout.flush()
667
 
            return result
668
 
        except IOError, e:
669
 
            if not hasattr(e, 'errno'):
670
 
                raise
671
 
            if e.errno != errno.EPIPE:
672
 
                # Win32 raises IOError with errno=0 on a broken pipe
673
 
                if sys.platform != 'win32' or e.errno != 0:
674
 
                    raise
675
 
            pass
676
 
        except KeyboardInterrupt:
677
 
            pass
678
 
    return ignore_pipe
 
990
            prof = hotshot.Profile(pfname)
 
991
            ret = prof.runcall(cmd_class, cmdopts, cmdargs) or 0
 
992
            prof.close()
 
993
 
 
994
            import hotshot.stats
 
995
            stats = hotshot.stats.load(pfname)
 
996
            #stats.strip_dirs()
 
997
            stats.sort_stats('time')
 
998
            ## XXX: Might like to write to stderr or the trace file instead but
 
999
            ## print_stats seems hardcoded to stdout
 
1000
            stats.print_stats(20)
 
1001
            
 
1002
            return ret.status
 
1003
 
 
1004
        finally:
 
1005
            os.close(pffileno)
 
1006
            os.remove(pfname)
 
1007
    else:
 
1008
        cmdobj = cmd_class(cmdopts, cmdargs).status 
 
1009
 
 
1010
 
 
1011
def _report_exception(summary, quiet=False):
 
1012
    import traceback
 
1013
    log_error('bzr: ' + summary)
 
1014
    bzrlib.trace.log_exception()
 
1015
 
 
1016
    if not quiet:
 
1017
        tb = sys.exc_info()[2]
 
1018
        exinfo = traceback.extract_tb(tb)
 
1019
        if exinfo:
 
1020
            sys.stderr.write('  at %s:%d in %s()\n' % exinfo[-1][:3])
 
1021
        sys.stderr.write('  see ~/.bzr.log for debug information\n')
 
1022
 
679
1023
 
680
1024
 
681
1025
def main(argv):
682
 
    import bzrlib.ui
683
 
    from bzrlib.ui.text import TextUIFactory
684
 
    bzrlib.ui.ui_factory = TextUIFactory()
685
 
    argv = [a.decode(bzrlib.user_encoding) for a in argv[1:]]
686
 
    ret = run_bzr_catch_errors(argv)
687
 
    mutter("return code %d", ret)
688
 
    return ret
689
 
 
690
 
 
691
 
def run_bzr_catch_errors(argv):
 
1026
    import errno
 
1027
    
 
1028
    bzrlib.open_tracefile(argv)
 
1029
 
692
1030
    try:
693
 
        return run_bzr(argv)
694
 
        # do this here inside the exception wrappers to catch EPIPE
695
 
        sys.stdout.flush()
696
 
    except Exception, e:
697
 
        # used to handle AssertionError and KeyboardInterrupt
698
 
        # specially here, but hopefully they're handled ok by the logger now
699
 
        bzrlib.trace.report_exception(sys.exc_info(), sys.stderr)
700
 
        if os.environ.get('BZR_PDB'):
701
 
            print '**** entering debugger'
702
 
            import pdb
703
 
            pdb.post_mortem(sys.exc_traceback)
704
 
        return 3
 
1031
        try:
 
1032
            try:
 
1033
                return run_bzr(argv)
 
1034
            finally:
 
1035
                # do this here inside the exception wrappers to catch EPIPE
 
1036
                sys.stdout.flush()
 
1037
        except BzrError, e:
 
1038
            quiet = isinstance(e, (BzrCommandError))
 
1039
            _report_exception('error: ' + e.args[0], quiet=quiet)
 
1040
            if len(e.args) > 1:
 
1041
                for h in e.args[1]:
 
1042
                    # some explanation or hints
 
1043
                    log_error('  ' + h)
 
1044
            return 1
 
1045
        except AssertionError, e:
 
1046
            msg = 'assertion failed'
 
1047
            if str(e):
 
1048
                msg += ': ' + str(e)
 
1049
            _report_exception(msg)
 
1050
            return 2
 
1051
        except KeyboardInterrupt, e:
 
1052
            _report_exception('interrupted', quiet=True)
 
1053
            return 2
 
1054
        except Exception, e:
 
1055
            quiet = False
 
1056
            if (isinstance(e, IOError) 
 
1057
                and hasattr(e, 'errno')
 
1058
                and e.errno == errno.EPIPE):
 
1059
                quiet = True
 
1060
                msg = 'broken pipe'
 
1061
            else:
 
1062
                msg = str(e).rstrip('\n')
 
1063
            _report_exception(msg, quiet)
 
1064
            return 2
 
1065
    finally:
 
1066
        bzrlib.trace.close_trace()
 
1067
 
705
1068
 
706
1069
if __name__ == '__main__':
707
1070
    sys.exit(main(sys.argv))