263
263
def get_delta(self, version_id):
264
264
"""See VersionedFile.get_delta."""
265
if not self.has_version(version_id):
266
raise RevisionNotPresent(version_id, self)
267
sha1 = self.get_sha1(version_id)
268
parents = self.get_parents(version_id)
271
parent_inclusions = set(self.get_ancestry(parent))
274
parent_inclusions = set()
275
# we want to emit start, finish, replacement_length, replacement_lines tuples.
277
current_hunk = [0, 0, 0, []] #start, finish, repl_length, repl_tuples
278
inclusions = set(self.get_ancestry(version_id))
282
last_parent_line = ''
283
for lineno, inserted, deletes, line in self._walk_internal(inclusions):
284
parent_active = inserted in parent_inclusions and not (deletes & parent_inclusions)
285
version_active = inserted in inclusions and not (deletes & inclusions)
286
if not parent_active and not version_active:
287
# unrelated line of ancestry
289
elif parent_active and version_active:
291
if current_hunk != [parent_linenum, parent_linenum, 0, []]:
292
diff_hunks.append(tuple(current_hunk))
294
current_hunk = [parent_linenum, parent_linenum, 0, []]
295
if len(line) and line[-1] != '\n':
297
elif parent_active and not version_active:
301
last_parent_line = line
302
elif not parent_active and version_active:
304
# noeol only occurs at the end of a file because we
305
# diff linewise. We want to show noeol changes as a
306
# empty diff unless the actual eol-less content changed.
307
if len(last_parent_line) and last_parent_line[-1] != '\n':
309
if len(line) and line[-1] != '\n':
312
if parent_noeol == noeol:
313
# no noeol toggle, so trust the weave
317
if last_parent_line != line[:-1]:
321
# append a eol so that it looks like
325
if last_parent_line != line:
330
current_hunk[3].append((inserted, line))
332
# last hunk last parent line is not eaten
265
return self.get_deltas([version_id])[version_id]
267
def get_deltas(self, version_ids):
268
"""See VersionedFile.get_deltas."""
269
version_ids = self.get_ancestry(version_ids)
270
for version_id in version_ids:
271
if not self.has_version(version_id):
272
raise RevisionNotPresent(version_id, self)
273
# try extracting all versions; parallel extraction is used
274
nv = self.num_versions()
280
last_parent_lines = {}
282
parent_inclusions = {}
287
# its simplest to generate a full set of prepared variables.
289
name = self._names[i]
290
sha1s[name] = self.get_sha1(name)
291
parents_list = self.get_parents(name)
293
parent = parents_list[0]
294
parents[name] = parent
295
parent_inclusions[name] = inclusions[parent]
298
parent_inclusions[name] = set()
299
# we want to emit start, finish, replacement_length, replacement_lines tuples.
300
diff_hunks[name] = []
301
current_hunks[name] = [0, 0, 0, []] # #start, finish, repl_length, repl_tuples
302
parent_linenums[name] = 0
304
parent_noeols[name] = False
305
last_parent_lines[name] = None
306
new_inc = set([name])
307
for p in self._parents[i]:
308
new_inc.update(inclusions[self._idx_to_name(p)])
309
# debug only, known good so far.
310
#assert set(new_inc) == set(self.get_ancestry(name)), \
311
# 'failed %s != %s' % (set(new_inc), set(self.get_ancestry(name)))
312
inclusions[name] = new_inc
314
nlines = len(self._weave)
316
for lineno, inserted, deletes, line in self._walk_internal():
317
# a line is active in a version if:
318
# insert is in the versions inclusions
320
# deleteset & the versions inclusions is an empty set.
321
# so - if we have a included by mapping - version is included by
322
# children, we get a list of children to examine for deletes affect
323
# ing them, which is less than the entire set of children.
324
for version_id in version_ids:
325
# The active inclusion must be an ancestor,
326
# and no ancestors must have deleted this line,
327
# because we don't support resurrection.
328
parent_inclusion = parent_inclusions[version_id]
329
inclusion = inclusions[version_id]
330
parent_active = inserted in parent_inclusion and not (deletes & parent_inclusion)
331
version_active = inserted in inclusion and not (deletes & inclusion)
332
if not parent_active and not version_active:
333
# unrelated line of ancestry
335
elif parent_active and version_active:
337
parent_linenum = parent_linenums[version_id]
338
if current_hunks[version_id] != [parent_linenum, parent_linenum, 0, []]:
339
diff_hunks[version_id].append(tuple(current_hunks[version_id]))
341
current_hunks[version_id] = [parent_linenum, parent_linenum, 0, []]
342
parent_linenums[version_id] = parent_linenum
345
noeols[version_id] = True
348
elif parent_active and not version_active:
350
current_hunks[version_id][1] += 1
351
parent_linenums[version_id] += 1
352
last_parent_lines[version_id] = line
353
elif not parent_active and version_active:
355
# noeol only occurs at the end of a file because we
356
# diff linewise. We want to show noeol changes as a
357
# empty diff unless the actual eol-less content changed.
360
if last_parent_lines[version_id][-1] != '\n':
361
parent_noeols[version_id] = True
362
except (TypeError, IndexError):
365
if theline[-1] != '\n':
366
noeols[version_id] = True
370
parent_should_go = False
372
if parent_noeols[version_id] == noeols[version_id]:
373
# no noeol toggle, so trust the weaves statement
374
# that this line is changed.
376
if parent_noeols[version_id]:
377
theline = theline + '\n'
378
elif parent_noeols[version_id]:
379
# parent has no eol, we do:
380
# our line is new, report as such..
382
elif noeols[version_id]:
383
# append a eol so that it looks like
385
theline = theline + '\n'
386
if parents[version_id] is not None:
387
#if last_parent_lines[version_id] is not None:
388
parent_should_go = True
389
if last_parent_lines[version_id] != theline:
392
#parent_should_go = False
394
current_hunks[version_id][2] += 1
395
current_hunks[version_id][3].append((inserted, theline))
397
# last hunk last parent line is not eaten
398
current_hunks[version_id][1] -= 1
399
if current_hunks[version_id][1] < 0:
400
current_hunks[version_id][1] = 0
401
# import pdb;pdb.set_trace()
402
# assert current_hunks[version_id][1] >= 0
334
404
# flush last hunk
335
if current_hunk != [0, 0, 0, []]:
336
diff_hunks.append(tuple(current_hunk))
337
return parent, sha1, noeol, diff_hunks
406
version = self._idx_to_name(i)
407
if current_hunks[version] != [0, 0, 0, []]:
408
diff_hunks[version].append(tuple(current_hunks[version]))
410
for version_id in version_ids:
411
result[version_id] = (
415
diff_hunks[version_id],
339
419
def get_parents(self, version_id):
340
420
"""See VersionedFile.get_parent."""
341
421
return map(self._idx_to_name, self._parents[self._lookup(version_id)])