160
166
each version; the parent's parents are implied.
163
List of hex SHA-1 of each version, or None if not recorded.
169
List of hex SHA-1 of each version.
172
List of symbolic names for each version. Each should be unique.
175
For each name, the version number.
178
Descriptive name of this weave; typically the filename if known.
166
__slots__ = ['_weave', '_parents', '_sha1s']
182
__slots__ = ['_weave', '_parents', '_sha1s', '_names', '_name_map',
185
def __init__(self, weave_name=None, access_mode='w'):
186
super(Weave, self).__init__(access_mode)
170
188
self._parents = []
192
self._weave_name = weave_name
195
return "Weave(%r)" % self._weave_name
198
"""Return a deep copy of self.
200
The copy can be modified without affecting the original weave."""
202
other._weave = self._weave[:]
203
other._parents = self._parents[:]
204
other._sha1s = self._sha1s[:]
205
other._names = self._names[:]
206
other._name_map = self._name_map.copy()
207
other._weave_name = self._weave_name
174
210
def __eq__(self, other):
175
211
if not isinstance(other, Weave):
177
213
return self._parents == other._parents \
178
and self._weave == other._weave
214
and self._weave == other._weave \
215
and self._sha1s == other._sha1s
181
217
def __ne__(self, other):
182
218
return not self.__eq__(other)
185
def add(self, parents, text):
220
@deprecated_method(zero_eight)
221
def idx_to_name(self, index):
222
"""Old public interface, the public interface is all names now."""
225
def _idx_to_name(self, version):
226
return self._names[version]
228
@deprecated_method(zero_eight)
229
def lookup(self, name):
230
"""Backwards compatability thunk:
232
Return name, as name is valid in the api now, and spew deprecation
237
def _lookup(self, name):
238
"""Convert symbolic version name to index."""
240
return self._name_map[name]
242
raise RevisionNotPresent(name, self._weave_name)
244
@deprecated_method(zero_eight)
245
def iter_names(self):
246
"""Deprecated convenience function, please see VersionedFile.names()."""
247
return iter(self.names())
249
@deprecated_method(zero_eight)
251
"""See Weave.versions for the current api."""
252
return self.versions()
255
"""See VersionedFile.versions."""
256
return self._names[:]
258
def has_version(self, version_id):
259
"""See VersionedFile.has_version."""
260
return self._name_map.has_key(version_id)
262
__contains__ = has_version
264
def get_delta(self, version_id):
265
"""See VersionedFile.get_delta."""
266
return self.get_deltas([version_id])[version_id]
268
def get_deltas(self, version_ids):
269
"""See VersionedFile.get_deltas."""
270
version_ids = self.get_ancestry(version_ids)
271
for version_id in version_ids:
272
if not self.has_version(version_id):
273
raise RevisionNotPresent(version_id, self)
274
# try extracting all versions; parallel extraction is used
275
nv = self.num_versions()
281
last_parent_lines = {}
283
parent_inclusions = {}
288
# its simplest to generate a full set of prepared variables.
290
name = self._names[i]
291
sha1s[name] = self.get_sha1(name)
292
parents_list = self.get_parents(name)
294
parent = parents_list[0]
295
parents[name] = parent
296
parent_inclusions[name] = inclusions[parent]
299
parent_inclusions[name] = set()
300
# we want to emit start, finish, replacement_length, replacement_lines tuples.
301
diff_hunks[name] = []
302
current_hunks[name] = [0, 0, 0, []] # #start, finish, repl_length, repl_tuples
303
parent_linenums[name] = 0
305
parent_noeols[name] = False
306
last_parent_lines[name] = None
307
new_inc = set([name])
308
for p in self._parents[i]:
309
new_inc.update(inclusions[self._idx_to_name(p)])
310
# debug only, known good so far.
311
#assert set(new_inc) == set(self.get_ancestry(name)), \
312
# 'failed %s != %s' % (set(new_inc), set(self.get_ancestry(name)))
313
inclusions[name] = new_inc
315
nlines = len(self._weave)
317
for lineno, inserted, deletes, line in self._walk_internal():
318
# a line is active in a version if:
319
# insert is in the versions inclusions
321
# deleteset & the versions inclusions is an empty set.
322
# so - if we have a included by mapping - version is included by
323
# children, we get a list of children to examine for deletes affect
324
# ing them, which is less than the entire set of children.
325
for version_id in version_ids:
326
# The active inclusion must be an ancestor,
327
# and no ancestors must have deleted this line,
328
# because we don't support resurrection.
329
parent_inclusion = parent_inclusions[version_id]
330
inclusion = inclusions[version_id]
331
parent_active = inserted in parent_inclusion and not (deletes & parent_inclusion)
332
version_active = inserted in inclusion and not (deletes & inclusion)
333
if not parent_active and not version_active:
334
# unrelated line of ancestry
336
elif parent_active and version_active:
338
parent_linenum = parent_linenums[version_id]
339
if current_hunks[version_id] != [parent_linenum, parent_linenum, 0, []]:
340
diff_hunks[version_id].append(tuple(current_hunks[version_id]))
342
current_hunks[version_id] = [parent_linenum, parent_linenum, 0, []]
343
parent_linenums[version_id] = parent_linenum
346
noeols[version_id] = True
349
elif parent_active and not version_active:
351
current_hunks[version_id][1] += 1
352
parent_linenums[version_id] += 1
353
last_parent_lines[version_id] = line
354
elif not parent_active and version_active:
356
# noeol only occurs at the end of a file because we
357
# diff linewise. We want to show noeol changes as a
358
# empty diff unless the actual eol-less content changed.
361
if last_parent_lines[version_id][-1] != '\n':
362
parent_noeols[version_id] = True
363
except (TypeError, IndexError):
366
if theline[-1] != '\n':
367
noeols[version_id] = True
371
parent_should_go = False
373
if parent_noeols[version_id] == noeols[version_id]:
374
# no noeol toggle, so trust the weaves statement
375
# that this line is changed.
377
if parent_noeols[version_id]:
378
theline = theline + '\n'
379
elif parent_noeols[version_id]:
380
# parent has no eol, we do:
381
# our line is new, report as such..
383
elif noeols[version_id]:
384
# append a eol so that it looks like
386
theline = theline + '\n'
387
if parents[version_id] is not None:
388
#if last_parent_lines[version_id] is not None:
389
parent_should_go = True
390
if last_parent_lines[version_id] != theline:
393
#parent_should_go = False
395
current_hunks[version_id][2] += 1
396
current_hunks[version_id][3].append((inserted, theline))
398
# last hunk last parent line is not eaten
399
current_hunks[version_id][1] -= 1
400
if current_hunks[version_id][1] < 0:
401
current_hunks[version_id][1] = 0
402
# import pdb;pdb.set_trace()
403
# assert current_hunks[version_id][1] >= 0
407
version = self._idx_to_name(i)
408
if current_hunks[version] != [0, 0, 0, []]:
409
diff_hunks[version].append(tuple(current_hunks[version]))
411
for version_id in version_ids:
412
result[version_id] = (
416
diff_hunks[version_id],
420
def get_parents(self, version_id):
421
"""See VersionedFile.get_parent."""
422
return map(self._idx_to_name, self._parents[self._lookup(version_id)])
424
def _check_repeated_add(self, name, parents, text, sha1):
425
"""Check that a duplicated add is OK.
427
If it is, return the (old) index; otherwise raise an exception.
429
idx = self._lookup(name)
430
if sorted(self._parents[idx]) != sorted(parents) \
431
or sha1 != self._sha1s[idx]:
432
raise RevisionAlreadyPresent(name, self._weave_name)
435
@deprecated_method(zero_eight)
436
def add_identical(self, old_rev_id, new_rev_id, parents):
437
"""Please use Weave.clone_text now."""
438
return self.clone_text(new_rev_id, old_rev_id, parents)
440
def _add_lines(self, version_id, parents, lines, parent_texts):
441
"""See VersionedFile.add_lines."""
442
return self._add(version_id, lines, map(self._lookup, parents))
444
@deprecated_method(zero_eight)
445
def add(self, name, parents, text, sha1=None):
446
"""See VersionedFile.add_lines for the non deprecated api."""
447
return self._add(name, text, map(self._maybe_lookup, parents), sha1)
449
def _add(self, version_id, lines, parents, sha1=None):
186
450
"""Add a single text on top of the weave.
188
452
Returns the index number of the newly added version.
455
Symbolic name for this version.
456
(Typically the revision-id of the revision that added it.)
191
459
List or set of direct parent version numbers.
194
Sequence of lines to be added in the new version."""
462
Sequence of lines to be added in the new version.
465
assert isinstance(version_id, basestring)
467
sha1 = sha_strings(lines)
468
if version_id in self._name_map:
469
return self._check_repeated_add(version_id, parents, lines, sha1)
196
471
self._check_versions(parents)
197
## self._check_lines(text)
472
## self._check_lines(lines)
198
473
new_version = len(self._parents)
206
# if we abort after here the weave will be corrupt
207
self._parents.append(frozenset(parents))
475
# if we abort after here the (in-memory) weave will be corrupt because only
476
# some fields are updated
477
# XXX: FIXME implement a succeed-or-fail of the rest of this routine.
478
# - Robert Collins 20060226
479
self._parents.append(parents[:])
208
480
self._sha1s.append(sha1)
481
self._names.append(version_id)
482
self._name_map[version_id] = new_version
447
assert isinstance(l, basestring)
787
assert l.__class__ in (str, unicode)
448
788
if isactive is None:
449
789
isactive = (not dset) and istack and (istack[-1] in included)
451
791
result.append((istack[-1], lineno, l))
455
raise WFE("unclosed insertion blocks at end of weave",
794
raise WeaveFormatError("unclosed insertion blocks "
795
"at end of weave: %s" % istack)
458
raise WFE("unclosed deletion blocks at end of weave",
465
def get_iter(self, version):
466
"""Yield lines for the specified version."""
467
for origin, lineno, line in self._extract([version]):
471
def get(self, index):
472
return list(self.get_iter(index))
475
def mash_iter(self, included):
476
"""Return composed version of multiple included versions."""
477
for origin, lineno, text in self._extract(included):
481
def dump(self, to_file):
482
from pprint import pprint
483
print >>to_file, "Weave._weave = ",
484
pprint(self._weave, to_file)
485
print >>to_file, "Weave._parents = ",
486
pprint(self._parents, to_file)
797
raise WeaveFormatError("unclosed deletion blocks at end of weave: %s"
801
@deprecated_method(zero_eight)
802
def get_iter(self, name_or_index):
803
"""Deprecated, please do not use. Lookups are not not needed.
805
Please use get_lines now.
807
return iter(self.get_lines(self._maybe_lookup(name_or_index)))
809
@deprecated_method(zero_eight)
810
def maybe_lookup(self, name_or_index):
811
"""Deprecated, please do not use. Lookups are not not needed."""
812
return self._maybe_lookup(name_or_index)
814
def _maybe_lookup(self, name_or_index):
815
"""Convert possible symbolic name to index, or pass through indexes.
819
if isinstance(name_or_index, (int, long)):
822
return self._lookup(name_or_index)
824
@deprecated_method(zero_eight)
825
def get(self, version_id):
826
"""Please use either Weave.get_text or Weave.get_lines as desired."""
827
return self.get_lines(version_id)
829
def get_lines(self, version_id):
830
"""See VersionedFile.get_lines()."""
831
int_index = self._maybe_lookup(version_id)
832
result = [line for (origin, lineno, line) in self._extract([int_index])]
833
expected_sha1 = self._sha1s[int_index]
834
measured_sha1 = sha_strings(result)
835
if measured_sha1 != expected_sha1:
836
raise errors.WeaveInvalidChecksum(
837
'file %s, revision %s, expected: %s, measured %s'
838
% (self._weave_name, version_id,
839
expected_sha1, measured_sha1))
842
def get_sha1(self, name):
843
"""Get the stored sha1 sum for the given revision.
845
:param name: The name of the version to lookup
847
return self._sha1s[self._lookup(name)]
849
@deprecated_method(zero_eight)
490
850
def numversions(self):
851
"""How many versions are in this weave?
853
Deprecated in favour of num_versions.
855
return self.num_versions()
857
def num_versions(self):
858
"""How many versions are in this weave?"""
491
859
l = len(self._parents)
492
860
assert l == len(self._sha1s)
497
return self.numversions()
863
__len__ = num_versions
500
865
def check(self, progress_bar=None):
501
# check no circular inclusions
502
for version in range(self.numversions()):
866
# TODO evaluate performance hit of using string sets in this routine.
867
# TODO: check no circular inclusions
868
# TODO: create a nested progress bar
869
for version in range(self.num_versions()):
503
870
inclusions = list(self._parents[version])
505
872
inclusions.sort()
507
874
raise WeaveFormatError("invalid included version %d for index %d"
508
875
% (inclusions[-1], version))
510
# try extracting all versions; this is a bit slow and parallel
511
# extraction could be used
513
nv = self.numversions()
514
for version in range(nv):
877
# try extracting all versions; parallel extraction is used
878
nv = self.num_versions()
883
# For creating the ancestry, IntSet is much faster (3.7s vs 0.17s)
884
# The problem is that set membership is much more expensive
885
name = self._idx_to_name(i)
886
sha1s[name] = sha.new()
888
new_inc = set([name])
889
for p in self._parents[i]:
890
new_inc.update(inclusions[self._idx_to_name(p)])
892
assert set(new_inc) == set(self.get_ancestry(name)), \
893
'failed %s != %s' % (set(new_inc), set(self.get_ancestry(name)))
894
inclusions[name] = new_inc
896
nlines = len(self._weave)
898
update_text = 'checking weave'
900
short_name = os.path.basename(self._weave_name)
901
update_text = 'checking %s' % (short_name,)
902
update_text = update_text[:25]
904
for lineno, insert, deleteset, line in self._walk_internal():
516
progress_bar.update('checking text', version, nv)
518
for l in self.get_iter(version):
521
expected = self._sha1s[version]
906
progress_bar.update(update_text, lineno, nlines)
908
for name, name_inclusions in inclusions.items():
909
# The active inclusion must be an ancestor,
910
# and no ancestors must have deleted this line,
911
# because we don't support resurrection.
912
if (insert in name_inclusions) and not (deleteset & name_inclusions):
913
sha1s[name].update(line)
916
version = self._idx_to_name(i)
917
hd = sha1s[version].hexdigest()
918
expected = self._sha1s[i]
522
919
if hd != expected:
523
raise WeaveError("mismatched sha1 for version %d; "
524
"got %s, expected %s"
525
% (version, hd, expected))
920
raise errors.WeaveInvalidChecksum(
921
"mismatched sha1 for version %s: "
922
"got %s, expected %s"
923
% (version, hd, expected))
527
925
# TODO: check insertions are properly nested, that there are
528
926
# no lines outside of insertion blocks, that deletions are
529
927
# properly paired, etc.
533
def merge(self, merge_versions):
534
"""Automerge and mark conflicts between versions.
536
This returns a sequence, each entry describing alternatives
537
for a chunk of the file. Each of the alternatives is given as
540
If there is a chunk of the file where there's no diagreement,
541
only one alternative is given.
544
# approach: find the included versions common to all the
546
raise NotImplementedError()
550
def _delta(self, included, lines):
551
"""Return changes from basis to new revision.
553
The old text for comparison is the union of included revisions.
555
This is used in inserting a new text.
557
Delta is returned as a sequence of
558
(weave1, weave2, newlines).
560
This indicates that weave1:weave2 of the old weave should be
561
replaced by the sequence of lines in newlines. Note that
562
these line numbers are positions in the total weave and don't
563
correspond to the lines in any extracted version, or even the
564
extracted union of included versions.
566
If line1=line2, this is a pure insert; if newlines=[] this is a
567
pure delete. (Similar to difflib.)
572
def plan_merge(self, ver_a, ver_b):
573
"""Return pseudo-annotation indicating how the two versions merge.
575
This is computed between versions a and b and their common
578
Weave lines present in none of them are skipped entirely.
580
inc_a = self.inclusions([ver_a])
581
inc_b = self.inclusions([ver_b])
582
inc_c = inc_a & inc_b
584
for lineno, insert, deleteset, line in self._walk():
585
if deleteset & inc_c:
586
# killed in parent; can't be in either a or b
587
# not relevant to our work
588
yield 'killed-base', line
589
elif insert in inc_c:
590
# was inserted in base
591
killed_a = bool(deleteset & inc_a)
592
killed_b = bool(deleteset & inc_b)
593
if killed_a and killed_b:
594
yield 'killed-both', line
596
yield 'killed-a', line
598
yield 'killed-b', line
600
yield 'unchanged', line
601
elif insert in inc_a:
602
if deleteset & inc_a:
603
yield 'ghost-a', line
607
elif insert in inc_b:
608
if deleteset & inc_b:
609
yield 'ghost-b', line
613
# not in either revision
614
yield 'irrelevant', line
616
yield 'unchanged', '' # terminator
620
def weave_merge(self, plan):
625
for state, line in plan:
626
if state == 'unchanged' or state == 'killed-both':
627
# resync and flush queued conflicts changes if any
628
if not lines_a and not lines_b:
630
elif ch_a and not ch_b:
632
for l in lines_a: yield l
633
elif ch_b and not ch_a:
634
for l in lines_b: yield l
635
elif lines_a == lines_b:
636
for l in lines_a: yield l
639
for l in lines_a: yield l
641
for l in lines_b: yield l
648
if state == 'unchanged':
651
elif state == 'killed-a':
654
elif state == 'killed-b':
657
elif state == 'new-a':
660
elif state == 'new-b':
664
assert state in ('irrelevant', 'ghost-a', 'ghost-b', 'killed-base',
675
"""Show some text information about the weave."""
676
print '%6s %40s %20s' % ('ver', 'sha1', 'parents')
677
for i in (6, 40, 20):
929
def _join(self, other, pb, msg, version_ids, ignore_missing):
930
"""Worker routine for join()."""
931
if not other.versions():
932
return # nothing to update, easy
935
for version_id in version_ids:
936
if not other.has_version(version_id) and not ignore_missing:
937
raise RevisionNotPresent(version_id, self._weave_name)
939
version_ids = other.versions()
941
# two loops so that we do not change ourselves before verifying it
943
# work through in index order to make sure we get all dependencies
946
# get the selected versions only that are in other.versions.
947
version_ids = set(other.versions()).intersection(set(version_ids))
948
# pull in the referenced graph.
949
version_ids = other.get_ancestry(version_ids)
950
pending_graph = [(version, other.get_parents(version)) for
951
version in version_ids]
952
for name in topo_sort(pending_graph):
953
other_idx = other._name_map[name]
954
# returns True if we have it, False if we need it.
955
if not self._check_version_consistent(other, other_idx, name):
956
names_to_join.append((other_idx, name))
965
for other_idx, name in names_to_join:
966
# TODO: If all the parents of the other version are already
967
# present then we can avoid some work by just taking the delta
968
# and adjusting the offsets.
969
new_parents = self._imported_parents(other, other_idx)
970
sha1 = other._sha1s[other_idx]
975
pb.update(msg, merged, len(names_to_join))
977
lines = other.get_lines(other_idx)
978
self._add(name, lines, new_parents, sha1)
980
mutter("merged = %d, processed = %d, file_id=%s; deltat=%d"%(
981
merged, processed, self._weave_name, time.time()-time0))
983
def _imported_parents(self, other, other_idx):
984
"""Return list of parents in self corresponding to indexes in other."""
986
for parent_idx in other._parents[other_idx]:
987
parent_name = other._names[parent_idx]
988
if parent_name not in self._name_map:
989
# should not be possible
990
raise WeaveError("missing parent {%s} of {%s} in %r"
991
% (parent_name, other._name_map[other_idx], self))
992
new_parents.append(self._name_map[parent_name])
995
def _check_version_consistent(self, other, other_idx, name):
996
"""Check if a version in consistent in this and other.
998
To be consistent it must have:
1001
* the same direct parents (by name, not index, and disregarding
1004
If present & correct return True;
1005
if not present in self return False;
1006
if inconsistent raise error."""
1007
this_idx = self._name_map.get(name, -1)
1009
if self._sha1s[this_idx] != other._sha1s[other_idx]:
1010
raise errors.WeaveTextDiffers(name, self, other)
1011
self_parents = self._parents[this_idx]
1012
other_parents = other._parents[other_idx]
1013
n1 = set([self._names[i] for i in self_parents])
1014
n2 = set([other._names[i] for i in other_parents])
1015
if not self._compatible_parents(n1, n2):
1016
raise WeaveParentMismatch("inconsistent parents "
1017
"for version {%s}: %s vs %s" % (name, n1, n2))
1023
@deprecated_method(zero_eight)
1024
def reweave(self, other, pb=None, msg=None):
1025
"""reweave has been superceded by plain use of join."""
1026
return self.join(other, pb, msg)
1028
def _reweave(self, other, pb, msg):
1029
"""Reweave self with other - internal helper for join().
1031
:param other: The other weave to merge
1032
:param pb: An optional progress bar, indicating how far done we are
1033
:param msg: An optional message for the progress
1035
new_weave = _reweave(self, other, pb=pb, msg=msg)
1036
self._copy_weave_content(new_weave)
1038
def _copy_weave_content(self, otherweave):
1039
"""adsorb the content from otherweave."""
1040
for attr in self.__slots__:
1041
if attr != '_weave_name':
1042
setattr(self, attr, copy(getattr(otherweave, attr)))
1045
class WeaveFile(Weave):
1046
"""A WeaveFile represents a Weave on disk and writes on change."""
1048
WEAVE_SUFFIX = '.weave'
1050
def __init__(self, name, transport, filemode=None, create=False, access_mode='w'):
1051
"""Create a WeaveFile.
1053
:param create: If not True, only open an existing knit.
1055
super(WeaveFile, self).__init__(name, access_mode)
1056
self._transport = transport
1057
self._filemode = filemode
1059
_read_weave_v5(self._transport.get(name + WeaveFile.WEAVE_SUFFIX), self)
1060
except errors.NoSuchFile:
1066
def _add_lines(self, version_id, parents, lines, parent_texts):
1067
"""Add a version and save the weave."""
1068
result = super(WeaveFile, self)._add_lines(version_id, parents, lines,
1073
def _clone_text(self, new_version_id, old_version_id, parents):
1074
"""See VersionedFile.clone_text."""
1075
super(WeaveFile, self)._clone_text(new_version_id, old_version_id, parents)
1078
def copy_to(self, name, transport):
1079
"""See VersionedFile.copy_to()."""
1080
# as we are all in memory always, just serialise to the new place.
1082
write_weave_v5(self, sio)
1084
transport.put(name + WeaveFile.WEAVE_SUFFIX, sio, self._filemode)
1086
def create_empty(self, name, transport, filemode=None):
1087
return WeaveFile(name, transport, filemode, create=True)
1090
"""Save the weave."""
1091
self._check_write_ok()
1093
write_weave_v5(self, sio)
1095
self._transport.put(self._weave_name + WeaveFile.WEAVE_SUFFIX,
1101
"""See VersionedFile.get_suffixes()."""
1102
return [WeaveFile.WEAVE_SUFFIX]
1104
def join(self, other, pb=None, msg=None, version_ids=None,
1105
ignore_missing=False):
1106
"""Join other into self and save."""
1107
super(WeaveFile, self).join(other, pb, msg, version_ids, ignore_missing)
1111
@deprecated_function(zero_eight)
1112
def reweave(wa, wb, pb=None, msg=None):
1113
"""reweaving is deprecation, please just use weave.join()."""
1114
_reweave(wa, wb, pb, msg)
1116
def _reweave(wa, wb, pb=None, msg=None):
1117
"""Combine two weaves and return the result.
1119
This works even if a revision R has different parents in
1120
wa and wb. In the resulting weave all the parents are given.
1122
This is done by just building up a new weave, maintaining ordering
1123
of the versions in the two inputs. More efficient approaches
1124
might be possible but it should only be necessary to do
1125
this operation rarely, when a new previously ghost version is
1128
:param pb: An optional progress bar, indicating how far done we are
1129
:param msg: An optional message for the progress
1133
queue_a = range(wa.num_versions())
1134
queue_b = range(wb.num_versions())
1135
# first determine combined parents of all versions
1136
# map from version name -> all parent names
1137
combined_parents = _reweave_parent_graphs(wa, wb)
1138
mutter("combined parents: %r", combined_parents)
1139
order = topo_sort(combined_parents.iteritems())
1140
mutter("order to reweave: %r", order)
1145
for idx, name in enumerate(order):
1147
pb.update(msg, idx, len(order))
1148
if name in wa._name_map:
1149
lines = wa.get_lines(name)
1150
if name in wb._name_map:
1151
lines_b = wb.get_lines(name)
1152
if lines != lines_b:
1153
mutter('Weaves differ on content. rev_id {%s}', name)
1154
mutter('weaves: %s, %s', wa._weave_name, wb._weave_name)
1156
lines = list(difflib.unified_diff(lines, lines_b,
1157
wa._weave_name, wb._weave_name))
1158
mutter('lines:\n%s', ''.join(lines))
1159
raise errors.WeaveTextDiffers(name, wa, wb)
1161
lines = wb.get_lines(name)
1162
wr._add(name, lines, [wr._lookup(i) for i in combined_parents[name]])
1165
def _reweave_parent_graphs(wa, wb):
1166
"""Return combined parent ancestry for two weaves.
1168
Returned as a list of (version_name, set(parent_names))"""
1170
for weave in [wa, wb]:
1171
for idx, name in enumerate(weave._names):
1172
p = combined.setdefault(name, set())
1173
p.update(map(weave._idx_to_name, weave._parents[idx]))
1178
"""Show the weave's table-of-contents"""
1179
print '%6s %50s %10s %10s' % ('ver', 'name', 'sha1', 'parents')
1180
for i in (6, 50, 10, 10):
680
for i in range(w.numversions()):
1183
for i in range(w.num_versions()):
681
1184
sha1 = w._sha1s[i]
682
print '%6d %40s %s' % (i, sha1, ' '.join(map(str, w._parents[i])))
686
def weave_stats(weave_file):
687
from bzrlib.progress import ProgressBar
1186
parent_str = ' '.join(map(str, w._parents[i]))
1187
print '%6d %-50.50s %10.10s %s' % (i, name, sha1, parent_str)
1191
def weave_stats(weave_file, pb):
688
1192
from bzrlib.weavefile import read_weave
692
1194
wf = file(weave_file, 'rb')
1195
w = read_weave(wf, WeaveVersionedFile)
694
1196
# FIXME: doesn't work on pipes
695
1197
weave_size = wf.tell()