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
23
from bzrlib.trace import mutter, note, log_error, warning
24
from bzrlib.errors import BzrError, BzrCheckError, BzrCommandError
28
25
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
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
26
from bzrlib import BZRDIR
27
from bzrlib.commands import Command
114
30
class cmd_status(Command):
115
31
"""Display status summary.
305
190
This is equivalent to creating the directory and then adding it.
308
192
takes_args = ['dir+']
309
encoding_type = 'replace'
311
194
def run(self, dir_list):
312
197
for d in dir_list:
314
wt, dd = WorkingTree.open_containing(d)
316
self.outf.write('added %s\n' % d)
200
b = Branch.open_containing(d)
319
205
class cmd_relpath(Command):
320
206
"""Show path of a file relative to root"""
322
207
takes_args = ['filename']
326
210
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')
211
print Branch.open_containing(filename).relpath(filename)
334
215
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']
216
"""Show inventory of the current working copy or a revision."""
217
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]
219
def run(self, revision=None, show_ids=False):
220
b = Branch.open_containing('.')
348
221
if revision is None:
349
inv = tree.read_working_inventory()
222
inv = b.read_working_inventory()
351
224
if len(revision) > 1:
352
225
raise BzrCommandError('bzr inventory --revision takes'
353
226
' exactly one revision identifier')
354
inv = tree.branch.repository.get_revision_inventory(
355
revision[0].in_history(tree.branch).rev_id)
227
inv = b.get_revision_inventory(revision[0].in_history(b).rev_id)
357
229
for path, entry in inv.entries():
358
if kind and kind != entry.kind:
361
self.outf.write('%-50s %s\n' % (path, entry.file_id))
231
print '%-50s %s' % (path, entry.file_id)
363
self.outf.write(path)
364
self.outf.write('\n')
236
class cmd_move(Command):
237
"""Move files to a different directory.
242
The destination must be a versioned directory in the same branch.
244
takes_args = ['source$', 'dest']
245
def run(self, source_list, dest):
246
b = Branch.open_containing('.')
248
# TODO: glob expansion on windows?
249
b.move([b.relpath(s) for s in source_list], b.relpath(dest))
252
class cmd_rename(Command):
253
"""Change the name of an entry.
256
bzr rename frob.c frobber.c
257
bzr rename src/frob.c lib/frob.c
259
It is an error if the destination name exists.
261
See also the 'move' command, which moves files into a different
262
directory without changing their name.
264
TODO: Some way to rename multiple files without invoking bzr for each
266
takes_args = ['from_name', 'to_name']
268
def run(self, from_name, to_name):
269
b = Branch.open_containing('.')
270
b.rename_one(b.relpath(from_name), b.relpath(to_name))
367
274
class cmd_mv(Command):
378
285
Files cannot be moved between branches.
381
287
takes_args = ['names*']
382
aliases = ['move', 'rename']
383
encoding_type = 'replace'
385
288
def run(self, names_list):
386
289
if len(names_list) < 2:
387
290
raise BzrCommandError("missing file argument")
388
tree, rel_names = tree_files(names_list)
291
b = Branch.open_containing(names_list[0])
293
rel_names = [b.relpath(x) for x in names_list]
390
295
if os.path.isdir(names_list[-1]):
391
296
# move into existing directory
392
for pair in tree.move(rel_names[:-1], rel_names[-1]):
393
self.outf.write("%s => %s\n" % pair)
297
for pair in b.move(rel_names[:-1], rel_names[-1]):
298
print "%s => %s" % pair
395
300
if len(names_list) != 2:
396
301
raise BzrCommandError('to mv multiple files the destination '
397
302
'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]))
303
b.rename_one(rel_names[0], rel_names[1])
304
print "%s => %s" % (rel_names[0], rel_names[1])
402
309
class cmd_pull(Command):
403
"""Turn this branch into a mirror of another branch.
310
"""Pull any changes from another branch into the current one.
312
If the location is omitted, the last-used location will be used.
313
Both the revision history and the working directory will be
405
316
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.
317
considered diverged if both branches have had commits without first
318
pulling from the other.
413
320
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.
321
from one into the other.
425
takes_options = ['remember', 'overwrite', 'revision', 'verbose']
426
323
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
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()
325
def run(self, location=None):
326
from bzrlib.merge import merge
328
from shutil import rmtree
331
br_to = Branch.open_containing('.')
332
stored_loc = br_to.get_parent()
438
333
if location is None:
439
334
if stored_loc is None:
440
335
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)
337
print "Using last location: %s" % stored_loc
338
location = stored_loc
339
cache_root = tempfile.mkdtemp()
340
from bzrlib.errors import DivergedBranches
341
br_from = Branch.open_containing(location)
342
location = br_from.base
343
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()
345
from bzrlib.errors import DivergedBranches
346
br_from = Branch.open(location)
347
br_from.setup_caching(cache_root)
348
location = br_from.base
349
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)
351
br_to.update_revisions(br_from)
576
352
except DivergedBranches:
577
353
raise BzrCommandError("These branches have diverged."
578
" Try a merge then push with overwrite.")
579
note('%d revision(s) pushed.' % (count,))
356
merge(('.', -1), ('.', old_revno), check_clean=False)
357
if location != stored_loc:
358
br_to.set_parent(location)
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
364
class cmd_branch(Command):
596
370
To retrieve the branch as of a particular revision, supply the --revision
597
371
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
373
takes_args = ['from_location', 'to_location?']
604
takes_options = ['revision', 'basis']
374
takes_options = ['revision']
605
375
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()
377
def run(self, from_location, to_location=None, revision=None):
378
from bzrlib.branch import copy_branch
381
from shutil import rmtree
382
cache_root = tempfile.mkdtemp()
386
elif len(revision) > 1:
387
raise BzrCommandError(
388
'bzr branch --revision takes exactly 1 revision value')
390
br_from = Branch.open(from_location)
392
if e.errno == errno.ENOENT:
393
raise BzrCommandError('Source location "%s" does not'
394
' exist.' % to_location)
397
br_from.setup_caching(cache_root)
636
398
if to_location is None:
637
399
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()
401
os.mkdir(to_location)
403
if e.errno == errno.EEXIST:
404
raise BzrCommandError('Target directory "%s" already'
405
' exists.' % to_location)
406
if e.errno == errno.ENOENT:
407
raise BzrCommandError('Parent of "%s" does not exist.' %
412
copy_branch(br_from, to_location, revision[0])
656
413
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
414
rmtree(to_location)
662
msg = "The branch %s cannot be used as a --basis" % (basis,)
415
msg = "The branch %s has no revision %d." % (from_location, revision[0])
663
416
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
421
class cmd_renames(Command):
760
422
"""Show list of renamed files.
424
TODO: Option to show renames between two historical versions.
426
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
428
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()
430
def run(self, dir='.'):
431
b = Branch.open_containing(dir)
432
old_inv = b.basis_tree().inventory
433
new_inv = b.read_working_inventory()
773
435
renames = list(bzrlib.tree.find_renames(old_inv, new_inv))
775
437
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()),))
438
print "%s => %s" % (old_name, new_name)
812
441
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],
442
"""Show statistical information about a branch."""
443
takes_args = ['branch?']
445
def run(self, branch=None):
448
b = Branch.open_containing(branch)
831
452
class cmd_remove(Command):
867
470
same through all revisions where the file exists, even when it is
868
471
moved or renamed.
872
474
takes_args = ['filename']
875
475
def run(self, filename):
876
tree, relpath = WorkingTree.open_containing(filename)
877
i = tree.inventory.path2id(relpath)
476
b = Branch.open_containing(filename)
477
i = b.inventory.path2id(b.relpath(filename))
879
479
raise BzrError("%r is not a versioned file" % filename)
881
self.outf.write(i + '\n')
884
484
class cmd_file_path(Command):
885
485
"""Print path of file_ids to a file or directory.
887
487
This prints one line for each directory down to the target,
888
starting at the branch root.
488
starting at the branch root."""
892
490
takes_args = ['filename']
895
491
def run(self, filename):
896
tree, relpath = WorkingTree.open_containing(filename)
898
fid = inv.path2id(relpath)
492
b = Branch.open_containing(filename)
494
fid = inv.path2id(b.relpath(filename))
900
496
raise BzrError("%r is not a versioned file" % filename)
901
497
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
501
class cmd_revision_history(Command):
932
502
"""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')
505
for patchid in Branch.open_containing('.').revision_history():
509
class cmd_directories(Command):
510
"""Display list of versioned directories in this branch."""
512
for name, ie in Branch.open_containing('.').read_working_inventory().directories():
959
519
class cmd_init(Command):
962
522
Use this to create an empty branch, or before importing an
963
523
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
525
Recipe for importing a tree of files:
978
530
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
533
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)
534
Branch.initialize('.')
1056
537
class cmd_diff(Command):
1059
540
If files are listed, only the changes in those files are listed.
1060
541
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".
543
TODO: Allow diff across branches.
545
TODO: Option to use external diff command; could be GNU diff, wdiff,
548
TODO: Python difflib is not exactly the same as unidiff; should
549
either fix it up or prefer to use an external diff.
551
TODO: If a directory is given, diff everything under that.
553
TODO: Selected-file diff is inefficient and doesn't show you
556
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
564
takes_args = ['file*']
1085
takes_options = ['revision', 'diff-options', 'prefix']
565
takes_options = ['revision', 'diff-options']
1086
566
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'):
568
def run(self, revision=None, file_list=None, diff_options=None):
569
from bzrlib.diff import show_diff
572
b = Branch.open_containing(file_list[0])
573
file_list = [b.relpath(f) for f in file_list]
574
if file_list == ['']:
575
# 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")
578
b = Branch.open_containing('.')
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
580
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)
581
if len(revision) == 1:
582
show_diff(b, revision[0], specific_files=file_list,
583
external_diff_options=diff_options)
1128
584
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)
585
show_diff(b, revision[0], specific_files=file_list,
586
external_diff_options=diff_options,
587
revision2=revision[1])
1133
589
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)
591
show_diff(b, None, specific_files=file_list,
592
external_diff_options=diff_options)
1145
597
class cmd_deleted(Command):
1146
598
"""List files deleted in the working tree.
600
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
602
def run(self, show_ids=False):
1158
tree = WorkingTree.open_containing(u'.')[0]
1159
old = tree.basis_tree()
603
b = Branch.open_containing('.')
605
new = b.working_tree()
607
## TODO: Much more efficient way to do this: read in new
608
## directories with readdir, rather than stating each one. Same
609
## level of effort but possibly much less IO. (Or possibly not,
610
## if the directories are very large...)
1160
612
for path, ie in old.inventory.iter_entries():
1161
if not tree.has_id(ie.file_id):
1162
self.outf.write(path)
613
if not new.has_id(ie.file_id):
1164
self.outf.write(' ')
1165
self.outf.write(ie.file_id)
1166
self.outf.write('\n')
615
print '%-50s %s' % (path, ie.file_id)
1169
620
class cmd_modified(Command):
1170
621
"""List files modified in working tree."""
1174
624
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')
626
b = Branch.open_containing('.')
627
td = compare_trees(b.basis_tree(), b.working_tree())
629
for path, id, kind in td.modified:
1183
634
class cmd_added(Command):
1184
635
"""List files added in working tree."""
1188
wt = WorkingTree.open_containing(u'.')[0]
1189
basis_inv = wt.basis_tree().inventory
638
b = Branch.open_containing('.')
639
wt = b.working_tree()
640
basis_inv = b.basis_tree().inventory
1190
641
inv = wt.inventory
1191
642
for file_id in inv:
1192
643
if file_id in basis_inv:
1194
645
path = inv.id2path(file_id)
1195
if not os.access(bzrlib.osutils.abspath(path), os.F_OK):
646
if not os.access(b.abspath(path), os.F_OK):
1197
self.outf.write(path + '\n')
1200
652
class cmd_root(Command):
1201
653
"""Show the tree root directory.
1203
655
The root is the nearest enclosing directory with a .bzr control
1205
657
takes_args = ['filename?']
1207
658
def run(self, filename=None):
1208
659
"""Print the branch root."""
1209
tree = WorkingTree.open_containing(filename)[0]
1210
self.outf.write(tree.basedir + '\n')
660
b = Branch.open_containing(filename)
1213
664
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
665
"""Show log of this branch.
667
To request a range of logs, you can use the command -r begin:end
668
-r revision requests a specific revision, -r :end or -r begin: are
1225
bzr log -r -10.. http://server/branch
671
--message allows you to give a regular expression, which will be evaluated
672
so that only matching entries will be displayed.
674
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',
678
takes_args = ['filename?']
679
takes_options = ['forward', 'timezone', 'verbose', 'show-ids', 'revision',
680
'long', 'message', 'short',]
682
def run(self, filename=None, timezone='original',
1257
690
from bzrlib.log import log_formatter, show_log
1258
assert message is None or isinstance(message, basestring), \
1259
"invalid message argument %r" % message
1260
693
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)
696
b = Branch.open_containing(filename)
697
fp = b.relpath(filename)
699
file_id = b.read_working_inventory().path2id(fp)
701
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()
703
b = Branch.open_containing('.')
1283
706
if revision is None:
1327
def get_log_format(long=False, short=False, line=False, default='long'):
1328
log_format = default
1332
log_format = 'short'
1338
748
class cmd_touching_revisions(Command):
1339
749
"""Return revision-ids which affected a particular file.
1341
A more user-friendly interface is "bzr log FILE".
751
A more user-friendly interface is "bzr log FILE"."""
1345
753
takes_args = ["filename"]
1348
754
def run(self, filename):
1349
tree, relpath = WorkingTree.open_containing(filename)
1351
inv = tree.read_working_inventory()
1352
file_id = inv.path2id(relpath)
755
b = Branch.open_containing(filename)
756
inv = b.read_working_inventory()
757
file_id = inv.path2id(b.relpath(filename))
1353
758
for revno, revision_id, what in bzrlib.log.find_touching_revisions(b, file_id):
1354
self.outf.write("%6d %s\n" % (revno, what))
759
print "%6d %s" % (revno, what)
1357
762
class cmd_ls(Command):
1358
763
"""List files in a tree.
765
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')
768
def run(self, revision=None, verbose=False):
769
b = Branch.open_containing('.')
771
tree = b.working_tree()
773
tree = b.revision_tree(revision.in_history(b).rev_id)
775
for fp, fc, kind, fid in tree.list_files():
777
if kind == 'directory':
1408
self.outf.write(fp + '\n')
784
print '%-8s %s%s' % (fc, fp, kindch)
1411
790
class cmd_unknowns(Command):
1412
791
"""List unknown files."""
1415
793
from bzrlib.osutils import quotefn
1416
for f in WorkingTree.open_containing(u'.')[0].unknowns():
1417
self.outf.write(quotefn(f) + '\n')
794
for f in Branch.open_containing('.').unknowns():
1420
799
class cmd_ignore(Command):
1600
958
A selected-file commit may fail in some cases where the committed
1601
959
tree would be invalid, such as trying to commit a file in a
1602
960
newly-added directory that is not itself committed.
962
TODO: Run hooks on tree to-be-committed, and after commit.
964
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
966
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 "
967
takes_options = ['message', 'file', 'verbose', 'unchanged']
1630
968
aliases = ['ci', 'checkin']
970
# TODO: Give better message for -s, --summary, used by tla people
1632
972
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)
974
from bzrlib.errors import PointlessCommit
975
from bzrlib.msgeditor import edit_commit_message
976
from bzrlib.status import show_status
977
from cStringIO import StringIO
979
b = Branch.open_containing('.')
981
selected_list = [b.relpath(s) for s in selected_list]
983
if not message and not file:
985
show_status(b, specific_files=selected_list,
987
message = edit_commit_message(catcher.getvalue())
1661
989
if message is None:
1662
990
raise BzrCommandError("please specify a commit message"
1663
991
" with either --message or --file")
1733
1039
if c.needs_write:
1737
1044
class cmd_upgrade(Command):
1738
1045
"""Upgrade branch storage to current format.
1740
1047
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):
1050
takes_args = ['dir?']
1052
def run(self, dir='.'):
1756
1053
from bzrlib.upgrade import upgrade
1758
format = get_format_type('default')
1759
upgrade(url, format)
1054
upgrade(Branch.open_containing(dir))
1762
1058
class cmd_whoami(Command):
1763
1059
"""Show bzr user id."""
1764
1060
takes_options = ['email']
1767
1062
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()
1064
b = bzrlib.branch.Branch.open_containing('.')
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):
1069
print bzrlib.osutils.user_email(b)
1071
print bzrlib.osutils.username(b)
1799
1074
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)
1075
"""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,
1077
takes_options = ['verbose', 'pattern']
1078
def run(self, verbose=False, pattern=".*"):
1857
1079
import bzrlib.ui
1858
from bzrlib.tests import selftest
1859
import bzrlib.benchmarks as benchmarks
1080
from bzrlib.selftest import selftest
1860
1081
# we don't want progress meters from the tests to go to the
1861
1082
# real output; and we don't want log messages cluttering up
1862
1083
# the real logs.
1863
1084
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
1085
bzrlib.trace.info('running tests...')
1869
1087
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)
1088
result = selftest(verbose=verbose, pattern=pattern)
1890
1090
bzrlib.trace.info('tests passed')
2033
1188
merge refuses to run if there are any uncommitted changes, unless
2034
1189
--force is given.
2036
The following merge types are available:
2038
1191
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):
1192
takes_options = ['revision', 'force', 'merge-type']
1194
def run(self, branch='.', revision=None, force=False,
1196
from bzrlib.merge import merge
1197
from bzrlib.merge_core import ApplyMerge3
2050
1198
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')
1199
merge_type = ApplyMerge3
2076
1201
if revision is None or len(revision) < 1:
2077
1202
base = [None, None]
2078
1203
other = [branch, -1]
2079
other_branch, path = Branch.open_containing(branch)
2081
1205
if len(revision) == 1:
2082
1206
base = [None, None]
2083
other_branch, path = Branch.open_containing(branch)
2084
revno = revision[0].in_history(other_branch).revno
2085
other = [branch, revno]
1207
other = [branch, revision[0].in_history(branch).revno]
2087
1209
assert len(revision) == 2
2088
1210
if None in revision:
2089
1211
raise BzrCommandError(
2090
1212
"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()
1213
from bzrlib.branch import Branch
1214
b = Branch.open(branch)
1216
base = [branch, revision[0].in_history(b).revno]
1217
other = [branch, revision[1].in_history(b).revno]
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:
1220
merge(other, base, check_clean=(not force), merge_type=merge_type)
2117
1221
except bzrlib.errors.AmbiguousBase, e:
2118
1222
m = ("sorry, bzr can't determine the right merge base yet\n"
2119
1223
"candidates are:\n "
2123
1227
"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
1231
class cmd_revert(Command):
2230
1232
"""Reverse all changes since the last commit.
2293
1293
aliases = ['s-c']
2297
1296
def run(self, context=None):
2298
1297
import shellcomplete
2299
1298
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
1301
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:
1302
"""What is missing in this branch relative to other branch.
1304
takes_args = ['remote?']
1305
aliases = ['mis', 'miss']
1306
# We don't have to add quiet to the list, because
1307
# unknown options are parsed as booleans
1308
takes_options = ['verbose', 'quiet']
1310
def run(self, remote=None, verbose=False, quiet=False):
1311
from bzrlib.errors import BzrCommandError
1312
from bzrlib.missing import show_missing
1314
if verbose and quiet:
1315
raise BzrCommandError('Cannot pass both quiet and verbose')
1317
b = Branch.open_containing('.')
1318
parent = b.get_parent()
2344
1321
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()
1324
print "Using last location: %s" % parent
1326
elif parent is None:
1327
# We only update parent if it did not exist, missing
1328
# should not change the parent
1329
b.set_parent(remote)
1330
br_remote = Branch.open_containing(remote)
1332
return show_missing(b, br_remote, verbose=verbose, quiet=quiet)
2398
1336
class cmd_plugins(Command):
2399
1337
"""List plugins"""
2403
1340
import bzrlib.plugin
2404
1341
from inspect import getdoc
2405
for name, plugin in bzrlib.plugin.all_plugins().items():
1342
for plugin in bzrlib.plugin.all_plugins:
2406
1343
if hasattr(plugin, '__path__'):
2407
1344
print plugin.__path__[0]
2408
1345
elif hasattr(plugin, '__file__'):
2415
1352
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