460
460
return file(self.abspath(filename), 'rb')
463
def annotate_iter(self, file_id, default_revision=CURRENT_REVISION):
463
def annotate_iter(self, file_id):
464
464
"""See Tree.annotate_iter
466
466
This implementation will use the basis tree implementation if possible.
494
494
old.append(list(tree.annotate_iter(file_id)))
495
495
return annotate.reannotate(old, self.get_file(file_id).readlines(),
500
def _get_ancestors(self, default_revision):
501
ancestors = set([default_revision])
502
for parent_id in self.get_parent_ids():
503
ancestors.update(self.branch.repository.get_ancestry(
504
parent_id, topo_sorted=False))
507
500
def get_parent_ids(self):
508
501
"""See Tree.get_parent_ids.
510
503
This implementation reads the pending merges list and last_revision
511
504
value and uses that to decide what the parents list should be.
513
last_rev = _mod_revision.ensure_null(self._last_revision())
514
if _mod_revision.NULL_REVISION == last_rev:
506
last_rev = self._last_revision()
517
510
parents = [last_rev]
623
616
# should probably put it back with the previous ID.
624
617
# the read and write working inventory should not occur in this
625
618
# function - they should be part of lock_write and unlock.
619
inv = self.read_working_inventory()
627
620
for f, file_id, kind in zip(files, ids, kinds):
628
621
assert kind is not None
629
622
if file_id is None:
632
625
file_id = osutils.safe_file_id(file_id)
633
626
inv.add_path(f, kind=kind, file_id=file_id)
634
self._inventory_is_modified = True
627
self._write_inventory(inv)
636
629
@needs_tree_write_lock
637
630
def _gather_kinds(self, files, kinds):
742
735
revision_ids = [osutils.safe_revision_id(r) for r in revision_ids]
743
736
self._check_parents_for_ghosts(revision_ids,
744
737
allow_leftmost_as_ghost=allow_leftmost_as_ghost)
745
for revision_id in revision_ids:
746
_mod_revision.check_not_reserved_id(revision_id)
748
739
if len(revision_ids) > 0:
749
740
self.set_last_revision(revision_ids[0])
751
self.set_last_revision(_mod_revision.NULL_REVISION)
742
self.set_last_revision(None)
753
744
self._set_merges_from_parent_ids(revision_ids)
756
747
def set_parent_trees(self, parents_list, allow_leftmost_as_ghost=False):
757
748
"""See MutableTree.set_parent_trees."""
758
749
parent_ids = [osutils.safe_revision_id(rev) for (rev, tree) in parents_list]
759
for revision_id in parent_ids:
760
_mod_revision.check_not_reserved_id(revision_id)
762
751
self._check_parents_for_ghosts(parent_ids,
763
752
allow_leftmost_as_ghost=allow_leftmost_as_ghost)
765
754
if len(parent_ids) == 0:
766
leftmost_parent_id = _mod_revision.NULL_REVISION
755
leftmost_parent_id = None
767
756
leftmost_parent_tree = None
769
758
leftmost_parent_id, leftmost_parent_tree = parents_list[0]
819
808
# local alterations
820
809
merger.check_basis(check_clean=True, require_commits=False)
821
810
if to_revision is None:
822
to_revision = _mod_revision.ensure_null(branch.last_revision())
811
to_revision = branch.last_revision()
824
813
to_revision = osutils.safe_revision_id(to_revision)
825
814
merger.other_rev_id = to_revision
826
if _mod_revision.is_null(merger.other_rev_id):
815
if merger.other_rev_id is None:
827
816
raise errors.NoCommits(branch)
828
817
self.branch.fetch(branch, last_revision=merger.other_rev_id)
829
818
merger.other_basis = merger.other_rev_id
1774
1763
@needs_tree_write_lock
1775
1764
def remove(self, files, verbose=False, to_file=None, keep_files=True,
1777
"""Remove nominated files from the working inventory.
1766
"""Remove nominated files from the working inventor.
1779
1768
:files: File paths relative to the basedir.
1780
1769
:keep_files: If true, the files will also be kept.
1815
1804
recurse_directory_to_add_files(filename)
1816
1805
files = [f for f in new_files]
1819
return # nothing to do
1821
1807
# Sort needed to first handle directory content before the directory
1822
1808
files.sort(reverse=True)
1823
1809
if not keep_files and not force:
1824
has_changed_files = len(unknown_files_in_directory) > 0
1825
if not has_changed_files:
1826
for (file_id, path, content_change, versioned, parent_id, name,
1827
kind, executable) in self._iter_changes(self.basis_tree(),
1828
include_unchanged=True, require_versioned=False,
1829
want_unversioned=True, specific_files=files):
1830
# check if it's unknown OR changed but not deleted:
1831
if (versioned == (False, False)
1832
or (content_change and kind[1] != None)):
1833
has_changed_files = True
1836
if has_changed_files:
1837
# make delta to show ALL applicable changes in error message.
1838
tree_delta = self.changes_from(self.basis_tree(),
1839
specific_files=files)
1840
for unknown_file in unknown_files_in_directory:
1841
tree_delta.unversioned.extend((unknown_file,))
1810
tree_delta = self.changes_from(self.basis_tree(),
1811
specific_files=files)
1812
for unknown_file in unknown_files_in_directory:
1813
tree_delta.unversioned.extend((unknown_file,))
1814
if bool(tree_delta.modified
1816
or tree_delta.renamed
1817
or tree_delta.kind_changed
1818
or tree_delta.unversioned):
1842
1819
raise errors.BzrRemoveChangedFilesError(tree_delta)
1844
1821
# do this before any modifications
1995
1972
raise NotImplementedError(self.unlock)
1997
def update(self, change_reporter=None):
1998
1975
"""Update a working tree along its branch.
2000
1977
This will update the branch if its bound too, which means we have
2030
2007
old_tip = self.branch.update()
2033
return self._update_tree(old_tip, change_reporter)
2010
return self._update_tree(old_tip)
2037
2014
@needs_tree_write_lock
2038
def _update_tree(self, old_tip=None, change_reporter=None):
2015
def _update_tree(self, old_tip=None):
2039
2016
"""Update a tree to the master branch.
2041
2018
:param old_tip: if supplied, the previous tip revision the branch,
2056
2033
last_rev = self.get_parent_ids()[0]
2057
2034
except IndexError:
2058
last_rev = _mod_revision.NULL_REVISION
2059
if last_rev != _mod_revision.ensure_null(self.branch.last_revision()):
2036
if last_rev != self.branch.last_revision():
2060
2037
# merge tree state up to new branch tip.
2061
2038
basis = self.basis_tree()
2062
2039
basis.lock_read()
2085
2061
for parent in merges:
2086
2062
parent_trees.append(
2087
2063
(parent, self.branch.repository.revision_tree(parent)))
2088
if (old_tip is not None and not _mod_revision.is_null(old_tip)):
2064
if old_tip is not None:
2089
2065
parent_trees.append(
2090
2066
(old_tip, self.branch.repository.revision_tree(old_tip)))
2091
2067
self.set_parent_trees(parent_trees)
2094
2070
# the working tree had the same last-revision as the master
2095
2071
# branch did. We may still have pivot local work from the local
2096
2072
# branch into old_tip:
2097
if (old_tip is not None and not _mod_revision.is_null(old_tip)):
2073
if old_tip is not None:
2098
2074
self.add_parent_tree_id(old_tip)
2099
if (old_tip is not None and not _mod_revision.is_null(old_tip)
2100
and old_tip != last_rev):
2075
if old_tip and old_tip != last_rev:
2101
2076
# our last revision was not the prior branch last revision
2102
2077
# and we have converted that last revision to a pending merge.
2103
2078
# base is somewhere between the branch tip now
2619
2593
if not isinstance(a_bzrdir.transport, LocalTransport):
2620
2594
raise errors.NotLocalUrl(a_bzrdir.transport.base)
2621
2595
branch = a_bzrdir.open_branch()
2622
if revision_id is None:
2623
revision_id = _mod_revision.ensure_null(branch.last_revision())
2596
if revision_id is not None:
2625
2597
revision_id = osutils.safe_revision_id(revision_id)
2628
branch.generate_revision_history(revision_id)
2600
revision_history = branch.revision_history()
2602
position = revision_history.index(revision_id)
2604
raise errors.NoSuchRevision(branch, revision_id)
2605
branch.set_revision_history(revision_history[:position + 1])
2608
revision = branch.last_revision()
2631
2609
inv = Inventory()
2632
2610
wt = WorkingTree2(a_bzrdir.root_transport.local_abspath('.'),
2635
2613
_internal=True,
2637
2615
_bzrdir=a_bzrdir)
2638
basis_tree = branch.repository.revision_tree(revision_id)
2616
basis_tree = branch.repository.revision_tree(revision)
2639
2617
if basis_tree.inventory.root is not None:
2640
2618
wt.set_root_id(basis_tree.inventory.root.file_id)
2641
2619
# set the parent list and cache the basis tree.
2642
if _mod_revision.is_null(revision_id):
2645
parent_trees = [(revision_id, basis_tree)]
2646
wt.set_parent_trees(parent_trees)
2620
wt.set_parent_trees([(revision, basis_tree)])
2647
2621
transform.build_tree(basis_tree, wt)
2720
2694
control_files.put_utf8('format', self.get_format_string())
2721
2695
branch = a_bzrdir.open_branch()
2722
2696
if revision_id is None:
2723
revision_id = _mod_revision.ensure_null(branch.last_revision())
2697
revision_id = branch.last_revision()
2725
2699
revision_id = osutils.safe_revision_id(revision_id)
2726
2700
# WorkingTree3 can handle an inventory which has a unique root id.
2798
2772
# and not independently creatable, so are not registered.
2799
2773
_legacy_formats = [WorkingTreeFormat2(),
2777
class WorkingTreeTestProviderAdapter(object):
2778
"""A tool to generate a suite testing multiple workingtree formats at once.
2780
This is done by copying the test once for each transport and injecting
2781
the transport_server, transport_readonly_server, and workingtree_format
2782
classes into each copy. Each copy is also given a new id() to make it
2786
def __init__(self, transport_server, transport_readonly_server, formats):
2787
self._transport_server = transport_server
2788
self._transport_readonly_server = transport_readonly_server
2789
self._formats = formats
2791
def _clone_test(self, test, bzrdir_format, workingtree_format, variation):
2792
"""Clone test for adaption."""
2793
new_test = deepcopy(test)
2794
new_test.transport_server = self._transport_server
2795
new_test.transport_readonly_server = self._transport_readonly_server
2796
new_test.bzrdir_format = bzrdir_format
2797
new_test.workingtree_format = workingtree_format
2798
def make_new_test_id():
2799
new_id = "%s(%s)" % (test.id(), variation)
2800
return lambda: new_id
2801
new_test.id = make_new_test_id()
2804
def adapt(self, test):
2805
from bzrlib.tests import TestSuite
2806
result = TestSuite()
2807
for workingtree_format, bzrdir_format in self._formats:
2808
new_test = self._clone_test(
2811
workingtree_format, workingtree_format.__class__.__name__)
2812
result.addTest(new_test)