19
import sys, os, time, os.path
22
23
from bzrlib.trace import mutter, note, log_error
23
24
from bzrlib.errors import bailout, BzrError, BzrCheckError, BzrCommandError
24
from bzrlib.osutils import quotefn
25
from bzrlib import Branch, Inventory, InventoryEntry, BZRDIR, \
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, \
34
37
assert cmd.startswith("cmd_")
35
38
return cmd[4:].replace('_','-')
37
def _parse_revision_str(revstr):
38
"""This handles a revision string -> revno.
40
There are several possibilities:
43
'234:345' -> [234, 345]
47
In the future we will also support:
48
'uuid:blah-blah-blah' -> ?
49
'hash:blahblahblah' -> ?
53
if revstr.find(':') != -1:
54
revs = revstr.split(':')
56
raise ValueError('More than 2 pieces not supported for --revision: %r' % revstr)
61
revs[0] = int(revs[0])
66
revs[1] = int(revs[1])
72
"""Find all python files which are plugins, and load their commands
73
to add to the list of "all commands"
75
The environment variable BZRPATH is considered a delimited set of
76
paths to look through. Each entry is searched for *.py files.
77
If a directory is found, it is also searched, but they are
78
not searched recursively. This allows you to revctl the plugins.
80
Inside the plugin should be a series of cmd_* function, which inherit from
81
the bzrlib.commands.Command class.
83
bzrpath = os.environ.get('BZRPLUGINPATH', '')
88
_platform_extensions = {
94
if _platform_extensions.has_key(sys.platform):
95
platform_extension = _platform_extensions[sys.platform]
97
platform_extension = None
98
for d in bzrpath.split(os.pathsep):
99
plugin_names = {} # This should really be a set rather than a dict
100
for f in os.listdir(d):
101
if f.endswith('.py'):
103
elif f.endswith('.pyc') or f.endswith('.pyo'):
105
elif platform_extension and f.endswith(platform_extension):
106
f = f[:-len(platform_extension)]
107
if f.endswidth('module'):
108
f = f[:-len('module')]
111
if not plugin_names.has_key(f):
112
plugin_names[f] = True
114
plugin_names = plugin_names.keys()
117
sys.path.insert(0, d)
118
for name in plugin_names:
122
if sys.modules.has_key(name):
123
old_module = sys.modules[name]
124
del sys.modules[name]
125
plugin = __import__(name, locals())
126
for k in dir(plugin):
127
if k.startswith('cmd_'):
128
k_unsquished = _unsquish_command_name(k)
129
if not plugin_cmds.has_key(k_unsquished):
130
plugin_cmds[k_unsquished] = getattr(plugin, k)
132
log_error('Two plugins defined the same command: %r' % k)
133
log_error('Not loading the one in %r in dir %r' % (name, d))
136
sys.modules[name] = old_module
137
except ImportError, e:
138
log_error('Unable to load plugin: %r from %r\n%s' % (name, d, e))
143
def _get_cmd_dict(include_plugins=True):
41
"""Return canonical name and class for all registered commands."""
145
42
for k, v in globals().iteritems():
146
43
if k.startswith("cmd_"):
147
d[_unsquish_command_name(k)] = v
149
d.update(_find_plugins())
152
def get_all_cmds(include_plugins=True):
153
"""Return canonical name and class for all registered commands."""
154
for k, v in _get_cmd_dict(include_plugins=include_plugins).iteritems():
158
def get_cmd_class(cmd,include_plugins=True):
44
yield _unsquish_command_name(k), v
46
def get_cmd_class(cmd):
159
47
"""Return the canonical name and command class for a command.
161
49
cmd = str(cmd) # not unicode
163
51
# first look up this command under the specified name
164
cmds = _get_cmd_dict(include_plugins=include_plugins)
166
return cmd, cmds[cmd]
53
return cmd, globals()[_squish_command_name(cmd)]
170
57
# look for any command which claims this as an alias
171
for cmdname, cmdclass in cmds.iteritems():
58
for cmdname, cmdclass in get_all_cmds():
172
59
if cmd in cmdclass.aliases:
173
60
return cmdname, cmdclass
175
cmdclass = ExternalCommand.find_command(cmd)
179
raise BzrCommandError("unknown command %r" % cmd)
182
class Command(object):
62
raise BzrCommandError("unknown command %r" % cmd)
183
66
"""Base class for commands.
185
68
The docstring for an actual command should give a single-line
231
class ExternalCommand(Command):
232
"""Class to wrap external commands.
234
We cheat a little here, when get_cmd_class() calls us we actually give it back
235
an object we construct that has the appropriate path, help, options etc for the
238
When run_bzr() tries to instantiate that 'class' it gets caught by the __call__
239
method, which we override to call the Command.__init__ method. That then calls
240
our run method which is pretty straight forward.
242
The only wrinkle is that we have to map bzr's dictionary of options and arguments
243
back into command line options and arguments for the script.
246
def find_command(cls, cmd):
248
bzrpath = os.environ.get('BZRPATH', '')
250
for dir in bzrpath.split(os.pathsep):
251
path = os.path.join(dir, cmd)
252
if os.path.isfile(path):
253
return ExternalCommand(path)
257
find_command = classmethod(find_command)
259
def __init__(self, path):
262
# TODO: If either of these fail, we should detect that and
263
# assume that path is not really a bzr plugin after all.
265
pipe = os.popen('%s --bzr-usage' % path, 'r')
266
self.takes_options = pipe.readline().split()
267
self.takes_args = pipe.readline().split()
270
pipe = os.popen('%s --bzr-help' % path, 'r')
271
self.__doc__ = pipe.read()
274
def __call__(self, options, arguments):
275
Command.__init__(self, options, arguments)
278
def run(self, **kargs):
286
if OPTIONS.has_key(name):
288
opts.append('--%s' % name)
289
if value is not None and value is not True:
290
opts.append(str(value))
292
# it's an arg, or arg list
293
if type(value) is not list:
299
self.status = os.spawnv(os.P_WAIT, self.path, [self.path] + opts + args)
303
115
class cmd_status(Command):
304
116
"""Display status summary.
306
This reports on versioned and unknown files, reporting them
307
grouped by state. Possible states are:
310
Versioned in the working copy but not in the previous revision.
313
Versioned in the previous revision but removed or deleted
317
Path of this file changed from the previous revision;
318
the text may also have changed. This includes files whose
319
parent directory was renamed.
322
Text has changed since the previous revision.
325
Nothing about this file has changed since the previous revision.
326
Only shown with --all.
329
Not versioned and not matching an ignore pattern.
331
To see ignored files use 'bzr ignored'. For details in the
332
changes to file texts, use 'bzr diff'.
334
If no arguments are specified, the status of the entire working
335
directory is shown. Otherwise, only the status of the specified
336
files or directories is reported. If a directory is given, status
337
is reported for everything inside that directory.
118
For each file there is a single line giving its file state and name.
119
The name is that in the current revision unless it is deleted or
120
missing, in which case the old name is shown.
339
takes_args = ['file*']
340
takes_options = ['all', 'show-ids']
122
takes_options = ['all']
341
123
aliases = ['st', 'stat']
343
def run(self, all=False, show_ids=False, file_list=None):
345
b = Branch(file_list[0])
346
file_list = [b.relpath(x) for x in file_list]
347
# special case: only one path was given and it's the root
349
if file_list == ['']:
354
status.show_status(b, show_unchanged=all, show_ids=show_ids,
355
specific_files=file_list)
125
def run(self, all=False):
126
#import bzrlib.status
127
#bzrlib.status.tree_status(Branch('.'))
128
Branch('.').show_status(show_all=all)
358
131
class cmd_cat_revision(Command):
395
168
recursively add that parent, rather than giving an error?
397
170
takes_args = ['file+']
398
takes_options = ['verbose', 'no-recurse']
171
takes_options = ['verbose']
400
def run(self, file_list, verbose=False, no_recurse=False):
401
bzrlib.add.smart_add(file_list, verbose, not no_recurse)
404
class cmd_relpath(Command):
173
def run(self, file_list, verbose=False):
174
bzrlib.add.smart_add(file_list, verbose)
177
def Relpath(Command):
405
178
"""Show path of a file relative to root"""
406
takes_args = ['filename']
179
takes_args = ('filename')
409
def run(self, filename):
410
print Branch(filename).relpath(filename)
182
print Branch(self.args['filename']).relpath(filename)
414
186
class cmd_inventory(Command):
415
187
"""Show inventory of the current working copy or a revision."""
416
takes_options = ['revision', 'show-ids']
188
takes_options = ['revision']
418
def run(self, revision=None, show_ids=False):
190
def run(self, revision=None):
420
192
if revision == None:
421
193
inv = b.read_working_inventory()
423
195
inv = b.get_revision_inventory(b.lookup_revision(revision))
425
for path, entry in inv.entries():
427
print '%-50s %s' % (path, entry.file_id)
197
for path, entry in inv.iter_entries():
198
print '%-50s %s' % (entry.file_id, path)
432
201
class cmd_move(Command):
471
class cmd_pull(Command):
472
"""Pull any changes from another branch into the current one.
474
If the location is omitted, the last-used location will be used.
475
Both the revision history and the working directory will be
478
This command only works on branches that have not diverged. Branches are
479
considered diverged if both branches have had commits without first
480
pulling from the other.
482
If branches have diverged, you can use 'bzr merge' to pull the text changes
483
from one into the other.
485
takes_args = ['location?']
487
def run(self, location=None):
488
from bzrlib.merge import merge
494
stored_loc = br_to.controlfile("x-pull", "rb").read().rstrip('\n')
496
if errno == errno.ENOENT:
499
location = stored_loc
501
raise BzrCommandError("No pull location known or specified.")
502
from branch import find_branch, DivergedBranches
503
br_from = find_branch(location)
504
location = pull_loc(br_from)
505
old_revno = br_to.revno()
507
br_to.update_revisions(br_from)
508
except DivergedBranches:
509
raise BzrCommandError("These branches have diverged. Try merge.")
511
merge(('.', -1), ('.', old_revno), check_clean=False)
512
if location != stored_loc:
513
br_to.controlfile("x-pull", "wb").write(location + "\n")
517
class cmd_branch(Command):
518
"""Create a new copy of a branch.
520
If the TO_LOCATION is omitted, the last component of the
521
FROM_LOCATION will be used. In other words,
522
"branch ../foo/bar" will attempt to create ./bar.
524
takes_args = ['from_location', 'to_location?']
526
def run(self, from_location, to_location=None):
528
from bzrlib.merge import merge
530
if to_location is None:
531
to_location = os.path.basename(from_location)
532
# FIXME: If there's a trailing slash, keep removing them
533
# until we find the right bit
536
os.mkdir(to_location)
538
if e.errno == errno.EEXIST:
539
raise BzrCommandError('Target directory "%s" already exists.' %
541
if e.errno == errno.ENOENT:
542
raise BzrCommandError('Parent of "%s" does not exist.' %
546
br_to = Branch(to_location, init=True)
547
from branch import find_branch, DivergedBranches
549
br_from = find_branch(from_location)
551
if e.errno == errno.ENOENT:
552
raise BzrCommandError('Source location "%s" does not exist.' %
557
from_location = pull_loc(br_from)
558
br_to.update_revisions(br_from)
559
merge((to_location, -1), (to_location, 0), this_dir=to_location,
561
br_to.controlfile("x-pull", "wb").write(from_location + "\n")
564
def pull_loc(branch):
565
# TODO: Should perhaps just make attribute be 'base' in
566
# RemoteBranch and Branch?
567
if hasattr(branch, "baseurl"):
568
return branch.baseurl
574
238
class cmd_renames(Command):
575
239
"""Show list of renamed files.
714
372
takes_args = ['file*']
715
takes_options = ['revision', 'diff-options']
716
aliases = ['di', 'dif']
373
takes_options = ['revision']
718
def run(self, revision=None, file_list=None, diff_options=None):
376
def run(self, revision=None, file_list=None):
719
377
from bzrlib.diff import show_diff
720
from bzrlib import find_branch
723
b = find_branch(file_list[0])
724
file_list = [b.relpath(f) for f in file_list]
725
if file_list == ['']:
726
# just pointing to top-of-tree
731
show_diff(b, revision, specific_files=file_list,
732
external_diff_options=diff_options)
379
show_diff(Branch('.'), revision, file_list)
738
382
class cmd_deleted(Command):
761
class cmd_modified(Command):
762
"""List files modified in working tree."""
767
inv = b.read_working_inventory()
768
sc = statcache.update_cache(b, inv)
769
basis = b.basis_tree()
770
basis_inv = basis.inventory
772
# We used to do this through iter_entries(), but that's slow
773
# when most of the files are unmodified, as is usually the
774
# case. So instead we iterate by inventory entry, and only
775
# calculate paths as necessary.
777
for file_id in basis_inv:
778
cacheentry = sc.get(file_id)
779
if not cacheentry: # deleted
781
ie = basis_inv[file_id]
782
if cacheentry[statcache.SC_SHA1] != ie.text_sha1:
783
path = inv.id2path(file_id)
788
class cmd_added(Command):
789
"""List files added in working tree."""
793
wt = b.working_tree()
794
basis_inv = b.basis_tree().inventory
797
if file_id in basis_inv:
799
path = inv.id2path(file_id)
800
if not os.access(b.abspath(path), os.F_OK):
806
404
class cmd_root(Command):
807
405
"""Show the tree root directory.
811
409
takes_args = ['filename?']
812
410
def run(self, filename=None):
813
411
"""Print the branch root."""
814
from branch import find_branch
815
b = find_branch(filename)
816
print getattr(b, 'base', None) or getattr(b, 'baseurl')
412
print bzrlib.branch.find_branch_root(filename)
819
416
class cmd_log(Command):
820
417
"""Show log of this branch.
822
To request a range of logs, you can use the command -r begin:end
823
-r revision requests a specific revision, -r :end or -r begin: are
826
TODO: Make --revision support uuid: and hash: [future tag:] notation.
419
TODO: Options to show ids; to limit range; etc.
830
takes_args = ['filename?']
831
takes_options = ['forward', 'timezone', 'verbose', 'show-ids', 'revision']
833
def run(self, filename=None, timezone='original',
838
from bzrlib import show_log, find_branch
841
direction = (forward and 'forward') or 'reverse'
844
b = find_branch(filename)
845
fp = b.relpath(filename)
847
file_id = b.read_working_inventory().path2id(fp)
849
file_id = None # points to branch root
855
revision = [None, None]
856
elif isinstance(revision, int):
857
revision = [revision, revision]
862
assert len(revision) == 2
864
mutter('encoding log as %r' % bzrlib.user_encoding)
866
# use 'replace' so that we don't abort if trying to write out
867
# in e.g. the default C locale.
868
outf = codecs.getwriter(bzrlib.user_encoding)(sys.stdout, errors='replace')
871
show_timezone=timezone,
876
start_revision=revision[0],
877
end_revision=revision[1])
881
class cmd_touching_revisions(Command):
882
"""Return revision-ids which affected a particular file.
884
A more user-friendly interface is "bzr log FILE"."""
886
takes_args = ["filename"]
887
def run(self, filename):
889
inv = b.read_working_inventory()
890
file_id = inv.path2id(b.relpath(filename))
891
for revno, revision_id, what in bzrlib.log.find_touching_revisions(b, file_id):
892
print "%6d %s" % (revno, what)
421
takes_options = ['timezone', 'verbose']
422
def run(self, timezone='original', verbose=False):
423
Branch('.', lock_mode='r').write_log(show_timezone=timezone, verbose=verbose)
895
426
class cmd_ls(Command):
931
462
class cmd_ignore(Command):
932
"""Ignore a command or pattern.
934
To remove patterns from the ignore list, edit the .bzrignore file.
936
If the pattern contains a slash, it is compared to the whole path
937
from the branch root. Otherwise, it is comapred to only the last
938
component of the path.
940
Ignore patterns are case-insensitive on case-insensitive systems.
942
Note: wildcards must be quoted from the shell on Unix.
945
bzr ignore ./Makefile
463
"""Ignore a command or pattern"""
948
464
takes_args = ['name_pattern']
950
466
def run(self, name_pattern):
951
from bzrlib.atomicfile import AtomicFile
955
ifn = b.abspath('.bzrignore')
957
if os.path.exists(ifn):
960
igns = f.read().decode('utf-8')
966
# TODO: If the file already uses crlf-style termination, maybe
967
# we should use that for the newly added lines?
969
if igns and igns[-1] != '\n':
971
igns += name_pattern + '\n'
974
f = AtomicFile(ifn, 'wt')
975
f.write(igns.encode('utf-8'))
469
# XXX: This will fail if it's a hardlink; should use an AtomicFile class.
470
f = open(b.abspath('.bzrignore'), 'at')
471
f.write(name_pattern + '\n')
980
474
inv = b.working_tree().inventory
981
475
if inv.path2id('.bzrignore'):
1058
550
class cmd_commit(Command):
1059
551
"""Commit changes into a new revision.
1061
If selected files are specified, only changes to those files are
1062
committed. If a directory is specified then its contents are also
1065
A selected-file commit may fail in some cases where the committed
1066
tree would be invalid, such as trying to commit a file in a
1067
newly-added directory that is not itself committed.
553
TODO: Commit only selected files.
1069
555
TODO: Run hooks on tree to-be-committed, and after commit.
1071
557
TODO: Strict commit that fails if there are unknown or deleted files.
1073
takes_args = ['selected*']
1074
takes_options = ['message', 'file', 'verbose']
559
takes_options = ['message', 'verbose']
1075
560
aliases = ['ci', 'checkin']
1077
def run(self, message=None, file=None, verbose=True, selected_list=None):
1078
from bzrlib.commit import commit
1080
## Warning: shadows builtin file()
1081
if not message and not file:
1082
raise BzrCommandError("please specify a commit message",
1083
["use either --message or --file"])
1084
elif message and file:
1085
raise BzrCommandError("please specify either --message or --file")
1089
message = codecs.open(file, 'rt', bzrlib.user_encoding).read()
1092
commit(b, message, verbose=verbose, specific_files=selected_list)
562
def run(self, message=None, verbose=False):
564
raise BzrCommandError("please specify a commit message")
565
Branch('.').commit(message, verbose=verbose)
1095
568
class cmd_check(Command):
1120
593
"""Run internal test suite"""
1123
from bzrlib.selftest import selftest
596
failures, tests = 0, 0
598
import doctest, bzrlib.store, bzrlib.tests
599
bzrlib.trace.verbose = False
601
for m in bzrlib.store, bzrlib.inventory, bzrlib.branch, bzrlib.osutils, \
602
bzrlib.tree, bzrlib.tests, bzrlib.commands, bzrlib.add:
603
mf, mt = doctest.testmod(m)
606
print '%-40s %3d tests' % (m.__name__, mt),
608
print '%3d FAILED!' % mf
612
print '%-40s %3d tests' % ('total', tests),
614
print '%3d FAILED!' % failures
1131
620
class cmd_version(Command):
1132
"""Show version of bzr."""
621
"""Show version of bzr"""
1136
625
def show_version():
1137
626
print "bzr (bazaar-ng) %s" % bzrlib.__version__
1138
# is bzrlib itself in a branch?
1139
bzrrev = bzrlib.get_bzr_revision()
1141
print " (bzr checkout, revision %d {%s})" % bzrrev
1142
627
print bzrlib.__copyright__
1143
628
print "http://bazaar-ng.org/"
1154
639
print "it sure does!"
1156
def parse_spec(spec):
1158
>>> parse_spec(None)
1160
>>> parse_spec("./")
1162
>>> parse_spec("../@")
1164
>>> parse_spec("../f/@35")
1170
parsed = spec.split('/@')
1171
assert len(parsed) == 2
1175
parsed[1] = int(parsed[1])
1176
assert parsed[1] >=0
1178
parsed = [spec, None]
1183
class cmd_merge(Command):
1184
"""Perform a three-way merge of trees.
1186
The SPEC parameters are working tree or revision specifiers. Working trees
1187
are specified using standard paths or urls. No component of a directory
1188
path may begin with '@'.
1190
Working tree examples: '.', '..', 'foo@', but NOT 'foo/@bar'
1192
Revisions are specified using a dirname/@revno pair, where dirname is the
1193
branch directory and revno is the revision within that branch. If no revno
1194
is specified, the latest revision is used.
1196
Revision examples: './@127', 'foo/@', '../@1'
1198
The OTHER_SPEC parameter is required. If the BASE_SPEC parameter is
1199
not supplied, the common ancestor of OTHER_SPEC the current branch is used
1202
merge refuses to run if there are any uncommitted changes, unless
1205
takes_args = ['other_spec', 'base_spec?']
1206
takes_options = ['force']
1208
def run(self, other_spec, base_spec=None, force=False):
1209
from bzrlib.merge import merge
1210
merge(parse_spec(other_spec), parse_spec(base_spec),
1211
check_clean=(not force))
1214
class cmd_revert(Command):
1215
"""Reverse all changes since the last commit.
1217
Only versioned files are affected.
1219
TODO: Store backups of any files that will be reverted, so
1220
that the revert can be undone.
1222
takes_options = ['revision']
1224
def run(self, revision=-1):
1225
from bzrlib.merge import merge
1226
merge(('.', revision), parse_spec('.'),
1231
642
class cmd_assert_fail(Command):
1232
643
"""Test reporting of assertion failures"""