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
17
"""builtin bzr commands"""
23
from shutil import rmtree
28
from bzrlib.branch import Branch
29
import bzrlib.bzrdir as bzrdir
30
from bzrlib.bundle.read_bundle import BundleReader
31
from bzrlib.bundle.apply_bundle import merge_bundle
32
from bzrlib.commands import Command, display_command
33
import bzrlib.errors as errors
34
from bzrlib.errors import (BzrError, BzrCheckError, BzrCommandError,
35
NotBranchError, DivergedBranches, NotConflicted,
36
NoSuchFile, NoWorkingTree, FileInWrongBranch,
37
NotVersionedError, NotABundle)
38
from bzrlib.log import show_one_log
39
from bzrlib.merge import Merge3Merger
40
from bzrlib.option import Option
42
from bzrlib.progress import DummyProgress, ProgressPhase
43
from bzrlib.revision import common_ancestor
44
from bzrlib.revisionspec import RevisionSpec
45
22
import bzrlib.trace
46
from bzrlib.trace import mutter, note, log_error, warning, is_quiet
47
from bzrlib.transport.local import LocalTransport
49
import bzrlib.urlutils as urlutils
50
from bzrlib.workingtree import WorkingTree
53
def tree_files(file_list, default_branch=u'.'):
55
return internal_tree_files(file_list, default_branch)
56
except FileInWrongBranch, e:
57
raise BzrCommandError("%s is not in the same branch as %s" %
58
(e.path, file_list[0]))
61
# XXX: Bad function name; should possibly also be a class method of
62
# WorkingTree rather than a function.
63
def internal_tree_files(file_list, default_branch=u'.'):
64
"""Convert command-line paths to a WorkingTree and relative paths.
66
This is typically used for command-line processors that take one or
67
more filenames, and infer the workingtree that contains them.
69
The filenames given are not required to exist.
71
:param file_list: Filenames to convert.
73
:param default_branch: Fallback tree path to use if file_list is empty or None.
75
:return: workingtree, [relative_paths]
77
if file_list is None or len(file_list) == 0:
78
return WorkingTree.open_containing(default_branch)[0], file_list
79
tree = WorkingTree.open_containing(file_list[0])[0]
81
for filename in file_list:
83
new_list.append(tree.relpath(filename))
84
except errors.PathNotChild:
85
raise FileInWrongBranch(tree.branch, filename)
89
def get_format_type(typestring):
90
"""Parse and return a format specifier."""
91
if typestring == "weave":
92
return bzrdir.BzrDirFormat6()
93
if typestring == "default":
94
return bzrdir.BzrDirMetaFormat1()
95
if typestring == "metaweave":
96
format = bzrdir.BzrDirMetaFormat1()
97
format.repository_format = bzrlib.repository.RepositoryFormat7()
99
if typestring == "knit":
100
format = bzrdir.BzrDirMetaFormat1()
101
format.repository_format = bzrlib.repository.RepositoryFormatKnit1()
103
msg = "Unknown bzr format %s. Current formats are: default, knit,\n" \
104
"metaweave and weave" % typestring
105
raise BzrCommandError(msg)
108
# TODO: Make sure no commands unconditionally use the working directory as a
109
# branch. If a filename argument is used, the first of them should be used to
110
# specify the branch. (Perhaps this can be factored out into some kind of
111
# Argument class, representing a file in a branch, where the first occurrence
23
from bzrlib.trace import mutter, note, log_error, warning
24
from bzrlib.errors import BzrError, BzrCheckError, BzrCommandError
25
from bzrlib.branch import find_branch
26
from bzrlib import BZRDIR
27
from bzrlib.commands import Command
114
30
class cmd_status(Command):
115
31
"""Display status summary.
147
63
files or directories is reported. If a directory is given, status
148
64
is reported for everything inside that directory.
150
If a revision argument is given, the status is calculated against
151
that revision, or between two revisions if two are provided.
66
If a revision is specified, the changes since that revision are shown.
154
# TODO: --no-recurse, --recurse options
156
68
takes_args = ['file*']
157
69
takes_options = ['all', 'show-ids', 'revision']
158
70
aliases = ['st', 'stat']
160
encoding_type = 'replace'
163
def run(self, all=False, show_ids=False, file_list=None, revision=None):
164
from bzrlib.status import show_tree_status
166
tree, file_list = tree_files(file_list)
72
def run(self, all=False, show_ids=False, file_list=None):
74
b = find_branch(file_list[0])
75
file_list = [b.relpath(x) for x in file_list]
76
# special case: only one path was given and it's the root
168
show_tree_status(tree, show_unchanged=all, show_ids=show_ids,
169
specific_files=file_list, revision=revision,
83
from bzrlib.status import show_status
84
show_status(b, show_unchanged=all, show_ids=show_ids,
85
specific_files=file_list)
173
88
class cmd_cat_revision(Command):
174
"""Write out metadata for a revision.
176
The revision to print can either be specified by a specific
177
revision identifier, or you can use --revision.
89
"""Write out metadata for a revision."""
181
takes_args = ['revision_id?']
182
takes_options = ['revision']
183
# cat-revision is more for frontends so should be exact
187
def run(self, revision_id=None, revision=None):
189
if revision_id is not None and revision is not None:
190
raise BzrCommandError('You can only supply one of revision_id or --revision')
191
if revision_id is None and revision is None:
192
raise BzrCommandError('You must supply either --revision or a revision_id')
193
b = WorkingTree.open_containing(u'.')[0].branch
195
# TODO: jam 20060112 should cat-revision always output utf-8?
196
if revision_id is not None:
197
self.outf.write(b.repository.get_revision_xml(revision_id).decode('utf-8'))
198
elif revision is not None:
201
raise BzrCommandError('You cannot specify a NULL revision.')
202
revno, rev_id = rev.in_history(b)
203
self.outf.write(b.repository.get_revision_xml(rev_id).decode('utf-8'))
92
takes_args = ['revision_id']
94
def run(self, revision_id):
95
from bzrlib.xml import pack_xml
96
pack_xml(find_branch('.').get_revision(revision_id), sys.stdout)
206
99
class cmd_revno(Command):
207
100
"""Show current revision number.
209
This is equal to the number of revisions on this branch.
212
takes_args = ['location?']
215
def run(self, location=u'.'):
216
self.outf.write(str(Branch.open_containing(location)[0].revno()))
217
self.outf.write('\n')
102
This is equal to the number of revisions on this branch."""
104
print find_branch('.').revno()
220
106
class cmd_revision_info(Command):
221
107
"""Show revision number and revision id for a given revision identifier.
305
163
This is equivalent to creating the directory and then adding it.
308
165
takes_args = ['dir+']
309
encoding_type = 'replace'
311
167
def run(self, dir_list):
312
170
for d in dir_list:
314
wt, dd = WorkingTree.open_containing(d)
316
self.outf.write('added %s\n' % d)
319
178
class cmd_relpath(Command):
320
179
"""Show path of a file relative to root"""
322
180
takes_args = ['filename']
326
183
def run(self, filename):
327
# TODO: jam 20050106 Can relpath return a munged path if
328
# sys.stdout encoding cannot represent it?
329
tree, relpath = WorkingTree.open_containing(filename)
330
self.outf.write(relpath)
331
self.outf.write('\n')
184
print find_branch(filename).relpath(filename)
334
188
class cmd_inventory(Command):
335
"""Show inventory of the current working copy or a revision.
337
It is possible to limit the output to a particular entry
338
type using the --kind option. For example; --kind file.
341
takes_options = ['revision', 'show-ids', 'kind']
189
"""Show inventory of the current working copy or a revision."""
190
takes_options = ['revision', 'show-ids']
344
def run(self, revision=None, show_ids=False, kind=None):
345
if kind and kind not in ['file', 'directory', 'symlink']:
346
raise BzrCommandError('invalid kind specified')
347
tree = WorkingTree.open_containing(u'.')[0]
349
inv = tree.read_working_inventory()
192
def run(self, revision=None, show_ids=False):
195
inv = b.read_working_inventory()
351
197
if len(revision) > 1:
352
198
raise BzrCommandError('bzr inventory --revision takes'
353
199
' exactly one revision identifier')
354
inv = tree.branch.repository.get_revision_inventory(
355
revision[0].in_history(tree.branch).rev_id)
200
inv = b.get_revision_inventory(b.lookup_revision(revision[0]))
357
202
for path, entry in inv.entries():
358
if kind and kind != entry.kind:
361
self.outf.write('%-50s %s\n' % (path, entry.file_id))
204
print '%-50s %s' % (path, entry.file_id)
363
self.outf.write(path)
364
self.outf.write('\n')
209
class cmd_move(Command):
210
"""Move files to a different directory.
215
The destination must be a versioned directory in the same branch.
217
takes_args = ['source$', 'dest']
218
def run(self, source_list, dest):
221
# TODO: glob expansion on windows?
222
b.move([b.relpath(s) for s in source_list], b.relpath(dest))
225
class cmd_rename(Command):
226
"""Change the name of an entry.
229
bzr rename frob.c frobber.c
230
bzr rename src/frob.c lib/frob.c
232
It is an error if the destination name exists.
234
See also the 'move' command, which moves files into a different
235
directory without changing their name.
237
TODO: Some way to rename multiple files without invoking bzr for each
239
takes_args = ['from_name', 'to_name']
241
def run(self, from_name, to_name):
243
b.rename_one(b.relpath(from_name), b.relpath(to_name))
367
247
class cmd_mv(Command):
378
258
Files cannot be moved between branches.
381
260
takes_args = ['names*']
382
aliases = ['move', 'rename']
383
encoding_type = 'replace'
385
261
def run(self, names_list):
386
262
if len(names_list) < 2:
387
263
raise BzrCommandError("missing file argument")
388
tree, rel_names = tree_files(names_list)
264
b = find_branch(names_list[0])
266
rel_names = [b.relpath(x) for x in names_list]
390
268
if os.path.isdir(names_list[-1]):
391
269
# move into existing directory
392
for pair in tree.move(rel_names[:-1], rel_names[-1]):
393
self.outf.write("%s => %s\n" % pair)
270
for pair in b.move(rel_names[:-1], rel_names[-1]):
271
print "%s => %s" % pair
395
273
if len(names_list) != 2:
396
274
raise BzrCommandError('to mv multiple files the destination '
397
275
'must be a versioned directory')
398
tree.rename_one(rel_names[0], rel_names[1])
399
self.outf.write("%s => %s\n" % (rel_names[0], rel_names[1]))
276
for pair in b.move(rel_names[0], rel_names[1]):
277
print "%s => %s" % pair
402
282
class cmd_pull(Command):
403
"""Turn this branch into a mirror of another branch.
283
"""Pull any changes from another branch into the current one.
285
If the location is omitted, the last-used location will be used.
286
Both the revision history and the working directory will be
405
289
This command only works on branches that have not diverged. Branches are
406
considered diverged if the destination branch's most recent commit is one
407
that has not been merged (directly or indirectly) into the parent.
409
If branches have diverged, you can use 'bzr merge' to integrate the changes
410
from one into the other. Once one branch has merged, the other should
411
be able to pull it again.
290
considered diverged if both branches have had commits without first
291
pulling from the other.
413
293
If branches have diverged, you can use 'bzr merge' to pull the text changes
414
from one into the other. Once one branch has merged, the other should
415
be able to pull it again.
417
If you want to forget your local changes and just update your branch to
418
match the remote one, use pull --overwrite.
420
If there is no default location set, the first pull will set it. After
421
that, you can omit the location to use the default. To change the
422
default, use --remember.
294
from one into the other.
425
takes_options = ['remember', 'overwrite', 'revision', 'verbose']
426
296
takes_args = ['location?']
427
encoding_type = 'replace'
429
def run(self, location=None, remember=False, overwrite=False, revision=None, verbose=False):
430
# FIXME: too much stuff is in the command class
298
def run(self, location=None):
299
from bzrlib.merge import merge
301
from shutil import rmtree
303
from bzrlib.branch import pull_loc
305
br_to = find_branch('.')
432
tree_to = WorkingTree.open_containing(u'.')[0]
433
branch_to = tree_to.branch
434
except NoWorkingTree:
436
branch_to = Branch.open_containing(u'.')[0]
437
stored_loc = branch_to.get_parent()
308
stored_loc = br_to.controlfile("x-pull", "rb").read().rstrip('\n')
310
if e.errno != errno.ENOENT:
438
312
if location is None:
439
313
if stored_loc is None:
440
314
raise BzrCommandError("No pull location known or specified.")
442
display_url = urlutils.unescape_for_display(stored_loc,
444
self.outf.write("Using saved location: %s\n" % display_url)
445
location = stored_loc
447
branch_from = Branch.open(location)
449
if branch_to.get_parent() is None or remember:
450
branch_to.set_parent(branch_from.base)
454
elif len(revision) == 1:
455
rev_id = revision[0].in_history(branch_from).rev_id
457
raise BzrCommandError('bzr pull --revision takes one value.')
459
old_rh = branch_to.revision_history()
460
if tree_to is not None:
461
count = tree_to.pull(branch_from, overwrite, rev_id)
463
count = branch_to.pull(branch_from, overwrite, rev_id)
464
note('%d revision(s) pulled.' % (count,))
467
new_rh = branch_to.revision_history()
470
from bzrlib.log import show_changed_revisions
471
show_changed_revisions(branch_to, old_rh, new_rh,
475
class cmd_push(Command):
476
"""Update a mirror of this branch.
478
The target branch will not have its working tree populated because this
479
is both expensive, and is not supported on remote file systems.
481
Some smart servers or protocols *may* put the working tree in place in
484
This command only works on branches that have not diverged. Branches are
485
considered diverged if the destination branch's most recent commit is one
486
that has not been merged (directly or indirectly) by the source branch.
488
If branches have diverged, you can use 'bzr push --overwrite' to replace
489
the other branch completely, discarding its unmerged changes.
491
If you want to ensure you have the different changes in the other branch,
492
do a merge (see bzr help merge) from the other branch, and commit that.
493
After that you will be able to do a push without '--overwrite'.
495
If there is no default push location set, the first push will set it.
496
After that, you can omit the location to use the default. To change the
497
default, use --remember.
500
takes_options = ['remember', 'overwrite', 'verbose',
501
Option('create-prefix',
502
help='Create the path leading up to the branch '
503
'if it does not already exist')]
504
takes_args = ['location?']
505
encoding_type = 'replace'
507
def run(self, location=None, remember=False, overwrite=False,
508
create_prefix=False, verbose=False):
509
# FIXME: Way too big! Put this into a function called from the
511
from bzrlib.transport import get_transport
513
br_from = Branch.open_containing('.')[0]
514
stored_loc = br_from.get_push_location()
516
if stored_loc is None:
517
raise BzrCommandError("No push location known or specified.")
519
display_url = urlutils.unescape_for_display(stored_loc,
521
self.outf.write("Using saved location: %s" % display_url)
522
location = stored_loc
524
transport = get_transport(location)
525
location_url = transport.base
526
if br_from.get_push_location() is None or remember:
527
br_from.set_push_location(location_url)
316
print "Using last location: %s" % stored_loc
317
location = stored_loc
318
cache_root = tempfile.mkdtemp()
319
from bzrlib.branch import DivergedBranches
320
br_from = find_branch(location)
321
location = pull_loc(br_from)
322
old_revno = br_to.revno()
531
dir_to = bzrlib.bzrdir.BzrDir.open(location_url)
532
br_to = dir_to.open_branch()
533
except NotBranchError:
535
transport = transport.clone('..')
536
if not create_prefix:
538
relurl = transport.relpath(location_url)
539
mutter('creating directory %s => %s', location_url, relurl)
540
transport.mkdir(relurl)
542
raise BzrCommandError("Parent directory of %s "
543
"does not exist." % location)
545
current = transport.base
546
needed = [(transport, transport.relpath(location_url))]
549
transport, relpath = needed[-1]
550
transport.mkdir(relpath)
553
new_transport = transport.clone('..')
554
needed.append((new_transport,
555
new_transport.relpath(transport.base)))
556
if new_transport.base == transport.base:
557
raise BzrCommandError("Could not create "
559
dir_to = br_from.bzrdir.clone(location_url,
560
revision_id=br_from.last_revision())
561
br_to = dir_to.open_branch()
562
count = len(br_to.revision_history())
564
old_rh = br_to.revision_history()
324
from branch import find_cached_branch, DivergedBranches
325
br_from = find_cached_branch(location, cache_root)
326
location = pull_loc(br_from)
327
old_revno = br_to.revno()
567
tree_to = dir_to.open_workingtree()
568
except errors.NotLocalUrl:
569
warning('This transport does not update the working '
570
'tree of: %s' % (br_to.base,))
571
count = br_to.pull(br_from, overwrite)
572
except NoWorkingTree:
573
count = br_to.pull(br_from, overwrite)
575
count = tree_to.pull(br_from, overwrite)
329
br_to.update_revisions(br_from)
576
330
except DivergedBranches:
577
331
raise BzrCommandError("These branches have diverged."
578
" Try a merge then push with overwrite.")
579
note('%d revision(s) pushed.' % (count,))
334
merge(('.', -1), ('.', old_revno), check_clean=False)
335
if location != stored_loc:
336
br_to.controlfile("x-pull", "wb").write(location + "\n")
582
new_rh = br_to.revision_history()
585
from bzrlib.log import show_changed_revisions
586
show_changed_revisions(br_to, old_rh, new_rh,
590
342
class cmd_branch(Command):
596
348
To retrieve the branch as of a particular revision, supply the --revision
597
349
parameter, as in "branch foo/bar -r 5".
599
--basis is to speed up branching from remote branches. When specified, it
600
copies all the file-contents, inventory and revision data from the basis
601
branch before copying anything from the remote branch.
603
351
takes_args = ['from_location', 'to_location?']
604
takes_options = ['revision', 'basis']
352
takes_options = ['revision']
605
353
aliases = ['get', 'clone']
607
def run(self, from_location, to_location=None, revision=None, basis=None):
608
from bzrlib.transport import get_transport
609
from bzrlib.osutils import rmtree
612
elif len(revision) > 1:
613
raise BzrCommandError(
614
'bzr branch --revision takes exactly 1 revision value')
616
br_from = Branch.open(from_location)
618
if e.errno == errno.ENOENT:
619
raise BzrCommandError('Source location "%s" does not'
620
' exist.' % to_location)
625
if basis is not None:
626
basis_dir = bzrdir.BzrDir.open_containing(basis)[0]
629
if len(revision) == 1 and revision[0] is not None:
630
revision_id = revision[0].in_history(br_from)[1]
632
# FIXME - wt.last_revision, fallback to branch, fall back to
633
# None or perhaps NULL_REVISION to mean copy nothing
635
revision_id = br_from.last_revision()
355
def run(self, from_location, to_location=None, revision=None):
356
from bzrlib.branch import copy_branch, find_cached_branch
359
from shutil import rmtree
360
cache_root = tempfile.mkdtemp()
364
elif len(revision) > 1:
365
raise BzrCommandError(
366
'bzr branch --revision takes exactly 1 revision value')
368
br_from = find_cached_branch(from_location, cache_root)
370
if e.errno == errno.ENOENT:
371
raise BzrCommandError('Source location "%s" does not'
372
' exist.' % to_location)
636
375
if to_location is None:
637
376
to_location = os.path.basename(from_location.rstrip("/\\"))
640
name = os.path.basename(to_location) + '\n'
642
to_transport = get_transport(to_location)
644
to_transport.mkdir('.')
645
except bzrlib.errors.FileExists:
646
raise BzrCommandError('Target directory "%s" already'
647
' exists.' % to_location)
648
except bzrlib.errors.NoSuchFile:
649
raise BzrCommandError('Parent of "%s" does not exist.' %
652
# preserve whatever source format we have.
653
dir = br_from.bzrdir.sprout(to_transport.base,
654
revision_id, basis_dir)
655
branch = dir.open_branch()
378
os.mkdir(to_location)
380
if e.errno == errno.EEXIST:
381
raise BzrCommandError('Target directory "%s" already'
382
' exists.' % to_location)
383
if e.errno == errno.ENOENT:
384
raise BzrCommandError('Parent of "%s" does not exist.' %
389
copy_branch(br_from, to_location, revision[0])
656
390
except bzrlib.errors.NoSuchRevision:
657
to_transport.delete_tree('.')
658
msg = "The branch %s has no revision %s." % (from_location, revision[0])
659
raise BzrCommandError(msg)
660
except bzrlib.errors.UnlistableBranch:
661
391
rmtree(to_location)
662
msg = "The branch %s cannot be used as a --basis" % (basis,)
392
msg = "The branch %s has no revision %d." % (from_location, revision[0])
663
393
raise BzrCommandError(msg)
665
branch.control_files.put_utf8('branch-name', name)
666
note('Branched %d revision(s).' % branch.revno())
671
class cmd_checkout(Command):
672
"""Create a new checkout of an existing branch.
674
If BRANCH_LOCATION is omitted, checkout will reconstitute a working tree for
675
the branch found in '.'. This is useful if you have removed the working tree
676
or if it was never created - i.e. if you pushed the branch to its current
679
If the TO_LOCATION is omitted, the last component of the BRANCH_LOCATION will
680
be used. In other words, "checkout ../foo/bar" will attempt to create ./bar.
682
To retrieve the branch as of a particular revision, supply the --revision
683
parameter, as in "checkout foo/bar -r 5". Note that this will be immediately
684
out of date [so you cannot commit] but it may be useful (i.e. to examine old
687
--basis is to speed up checking out from remote branches. When specified, it
688
uses the inventory and file contents from the basis branch in preference to the
689
branch being checked out.
691
takes_args = ['branch_location?', 'to_location?']
692
takes_options = ['revision', # , 'basis']
693
Option('lightweight',
694
help="perform a lightweight checkout. Lightweight "
695
"checkouts depend on access to the branch for "
696
"every operation. Normal checkouts can perform "
697
"common operations like diff and status without "
698
"such access, and also support local commits."
702
def run(self, branch_location=None, to_location=None, revision=None, basis=None,
706
elif len(revision) > 1:
707
raise BzrCommandError(
708
'bzr checkout --revision takes exactly 1 revision value')
709
if branch_location is None:
710
branch_location = bzrlib.osutils.getcwd()
711
to_location = branch_location
712
source = Branch.open(branch_location)
713
if len(revision) == 1 and revision[0] is not None:
714
revision_id = revision[0].in_history(source)[1]
717
if to_location is None:
718
to_location = os.path.basename(branch_location.rstrip("/\\"))
719
# if the source and to_location are the same,
720
# and there is no working tree,
721
# then reconstitute a branch
722
if (bzrlib.osutils.abspath(to_location) ==
723
bzrlib.osutils.abspath(branch_location)):
725
source.bzrdir.open_workingtree()
726
except errors.NoWorkingTree:
727
source.bzrdir.create_workingtree()
730
os.mkdir(to_location)
732
if e.errno == errno.EEXIST:
733
raise BzrCommandError('Target directory "%s" already'
734
' exists.' % to_location)
735
if e.errno == errno.ENOENT:
736
raise BzrCommandError('Parent of "%s" does not exist.' %
740
old_format = bzrlib.bzrdir.BzrDirFormat.get_default_format()
741
bzrlib.bzrdir.BzrDirFormat.set_default_format(bzrdir.BzrDirMetaFormat1())
744
checkout = bzrdir.BzrDirMetaFormat1().initialize(to_location)
745
bzrlib.branch.BranchReferenceFormat().initialize(checkout, source)
747
checkout_branch = bzrlib.bzrdir.BzrDir.create_branch_convenience(
748
to_location, force_new_tree=False)
749
checkout = checkout_branch.bzrdir
750
checkout_branch.bind(source)
751
if revision_id is not None:
752
rh = checkout_branch.revision_history()
753
checkout_branch.set_revision_history(rh[:rh.index(revision_id) + 1])
754
checkout.create_workingtree(revision_id)
756
bzrlib.bzrdir.BzrDirFormat.set_default_format(old_format)
759
398
class cmd_renames(Command):
760
399
"""Show list of renamed files.
401
TODO: Option to show renames between two historical versions.
403
TODO: Only show renames under dir, rather than in the whole branch.
762
# TODO: Option to show renames between two historical versions.
764
# TODO: Only show renames under dir, rather than in the whole branch.
765
405
takes_args = ['dir?']
768
def run(self, dir=u'.'):
769
tree = WorkingTree.open_containing(dir)[0]
770
old_inv = tree.basis_tree().inventory
771
new_inv = tree.read_working_inventory()
407
def run(self, dir='.'):
409
old_inv = b.basis_tree().inventory
410
new_inv = b.read_working_inventory()
773
412
renames = list(bzrlib.tree.find_renames(old_inv, new_inv))
775
414
for old_name, new_name in renames:
776
self.outf.write("%s => %s\n" % (old_name, new_name))
779
class cmd_update(Command):
780
"""Update a tree to have the latest code committed to its branch.
782
This will perform a merge into the working tree, and may generate
783
conflicts. If you have any local changes, you will still
784
need to commit them after the update for the update to be complete.
786
If you want to discard your local changes, you can just do a
787
'bzr revert' instead of 'bzr commit' after the update.
789
takes_args = ['dir?']
791
def run(self, dir='.'):
792
tree = WorkingTree.open_containing(dir)[0]
795
if tree.last_revision() == tree.branch.last_revision():
796
# may be up to date, check master too.
797
master = tree.branch.get_master_branch()
798
if master is None or master.last_revision == tree.last_revision():
799
note("Tree is up to date.")
801
conflicts = tree.update()
802
note('Updated to revision %d.' %
803
(tree.branch.revision_id_to_revno(tree.last_revision()),))
415
print "%s => %s" % (old_name, new_name)
812
418
class cmd_info(Command):
813
"""Show information about a working tree, branch or repository.
815
This command will show all known locations and formats associated to the
816
tree, branch or repository. Statistical information is included with
819
Branches and working trees will also report any missing revisions.
821
takes_args = ['location?']
822
takes_options = ['verbose']
825
def run(self, location=None, verbose=False):
826
from bzrlib.info import show_bzrdir_info
827
show_bzrdir_info(bzrdir.BzrDir.open_containing(location)[0],
419
"""Show statistical information about a branch."""
420
takes_args = ['branch?']
422
def run(self, branch=None):
425
b = find_branch(branch)
831
429
class cmd_remove(Command):
867
447
same through all revisions where the file exists, even when it is
868
448
moved or renamed.
872
451
takes_args = ['filename']
875
452
def run(self, filename):
876
tree, relpath = WorkingTree.open_containing(filename)
877
i = tree.inventory.path2id(relpath)
453
b = find_branch(filename)
454
i = b.inventory.path2id(b.relpath(filename))
879
456
raise BzrError("%r is not a versioned file" % filename)
881
self.outf.write(i + '\n')
884
461
class cmd_file_path(Command):
885
462
"""Print path of file_ids to a file or directory.
887
464
This prints one line for each directory down to the target,
888
starting at the branch root.
465
starting at the branch root."""
892
467
takes_args = ['filename']
895
468
def run(self, filename):
896
tree, relpath = WorkingTree.open_containing(filename)
898
fid = inv.path2id(relpath)
469
b = find_branch(filename)
471
fid = inv.path2id(b.relpath(filename))
900
473
raise BzrError("%r is not a versioned file" % filename)
901
474
for fip in inv.get_idpath(fid):
902
self.outf.write(fip + '\n')
905
class cmd_reconcile(Command):
906
"""Reconcile bzr metadata in a branch.
908
This can correct data mismatches that may have been caused by
909
previous ghost operations or bzr upgrades. You should only
910
need to run this command if 'bzr check' or a bzr developer
911
advises you to run it.
913
If a second branch is provided, cross-branch reconciliation is
914
also attempted, which will check that data like the tree root
915
id which was not present in very early bzr versions is represented
916
correctly in both branches.
918
At the same time it is run it may recompress data resulting in
919
a potential saving in disk space or performance gain.
921
The branch *MUST* be on a listable system such as local disk or sftp.
923
takes_args = ['branch?']
925
def run(self, branch="."):
926
from bzrlib.reconcile import reconcile
927
dir = bzrlib.bzrdir.BzrDir.open(branch)
931
478
class cmd_revision_history(Command):
932
479
"""Display list of revision ids on this branch."""
937
branch = WorkingTree.open_containing(u'.')[0].branch
938
for patchid in branch.revision_history():
939
self.outf.write(patchid)
940
self.outf.write('\n')
943
class cmd_ancestry(Command):
944
"""List all revisions merged into this branch."""
949
tree = WorkingTree.open_containing(u'.')[0]
951
# FIXME. should be tree.last_revision
952
revision_ids = b.repository.get_ancestry(b.last_revision())
953
assert revision_ids[0] == None
955
for revision_id in revision_ids:
956
self.outf.write(revision_id + '\n')
482
for patchid in find_branch('.').revision_history():
486
class cmd_directories(Command):
487
"""Display list of versioned directories in this branch."""
489
for name, ie in find_branch('.').read_working_inventory().directories():
959
496
class cmd_init(Command):
962
499
Use this to create an empty branch, or before importing an
963
500
existing project.
965
If there is a repository in a parent directory of the location, then
966
the history of the branch will be stored in the repository. Otherwise
967
init creates a standalone branch which carries its own history in
970
If there is already a branch at the location but it has no working tree,
971
the tree can be populated with 'bzr checkout'.
973
502
Recipe for importing a tree of files:
978
507
bzr commit -m 'imported project'
980
takes_args = ['location?']
983
help='Specify a format for this branch. Current'
984
' formats are: default, knit, metaweave and'
985
' weave. Default is knit; metaweave and'
986
' weave are deprecated',
987
type=get_format_type),
989
def run(self, location=None, format=None):
990
510
from bzrlib.branch import Branch
992
format = get_format_type('default')
996
# The path has to exist to initialize a
997
# branch inside of it.
998
# Just using os.mkdir, since I don't
999
# believe that we want to create a bunch of
1000
# locations if the user supplies an extended path
1001
if not os.path.exists(location):
1004
existing_bzrdir = bzrdir.BzrDir.open(location)
1005
except NotBranchError:
1006
# really a NotBzrDir error...
1007
bzrdir.BzrDir.create_branch_convenience(location, format=format)
1009
if existing_bzrdir.has_branch():
1010
if existing_bzrdir.has_workingtree():
1011
raise errors.AlreadyBranchError(location)
1013
raise errors.BranchExistsWithoutWorkingTree(location)
1015
existing_bzrdir.create_branch()
1016
existing_bzrdir.create_workingtree()
1019
class cmd_init_repository(Command):
1020
"""Create a shared repository to hold branches.
1022
New branches created under the repository directory will store their revisions
1023
in the repository, not in the branch directory, if the branch format supports
1029
bzr checkout --lightweight repo/trunk trunk-checkout
1033
takes_args = ["location"]
1034
takes_options = [Option('format',
1035
help='Specify a format for this repository.'
1036
' Current formats are: default, knit,'
1037
' metaweave and weave. Default is knit;'
1038
' metaweave and weave are deprecated',
1039
type=get_format_type),
1041
help='Allows branches in repository to have'
1043
aliases = ["init-repo"]
1044
def run(self, location, format=None, trees=False):
1045
from bzrlib.transport import get_transport
1047
format = get_format_type('default')
1048
transport = get_transport(location)
1049
if not transport.has('.'):
1051
newdir = format.initialize_on_transport(transport)
1052
repo = newdir.create_repository(shared=True)
1053
repo.set_make_working_trees(trees)
511
Branch('.', init=True)
1056
514
class cmd_diff(Command):
1059
517
If files are listed, only the changes in those files are listed.
1060
518
Otherwise, all changes for the tree are listed.
1062
"bzr diff -p1" is equivalent to "bzr diff --prefix old/:new/", and
1063
produces patches suitable for "patch -p1".
520
TODO: Allow diff across branches.
522
TODO: Option to use external diff command; could be GNU diff, wdiff,
525
TODO: Python difflib is not exactly the same as unidiff; should
526
either fix it up or prefer to use an external diff.
528
TODO: If a directory is given, diff everything under that.
530
TODO: Selected-file diff is inefficient and doesn't show you
533
TODO: This probably handles non-Unix newlines poorly.
1069
bzr diff --diff-prefix old/:new/
1070
bzr diff bzr.mine bzr.dev
1073
# TODO: Option to use external diff command; could be GNU diff, wdiff,
1074
# or a graphical diff.
1076
# TODO: Python difflib is not exactly the same as unidiff; should
1077
# either fix it up or prefer to use an external diff.
1079
# TODO: Selected-file diff is inefficient and doesn't show you
1082
# TODO: This probably handles non-Unix newlines poorly.
1084
541
takes_args = ['file*']
1085
takes_options = ['revision', 'diff-options', 'prefix']
542
takes_options = ['revision', 'diff-options']
1086
543
aliases = ['di', 'dif']
1087
encoding_type = 'exact'
1090
def run(self, revision=None, file_list=None, diff_options=None,
1092
from bzrlib.diff import diff_cmd_helper, show_diff_trees
1094
if (prefix is None) or (prefix == '0'):
545
def run(self, revision=None, file_list=None, diff_options=None):
546
from bzrlib.diff import show_diff
549
b = find_branch(file_list[0])
550
file_list = [b.relpath(f) for f in file_list]
551
if file_list == ['']:
552
# just pointing to top-of-tree
1102
if not ':' in prefix:
1103
raise BzrError("--diff-prefix expects two values separated by a colon")
1104
old_label, new_label = prefix.split(":")
1107
tree1, file_list = internal_tree_files(file_list)
1111
except FileInWrongBranch:
1112
if len(file_list) != 2:
1113
raise BzrCommandError("Files are in different branches")
1115
tree1, file1 = WorkingTree.open_containing(file_list[0])
1116
tree2, file2 = WorkingTree.open_containing(file_list[1])
1117
if file1 != "" or file2 != "":
1118
# FIXME diff those two files. rbc 20051123
1119
raise BzrCommandError("Files are in different branches")
1121
557
if revision is not None:
1122
if tree2 is not None:
1123
raise BzrCommandError("Can't specify -r with two branches")
1124
if (len(revision) == 1) or (revision[1].spec is None):
1125
return diff_cmd_helper(tree1, file_list, diff_options,
1127
old_label=old_label, new_label=new_label)
558
if len(revision) == 1:
559
show_diff(b, revision[0], specific_files=file_list,
560
external_diff_options=diff_options)
1128
561
elif len(revision) == 2:
1129
return diff_cmd_helper(tree1, file_list, diff_options,
1130
revision[0], revision[1],
1131
old_label=old_label, new_label=new_label)
562
show_diff(b, revision[0], specific_files=file_list,
563
external_diff_options=diff_options,
564
revision2=revision[1])
1133
566
raise BzrCommandError('bzr diff --revision takes exactly one or two revision identifiers')
1135
if tree2 is not None:
1136
return show_diff_trees(tree1, tree2, sys.stdout,
1137
specific_files=file_list,
1138
external_diff_options=diff_options,
1139
old_label=old_label, new_label=new_label)
1141
return diff_cmd_helper(tree1, file_list, diff_options,
1142
old_label=old_label, new_label=new_label)
568
show_diff(b, None, specific_files=file_list,
569
external_diff_options=diff_options)
1145
574
class cmd_deleted(Command):
1146
575
"""List files deleted in the working tree.
577
TODO: Show files deleted since a previous revision, or between two revisions.
1148
# TODO: Show files deleted since a previous revision, or
1149
# between two revisions.
1150
# TODO: Much more efficient way to do this: read in new
1151
# directories with readdir, rather than stating each one. Same
1152
# level of effort but possibly much less IO. (Or possibly not,
1153
# if the directories are very large...)
1154
takes_options = ['show-ids']
1157
579
def run(self, show_ids=False):
1158
tree = WorkingTree.open_containing(u'.')[0]
1159
old = tree.basis_tree()
582
new = b.working_tree()
584
## TODO: Much more efficient way to do this: read in new
585
## directories with readdir, rather than stating each one. Same
586
## level of effort but possibly much less IO. (Or possibly not,
587
## if the directories are very large...)
1160
589
for path, ie in old.inventory.iter_entries():
1161
if not tree.has_id(ie.file_id):
1162
self.outf.write(path)
590
if not new.has_id(ie.file_id):
1164
self.outf.write(' ')
1165
self.outf.write(ie.file_id)
1166
self.outf.write('\n')
592
print '%-50s %s' % (path, ie.file_id)
1169
597
class cmd_modified(Command):
1170
598
"""List files modified in working tree."""
1174
601
from bzrlib.delta import compare_trees
1176
tree = WorkingTree.open_containing(u'.')[0]
1177
td = compare_trees(tree.basis_tree(), tree)
1179
for path, id, kind, text_modified, meta_modified in td.modified:
1180
self.outf.write(path + '\n')
604
td = compare_trees(b.basis_tree(), b.working_tree())
606
for path, id, kind in td.modified:
1183
611
class cmd_added(Command):
1184
612
"""List files added in working tree."""
1188
wt = WorkingTree.open_containing(u'.')[0]
1189
basis_inv = wt.basis_tree().inventory
616
wt = b.working_tree()
617
basis_inv = b.basis_tree().inventory
1190
618
inv = wt.inventory
1191
619
for file_id in inv:
1192
620
if file_id in basis_inv:
1194
622
path = inv.id2path(file_id)
1195
if not os.access(bzrlib.osutils.abspath(path), os.F_OK):
623
if not os.access(b.abspath(path), os.F_OK):
1197
self.outf.write(path + '\n')
1200
629
class cmd_root(Command):
1201
630
"""Show the tree root directory.
1203
632
The root is the nearest enclosing directory with a .bzr control
1205
634
takes_args = ['filename?']
1207
635
def run(self, filename=None):
1208
636
"""Print the branch root."""
1209
tree = WorkingTree.open_containing(filename)[0]
1210
self.outf.write(tree.basedir + '\n')
637
b = find_branch(filename)
638
print getattr(b, 'base', None) or getattr(b, 'baseurl')
1213
641
class cmd_log(Command):
1214
"""Show log of a branch, file, or directory.
1216
By default show the log of the branch containing the working directory.
1218
To request a range of logs, you can use the command -r begin..end
1219
-r revision requests a specific revision, -r ..end or -r begin.. are
642
"""Show log of this branch.
644
To request a range of logs, you can use the command -r begin:end
645
-r revision requests a specific revision, -r :end or -r begin: are
1225
bzr log -r -10.. http://server/branch
648
--message allows you to give a regular expression, which will be evaluated
649
so that only matching entries will be displayed.
651
TODO: Make --revision support uuid: and hash: [future tag:] notation.
1228
# TODO: Make --revision support uuid: and hash: [future tag:] notation.
1230
takes_args = ['location?']
1231
takes_options = [Option('forward',
1232
help='show from oldest to newest'),
1235
help='show files changed in each revision'),
1236
'show-ids', 'revision',
1240
help='show revisions whose message matches this regexp',
1244
encoding_type = 'replace'
1247
def run(self, location=None, timezone='original',
655
takes_args = ['filename?']
656
takes_options = ['forward', 'timezone', 'verbose', 'show-ids', 'revision',
657
'long', 'message', 'short',]
659
def run(self, filename=None, timezone='original',
1257
667
from bzrlib.log import log_formatter, show_log
1258
assert message is None or isinstance(message, basestring), \
1259
"invalid message argument %r" % message
1260
670
direction = (forward and 'forward') or 'reverse'
1265
# find the file id to log:
1267
dir, fp = bzrdir.BzrDir.open_containing(location)
1268
b = dir.open_branch()
1272
inv = dir.open_workingtree().inventory
1273
except (errors.NotBranchError, errors.NotLocalUrl):
1274
# either no tree, or is remote.
1275
inv = b.basis_tree().inventory
1276
file_id = inv.path2id(fp)
673
b = find_branch(filename)
674
fp = b.relpath(filename)
676
file_id = b.read_working_inventory().path2id(fp)
678
file_id = None # points to branch root
1279
# FIXME ? log the current subdir only RBC 20060203
1280
dir, relpath = bzrdir.BzrDir.open_containing('.')
1281
b = dir.open_branch()
1283
683
if revision is None:
1286
686
elif len(revision) == 1:
1287
rev1 = rev2 = revision[0].in_history(b).revno
687
rev1 = rev2 = b.get_revision_info(revision[0])[0]
1288
688
elif len(revision) == 2:
1289
if revision[0].spec is None:
1290
# missing begin-range means first revision
1293
rev1 = revision[0].in_history(b).revno
1295
if revision[1].spec is None:
1296
# missing end-range means last known revision
1299
rev2 = revision[1].in_history(b).revno
689
rev1 = b.get_revision_info(revision[0])[0]
690
rev2 = b.get_revision_info(revision[1])[0]
1301
692
raise BzrCommandError('bzr log --revision takes one or two values.')
1303
# By this point, the revision numbers are converted to the +ve
1304
# form if they were supplied in the -ve form, so we can do
1305
# this comparison in relative safety
1307
(rev2, rev1) = (rev1, rev2)
1309
if (log_format == None):
1310
default = bzrlib.config.BranchConfig(b).log_format()
1311
log_format = get_log_format(long=long, short=short, line=line, default=default)
699
mutter('encoding log as %r' % bzrlib.user_encoding)
701
# use 'replace' so that we don't abort if trying to write out
702
# in e.g. the default C locale.
703
outf = codecs.getwriter(bzrlib.user_encoding)(sys.stdout, errors='replace')
1312
709
lf = log_formatter(log_format,
1313
710
show_ids=show_ids,
1315
712
show_timezone=timezone)
1327
def get_log_format(long=False, short=False, line=False, default='long'):
1328
log_format = default
1332
log_format = 'short'
1338
725
class cmd_touching_revisions(Command):
1339
726
"""Return revision-ids which affected a particular file.
1341
A more user-friendly interface is "bzr log FILE".
728
A more user-friendly interface is "bzr log FILE"."""
1345
730
takes_args = ["filename"]
1348
731
def run(self, filename):
1349
tree, relpath = WorkingTree.open_containing(filename)
1351
inv = tree.read_working_inventory()
1352
file_id = inv.path2id(relpath)
732
b = find_branch(filename)
733
inv = b.read_working_inventory()
734
file_id = inv.path2id(b.relpath(filename))
1353
735
for revno, revision_id, what in bzrlib.log.find_touching_revisions(b, file_id):
1354
self.outf.write("%6d %s\n" % (revno, what))
736
print "%6d %s" % (revno, what)
1357
739
class cmd_ls(Command):
1358
740
"""List files in a tree.
742
TODO: Take a revision or remote path and list that tree instead.
1360
# TODO: Take a revision or remote path and list that tree instead.
1362
takes_options = ['verbose', 'revision',
1363
Option('non-recursive',
1364
help='don\'t recurse into sub-directories'),
1366
help='Print all paths from the root of the branch.'),
1367
Option('unknown', help='Print unknown files'),
1368
Option('versioned', help='Print versioned files'),
1369
Option('ignored', help='Print ignored files'),
1371
Option('null', help='Null separate the files'),
1374
def run(self, revision=None, verbose=False,
1375
non_recursive=False, from_root=False,
1376
unknown=False, versioned=False, ignored=False,
1379
if verbose and null:
1380
raise BzrCommandError('Cannot set both --verbose and --null')
1381
all = not (unknown or versioned or ignored)
1383
selection = {'I':ignored, '?':unknown, 'V':versioned}
1385
tree, relpath = WorkingTree.open_containing(u'.')
1390
if revision is not None:
1391
tree = tree.branch.repository.revision_tree(
1392
revision[0].in_history(tree.branch).rev_id)
1394
for fp, fc, kind, fid, entry in tree.list_files():
1395
if fp.startswith(relpath):
1396
fp = fp[len(relpath):]
1397
if non_recursive and '/' in fp:
1399
if not all and not selection[fc]:
1402
kindch = entry.kind_character()
1403
self.outf.write('%-8s %s%s\n' % (fc, fp, kindch))
1405
self.outf.write(fp + '\0')
745
def run(self, revision=None, verbose=False):
748
tree = b.working_tree()
750
tree = b.revision_tree(b.lookup_revision(revision))
752
for fp, fc, kind, fid in tree.list_files():
754
if kind == 'directory':
1408
self.outf.write(fp + '\n')
761
print '%-8s %s%s' % (fc, fp, kindch)
1411
767
class cmd_unknowns(Command):
1412
768
"""List unknown files."""
1415
770
from bzrlib.osutils import quotefn
1416
for f in WorkingTree.open_containing(u'.')[0].unknowns():
1417
self.outf.write(quotefn(f) + '\n')
771
for f in find_branch('.').unknowns():
1420
776
class cmd_ignore(Command):
1600
935
A selected-file commit may fail in some cases where the committed
1601
936
tree would be invalid, such as trying to commit a file in a
1602
937
newly-added directory that is not itself committed.
939
TODO: Run hooks on tree to-be-committed, and after commit.
941
TODO: Strict commit that fails if there are unknown or deleted files.
1604
# TODO: Run hooks on tree to-be-committed, and after commit.
1606
# TODO: Strict commit that fails if there are deleted files.
1607
# (what does "deleted files" mean ??)
1609
# TODO: Give better message for -s, --summary, used by tla people
1611
# XXX: verbose currently does nothing
1613
943
takes_args = ['selected*']
1614
takes_options = ['message', 'verbose',
1616
help='commit even if nothing has changed'),
1617
Option('file', type=str,
1619
help='file containing commit message'),
1621
help="refuse to commit if there are unknown "
1622
"files in the working tree."),
1624
help="perform a local only commit in a bound "
1625
"branch. Such commits are not pushed to "
1626
"the master branch until a normal commit "
944
takes_options = ['message', 'file', 'verbose', 'unchanged']
1630
945
aliases = ['ci', 'checkin']
947
# TODO: Give better message for -s, --summary, used by tla people
1632
949
def run(self, message=None, file=None, verbose=True, selected_list=None,
1633
unchanged=False, strict=False, local=False):
1634
from bzrlib.commit import (NullCommitReporter, ReportCommitToLog)
1635
from bzrlib.errors import (PointlessCommit, ConflictsInTree,
1637
from bzrlib.msgeditor import edit_commit_message, \
1638
make_commit_message_template
1639
from tempfile import TemporaryFile
1641
# TODO: Need a blackbox test for invoking the external editor; may be
1642
# slightly problematic to run this cross-platform.
1644
# TODO: do more checks that the commit will succeed before
1645
# spending the user's valuable time typing a commit message.
1647
# TODO: if the commit *does* happen to fail, then save the commit
1648
# message to a temporary file where it can be recovered
1649
tree, selected_list = tree_files(selected_list)
1650
if selected_list == ['']:
1651
# workaround - commit of root of tree should be exactly the same
1652
# as just default commit in that tree, and succeed even though
1653
# selected-file merge commit is not done yet
1656
if local and not tree.branch.get_bound_location():
1657
raise errors.LocalRequiresBoundBranch()
1658
if message is None and not file:
1659
template = make_commit_message_template(tree, selected_list)
1660
message = edit_commit_message(template)
951
from bzrlib.errors import PointlessCommit
952
from bzrlib.msgeditor import edit_commit_message
953
from bzrlib.status import show_status
954
from cStringIO import StringIO
958
selected_list = [b.relpath(s) for s in selected_list]
960
if not message and not file:
962
show_status(b, specific_files=selected_list,
964
message = edit_commit_message(catcher.getvalue())
1661
966
if message is None:
1662
967
raise BzrCommandError("please specify a commit message"
1663
968
" with either --message or --file")
1733
1019
if c.needs_write:
1737
1024
class cmd_upgrade(Command):
1738
1025
"""Upgrade branch storage to current format.
1740
1027
The check command or bzr developers may sometimes advise you to run
1741
this command. When the default format has changed you may also be warned
1742
during other operations to upgrade.
1744
takes_args = ['url?']
1747
help='Upgrade to a specific format. Current formats'
1748
' are: default, knit, metaweave and weave.'
1749
' Default is knit; metaweave and weave are'
1751
type=get_format_type),
1755
def run(self, url='.', format=None):
1030
takes_args = ['dir?']
1032
def run(self, dir='.'):
1756
1033
from bzrlib.upgrade import upgrade
1758
format = get_format_type('default')
1759
upgrade(url, format)
1034
upgrade(find_branch(dir))
1762
1038
class cmd_whoami(Command):
1763
1039
"""Show bzr user id."""
1764
1040
takes_options = ['email']
1767
1042
def run(self, email=False):
1769
b = WorkingTree.open_containing(u'.')[0].branch
1770
config = bzrlib.config.BranchConfig(b)
1771
except NotBranchError:
1772
config = bzrlib.config.GlobalConfig()
1044
b = bzrlib.branch.find_branch('.')
1775
print config.user_email()
1777
print config.username()
1780
class cmd_nick(Command):
1781
"""Print or set the branch nickname.
1783
If unset, the tree root directory name is used as the nickname
1784
To print the current nickname, execute with no argument.
1786
takes_args = ['nickname?']
1787
def run(self, nickname=None):
1788
branch = Branch.open_containing(u'.')[0]
1789
if nickname is None:
1790
self.printme(branch)
1792
branch.nick = nickname
1795
def printme(self, branch):
1049
print bzrlib.osutils.user_email(b)
1051
print bzrlib.osutils.username(b)
1799
1054
class cmd_selftest(Command):
1800
"""Run internal test suite.
1802
This creates temporary test directories in the working directory,
1803
but not existing data is affected. These directories are deleted
1804
if the tests pass, or left behind to help in debugging if they
1805
fail and --keep-output is specified.
1807
If arguments are given, they are regular expressions that say
1808
which tests should run.
1810
If the global option '--no-plugins' is given, plugins are not loaded
1811
before running the selftests. This has two effects: features provided or
1812
modified by plugins will not be tested, and tests provided by plugins will
1817
bzr --no-plugins selftest -v
1819
# TODO: --list should give a list of all available tests
1821
# NB: this is used from the class without creating an instance, which is
1822
# why it does not have a self parameter.
1823
def get_transport_type(typestring):
1824
"""Parse and return a transport specifier."""
1825
if typestring == "sftp":
1826
from bzrlib.transport.sftp import SFTPAbsoluteServer
1827
return SFTPAbsoluteServer
1828
if typestring == "memory":
1829
from bzrlib.transport.memory import MemoryServer
1831
if typestring == "fakenfs":
1832
from bzrlib.transport.fakenfs import FakeNFSServer
1833
return FakeNFSServer
1834
msg = "No known transport type %s. Supported types are: sftp\n" %\
1836
raise BzrCommandError(msg)
1055
"""Run internal test suite"""
1839
takes_args = ['testspecs*']
1840
takes_options = ['verbose',
1841
Option('one', help='stop when one test fails'),
1842
Option('keep-output',
1843
help='keep output directories when tests fail'),
1845
help='Use a different transport by default '
1846
'throughout the test suite.',
1847
type=get_transport_type),
1848
Option('benchmark', help='run the bzr bencharks.'),
1849
Option('lsprof-timed',
1850
help='generate lsprof output for benchmarked'
1851
' sections of code.'),
1854
def run(self, testspecs_list=None, verbose=None, one=False,
1855
keep_output=False, transport=None, benchmark=None,
1057
takes_options = ['verbose', 'pattern']
1058
def run(self, verbose=False, pattern=".*"):
1857
1059
import bzrlib.ui
1858
from bzrlib.tests import selftest
1859
import bzrlib.benchmarks as benchmarks
1060
from bzrlib.selftest import selftest
1860
1061
# we don't want progress meters from the tests to go to the
1861
1062
# real output; and we don't want log messages cluttering up
1862
1063
# the real logs.
1863
1064
save_ui = bzrlib.ui.ui_factory
1864
print '%10s: %s' % ('bzr', bzrlib.osutils.realpath(sys.argv[0]))
1865
print '%10s: %s' % ('bzrlib', bzrlib.__path__[0])
1867
1065
bzrlib.trace.info('running tests...')
1869
1067
bzrlib.ui.ui_factory = bzrlib.ui.SilentUIFactory()
1870
if testspecs_list is not None:
1871
pattern = '|'.join(testspecs_list)
1875
test_suite_factory = benchmarks.test_suite
1879
test_suite_factory = None
1882
result = selftest(verbose=verbose,
1884
stop_on_failure=one,
1885
keep_output=keep_output,
1886
transport=transport,
1887
test_suite_factory=test_suite_factory,
1888
lsprof_timed=lsprof_timed)
1068
result = selftest(verbose=verbose, pattern=pattern)
1890
1070
bzrlib.trace.info('tests passed')
2033
1168
merge refuses to run if there are any uncommitted changes, unless
2034
1169
--force is given.
2036
The following merge types are available:
2038
1171
takes_args = ['branch?']
2039
takes_options = ['revision', 'force', 'merge-type', 'reprocess', 'remember',
2040
Option('show-base', help="Show base revision text in "
2044
from merge import merge_type_help
2045
from inspect import getdoc
2046
return getdoc(self) + '\n' + merge_type_help()
2048
def run(self, branch=None, revision=None, force=False, merge_type=None,
2049
show_base=False, reprocess=False, remember=False):
1172
takes_options = ['revision', 'force', 'merge-type']
1174
def run(self, branch='.', revision=None, force=False,
1176
from bzrlib.merge import merge
1177
from bzrlib.merge_core import ApplyMerge3
2050
1178
if merge_type is None:
2051
merge_type = Merge3Merger
2053
tree = WorkingTree.open_containing(u'.')[0]
2056
if branch is not None:
2057
reader = BundleReader(file(branch, 'rb'))
2061
if e.errno not in (errno.ENOENT, errno.EISDIR):
2066
if reader is not None:
2067
conflicts = merge_bundle(reader, tree, not force, merge_type,
2068
reprocess, show_base)
2074
branch = self._get_remembered_parent(tree, branch, 'Merging from')
1179
merge_type = ApplyMerge3
2076
1181
if revision is None or len(revision) < 1:
2077
1182
base = [None, None]
2078
1183
other = [branch, -1]
2079
other_branch, path = Branch.open_containing(branch)
2081
1185
if len(revision) == 1:
1186
other = [branch, revision[0]]
2082
1187
base = [None, None]
2083
other_branch, path = Branch.open_containing(branch)
2084
revno = revision[0].in_history(other_branch).revno
2085
other = [branch, revno]
2087
1189
assert len(revision) == 2
2088
1190
if None in revision:
2089
1191
raise BzrCommandError(
2090
1192
"Merge doesn't permit that revision specifier.")
2091
other_branch, path = Branch.open_containing(branch)
2093
base = [branch, revision[0].in_history(other_branch).revno]
2094
other = [branch, revision[1].in_history(other_branch).revno]
2096
if tree.branch.get_parent() is None or remember:
2097
tree.branch.set_parent(other_branch.base)
2100
interesting_files = [path]
2102
interesting_files = None
2103
pb = bzrlib.ui.ui_factory.nested_progress_bar()
1193
base = [branch, revision[0]]
1194
other = [branch, revision[1]]
2106
conflict_count = merge(other, base, check_clean=(not force),
2107
merge_type=merge_type,
2108
reprocess=reprocess,
2109
show_base=show_base,
2110
pb=pb, file_list=interesting_files)
2113
if conflict_count != 0:
1197
merge(other, base, check_clean=(not force), merge_type=merge_type)
2117
1198
except bzrlib.errors.AmbiguousBase, e:
2118
m = ("sorry, bzr can't determine the right merge base yet\n"
1199
m = ("sorry, bzr can't determine the write merge base yet\n"
2119
1200
"candidates are:\n "
2120
1201
+ "\n ".join(e.bases)
2123
1204
"and (if you want) report this to the bzr developers\n")
2126
# TODO: move up to common parent; this isn't merge-specific anymore.
2127
def _get_remembered_parent(self, tree, supplied_location, verb_string):
2128
"""Use tree.branch's parent if none was supplied.
2130
Report if the remembered location was used.
2132
if supplied_location is not None:
2133
return supplied_location
2134
stored_location = tree.branch.get_parent()
2135
mutter("%s", stored_location)
2136
if stored_location is None:
2137
raise BzrCommandError("No location specified or remembered")
2138
display_url = urlutils.unescape_for_display(stored_location, self.outf.encoding)
2139
self.outf.write("%s remembered location %s\n" % (verb_string, display_url))
2140
return stored_location
2143
class cmd_remerge(Command):
2146
Use this if you want to try a different merge technique while resolving
2147
conflicts. Some merge techniques are better than others, and remerge
2148
lets you try different ones on different files.
2150
The options for remerge have the same meaning and defaults as the ones for
2151
merge. The difference is that remerge can (only) be run when there is a
2152
pending merge, and it lets you specify particular files.
2155
$ bzr remerge --show-base
2156
Re-do the merge of all conflicted files, and show the base text in
2157
conflict regions, in addition to the usual THIS and OTHER texts.
2159
$ bzr remerge --merge-type weave --reprocess foobar
2160
Re-do the merge of "foobar", using the weave merge algorithm, with
2161
additional processing to reduce the size of conflict regions.
2163
The following merge types are available:"""
2164
takes_args = ['file*']
2165
takes_options = ['merge-type', 'reprocess',
2166
Option('show-base', help="Show base revision text in "
2170
from merge import merge_type_help
2171
from inspect import getdoc
2172
return getdoc(self) + '\n' + merge_type_help()
2174
def run(self, file_list=None, merge_type=None, show_base=False,
2176
from bzrlib.merge import merge_inner, transform_tree
2177
if merge_type is None:
2178
merge_type = Merge3Merger
2179
tree, file_list = tree_files(file_list)
2182
pending_merges = tree.pending_merges()
2183
if len(pending_merges) != 1:
2184
raise BzrCommandError("Sorry, remerge only works after normal"
2185
+ " merges. Not cherrypicking or"
2187
repository = tree.branch.repository
2188
base_revision = common_ancestor(tree.branch.last_revision(),
2189
pending_merges[0], repository)
2190
base_tree = repository.revision_tree(base_revision)
2191
other_tree = repository.revision_tree(pending_merges[0])
2192
interesting_ids = None
2193
if file_list is not None:
2194
interesting_ids = set()
2195
for filename in file_list:
2196
file_id = tree.path2id(filename)
2198
raise NotVersionedError(filename)
2199
interesting_ids.add(file_id)
2200
if tree.kind(file_id) != "directory":
2203
for name, ie in tree.inventory.iter_entries(file_id):
2204
interesting_ids.add(ie.file_id)
2205
transform_tree(tree, tree.basis_tree(), interesting_ids)
2206
if file_list is None:
2207
restore_files = list(tree.iter_conflicts())
2209
restore_files = file_list
2210
for filename in restore_files:
2212
restore(tree.abspath(filename))
2213
except NotConflicted:
2215
conflicts = merge_inner(tree.branch, other_tree, base_tree,
2217
interesting_ids = interesting_ids,
2218
other_rev_id=pending_merges[0],
2219
merge_type=merge_type,
2220
show_base=show_base,
2221
reprocess=reprocess)
2229
1208
class cmd_revert(Command):
2230
1209
"""Reverse all changes since the last commit.
2293
1267
aliases = ['s-c']
2297
1270
def run(self, context=None):
2298
1271
import shellcomplete
2299
1272
shellcomplete.shellcomplete(context)
2302
class cmd_fetch(Command):
2303
"""Copy in history from another branch but don't merge it.
2305
This is an internal method used for pull and merge."""
2307
takes_args = ['from_branch', 'to_branch']
2308
def run(self, from_branch, to_branch):
2309
from bzrlib.fetch import Fetcher
2310
from bzrlib.branch import Branch
2311
from_b = Branch.open(from_branch)
2312
to_b = Branch.open(to_branch)
2313
Fetcher(to_b, from_b)
2316
1275
class cmd_missing(Command):
2317
"""Show unmerged/unpulled revisions between two branches.
2319
OTHER_BRANCH may be local or remote."""
2320
takes_args = ['other_branch?']
2321
takes_options = [Option('reverse', 'Reverse the order of revisions'),
2323
'Display changes in the local branch only'),
2324
Option('theirs-only',
2325
'Display changes in the remote branch only'),
2334
def run(self, other_branch=None, reverse=False, mine_only=False,
2335
theirs_only=False, log_format=None, long=False, short=False, line=False,
2336
show_ids=False, verbose=False):
2337
from bzrlib.missing import find_unmerged, iter_log_data
2338
from bzrlib.log import log_formatter
2339
local_branch = bzrlib.branch.Branch.open_containing(u".")[0]
2340
parent = local_branch.get_parent()
2341
if other_branch is None:
2342
other_branch = parent
2343
if other_branch is None:
1276
"""What is missing in this branch relative to other branch.
1278
takes_args = ['remote?']
1279
aliases = ['mis', 'miss']
1280
# We don't have to add quiet to the list, because
1281
# unknown options are parsed as booleans
1282
takes_options = ['verbose', 'quiet']
1284
def run(self, remote=None, verbose=False, quiet=False):
1285
from bzrlib.errors import BzrCommandError
1286
from bzrlib.missing import show_missing
1288
if verbose and quiet:
1289
raise BzrCommandError('Cannot pass both quiet and verbose')
1291
b = find_branch('.')
1292
parent = b.get_parent()
2344
1295
raise BzrCommandError("No missing location known or specified.")
2345
print "Using last location: " + local_branch.get_parent()
2346
remote_branch = bzrlib.branch.Branch.open(other_branch)
2347
if remote_branch.base == local_branch.base:
2348
remote_branch = local_branch
2349
local_branch.lock_read()
2351
remote_branch.lock_read()
2353
local_extra, remote_extra = find_unmerged(local_branch, remote_branch)
2354
if (log_format == None):
2355
default = bzrlib.config.BranchConfig(local_branch).log_format()
2356
log_format = get_log_format(long=long, short=short, line=line, default=default)
2357
lf = log_formatter(log_format, sys.stdout,
2359
show_timezone='original')
2360
if reverse is False:
2361
local_extra.reverse()
2362
remote_extra.reverse()
2363
if local_extra and not theirs_only:
2364
print "You have %d extra revision(s):" % len(local_extra)
2365
for data in iter_log_data(local_extra, local_branch.repository,
2368
printed_local = True
2370
printed_local = False
2371
if remote_extra and not mine_only:
2372
if printed_local is True:
2374
print "You are missing %d revision(s):" % len(remote_extra)
2375
for data in iter_log_data(remote_extra, remote_branch.repository,
2378
if not remote_extra and not local_extra:
2380
print "Branches are up to date."
2384
remote_branch.unlock()
2386
local_branch.unlock()
2387
if not status_code and parent is None and other_branch is not None:
2388
local_branch.lock_write()
2390
# handle race conditions - a parent might be set while we run.
2391
if local_branch.get_parent() is None:
2392
local_branch.set_parent(remote_branch.base)
2394
local_branch.unlock()
1298
print "Using last location: %s" % parent
1300
elif parent is None:
1301
# We only update x-pull if it did not exist, missing should not change the parent
1302
b.controlfile('x-pull', 'wb').write(remote + '\n')
1303
br_remote = find_branch(remote)
1305
return show_missing(b, br_remote, verbose=verbose, quiet=quiet)
2398
1309
class cmd_plugins(Command):
2399
1310
"""List plugins"""
2403
1313
import bzrlib.plugin
2404
1314
from inspect import getdoc
2405
for name, plugin in bzrlib.plugin.all_plugins().items():
1315
for plugin in bzrlib.plugin.all_plugins:
2406
1316
if hasattr(plugin, '__path__'):
2407
1317
print plugin.__path__[0]
2408
1318
elif hasattr(plugin, '__file__'):
2415
1325
print '\t', d.split('\n')[0]
2418
class cmd_testament(Command):
2419
"""Show testament (signing-form) of a revision."""
2420
takes_options = ['revision', 'long']
2421
takes_args = ['branch?']
2423
def run(self, branch=u'.', revision=None, long=False):
2424
from bzrlib.testament import Testament
2425
b = WorkingTree.open_containing(branch)[0].branch
2428
if revision is None:
2429
rev_id = b.last_revision()
2431
rev_id = revision[0].in_history(b).rev_id
2432
t = Testament.from_revision(b.repository, rev_id)
2434
sys.stdout.writelines(t.as_text_lines())
2436
sys.stdout.write(t.as_short_text())
2441
class cmd_annotate(Command):
2442
"""Show the origin of each line in a file.
2444
This prints out the given file with an annotation on the left side
2445
indicating which revision, author and date introduced the change.
2447
If the origin is the same for a run of consecutive lines, it is
2448
shown only at the top, unless the --all option is given.
2450
# TODO: annotate directories; showing when each file was last changed
2451
# TODO: if the working copy is modified, show annotations on that
2452
# with new uncommitted lines marked
2453
aliases = ['blame', 'praise']
2454
takes_args = ['filename']
2455
takes_options = [Option('all', help='show annotations on all lines'),
2456
Option('long', help='show date in annotations'),
2461
def run(self, filename, all=False, long=False, revision=None):
2462
from bzrlib.annotate import annotate_file
2463
tree, relpath = WorkingTree.open_containing(filename)
2464
branch = tree.branch
2467
if revision is None:
2468
revision_id = branch.last_revision()
2469
elif len(revision) != 1:
2470
raise BzrCommandError('bzr annotate --revision takes exactly 1 argument')
2472
revision_id = revision[0].in_history(branch).rev_id
2473
file_id = tree.inventory.path2id(relpath)
2474
tree = branch.repository.revision_tree(revision_id)
2475
file_version = tree.inventory[file_id].revision
2476
annotate_file(branch, file_version, file_id, long, all, sys.stdout)
2481
class cmd_re_sign(Command):
2482
"""Create a digital signature for an existing revision."""
2483
# TODO be able to replace existing ones.
2485
hidden = True # is this right ?
2486
takes_args = ['revision_id*']
2487
takes_options = ['revision']
2489
def run(self, revision_id_list=None, revision=None):
2490
import bzrlib.config as config
2491
import bzrlib.gpg as gpg
2492
if revision_id_list is not None and revision is not None:
2493
raise BzrCommandError('You can only supply one of revision_id or --revision')
2494
if revision_id_list is None and revision is None:
2495
raise BzrCommandError('You must supply either --revision or a revision_id')
2496
b = WorkingTree.open_containing(u'.')[0].branch
2497
gpg_strategy = gpg.GPGStrategy(config.BranchConfig(b))
2498
if revision_id_list is not None:
2499
for revision_id in revision_id_list:
2500
b.repository.sign_revision(revision_id, gpg_strategy)
2501
elif revision is not None:
2502
if len(revision) == 1:
2503
revno, rev_id = revision[0].in_history(b)
2504
b.repository.sign_revision(rev_id, gpg_strategy)
2505
elif len(revision) == 2:
2506
# are they both on rh- if so we can walk between them
2507
# might be nice to have a range helper for arbitrary
2508
# revision paths. hmm.
2509
from_revno, from_revid = revision[0].in_history(b)
2510
to_revno, to_revid = revision[1].in_history(b)
2511
if to_revid is None:
2512
to_revno = b.revno()
2513
if from_revno is None or to_revno is None:
2514
raise BzrCommandError('Cannot sign a range of non-revision-history revisions')
2515
for revno in range(from_revno, to_revno + 1):
2516
b.repository.sign_revision(b.get_rev_id(revno),
2519
raise BzrCommandError('Please supply either one revision, or a range.')
2522
class cmd_bind(Command):
2523
"""Bind the current branch to a master branch.
2525
After binding, commits must succeed on the master branch
2526
before they are executed on the local one.
2529
takes_args = ['location']
2532
def run(self, location=None):
2533
b, relpath = Branch.open_containing(u'.')
2534
b_other = Branch.open(location)
2537
except DivergedBranches:
2538
raise BzrCommandError('These branches have diverged.'
2539
' Try merging, and then bind again.')
2542
class cmd_unbind(Command):
2543
"""Unbind the current branch from its master branch.
2545
After unbinding, the local branch is considered independent.
2546
All subsequent commits will be local.
2553
b, relpath = Branch.open_containing(u'.')
2555
raise BzrCommandError('Local branch is not bound')
2558
class cmd_uncommit(bzrlib.commands.Command):
2559
"""Remove the last committed revision.
2561
--verbose will print out what is being removed.
2562
--dry-run will go through all the motions, but not actually
2565
In the future, uncommit will create a revision bundle, which can then
2569
# TODO: jam 20060108 Add an option to allow uncommit to remove
2570
# unreferenced information in 'branch-as-repository' branches.
2571
# TODO: jam 20060108 Add the ability for uncommit to remove unreferenced
2572
# information in shared branches as well.
2573
takes_options = ['verbose', 'revision',
2574
Option('dry-run', help='Don\'t actually make changes'),
2575
Option('force', help='Say yes to all questions.')]
2576
takes_args = ['location?']
2579
def run(self, location=None,
2580
dry_run=False, verbose=False,
2581
revision=None, force=False):
2582
from bzrlib.branch import Branch
2583
from bzrlib.log import log_formatter
2585
from bzrlib.uncommit import uncommit
2587
if location is None:
2589
control, relpath = bzrdir.BzrDir.open_containing(location)
2591
tree = control.open_workingtree()
2593
except (errors.NoWorkingTree, errors.NotLocalUrl):
2595
b = control.open_branch()
2597
if revision is None:
2599
rev_id = b.last_revision()
2601
revno, rev_id = revision[0].in_history(b)
2603
print 'No revisions to uncommit.'
2605
for r in range(revno, b.revno()+1):
2606
rev_id = b.get_rev_id(r)
2607
lf = log_formatter('short', to_file=sys.stdout,show_timezone='original')
2608
lf.show(r, b.repository.get_revision(rev_id), None)
2611
print 'Dry-run, pretending to remove the above revisions.'
2613
val = raw_input('Press <enter> to continue')
2615
print 'The above revision(s) will be removed.'
2617
val = raw_input('Are you sure [y/N]? ')
2618
if val.lower() not in ('y', 'yes'):
2622
uncommit(b, tree=tree, dry_run=dry_run, verbose=verbose,
2626
class cmd_break_lock(Command):
2627
"""Break a dead lock on a repository, branch or working directory.
2629
CAUTION: Locks should only be broken when you are sure that the process
2630
holding the lock has been stopped.
2632
You can get information on what locks are open via the 'bzr info' command.
2637
takes_args = ['location?']
2639
def run(self, location=None, show=False):
2640
if location is None:
2642
control, relpath = bzrdir.BzrDir.open_containing(location)
2644
control.break_lock()
2645
except NotImplementedError:
2650
# command-line interpretation helper for merge-related commands
2651
def merge(other_revision, base_revision,
2652
check_clean=True, ignore_zero=False,
2653
this_dir=None, backup_files=False, merge_type=Merge3Merger,
2654
file_list=None, show_base=False, reprocess=False,
2655
pb=DummyProgress()):
2656
"""Merge changes into a tree.
2659
list(path, revno) Base for three-way merge.
2660
If [None, None] then a base will be automatically determined.
2662
list(path, revno) Other revision for three-way merge.
2664
Directory to merge changes into; '.' by default.
2666
If true, this_dir must have no uncommitted changes before the
2668
ignore_zero - If true, suppress the "zero conflicts" message when
2669
there are no conflicts; should be set when doing something we expect
2670
to complete perfectly.
2671
file_list - If supplied, merge only changes to selected files.
2673
All available ancestors of other_revision and base_revision are
2674
automatically pulled into the branch.
2676
The revno may be -1 to indicate the last revision on the branch, which is
2679
This function is intended for use from the command line; programmatic
2680
clients might prefer to call merge.merge_inner(), which has less magic
2683
from bzrlib.merge import Merger
2684
if this_dir is None:
2686
this_tree = WorkingTree.open_containing(this_dir)[0]
2687
if show_base and not merge_type is Merge3Merger:
2688
raise BzrCommandError("Show-base is not supported for this merge"
2689
" type. %s" % merge_type)
2690
if reprocess and not merge_type.supports_reprocess:
2691
raise BzrCommandError("Conflict reduction is not supported for merge"
2692
" type %s." % merge_type)
2693
if reprocess and show_base:
2694
raise BzrCommandError("Cannot do conflict reduction and show base.")
2696
merger = Merger(this_tree.branch, this_tree=this_tree, pb=pb)
2697
merger.pp = ProgressPhase("Merge phase", 5, pb)
2698
merger.pp.next_phase()
2699
merger.check_basis(check_clean)
2700
merger.set_other(other_revision)
2701
merger.pp.next_phase()
2702
merger.set_base(base_revision)
2703
if merger.base_rev_id == merger.other_rev_id:
2704
note('Nothing to do.')
2706
merger.backup_files = backup_files
2707
merger.merge_type = merge_type
2708
merger.set_interesting_files(file_list)
2709
merger.show_base = show_base
2710
merger.reprocess = reprocess
2711
conflicts = merger.do_merge()
2712
if file_list is None:
2713
merger.set_pending()
2719
# these get imported and then picked up by the scan for cmd_*
2720
# TODO: Some more consistent way to split command definitions across files;
2721
# we do need to load at least some information about them to know of
2722
# aliases. ideally we would avoid loading the implementation until the
2723
# details were needed.
2724
from bzrlib.conflicts import cmd_resolve, cmd_conflicts, restore
2725
from bzrlib.bundle.commands import cmd_bundle_revisions
2726
from bzrlib.sign_my_commits import cmd_sign_my_commits
2727
from bzrlib.weave_commands import cmd_weave_list, cmd_weave_join, \
2728
cmd_weave_plan_merge, cmd_weave_merge_text