142
143
return False, False
145
def diff(self, text_diff, from_label, tree, to_label, to_entry, to_tree,
146
output_to, reverse=False):
147
"""Perform a diff from this to to_entry.
149
text_diff will be used for textual difference calculation.
150
This is a template method, override _diff in child classes.
152
self._read_tree_state(tree.id2path(self.file_id), tree)
154
# cannot diff from one kind to another - you must do a removal
155
# and an addif they do not match.
156
assert self.kind == to_entry.kind
157
to_entry._read_tree_state(to_tree.id2path(to_entry.file_id),
159
self._diff(text_diff, from_label, tree, to_label, to_entry, to_tree,
144
162
def _diff(self, text_diff, from_label, tree, to_label, to_entry, to_tree,
145
163
output_to, reverse=False):
146
164
"""Perform a diff between two entries of the same kind."""
171
190
ie.executable = False
172
191
except AttributeError:
193
# must now be the same.
194
assert candidates[ie.revision] == ie
175
196
# add this revision as a candidate.
176
197
candidates[ie.revision] = ie
177
198
return candidates
179
@deprecated_method(deprecated_in((1, 6, 0)))
200
@deprecated_method(zero_ninetyone)
201
def find_previous_heads(self, previous_inventories,
202
versioned_file_store,
205
"""Return the revisions and entries that directly precede this.
207
Returned as a map from revision to inventory entry.
209
This is a map containing the file revisions in all parents
210
for which the file exists, and its revision is not a parent of
211
any other. If the file is new, the set will be empty.
213
:param versioned_file_store: A store where ancestry data on this
214
file id can be queried.
215
:param transaction: The transaction that queries to the versioned
216
file store should be completed under.
217
:param entry_vf: The entry versioned file, if its already available.
219
candidates = self.parent_candidates(previous_inventories)
221
# revision:ie mapping with one revision for each head.
223
# common case optimisation
224
if len(candidates) == 1:
225
# if there is only one candidate revision found
226
# then we can avoid opening the versioned file to access ancestry:
227
# there cannot be any ancestors to eliminate when there is
228
# only one revision available.
231
# --- what follows is now encapsulated in repository.get_graph.heads(),
232
# but that is not accessible from here as we have no repository
233
# pointer. Note that the repository.get_graph.heads() call can return
234
# different results *at the moment* because of the kind-changing check
235
# we have in parent_candidates().
237
# eliminate ancestors amongst the available candidates:
238
# heads are those that are not an ancestor of any other candidate
239
# - this provides convergence at a per-file level.
240
def get_ancestors(weave, entry):
241
return set(weave.get_ancestry(entry.revision, topo_sorted=False))
242
# revision: ancestor list for each head
244
for ie in candidates.values():
245
# may be an ancestor of a known head:
246
already_present = 0 != len(
247
[head for head in heads
248
if ie.revision in head_ancestors[head]])
250
# an ancestor of an analyzed candidate.
252
# not an ancestor of a known head:
253
# load the versioned file for this file id if needed
255
entry_vf = versioned_file_store.get_weave_or_empty(
256
self.file_id, transaction)
257
ancestors = get_ancestors(entry_vf, ie)
258
# may knock something else out:
259
check_heads = list(heads.keys())
260
for head in check_heads:
261
if head in ancestors:
262
# this previously discovered 'head' is not
263
# really a head - its an ancestor of the newly
266
head_ancestors[ie.revision] = ancestors
267
heads[ie.revision] = ie
180
270
def get_tar_item(self, root, dp, now, tree):
181
271
"""Get a tarfile item and a file stream for its content."""
182
272
item = tarfile.TarInfo(osutils.pathjoin(root, dp).encode('utf8'))
475
567
def _check(self, checker, tree_revision_id, tree):
476
568
"""See InventoryEntry._check"""
477
key = (self.file_id, self.revision)
478
if key in checker.checked_texts:
479
prev_sha = checker.checked_texts[key]
569
t = (self.file_id, self.revision)
570
if t in checker.checked_texts:
571
prev_sha = checker.checked_texts[t]
480
572
if prev_sha != self.text_sha1:
481
573
raise BzrCheckError(
482
574
'mismatched sha1 on {%s} in {%s} (%s != %s) %r' %
486
578
checker.repeated_text_cnt += 1
581
if self.file_id not in checker.checked_weaves:
582
mutter('check weave {%s}', self.file_id)
583
w = tree._get_weave(self.file_id)
584
# Not passing a progress bar, because it creates a new
585
# progress, which overwrites the current progress,
586
# and doesn't look nice
588
checker.checked_weaves[self.file_id] = True
590
w = tree._get_weave(self.file_id)
489
592
mutter('check version {%s} of {%s}', tree_revision_id, self.file_id)
490
593
checker.checked_text_cnt += 1
491
594
# We can't check the length, because Weave doesn't store that
492
595
# information, and the whole point of looking at the weave's
493
596
# sha1sum is that we don't have to extract the text.
494
if (self.text_sha1 != tree._repository.texts.get_sha1s([key])[key]):
495
raise BzrCheckError('text {%s} version {%s} wrong sha1' % key)
496
checker.checked_texts[key] = self.text_sha1
597
if self.text_sha1 != w.get_sha1(self.revision):
598
raise BzrCheckError('text {%s} version {%s} wrong sha1'
599
% (self.file_id, self.revision))
600
checker.checked_texts[t] = self.text_sha1
499
603
other = InventoryFile(self.file_id, self.name, self.parent_id)
513
619
def _diff(self, text_diff, from_label, tree, to_label, to_entry, to_tree,
514
620
output_to, reverse=False):
515
621
"""See InventoryEntry._diff."""
516
from bzrlib.diff import DiffText
517
from_file_id = self.file_id
519
to_file_id = to_entry.file_id
523
to_file_id, from_file_id = from_file_id, to_file_id
524
tree, to_tree = to_tree, tree
525
from_label, to_label = to_label, from_label
526
differ = DiffText(tree, to_tree, output_to, 'utf-8', '', '',
528
return differ.diff_text(from_file_id, to_file_id, from_label, to_label)
623
from_text = tree.get_file(self.file_id).readlines()
625
to_text = to_tree.get_file(to_entry.file_id).readlines()
629
text_diff(from_label, from_text,
630
to_label, to_text, output_to)
632
text_diff(to_label, to_text,
633
from_label, from_text, output_to)
634
except errors.BinaryFile:
636
label_pair = (to_label, from_label)
638
label_pair = (from_label, to_label)
640
("Binary files %s and %s differ\n" % label_pair).encode('utf8'))
530
642
def has_text(self):
531
643
"""See InventoryEntry.has_text."""
623
735
def _diff(self, text_diff, from_label, tree, to_label, to_entry, to_tree,
624
736
output_to, reverse=False):
625
737
"""See InventoryEntry._diff."""
626
from bzrlib.diff import DiffSymlink
627
old_target = self.symlink_target
738
from_text = self.symlink_target
628
739
if to_entry is not None:
629
new_target = to_entry.symlink_target
638
new_target, old_target = old_target, new_target
639
differ = DiffSymlink(old_tree, new_tree, output_to)
640
return differ.diff_symlink(old_target, new_target)
740
to_text = to_entry.symlink_target
745
output_to.write('=== target changed %r => %r\n' % (from_text, to_text))
748
output_to.write('=== target was %r\n' % self.symlink_target)
750
output_to.write('=== target is %r\n' % self.symlink_target)
642
752
def __init__(self, file_id, name, parent_id):
643
753
super(InventoryLink, self).__init__(file_id, name, parent_id)
815
926
# adds come later
817
928
# Preserve unaltered children of file_id for later reinsertion.
818
file_id_children = getattr(self[file_id], 'children', {})
819
if len(file_id_children):
820
children[file_id] = file_id_children
929
children[file_id] = getattr(self[file_id], 'children', {})
821
930
# Remove file_id and the unaltered children. If file_id is not
822
931
# being deleted it will be reinserted back later.
823
932
self.remove_recursive_id(file_id)
829
938
for new_path, new_entry in sorted((np, e) for op, np, f, e in
830
939
delta if np is not None):
831
940
if new_entry.kind == 'directory':
832
# Pop the child which to allow detection of children whose
833
# parents were deleted and which were not reattached to a new
835
new_entry.children = children.pop(new_entry.file_id, {})
941
new_entry.children = children.get(new_entry.file_id, {})
836
942
self.add(new_entry)
838
# Get the parent id that was deleted
839
parent_id, children = children.popitem()
840
raise errors.InconsistentDelta("<deleted>", parent_id,
841
"The file id was deleted but its children were not deleted.")
843
944
def _set_root(self, ie):