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, NotBranchError
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
193
This is equivalent to creating the directory and then adding it.
308
195
takes_args = ['dir+']
309
encoding_type = 'replace'
311
197
def run(self, dir_list):
312
200
for d in dir_list:
314
wt, dd = WorkingTree.open_containing(d)
316
self.outf.write('added %s\n' % d)
203
b = Branch.open_containing(d)
319
208
class cmd_relpath(Command):
320
209
"""Show path of a file relative to root"""
322
210
takes_args = ['filename']
326
213
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')
214
print Branch.open_containing(filename).relpath(filename)
334
218
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']
219
"""Show inventory of the current working copy or a revision."""
220
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]
222
def run(self, revision=None, show_ids=False):
223
b = Branch.open_containing('.')
348
224
if revision is None:
349
inv = tree.read_working_inventory()
225
inv = b.read_working_inventory()
351
227
if len(revision) > 1:
352
228
raise BzrCommandError('bzr inventory --revision takes'
353
229
' exactly one revision identifier')
354
inv = tree.branch.repository.get_revision_inventory(
355
revision[0].in_history(tree.branch).rev_id)
230
inv = b.get_revision_inventory(revision[0].in_history(b).rev_id)
357
232
for path, entry in inv.entries():
358
if kind and kind != entry.kind:
361
self.outf.write('%-50s %s\n' % (path, entry.file_id))
234
print '%-50s %s' % (path, entry.file_id)
363
self.outf.write(path)
364
self.outf.write('\n')
239
class cmd_move(Command):
240
"""Move files to a different directory.
245
The destination must be a versioned directory in the same branch.
247
takes_args = ['source$', 'dest']
248
def run(self, source_list, dest):
249
b = Branch.open_containing('.')
251
# TODO: glob expansion on windows?
252
b.move([b.relpath(s) for s in source_list], b.relpath(dest))
255
class cmd_rename(Command):
256
"""Change the name of an entry.
259
bzr rename frob.c frobber.c
260
bzr rename src/frob.c lib/frob.c
262
It is an error if the destination name exists.
264
See also the 'move' command, which moves files into a different
265
directory without changing their name.
267
TODO: Some way to rename multiple files without invoking bzr for each
269
takes_args = ['from_name', 'to_name']
271
def run(self, from_name, to_name):
272
b = Branch.open_containing('.')
273
b.rename_one(b.relpath(from_name), b.relpath(to_name))
367
277
class cmd_mv(Command):
378
288
Files cannot be moved between branches.
381
290
takes_args = ['names*']
382
aliases = ['move', 'rename']
383
encoding_type = 'replace'
385
291
def run(self, names_list):
386
292
if len(names_list) < 2:
387
293
raise BzrCommandError("missing file argument")
388
tree, rel_names = tree_files(names_list)
294
b = Branch.open_containing(names_list[0])
296
rel_names = [b.relpath(x) for x in names_list]
390
298
if os.path.isdir(names_list[-1]):
391
299
# move into existing directory
392
for pair in tree.move(rel_names[:-1], rel_names[-1]):
393
self.outf.write("%s => %s\n" % pair)
300
for pair in b.move(rel_names[:-1], rel_names[-1]):
301
print "%s => %s" % pair
395
303
if len(names_list) != 2:
396
304
raise BzrCommandError('to mv multiple files the destination '
397
305
'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]))
306
b.rename_one(rel_names[0], rel_names[1])
307
print "%s => %s" % (rel_names[0], rel_names[1])
402
312
class cmd_pull(Command):
403
"""Turn this branch into a mirror of another branch.
313
"""Pull any changes from another branch into the current one.
315
If the location is omitted, the last-used location will be used.
316
Both the revision history and the working directory will be
405
319
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.
320
considered diverged if both branches have had commits without first
321
pulling from the other.
413
323
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.
324
from one into the other.
425
takes_options = ['remember', 'overwrite', 'revision', 'verbose']
326
takes_options = ['remember']
426
327
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()
329
def run(self, location=None, remember=False):
330
from bzrlib.merge import merge
332
from shutil import rmtree
335
br_to = Branch.open_containing('.')
336
stored_loc = br_to.get_parent()
438
337
if location is None:
439
338
if stored_loc is None:
440
339
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)
341
print "Using saved location: %s" % stored_loc
342
location = stored_loc
343
cache_root = tempfile.mkdtemp()
344
from bzrlib.errors import DivergedBranches
345
br_from = Branch.open_containing(location)
346
location = br_from.base
347
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()
349
from bzrlib.errors import DivergedBranches
350
br_from = Branch.open(location)
351
br_from.setup_caching(cache_root)
352
location = br_from.base
353
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)
355
br_to.update_revisions(br_from)
576
356
except DivergedBranches:
577
357
raise BzrCommandError("These branches have diverged."
578
" Try a merge then push with overwrite.")
579
note('%d revision(s) pushed.' % (count,))
360
merge(('.', -1), ('.', old_revno), check_clean=False)
361
if stored_loc is None or remember:
362
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
368
class cmd_branch(Command):
605
383
aliases = ['get', 'clone']
607
385
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)
386
from bzrlib.clone import copy_branch
389
from shutil import rmtree
390
cache_root = tempfile.mkdtemp()
394
elif len(revision) > 1:
395
raise BzrCommandError(
396
'bzr branch --revision takes exactly 1 revision value')
398
br_from = Branch.open(from_location)
400
if e.errno == errno.ENOENT:
401
raise BzrCommandError('Source location "%s" does not'
402
' exist.' % to_location)
405
br_from.setup_caching(cache_root)
625
406
if basis is not None:
626
basis_dir = bzrdir.BzrDir.open_containing(basis)[0]
407
basis_branch = Branch.open_containing(basis)
629
410
if len(revision) == 1 and revision[0] is not None:
630
411
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()
636
414
if to_location is None:
637
415
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()
417
os.mkdir(to_location)
419
if e.errno == errno.EEXIST:
420
raise BzrCommandError('Target directory "%s" already'
421
' exists.' % to_location)
422
if e.errno == errno.ENOENT:
423
raise BzrCommandError('Parent of "%s" does not exist.' %
428
copy_branch(br_from, to_location, revision_id, basis_branch)
656
429
except bzrlib.errors.NoSuchRevision:
657
to_transport.delete_tree('.')
658
msg = "The branch %s has no revision %s." % (from_location, revision[0])
431
msg = "The branch %s has no revision %d." % (from_location, revision[0])
659
432
raise BzrCommandError(msg)
660
433
except bzrlib.errors.UnlistableBranch:
662
msg = "The branch %s cannot be used as a --basis" % (basis,)
663
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)
434
msg = "The branch %s cannot be used as a --basis"
759
439
class cmd_renames(Command):
760
440
"""Show list of renamed files.
442
TODO: Option to show renames between two historical versions.
444
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
446
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()
448
def run(self, dir='.'):
449
b = Branch.open_containing(dir)
450
old_inv = b.basis_tree().inventory
451
new_inv = b.read_working_inventory()
773
453
renames = list(bzrlib.tree.find_renames(old_inv, new_inv))
775
455
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()),))
456
print "%s => %s" % (old_name, new_name)
812
459
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],
460
"""Show statistical information about a branch."""
461
takes_args = ['branch?']
463
def run(self, branch=None):
466
b = Branch.open_containing(branch)
831
470
class cmd_remove(Command):
867
489
same through all revisions where the file exists, even when it is
868
490
moved or renamed.
872
493
takes_args = ['filename']
875
494
def run(self, filename):
876
tree, relpath = WorkingTree.open_containing(filename)
877
i = tree.inventory.path2id(relpath)
495
b = Branch.open_containing(filename)
496
i = b.inventory.path2id(b.relpath(filename))
879
498
raise BzrError("%r is not a versioned file" % filename)
881
self.outf.write(i + '\n')
884
503
class cmd_file_path(Command):
885
504
"""Print path of file_ids to a file or directory.
887
506
This prints one line for each directory down to the target,
888
starting at the branch root.
507
starting at the branch root."""
892
509
takes_args = ['filename']
895
510
def run(self, filename):
896
tree, relpath = WorkingTree.open_containing(filename)
898
fid = inv.path2id(relpath)
511
b = Branch.open_containing(filename)
513
fid = inv.path2id(b.relpath(filename))
900
515
raise BzrError("%r is not a versioned file" % filename)
901
516
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
520
class cmd_revision_history(Command):
932
521
"""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')
524
for patchid in Branch.open_containing('.').revision_history():
943
528
class cmd_ancestry(Command):
944
529
"""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')
533
for revision_id in b.get_ancestry(b.last_revision()):
537
class cmd_directories(Command):
538
"""Display list of versioned directories in this branch."""
540
for name, ie in Branch.open_containing('.').read_working_inventory().directories():
959
547
class cmd_init(Command):
962
550
Use this to create an empty branch, or before importing an
963
551
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
553
Recipe for importing a tree of files:
978
558
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
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)
561
Branch.initialize('.')
1056
564
class cmd_diff(Command):
1059
567
If files are listed, only the changes in those files are listed.
1060
568
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".
570
TODO: Allow diff across branches.
572
TODO: Option to use external diff command; could be GNU diff, wdiff,
575
TODO: Python difflib is not exactly the same as unidiff; should
576
either fix it up or prefer to use an external diff.
578
TODO: If a directory is given, diff everything under that.
580
TODO: Selected-file diff is inefficient and doesn't show you
583
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
591
takes_args = ['file*']
1085
takes_options = ['revision', 'diff-options', 'prefix']
592
takes_options = ['revision', 'diff-options']
1086
593
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'):
595
def run(self, revision=None, file_list=None, diff_options=None):
596
from bzrlib.diff import show_diff
599
b = Branch.open_containing(file_list[0])
600
file_list = [b.relpath(f) for f in file_list]
601
if file_list == ['']:
602
# 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")
605
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
607
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)
608
if len(revision) == 1:
609
show_diff(b, revision[0], specific_files=file_list,
610
external_diff_options=diff_options)
1128
611
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)
612
show_diff(b, revision[0], specific_files=file_list,
613
external_diff_options=diff_options,
614
revision2=revision[1])
1133
616
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)
618
show_diff(b, None, specific_files=file_list,
619
external_diff_options=diff_options)
1145
624
class cmd_deleted(Command):
1146
625
"""List files deleted in the working tree.
627
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
629
def run(self, show_ids=False):
1158
tree = WorkingTree.open_containing(u'.')[0]
1159
old = tree.basis_tree()
630
b = Branch.open_containing('.')
632
new = b.working_tree()
634
## TODO: Much more efficient way to do this: read in new
635
## directories with readdir, rather than stating each one. Same
636
## level of effort but possibly much less IO. (Or possibly not,
637
## if the directories are very large...)
1160
639
for path, ie in old.inventory.iter_entries():
1161
if not tree.has_id(ie.file_id):
1162
self.outf.write(path)
640
if not new.has_id(ie.file_id):
1164
self.outf.write(' ')
1165
self.outf.write(ie.file_id)
1166
self.outf.write('\n')
642
print '%-50s %s' % (path, ie.file_id)
1169
647
class cmd_modified(Command):
1170
648
"""List files modified in working tree."""
1174
651
from bzrlib.delta import compare_trees
1176
tree = WorkingTree.open_containing(u'.')[0]
1177
td = compare_trees(tree.basis_tree(), tree)
653
b = Branch.open_containing('.')
654
td = compare_trees(b.basis_tree(), b.working_tree())
1179
656
for path, id, kind, text_modified, meta_modified in td.modified:
1180
self.outf.write(path + '\n')
1183
661
class cmd_added(Command):
1184
662
"""List files added in working tree."""
1188
wt = WorkingTree.open_containing(u'.')[0]
1189
basis_inv = wt.basis_tree().inventory
665
b = Branch.open_containing('.')
666
wt = b.working_tree()
667
basis_inv = b.basis_tree().inventory
1190
668
inv = wt.inventory
1191
669
for file_id in inv:
1192
670
if file_id in basis_inv:
1194
672
path = inv.id2path(file_id)
1195
if not os.access(bzrlib.osutils.abspath(path), os.F_OK):
673
if not os.access(b.abspath(path), os.F_OK):
1197
self.outf.write(path + '\n')
1200
679
class cmd_root(Command):
1201
680
"""Show the tree root directory.
1203
682
The root is the nearest enclosing directory with a .bzr control
1205
684
takes_args = ['filename?']
1207
685
def run(self, filename=None):
1208
686
"""Print the branch root."""
1209
tree = WorkingTree.open_containing(filename)[0]
1210
self.outf.write(tree.basedir + '\n')
687
b = Branch.open_containing(filename)
1213
691
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
692
"""Show log of this branch.
694
To request a range of logs, you can use the command -r begin:end
695
-r revision requests a specific revision, -r :end or -r begin: are
1225
bzr log -r -10.. http://server/branch
698
--message allows you to give a regular expression, which will be evaluated
699
so that only matching entries will be displayed.
1228
702
# 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',
704
takes_args = ['filename?']
705
takes_options = ['forward', 'timezone', 'verbose', 'show-ids', 'revision',
706
'long', 'message', 'short',]
708
def run(self, filename=None, timezone='original',
1257
716
from bzrlib.log import log_formatter, show_log
1258
assert message is None or isinstance(message, basestring), \
1259
"invalid message argument %r" % message
1260
719
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)
722
b = Branch.open_containing(filename)
723
fp = b.relpath(filename)
725
file_id = b.read_working_inventory().path2id(fp)
727
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()
729
b = Branch.open_containing('.')
1283
732
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
774
class cmd_touching_revisions(Command):
1339
775
"""Return revision-ids which affected a particular file.
1341
A more user-friendly interface is "bzr log FILE".
777
A more user-friendly interface is "bzr log FILE"."""
1345
779
takes_args = ["filename"]
1348
780
def run(self, filename):
1349
tree, relpath = WorkingTree.open_containing(filename)
1351
inv = tree.read_working_inventory()
1352
file_id = inv.path2id(relpath)
781
b = Branch.open_containing(filename)
782
inv = b.read_working_inventory()
783
file_id = inv.path2id(b.relpath(filename))
1353
784
for revno, revision_id, what in bzrlib.log.find_touching_revisions(b, file_id):
1354
self.outf.write("%6d %s\n" % (revno, what))
785
print "%6d %s" % (revno, what)
1357
788
class cmd_ls(Command):
1358
789
"""List files in a tree.
791
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)
794
def run(self, revision=None, verbose=False):
795
b = Branch.open_containing('.')
797
tree = b.working_tree()
799
tree = b.revision_tree(revision.in_history(b).rev_id)
1394
800
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')
1408
self.outf.write(fp + '\n')
802
kindch = entry.kind_character()
803
print '%-8s %s%s' % (fc, fp, kindch)
1411
809
class cmd_unknowns(Command):
1412
810
"""List unknown files."""
1415
812
from bzrlib.osutils import quotefn
1416
for f in WorkingTree.open_containing(u'.')[0].unknowns():
1417
self.outf.write(quotefn(f) + '\n')
813
for f in Branch.open_containing('.').unknowns():
1420
818
class cmd_ignore(Command):
1600
984
A selected-file commit may fail in some cases where the committed
1601
985
tree would be invalid, such as trying to commit a file in a
1602
986
newly-added directory that is not itself committed.
988
TODO: Run hooks on tree to-be-committed, and after commit.
990
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 ??)
992
takes_args = ['selected*']
993
takes_options = ['message', 'file', 'verbose', 'unchanged']
994
aliases = ['ci', 'checkin']
1609
996
# TODO: Give better message for -s, --summary, used by tla people
1611
998
# XXX: verbose currently does nothing
1613
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 "
1630
aliases = ['ci', 'checkin']
1632
1000
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()
1002
from bzrlib.errors import PointlessCommit, ConflictsInTree
1003
from bzrlib.msgeditor import edit_commit_message
1004
from bzrlib.status import show_status
1005
from cStringIO import StringIO
1007
b = Branch.open_containing('.')
1009
selected_list = [b.relpath(s) for s in selected_list]
1658
1012
if message is None and not file:
1659
template = make_commit_message_template(tree, selected_list)
1660
message = edit_commit_message(template)
1013
catcher = StringIO()
1014
show_status(b, specific_files=selected_list,
1016
message = edit_commit_message(catcher.getvalue())
1661
1018
if message is None:
1662
1019
raise BzrCommandError("please specify a commit message"
1663
1020
" with either --message or --file")
1802
1114
This creates temporary test directories in the working directory,
1803
1115
but not existing data is affected. These directories are deleted
1804
1116
if the tests pass, or left behind to help in debugging if they
1805
fail and --keep-output is specified.
1807
1119
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
1120
which tests should run."""
1819
1121
# 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)
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,
1123
takes_args = ['testnames*']
1124
takes_options = ['verbose', 'pattern']
1125
def run(self, testnames_list=None, verbose=False, pattern=".*"):
1857
1126
import bzrlib.ui
1858
from bzrlib.tests import selftest
1859
import bzrlib.benchmarks as benchmarks
1127
from bzrlib.selftest import selftest
1860
1128
# we don't want progress meters from the tests to go to the
1861
1129
# real output; and we don't want log messages cluttering up
1862
1130
# the real logs.
1863
1131
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
1132
bzrlib.trace.info('running tests...')
1869
1134
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
1135
result = selftest(verbose=verbose,
1883
1136
pattern=pattern,
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)
1137
testnames=testnames_list)
1890
1139
bzrlib.trace.info('tests passed')
2033
1237
merge refuses to run if there are any uncommitted changes, unless
2034
1238
--force is given.
2036
The following merge types are available:
2038
1240
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):
1241
takes_options = ['revision', 'force', 'merge-type']
1243
def run(self, branch=None, revision=None, force=False,
1245
from bzrlib.merge import merge
1246
from bzrlib.merge_core import ApplyMerge3
2050
1247
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')
1248
merge_type = ApplyMerge3
1250
branch = Branch.open_containing('.').get_parent()
1252
raise BzrCommandError("No merge location known or specified.")
1254
print "Using saved location: %s" % branch
2076
1255
if revision is None or len(revision) < 1:
2077
1256
base = [None, None]
2078
1257
other = [branch, -1]
2079
other_branch, path = Branch.open_containing(branch)
2081
1259
if len(revision) == 1:
2082
1260
base = [None, None]
2083
other_branch, path = Branch.open_containing(branch)
2084
revno = revision[0].in_history(other_branch).revno
2085
other = [branch, revno]
1261
other = [branch, revision[0].in_history(branch).revno]
2087
1263
assert len(revision) == 2
2088
1264
if None in revision:
2089
1265
raise BzrCommandError(
2090
1266
"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()
1267
b = Branch.open(branch)
1269
base = [branch, revision[0].in_history(b).revno]
1270
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:
1273
merge(other, base, check_clean=(not force), merge_type=merge_type)
2117
1274
except bzrlib.errors.AmbiguousBase, e:
2118
1275
m = ("sorry, bzr can't determine the right merge base yet\n"
2119
1276
"candidates are:\n "
2123
1280
"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
1284
class cmd_revert(Command):
2230
1285
"""Reverse all changes since the last commit.
2308
1359
def run(self, from_branch, to_branch):
2309
1360
from bzrlib.fetch import Fetcher
2310
1361
from bzrlib.branch import Branch
2311
from_b = Branch.open(from_branch)
2312
to_b = Branch.open(to_branch)
1362
from_b = Branch(from_branch)
1363
to_b = Branch(to_branch)
2313
1364
Fetcher(to_b, from_b)
2316
1368
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:
1369
"""What is missing in this branch relative to other branch.
1371
# TODO: rewrite this in terms of ancestry so that it shows only
1374
takes_args = ['remote?']
1375
aliases = ['mis', 'miss']
1376
# We don't have to add quiet to the list, because
1377
# unknown options are parsed as booleans
1378
takes_options = ['verbose', 'quiet']
1380
def run(self, remote=None, verbose=False, quiet=False):
1381
from bzrlib.errors import BzrCommandError
1382
from bzrlib.missing import show_missing
1384
if verbose and quiet:
1385
raise BzrCommandError('Cannot pass both quiet and verbose')
1387
b = Branch.open_containing('.')
1388
parent = b.get_parent()
2344
1391
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()
1394
print "Using last location: %s" % parent
1396
elif parent is None:
1397
# We only update parent if it did not exist, missing
1398
# should not change the parent
1399
b.set_parent(remote)
1400
br_remote = Branch.open_containing(remote)
1401
return show_missing(b, br_remote, verbose=verbose, quiet=quiet)
2398
1404
class cmd_plugins(Command):
2399
1405
"""List plugins"""
2403
1408
import bzrlib.plugin
2404
1409
from inspect import getdoc
2405
for name, plugin in bzrlib.plugin.all_plugins().items():
1410
for plugin in bzrlib.plugin.all_plugins:
2406
1411
if hasattr(plugin, '__path__'):
2407
1412
print plugin.__path__[0]
2408
1413
elif hasattr(plugin, '__file__'):
2415
1420
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