77
94
if len(revno_str) > max_revno_len:
78
95
revno_str = revno_str[:max_revno_len-1] + '>'
79
96
anno = "%-*s %-7s " % (max_revno_len, revno_str, author[:7])
81
if anno.lstrip() == "" and full: anno = prevanno
82
print >>to_file, '%s| %s' % (anno, text)
97
if anno.lstrip() == "" and full:
101
except UnicodeEncodeError:
102
# cmd_annotate should be passing in an 'exact' object, which means
103
# we have a direct handle to sys.stdout or equivalent. It may not
104
# be able to handle the exact Unicode characters, but 'annotate' is
105
# a user function (non-scripting), so shouldn't die because of
106
# unrepresentable annotation characters. So encode using 'replace',
107
# and write them again.
108
to_file.write(anno.encode(encoding, 'replace'))
109
to_file.write('| %s\n' % (text,))
113
def _annotations(repo, file_id, rev_id):
114
"""Return the list of (origin,text) for a revision of a file in a repository."""
115
w = repo.weave_store.get_weave(file_id, repo.get_transaction())
116
return list(w.annotate_iter(rev_id))
86
119
def _annotate_file(branch, rev_id, file_id):
87
120
"""Yield the origins for each line of a file.
89
This includes detailed information, such as the committer name, and
122
This includes detailed information, such as the author name, and
90
123
date string for the commit, rather than just the revision id.
92
125
revision_id_to_revno = branch.get_revision_id_to_revno_map()
93
w = branch.repository.weave_store.get_weave(file_id,
94
branch.repository.get_transaction())
126
annotations = _annotations(branch.repository, file_id, rev_id)
95
127
last_origin = None
96
annotations = list(w.annotate_iter(rev_id))
97
128
revision_ids = set(o for o, t in annotations)
98
129
revision_ids = [o for o in revision_ids if
99
130
branch.repository.has_revision(o)]
124
155
yield (revno_str, author, date_str, origin, text)
127
def reannotate(parents_lines, new_lines, new_revision_id):
158
def reannotate(parents_lines, new_lines, new_revision_id,
159
_left_matching_blocks=None,
160
heads_provider=None):
128
161
"""Create a new annotated version from new lines and parent annotations.
130
163
:param parents_lines: List of annotated lines for all parents
131
164
:param new_lines: The un-annotated new lines
132
165
:param new_revision_id: The revision-id to associate with new lines
133
166
(will often be CURRENT_REVISION)
167
:param left_matching_blocks: a hint about which areas are common
168
between the text and its left-hand-parent. The format is
169
the SequenceMatcher.get_matching_blocks format
170
(start_left, start_right, length_of_match).
171
:param heads_provider: An object which provids a .heads() call to resolve
172
if any revision ids are children of others.
173
If None, then any ancestry disputes will be resolved with
135
if len(parents_lines) == 1:
136
for data in _reannotate(parents_lines[0], new_lines, new_revision_id):
176
if len(parents_lines) == 0:
177
lines = [(new_revision_id, line) for line in new_lines]
178
elif len(parents_lines) == 1:
179
lines = _reannotate(parents_lines[0], new_lines, new_revision_id,
180
_left_matching_blocks)
181
elif len(parents_lines) == 2:
182
left = _reannotate(parents_lines[0], new_lines, new_revision_id,
183
_left_matching_blocks)
184
lines = _reannotate_annotated(parents_lines[1], new_lines,
185
new_revision_id, left,
139
reannotations = [list(_reannotate(p, new_lines, new_revision_id)) for
188
reannotations = [_reannotate(parents_lines[0], new_lines,
189
new_revision_id, _left_matching_blocks)]
190
reannotations.extend(_reannotate(p, new_lines, new_revision_id)
191
for p in parents_lines[1:])
141
193
for annos in zip(*reannotations):
142
194
origins = set(a for a, l in annos)
144
195
if len(origins) == 1:
145
yield iter(origins).next(), line
146
elif len(origins) == 2 and new_revision_id in origins:
147
yield (x for x in origins if x != new_revision_id).next(), line
196
# All the parents agree, so just return the first one
197
lines.append(annos[0])
149
yield new_revision_id, line
152
def _reannotate(parent_lines, new_lines, new_revision_id):
153
plain_parent_lines = [l for r, l in parent_lines]
154
matcher = patiencediff.PatienceSequenceMatcher(None, plain_parent_lines,
200
if len(origins) == 2 and new_revision_id in origins:
201
origins.remove(new_revision_id)
202
if len(origins) == 1:
203
lines.append((origins.pop(), line))
205
lines.append((new_revision_id, line))
209
def _reannotate(parent_lines, new_lines, new_revision_id,
210
matching_blocks=None):
157
for i, j, n in matcher.get_matching_blocks():
212
if matching_blocks is None:
213
plain_parent_lines = [l for r, l in parent_lines]
214
matcher = patiencediff.PatienceSequenceMatcher(None,
215
plain_parent_lines, new_lines)
216
matching_blocks = matcher.get_matching_blocks()
218
for i, j, n in matching_blocks:
158
219
for line in new_lines[new_cur:j]:
159
yield new_revision_id, line
160
for data in parent_lines[i:i+n]:
220
lines.append((new_revision_id, line))
221
lines.extend(parent_lines[i:i+n])
226
def _get_matching_blocks(old, new):
227
matcher = patiencediff.PatienceSequenceMatcher(None,
229
return matcher.get_matching_blocks()
232
def _find_matching_unannotated_lines(output_lines, plain_child_lines,
233
child_lines, start_child, end_child,
234
right_lines, start_right, end_right,
235
heads_provider, revision_id):
236
"""Find lines in plain_right_lines that match the existing lines.
238
:param output_lines: Append final annotated lines to this list
239
:param plain_child_lines: The unannotated new lines for the child text
240
:param child_lines: Lines for the child text which have been annotated
242
:param start_child: Position in plain_child_lines and child_lines to start the
244
:param end_child: Last position in plain_child_lines and child_lines to search
246
:param right_lines: The annotated lines for the whole text for the right
248
:param start_right: Position in right_lines to start the match
249
:param end_right: Last position in right_lines to search for a match
250
:param heads_provider: When parents disagree on the lineage of a line, we
251
need to check if one side supersedes the other
252
:param revision_id: The label to give if a line should be labeled 'tip'
254
output_extend = output_lines.extend
255
output_append = output_lines.append
256
# We need to see if any of the unannotated lines match
257
plain_right_subset = [l for a,l in right_lines[start_right:end_right]]
258
plain_child_subset = plain_child_lines[start_child:end_child]
259
match_blocks = _get_matching_blocks(plain_right_subset, plain_child_subset)
263
for right_idx, child_idx, match_len in match_blocks:
264
# All the lines that don't match are just passed along
265
if child_idx > last_child_idx:
266
output_extend(child_lines[start_child + last_child_idx
267
:start_child + child_idx])
268
for offset in xrange(match_len):
269
left = child_lines[start_child+child_idx+offset]
270
right = right_lines[start_right+right_idx+offset]
271
if left[0] == right[0]:
272
# The annotations match, just return the left one
274
elif left[0] == revision_id:
275
# The left parent marked this as unmatched, so let the
276
# right parent claim it
279
# Left and Right both claim this line
280
if heads_provider is None:
281
output_append((revision_id, left[1]))
283
heads = heads_provider.heads((left[0], right[0]))
285
output_append((iter(heads).next(), left[1]))
287
# Both claim different origins
288
output_append((revision_id, left[1]))
289
# We know that revision_id is the head for
290
# left and right, so cache it
291
heads_provider.cache(
292
(revision_id, left[0]),
294
heads_provider.cache(
295
(revision_id, right[0]),
297
last_child_idx = child_idx + match_len
300
def _reannotate_annotated(right_parent_lines, new_lines, new_revision_id,
301
annotated_lines, heads_provider):
302
"""Update the annotations for a node based on another parent.
304
:param right_parent_lines: A list of annotated lines for the right-hand
306
:param new_lines: The unannotated new lines.
307
:param new_revision_id: The revision_id to attribute to lines which are not
308
present in either parent.
309
:param annotated_lines: A list of annotated lines. This should be the
310
annotation of new_lines based on parents seen so far.
311
:param heads_provider: When parents disagree on the lineage of a line, we
312
need to check if one side supersedes the other.
314
assert len(new_lines) == len(annotated_lines)
315
# First compare the newly annotated lines with the right annotated lines.
316
# Lines which were not changed in left or right should match. This tends to
317
# be the bulk of the lines, and they will need no further processing.
319
lines_extend = lines.extend
320
last_right_idx = 0 # The line just after the last match from the right side
322
matching_left_and_right = _get_matching_blocks(right_parent_lines,
324
for right_idx, left_idx, match_len in matching_left_and_right:
325
# annotated lines from last_left_idx to left_idx did not match the lines from
327
# to right_idx, the raw lines should be compared to determine what annotations
329
if last_right_idx == right_idx or last_left_idx == left_idx:
330
# One of the sides is empty, so this is a pure insertion
331
lines_extend(annotated_lines[last_left_idx:left_idx])
333
# We need to see if any of the unannotated lines match
334
_find_matching_unannotated_lines(lines,
335
new_lines, annotated_lines,
336
last_left_idx, left_idx,
338
last_right_idx, right_idx,
341
last_right_idx = right_idx + match_len
342
last_left_idx = left_idx + match_len
343
# If left and right agree on a range, just push that into the output
344
assert len(lines) == left_idx
345
lines_extend(annotated_lines[left_idx:left_idx + match_len])