/brz/remove-bazaar

To get this branch, use:
bzr branch http://gegoxaren.bato24.eu/bzr/brz/remove-bazaar

« 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 codecs
32
 
import errno
33
 
import os
34
 
from warnings import warn
35
 
import sys
 
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 import option
44
 
from bzrlib.option import Option
45
 
import bzrlib.osutils
46
 
from bzrlib.symbol_versioning import (deprecated_method, zero_eight)
47
 
import bzrlib.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 k_unsquished not in plugin_cmds:
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
 
    # We want only 'ascii' command names, but the user may have typed
133
 
    # in a Unicode name. In that case, they should just get a
134
 
    # 'command not found' error later.
135
 
    # In the future, we may actually support Unicode command names.
 
49
    cmd = str(cmd)                      # not unicode
136
50
 
137
51
    # first look up this command under the specified name
138
 
    cmds = _get_cmd_dict(plugins_override=plugins_override)
139
52
    try:
140
 
        return cmds[cmd_name]()
 
53
        return cmd, globals()[_squish_command_name(cmd)]
141
54
    except KeyError:
142
55
        pass
143
56
 
144
57
    # look for any command which claims this as an alias
145
 
    for real_cmd_name, cmd_class in cmds.iteritems():
146
 
        if cmd_name in cmd_class.aliases:
147
 
            return cmd_class()
148
 
 
149
 
    cmd_obj = ExternalCommand.find_command(cmd_name)
150
 
    if cmd_obj:
151
 
        return cmd_obj
152
 
 
153
 
    raise BzrCommandError('unknown command "%s"' % cmd_name)
154
 
 
155
 
 
156
 
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:
157
70
    """Base class for commands.
158
71
 
159
 
    Commands are the heart of the command-line bzr interface.
160
 
 
161
 
    The command object mostly handles the mapping of command-line
162
 
    parameters into one or more bzrlib operations, and of the results
163
 
    into textual output.
164
 
 
165
 
    Commands normally don't have any state.  All their arguments are
166
 
    passed in to the run method.  (Subclasses may take a different
167
 
    policy if the behaviour of the instance needs to depend on e.g. a
168
 
    shell plugin and not just its Python class.)
169
 
 
170
72
    The docstring for an actual command should give a single-line
171
73
    summary, then a complete description of the command.  A grammar
172
74
    description will be inserted.
173
75
 
174
 
    aliases
175
 
        Other accepted names for this command.
176
 
 
177
76
    takes_args
178
77
        List of argument forms, marked with whether they are optional,
179
78
        repeated, etc.
180
79
 
181
 
                Examples:
182
 
 
183
 
                ['to_location', 'from_branch?', 'file*']
184
 
 
185
 
                'to_location' is required
186
 
                'from_branch' is optional
187
 
                'file' can be specified 0 or more times
188
 
 
189
80
    takes_options
190
 
        List of options that may be given for this command.  These can
191
 
        be either strings, referring to globally-defined options,
192
 
        or option objects.  Retrieve through options().
 
81
        List of options that may be given for this command.
193
82
 
194
83
    hidden
195
 
        If true, this command isn't advertised.  This is typically
196
 
        for commands intended for expert users.
197
 
 
198
 
    encoding_type
199
 
        Command objects will get a 'outf' attribute, which has been
200
 
        setup to properly handle encoding of unicode strings.
201
 
        encoding_type determines what will happen when characters cannot
202
 
        be encoded
203
 
            strict - abort if we cannot decode
204
 
            replace - put in a bogus character (typically '?')
205
 
            exact - do not encode sys.stdout
206
 
 
 
84
        If true, this command isn't advertised.
207
85
    """
208
86
    aliases = []
 
87
    
209
88
    takes_args = []
210
89
    takes_options = []
211
 
    encoding_type = 'strict'
212
90
 
213
91
    hidden = False
214
92
    
215
 
    def __init__(self):
216
 
        """Construct an instance of this command."""
217
 
        if self.__doc__ == Command.__doc__:
218
 
            warn("No help message set for %r" % self)
219
 
 
220
 
    def options(self):
221
 
        """Return dict of valid options for this command.
222
 
 
223
 
        Maps from long option name to option object."""
224
 
        r = dict()
225
 
        r['help'] = Option.OPTIONS['help']
226
 
        for o in self.takes_options:
227
 
            if isinstance(o, basestring):
228
 
                o = Option.OPTIONS[o]
229
 
            r[o.name] = o
230
 
        return r
231
 
 
232
 
    def _setup_outf(self):
233
 
        """Return a file linked to stdout, which has proper encoding."""
234
 
        assert self.encoding_type in ['strict', 'exact', 'replace']
235
 
 
236
 
        # Originally I was using self.stdout, but that looks
237
 
        # *way* too much like sys.stdout
238
 
        if self.encoding_type == 'exact':
239
 
            self.outf = sys.stdout
240
 
            return
241
 
 
242
 
        output_encoding = bzrlib.osutils.get_terminal_encoding()
243
 
 
244
 
        # use 'replace' so that we don't abort if trying to write out
245
 
        # in e.g. the default C locale.
246
 
        self.outf = codecs.getwriter(output_encoding)(sys.stdout, errors=self.encoding_type)
247
 
        # For whatever reason codecs.getwriter() does not advertise its encoding
248
 
        # it just returns the encoding of the wrapped file, which is completely
249
 
        # bogus. So set the attribute, so we can find the correct encoding later.
250
 
        self.outf.encoding = output_encoding
251
 
 
252
 
    @deprecated_method(zero_eight)
253
 
    def run_argv(self, argv):
254
 
        """Parse command line and run.
255
 
        
256
 
        See run_argv_aliases for the 0.8 and beyond api.
257
 
        """
258
 
        return self.run_argv_aliases(argv)
259
 
 
260
 
    def run_argv_aliases(self, argv, alias_argv=None):
261
 
        """Parse the command line and run with extra aliases in alias_argv."""
262
 
        if argv is None:
263
 
            warn("Passing None for [] is deprecated from bzrlib 0.10", 
264
 
                 DeprecationWarning, stacklevel=2)
265
 
            argv = []
266
 
        args, opts = parse_args(self, argv, alias_argv)
267
 
        if 'help' in opts:  # e.g. bzr add --help
268
 
            from bzrlib.help import help_on_command
269
 
            help_on_command(self.name())
270
 
            return 0
271
 
        # mix arguments and options into one dictionary
272
 
        cmdargs = _match_argform(self.name(), self.takes_args, args)
273
 
        cmdopts = {}
274
 
        for k, v in opts.items():
275
 
            cmdopts[k.replace('-', '_')] = v
276
 
 
277
 
        all_cmd_args = cmdargs.copy()
278
 
        all_cmd_args.update(cmdopts)
279
 
 
280
 
        self._setup_outf()
281
 
 
282
 
        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
 
283
105
    
284
106
    def run(self):
285
 
        """Actually run the command.
 
107
        """Override this in sub-classes.
286
108
 
287
109
        This is invoked with the options and arguments bound to
288
110
        keyword parameters.
289
111
 
290
 
        Return 0 or None if the command was successful, or a non-zero
291
 
        shell error code if not.  It's OK for this method to allow
292
 
        an exception to raise up.
293
 
        """
294
 
        raise NotImplementedError('no implementation of command %r' 
295
 
                                  % self.name())
296
 
 
297
 
    def help(self):
298
 
        """Return help message for this class."""
299
 
        from inspect import getdoc
300
 
        if self.__doc__ is Command.__doc__:
301
 
            return None
302
 
        return getdoc(self)
303
 
 
304
 
    def name(self):
305
 
        return _unsquish_command_name(self.__class__.__name__)
306
 
 
307
 
    def plugin_name(self):
308
 
        """Get the name of the plugin that provides this command.
309
 
 
310
 
        :return: The name of the plugin or None if the command is builtin.
311
 
        """
312
 
        mod_parts = self.__module__.split('.')
313
 
        if len(mod_parts) >= 3 and mod_parts[1] == 'plugins':
314
 
            return mod_parts[2]
315
 
        else:
316
 
            return None
317
 
 
318
 
 
319
 
def parse_spec(spec):
320
 
    """
321
 
    >>> parse_spec(None)
322
 
    [None, None]
323
 
    >>> parse_spec("./")
324
 
    ['./', None]
325
 
    >>> parse_spec("../@")
326
 
    ['..', -1]
327
 
    >>> parse_spec("../f/@35")
328
 
    ['../f', 35]
329
 
    >>> parse_spec('./@revid:john@arbash-meinel.com-20050711044610-3ca0327c6a222f67')
330
 
    ['.', 'revid:john@arbash-meinel.com-20050711044610-3ca0327c6a222f67']
331
 
    """
332
 
    if spec is None:
333
 
        return [None, None]
334
 
    if '/@' in spec:
335
 
        parsed = spec.split('/@')
336
 
        assert len(parsed) == 2
337
 
        if parsed[1] == "":
338
 
            parsed[1] = -1
339
 
        else:
340
 
            try:
341
 
                parsed[1] = int(parsed[1])
342
 
            except ValueError:
343
 
                pass # We can allow stuff like ./@revid:blahblahblah
344
 
            else:
345
 
                assert parsed[1] >=0
346
 
    else:
347
 
        parsed = [spec, None]
348
 
    return parsed
349
 
 
350
 
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):
351
828
    """Parse command line.
352
829
    
353
830
    Arguments and options are parsed at this level before being passed
354
831
    down to specific command handlers.  This routine knows, from a
355
832
    lookup table, something about the available options, what optargs
356
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'})
357
843
    """
358
 
    # TODO: make it a method of the Command?
359
 
    parser = option.get_optparser(command.options())
360
 
    if alias_argv is not None:
361
 
        args = alias_argv + argv
362
 
    else:
363
 
        args = argv
364
 
 
365
 
    options, args = parser.parse_args(args)
366
 
    opts = dict([(k, v) for k, v in options.__dict__.iteritems() if 
367
 
                 v is not option.OptionParser.DEFAULT_VALUE])
 
844
    args = []
 
845
    opts = {}
 
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
 
368
888
    return args, opts
369
889
 
370
890
 
 
891
 
 
892
 
371
893
def _match_argform(cmd, takes_args, args):
372
894
    argdict = {}
373
895
 
395
917
                raise BzrCommandError("command %r needs one or more %s"
396
918
                        % (cmd, argname.upper()))
397
919
            argdict[argname + '_list'] = args[:-1]
398
 
            args[:-1] = []
 
920
            args[:-1] = []                
399
921
        else:
400
922
            # just a plain arg
401
923
            argname = ap
413
935
 
414
936
 
415
937
 
416
 
def apply_profiled(the_callable, *args, **kwargs):
417
 
    import hotshot
418
 
    import tempfile
419
 
    import hotshot.stats
420
 
    pffileno, pfname = tempfile.mkstemp()
421
 
    try:
422
 
        prof = hotshot.Profile(pfname)
423
 
        try:
424
 
            ret = prof.runcall(the_callable, *args, **kwargs) or 0
425
 
        finally:
426
 
            prof.close()
427
 
        stats = hotshot.stats.load(pfname)
428
 
        stats.strip_dirs()
429
 
        stats.sort_stats('cum')   # 'time'
430
 
        ## XXX: Might like to write to stderr or the trace file instead but
431
 
        ## print_stats seems hardcoded to stdout
432
 
        stats.print_stats(20)
433
 
        return ret
434
 
    finally:
435
 
        os.close(pffileno)
436
 
        os.remove(pfname)
437
 
 
438
 
 
439
 
def apply_lsprofiled(filename, the_callable, *args, **kwargs):
440
 
    from bzrlib.lsprof import profile
441
 
    import cPickle
442
 
    ret, stats = profile(the_callable, *args, **kwargs)
443
 
    stats.sort()
444
 
    if filename is None:
445
 
        stats.pprint()
446
 
    else:
447
 
        stats.freeze()
448
 
        cPickle.dump(stats, open(filename, 'w'), 2)
449
 
        print 'Profile data written to %r.' % filename
450
 
    return ret
451
 
 
452
 
 
453
 
def get_alias(cmd):
454
 
    """Return an expanded alias, or None if no alias exists"""
455
 
    import bzrlib.config
456
 
    alias = bzrlib.config.GlobalConfig().get_alias(cmd)
457
 
    if (alias):
458
 
        return alias.split(' ')
459
 
    return None
460
 
 
461
 
 
462
938
def run_bzr(argv):
463
939
    """Execute a command.
464
940
 
465
941
    This is similar to main(), but without all the trappings for
466
942
    logging and error handling.  
467
 
    
468
 
    argv
469
 
       The command-line arguments, without the program name from argv[0]
470
 
       These should already be decoded. All library/test code calling
471
 
       run_bzr should be passing valid strings (don't need decoding).
472
 
    
473
 
    Returns a command status or raises an exception.
474
 
 
475
 
    Special master options: these must come before the command because
476
 
    they control how the command is interpreted.
477
 
 
478
 
    --no-plugins
479
 
        Do not load plugin modules at all
480
 
 
481
 
    --no-aliases
482
 
        Do not allow aliases
483
 
 
484
 
    --builtin
485
 
        Only use builtin commands.  (Plugins are still allowed to change
486
 
        other behaviour.)
487
 
 
488
 
    --profile
489
 
        Run under the Python hotshot profiler.
490
 
 
491
 
    --lsprof
492
 
        Run under the Python lsprof profiler.
493
943
    """
494
 
    argv = list(argv)
495
 
 
496
 
    opt_lsprof = opt_profile = opt_no_plugins = opt_builtin =  \
497
 
                opt_no_aliases = False
498
 
    opt_lsprof_file = None
499
 
 
500
 
    # --no-plugins is handled specially at a very early stage. We need
501
 
    # to load plugins before doing other command parsing so that they
502
 
    # can override commands, but this needs to happen first.
503
 
 
504
 
    argv_copy = []
505
 
    i = 0
506
 
    while i < len(argv):
507
 
        a = argv[i]
508
 
        if a == '--profile':
509
 
            opt_profile = True
510
 
        elif a == '--lsprof':
511
 
            opt_lsprof = True
512
 
        elif a == '--lsprof-file':
513
 
            opt_lsprof = True
514
 
            opt_lsprof_file = argv[i + 1]
515
 
            i += 1
516
 
        elif a == '--no-plugins':
517
 
            opt_no_plugins = True
518
 
        elif a == '--no-aliases':
519
 
            opt_no_aliases = True
520
 
        elif a == '--builtin':
521
 
            opt_builtin = True
522
 
        elif a in ('--quiet', '-q'):
523
 
            be_quiet()
524
 
        else:
525
 
            argv_copy.append(a)
526
 
        i += 1
527
 
 
528
 
    argv = argv_copy
529
 
    if (not argv):
530
 
        from bzrlib.builtins import cmd_help
531
 
        cmd_help().run_argv_aliases([])
532
 
        return 0
533
 
 
534
 
    if argv[0] == '--version':
535
 
        from bzrlib.version import show_version
536
 
        show_version()
537
 
        return 0
538
 
        
539
 
    if not opt_no_plugins:
540
 
        from bzrlib.plugin import load_plugins
541
 
        load_plugins()
542
 
    else:
543
 
        from bzrlib.plugin import disable_plugins
544
 
        disable_plugins()
545
 
 
546
 
    alias_argv = None
547
 
 
548
 
    if not opt_no_aliases:
549
 
        alias_argv = get_alias(argv[0])
550
 
        if alias_argv:
551
 
            alias_argv = [a.decode(bzrlib.user_encoding) for a in alias_argv]
552
 
            argv[0] = alias_argv.pop(0)
553
 
 
554
 
    cmd = argv.pop(0)
555
 
    # We want only 'ascii' command names, but the user may have typed
556
 
    # in a Unicode name. In that case, they should just get a
557
 
    # 'command not found' error later.
558
 
 
559
 
    cmd_obj = get_cmd_object(cmd, plugins_override=not opt_builtin)
560
 
    if not getattr(cmd_obj.run_argv, 'is_deprecated', False):
561
 
        run = cmd_obj.run_argv
562
 
        run_argv = [argv]
563
 
    else:
564
 
        run = cmd_obj.run_argv_aliases
565
 
        run_argv = [argv, alias_argv]
566
 
 
 
944
    argv = [a.decode(bzrlib.user_encoding) for a in argv]
 
945
    
567
946
    try:
568
 
        if opt_lsprof:
569
 
            ret = apply_lsprofiled(opt_lsprof_file, run, *run_argv)
570
 
        elif opt_profile:
571
 
            ret = apply_profiled(run, *run_argv)
572
 
        else:
573
 
            ret = run(*run_argv)
574
 
        return ret or 0
575
 
    finally:
576
 
        # reset, in case we may do other commands later within the same process
577
 
        be_quiet(False)
578
 
 
579
 
def display_command(func):
580
 
    """Decorator that suppresses pipe/interrupt errors."""
581
 
    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()
582
989
        try:
583
 
            result = func(*args, **kwargs)
584
 
            sys.stdout.flush()
585
 
            return result
586
 
        except IOError, e:
587
 
            if getattr(e, 'errno', None) is None:
588
 
                raise
589
 
            if e.errno != errno.EPIPE:
590
 
                # Win32 raises IOError with errno=0 on a broken pipe
591
 
                if sys.platform != 'win32' or e.errno != 0:
592
 
                    raise
593
 
            pass
594
 
        except KeyboardInterrupt:
595
 
            pass
596
 
    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
 
597
1023
 
598
1024
 
599
1025
def main(argv):
600
 
    import bzrlib.ui
601
 
    from bzrlib.ui.text import TextUIFactory
602
 
    bzrlib.ui.ui_factory = TextUIFactory()
603
 
    argv = [a.decode(bzrlib.user_encoding) for a in argv[1:]]
604
 
    ret = run_bzr_catch_errors(argv)
605
 
    mutter("return code %d", ret)
606
 
    return ret
607
 
 
608
 
 
609
 
def run_bzr_catch_errors(argv):
 
1026
    import errno
 
1027
    
 
1028
    bzrlib.open_tracefile(argv)
 
1029
 
610
1030
    try:
611
 
        return run_bzr(argv)
612
 
        # do this here inside the exception wrappers to catch EPIPE
613
 
        sys.stdout.flush()
614
 
    except Exception, e:
615
 
        # used to handle AssertionError and KeyboardInterrupt
616
 
        # specially here, but hopefully they're handled ok by the logger now
617
 
        bzrlib.trace.report_exception(sys.exc_info(), sys.stderr)
618
 
        if os.environ.get('BZR_PDB'):
619
 
            print '**** entering debugger'
620
 
            import pdb
621
 
            pdb.post_mortem(sys.exc_traceback)
622
 
        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
 
623
1068
 
624
1069
if __name__ == '__main__':
625
1070
    sys.exit(main(sys.argv))