diff-lcs 1.3 → 1.5.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- checksums.yaml +5 -5
- data/Contributing.md +84 -48
- data/History.md +334 -154
- data/Manifest.txt +23 -1
- data/README.rdoc +10 -10
- data/Rakefile +85 -21
- data/bin/htmldiff +7 -4
- data/bin/ldiff +4 -1
- data/lib/diff/lcs/array.rb +1 -1
- data/lib/diff/lcs/backports.rb +9 -0
- data/lib/diff/lcs/block.rb +1 -1
- data/lib/diff/lcs/callbacks.rb +15 -12
- data/lib/diff/lcs/change.rb +30 -37
- data/lib/diff/lcs/htmldiff.rb +17 -16
- data/lib/diff/lcs/hunk.rb +156 -74
- data/lib/diff/lcs/internals.rb +43 -42
- data/lib/diff/lcs/ldiff.rb +46 -42
- data/lib/diff/lcs/string.rb +1 -1
- data/lib/diff/lcs.rb +188 -174
- data/lib/diff-lcs.rb +1 -1
- data/spec/change_spec.rb +31 -7
- data/spec/diff_spec.rb +16 -12
- data/spec/fixtures/aX +1 -0
- data/spec/fixtures/bXaX +1 -0
- data/spec/fixtures/ldiff/output.diff +4 -0
- data/spec/fixtures/ldiff/output.diff-c +7 -0
- data/spec/fixtures/ldiff/output.diff-e +3 -0
- data/spec/fixtures/ldiff/output.diff-f +3 -0
- data/spec/fixtures/ldiff/output.diff-u +5 -0
- data/spec/fixtures/ldiff/output.diff.chef +4 -0
- data/spec/fixtures/ldiff/output.diff.chef-c +15 -0
- data/spec/fixtures/ldiff/output.diff.chef-e +3 -0
- data/spec/fixtures/ldiff/output.diff.chef-f +3 -0
- data/spec/fixtures/ldiff/output.diff.chef-u +9 -0
- data/spec/fixtures/ldiff/output.diff.chef2 +7 -0
- data/spec/fixtures/ldiff/output.diff.chef2-c +20 -0
- data/spec/fixtures/ldiff/output.diff.chef2-d +7 -0
- data/spec/fixtures/ldiff/output.diff.chef2-e +7 -0
- data/spec/fixtures/ldiff/output.diff.chef2-f +7 -0
- data/spec/fixtures/ldiff/output.diff.chef2-u +16 -0
- data/spec/fixtures/new-chef +4 -0
- data/spec/fixtures/new-chef2 +17 -0
- data/spec/fixtures/old-chef +4 -0
- data/spec/fixtures/old-chef2 +14 -0
- data/spec/hunk_spec.rb +37 -26
- data/spec/issues_spec.rb +115 -10
- data/spec/lcs_spec.rb +10 -10
- data/spec/ldiff_spec.rb +71 -31
- data/spec/patch_spec.rb +93 -99
- data/spec/sdiff_spec.rb +89 -89
- data/spec/spec_helper.rb +118 -65
- data/spec/traverse_balanced_spec.rb +173 -173
- data/spec/traverse_sequences_spec.rb +29 -31
- metadata +54 -33
- data/autotest/discover.rb +0 -1
data/lib/diff/lcs/hunk.rb
CHANGED
@@ -1,30 +1,43 @@
|
|
1
|
-
#
|
1
|
+
# frozen_string_literal: true
|
2
2
|
|
3
3
|
require 'diff/lcs/block'
|
4
4
|
|
5
|
-
# A Hunk is a group of Blocks which overlap because of the context
|
6
|
-
#
|
7
|
-
#
|
5
|
+
# A Hunk is a group of Blocks which overlap because of the context surrounding
|
6
|
+
# each block. (So if we're not using context, every hunk will contain one
|
7
|
+
# block.) Used in the diff program (bin/ldiff).
|
8
8
|
class Diff::LCS::Hunk
|
9
|
-
|
10
|
-
|
9
|
+
OLD_DIFF_OP_ACTION = { '+' => 'a', '-' => 'd', '!' => 'c' }.freeze #:nodoc:
|
10
|
+
ED_DIFF_OP_ACTION = { '+' => 'a', '-' => 'd', '!' => 'c' }.freeze #:nodoc:
|
11
|
+
|
12
|
+
private_constant :OLD_DIFF_OP_ACTION, :ED_DIFF_OP_ACTION if respond_to?(:private_constant)
|
13
|
+
|
14
|
+
# Create a hunk using references to both the old and new data, as well as the
|
15
|
+
# piece of data.
|
11
16
|
def initialize(data_old, data_new, piece, flag_context, file_length_difference)
|
12
17
|
# At first, a hunk will have just one Block in it
|
13
|
-
@blocks = [
|
18
|
+
@blocks = [Diff::LCS::Block.new(piece)]
|
19
|
+
|
20
|
+
if @blocks[0].remove.empty? && @blocks[0].insert.empty?
|
21
|
+
fail "Cannot build a hunk from #{piece.inspect}; has no add or remove actions"
|
22
|
+
end
|
23
|
+
|
14
24
|
if String.method_defined?(:encoding)
|
15
|
-
@preferred_data_encoding = data_old.fetch(0, data_new.fetch(0,'')
|
25
|
+
@preferred_data_encoding = data_old.fetch(0, data_new.fetch(0, '')).encoding
|
16
26
|
end
|
27
|
+
|
17
28
|
@data_old = data_old
|
18
29
|
@data_new = data_new
|
19
30
|
|
20
31
|
before = after = file_length_difference
|
21
32
|
after += @blocks[0].diff_size
|
22
33
|
@file_length_difference = after # The caller must get this manually
|
34
|
+
@max_diff_size = @blocks.map { |e| e.diff_size.abs }.max
|
35
|
+
|
23
36
|
|
24
37
|
# Save the start & end of each array. If the array doesn't exist (e.g.,
|
25
|
-
# we're only adding items in this block), then figure out the line
|
26
|
-
#
|
27
|
-
#
|
38
|
+
# we're only adding items in this block), then figure out the line number
|
39
|
+
# based on the line number of the other file and the current difference in
|
40
|
+
# file lengths.
|
28
41
|
if @blocks[0].remove.empty?
|
29
42
|
a1 = a2 = nil
|
30
43
|
else
|
@@ -54,20 +67,27 @@ class Diff::LCS::Hunk
|
|
54
67
|
|
55
68
|
# Change the "start" and "end" fields to note that context should be added
|
56
69
|
# to this hunk.
|
57
|
-
attr_accessor :flag_context
|
58
|
-
undef :flag_context
|
59
|
-
def flag_context=(context) #:nodoc:
|
70
|
+
attr_accessor :flag_context # rubocop:disable Layout/EmptyLinesAroundAttributeAccessor
|
71
|
+
undef :flag_context=
|
72
|
+
def flag_context=(context) #:nodoc: # rubocop:disable Lint/DuplicateMethods
|
60
73
|
return if context.nil? or context.zero?
|
61
74
|
|
62
|
-
add_start =
|
75
|
+
add_start = context > @start_old ? @start_old : context
|
76
|
+
|
63
77
|
@start_old -= add_start
|
64
78
|
@start_new -= add_start
|
65
79
|
|
66
|
-
|
67
|
-
|
68
|
-
|
69
|
-
|
70
|
-
|
80
|
+
old_size = @data_old.size
|
81
|
+
|
82
|
+
add_end =
|
83
|
+
if (@end_old + context) > old_size
|
84
|
+
old_size - @end_old
|
85
|
+
else
|
86
|
+
context
|
87
|
+
end
|
88
|
+
|
89
|
+
add_end = @max_diff_size if add_end >= old_size
|
90
|
+
|
71
91
|
@end_old += add_end
|
72
92
|
@end_new += add_end
|
73
93
|
end
|
@@ -76,15 +96,13 @@ class Diff::LCS::Hunk
|
|
76
96
|
# a truthy value so that if there is no overlap, you can know the merge
|
77
97
|
# was skipped.
|
78
98
|
def merge(hunk)
|
79
|
-
|
80
|
-
|
81
|
-
|
82
|
-
|
83
|
-
|
84
|
-
nil
|
85
|
-
end
|
99
|
+
return unless overlaps?(hunk)
|
100
|
+
|
101
|
+
@start_old = hunk.start_old
|
102
|
+
@start_new = hunk.start_new
|
103
|
+
blocks.unshift(*hunk.blocks)
|
86
104
|
end
|
87
|
-
|
105
|
+
alias unshift merge
|
88
106
|
|
89
107
|
# Determines whether there is an overlap between this hunk and the
|
90
108
|
# provided hunk. This will be true if the difference between the two hunks
|
@@ -95,47 +113,53 @@ class Diff::LCS::Hunk
|
|
95
113
|
end
|
96
114
|
|
97
115
|
# Returns a diff string based on a format.
|
98
|
-
def diff(format)
|
116
|
+
def diff(format, last = false)
|
99
117
|
case format
|
100
118
|
when :old
|
101
|
-
old_diff
|
119
|
+
old_diff(last)
|
102
120
|
when :unified
|
103
|
-
unified_diff
|
121
|
+
unified_diff(last)
|
104
122
|
when :context
|
105
|
-
context_diff
|
123
|
+
context_diff(last)
|
106
124
|
when :ed
|
107
125
|
self
|
108
126
|
when :reverse_ed, :ed_finish
|
109
|
-
ed_diff(format)
|
127
|
+
ed_diff(format, last)
|
110
128
|
else
|
111
|
-
|
129
|
+
fail "Unknown diff format #{format}."
|
112
130
|
end
|
113
131
|
end
|
114
132
|
|
115
133
|
# Note that an old diff can't have any context. Therefore, we know that
|
116
134
|
# there's only one block in the hunk.
|
117
|
-
def old_diff
|
118
|
-
warn
|
119
|
-
op_act = { "+" => 'a', "-" => 'd', "!" => "c" }
|
135
|
+
def old_diff(_last = false)
|
136
|
+
warn 'Expecting only one block in an old diff hunk!' if @blocks.size > 1
|
120
137
|
|
121
138
|
block = @blocks[0]
|
122
139
|
|
123
140
|
# Calculate item number range. Old diff range is just like a context
|
124
141
|
# diff range, except the ranges are on one line with the action between
|
125
142
|
# them.
|
126
|
-
s = encode("#{context_range(:old)}#{
|
143
|
+
s = encode("#{context_range(:old, ',')}#{OLD_DIFF_OP_ACTION[block.op]}#{context_range(:new, ',')}\n")
|
127
144
|
# If removing anything, just print out all the remove lines in the hunk
|
128
145
|
# which is just all the remove lines in the block.
|
129
|
-
|
130
|
-
|
131
|
-
|
146
|
+
unless block.remove.empty?
|
147
|
+
@data_old[@start_old..@end_old].each { |e| s << encode('< ') + e.chomp + encode("\n") }
|
148
|
+
end
|
149
|
+
|
150
|
+
s << encode("---\n") if block.op == '!'
|
151
|
+
|
152
|
+
unless block.insert.empty?
|
153
|
+
@data_new[@start_new..@end_new].each { |e| s << encode('> ') + e.chomp + encode("\n") }
|
154
|
+
end
|
155
|
+
|
132
156
|
s
|
133
157
|
end
|
134
158
|
private :old_diff
|
135
159
|
|
136
|
-
def unified_diff
|
160
|
+
def unified_diff(last = false)
|
137
161
|
# Calculate item number range.
|
138
|
-
s = encode("@@ -#{unified_range(:old)} +#{unified_range(:new)} @@\n")
|
162
|
+
s = encode("@@ -#{unified_range(:old, last)} +#{unified_range(:new, last)} @@\n")
|
139
163
|
|
140
164
|
# Outlist starts containing the hunk of the old file. Removing an item
|
141
165
|
# just means putting a '-' in front of it. Inserting an item requires
|
@@ -148,7 +172,14 @@ class Diff::LCS::Hunk
|
|
148
172
|
# file -- don't take removed items into account.
|
149
173
|
lo, hi, num_added, num_removed = @start_old, @end_old, 0, 0
|
150
174
|
|
151
|
-
outlist = @data_old[lo
|
175
|
+
outlist = @data_old[lo..hi].map { |e| String.new("#{encode(' ')}#{e.chomp}") }
|
176
|
+
|
177
|
+
last_block = blocks[-1]
|
178
|
+
|
179
|
+
if last
|
180
|
+
old_missing_newline = missing_last_newline?(@data_old)
|
181
|
+
new_missing_newline = missing_last_newline?(@data_new)
|
182
|
+
end
|
152
183
|
|
153
184
|
@blocks.each do |block|
|
154
185
|
block.remove.each do |item|
|
@@ -157,66 +188,100 @@ class Diff::LCS::Hunk
|
|
157
188
|
outlist[offset][0, 1] = encode(op)
|
158
189
|
num_removed += 1
|
159
190
|
end
|
191
|
+
|
192
|
+
if last && block == last_block && old_missing_newline && !new_missing_newline
|
193
|
+
outlist << encode('\')
|
194
|
+
num_removed += 1
|
195
|
+
end
|
196
|
+
|
160
197
|
block.insert.each do |item|
|
161
198
|
op = item.action.to_s # +
|
162
199
|
offset = item.position - @start_new + num_removed
|
163
|
-
outlist[offset, 0] = encode(op) + @data_new[item.position]
|
200
|
+
outlist[offset, 0] = encode(op) + @data_new[item.position].chomp
|
164
201
|
num_added += 1
|
165
202
|
end
|
166
203
|
end
|
167
204
|
|
205
|
+
outlist << encode('\') if last && new_missing_newline
|
206
|
+
|
168
207
|
s << outlist.join(encode("\n"))
|
208
|
+
|
209
|
+
s
|
169
210
|
end
|
170
211
|
private :unified_diff
|
171
212
|
|
172
|
-
def context_diff
|
213
|
+
def context_diff(last = false)
|
173
214
|
s = encode("***************\n")
|
174
|
-
s << encode("*** #{context_range(:old)} ****\n")
|
175
|
-
r = context_range(:new)
|
215
|
+
s << encode("*** #{context_range(:old, ',', last)} ****\n")
|
216
|
+
r = context_range(:new, ',', last)
|
217
|
+
|
218
|
+
if last
|
219
|
+
old_missing_newline = missing_last_newline?(@data_old)
|
220
|
+
new_missing_newline = missing_last_newline?(@data_new)
|
221
|
+
end
|
176
222
|
|
177
223
|
# Print out file 1 part for each block in context diff format if there
|
178
224
|
# are any blocks that remove items
|
179
225
|
lo, hi = @start_old, @end_old
|
180
|
-
removes = @blocks.
|
181
|
-
|
182
|
-
|
226
|
+
removes = @blocks.reject { |e| e.remove.empty? }
|
227
|
+
|
228
|
+
unless removes.empty?
|
229
|
+
outlist = @data_old[lo..hi].map { |e| String.new("#{encode(' ')}#{e.chomp}") }
|
230
|
+
|
231
|
+
last_block = removes[-1]
|
183
232
|
|
184
233
|
removes.each do |block|
|
185
234
|
block.remove.each do |item|
|
186
235
|
outlist[item.position - lo][0, 1] = encode(block.op) # - or !
|
187
236
|
end
|
237
|
+
|
238
|
+
if last && block == last_block && old_missing_newline
|
239
|
+
outlist << encode('\')
|
240
|
+
end
|
188
241
|
end
|
189
|
-
|
242
|
+
|
243
|
+
s << outlist.join(encode("\n")) << encode("\n")
|
190
244
|
end
|
191
245
|
|
192
|
-
s << encode("
|
246
|
+
s << encode("--- #{r} ----\n")
|
193
247
|
lo, hi = @start_new, @end_new
|
194
|
-
inserts = @blocks.
|
195
|
-
|
196
|
-
|
248
|
+
inserts = @blocks.reject { |e| e.insert.empty? }
|
249
|
+
|
250
|
+
unless inserts.empty?
|
251
|
+
outlist = @data_new[lo..hi].map { |e| String.new("#{encode(' ')}#{e.chomp}") }
|
252
|
+
|
253
|
+
last_block = inserts[-1]
|
254
|
+
|
197
255
|
inserts.each do |block|
|
198
256
|
block.insert.each do |item|
|
199
257
|
outlist[item.position - lo][0, 1] = encode(block.op) # + or !
|
200
258
|
end
|
259
|
+
|
260
|
+
if last && block == last_block && new_missing_newline
|
261
|
+
outlist << encode('\')
|
262
|
+
end
|
201
263
|
end
|
202
|
-
s << outlist.join("\n")
|
264
|
+
s << outlist.join(encode("\n"))
|
203
265
|
end
|
266
|
+
|
204
267
|
s
|
205
268
|
end
|
206
269
|
private :context_diff
|
207
270
|
|
208
|
-
def ed_diff(format)
|
209
|
-
|
210
|
-
warn "Expecting only one block in an old diff hunk!" if @blocks.size > 1
|
271
|
+
def ed_diff(format, _last = false)
|
272
|
+
warn 'Expecting only one block in an old diff hunk!' if @blocks.size > 1
|
211
273
|
|
212
|
-
|
213
|
-
|
214
|
-
|
215
|
-
|
216
|
-
|
274
|
+
s =
|
275
|
+
if format == :reverse_ed
|
276
|
+
encode("#{ED_DIFF_OP_ACTION[@blocks[0].op]}#{context_range(:old, ',')}\n")
|
277
|
+
else
|
278
|
+
encode("#{context_range(:old, ' ')}#{ED_DIFF_OP_ACTION[@blocks[0].op]}\n")
|
279
|
+
end
|
217
280
|
|
218
281
|
unless @blocks[0].insert.empty?
|
219
|
-
@data_new[@start_new
|
282
|
+
@data_new[@start_new..@end_new].each do |e|
|
283
|
+
s << e.chomp + encode("\n")
|
284
|
+
end
|
220
285
|
s << encode(".\n")
|
221
286
|
end
|
222
287
|
s
|
@@ -225,7 +290,7 @@ class Diff::LCS::Hunk
|
|
225
290
|
|
226
291
|
# Generate a range of item numbers to print. Only print 1 number if the
|
227
292
|
# range has only one item in it. Otherwise, it's 'start,end'
|
228
|
-
def context_range(mode, op =
|
293
|
+
def context_range(mode, op, last = false)
|
229
294
|
case mode
|
230
295
|
when :old
|
231
296
|
s, e = (@start_old + 1), (@end_old + 1)
|
@@ -233,14 +298,17 @@ class Diff::LCS::Hunk
|
|
233
298
|
s, e = (@start_new + 1), (@end_new + 1)
|
234
299
|
end
|
235
300
|
|
236
|
-
|
301
|
+
e -= 1 if last
|
302
|
+
e = 1 if e.zero?
|
303
|
+
|
304
|
+
s < e ? "#{s}#{op}#{e}" : e.to_s
|
237
305
|
end
|
238
306
|
private :context_range
|
239
307
|
|
240
308
|
# Generate a range of item numbers to print for unified diff. Print number
|
241
309
|
# where block starts, followed by number of lines in the block
|
242
310
|
# (don't print number of lines if it's 1)
|
243
|
-
def unified_range(mode)
|
311
|
+
def unified_range(mode, last)
|
244
312
|
case mode
|
245
313
|
when :old
|
246
314
|
s, e = (@start_old + 1), (@end_old + 1)
|
@@ -248,12 +316,25 @@ class Diff::LCS::Hunk
|
|
248
316
|
s, e = (@start_new + 1), (@end_new + 1)
|
249
317
|
end
|
250
318
|
|
251
|
-
length = e - s + 1
|
252
|
-
|
253
|
-
|
319
|
+
length = e - s + (last ? 0 : 1)
|
320
|
+
|
321
|
+
first = length < 2 ? e : s # "strange, but correct"
|
322
|
+
length <= 1 ? first.to_s : "#{first},#{length}"
|
254
323
|
end
|
255
324
|
private :unified_range
|
256
325
|
|
326
|
+
def missing_last_newline?(data)
|
327
|
+
newline = encode("\n")
|
328
|
+
|
329
|
+
if data[-2]
|
330
|
+
data[-2].end_with?(newline) && !data[-1].end_with?(newline)
|
331
|
+
elsif data[-1]
|
332
|
+
!data[-1].end_with?(newline)
|
333
|
+
else
|
334
|
+
true
|
335
|
+
end
|
336
|
+
end
|
337
|
+
|
257
338
|
if String.method_defined?(:encoding)
|
258
339
|
def encode(literal, target_encoding = @preferred_data_encoding)
|
259
340
|
literal.encode target_encoding
|
@@ -263,10 +344,11 @@ class Diff::LCS::Hunk
|
|
263
344
|
args.map { |arg| arg.encode(string.encoding) }
|
264
345
|
end
|
265
346
|
else
|
266
|
-
def encode(literal,
|
347
|
+
def encode(literal, _target_encoding = nil)
|
267
348
|
literal
|
268
349
|
end
|
269
|
-
|
350
|
+
|
351
|
+
def encode_as(_string, *args)
|
270
352
|
args
|
271
353
|
end
|
272
354
|
end
|
data/lib/diff/lcs/internals.rb
CHANGED
@@ -1,4 +1,4 @@
|
|
1
|
-
#
|
1
|
+
# frozen_string_literal: true
|
2
2
|
|
3
3
|
class << Diff::LCS
|
4
4
|
def diff_traversal(method, seq1, seq2, callbacks, &block)
|
@@ -44,47 +44,49 @@ class << Diff::LCS::Internals
|
|
44
44
|
b_finish = b.size - 1
|
45
45
|
vector = []
|
46
46
|
|
47
|
-
#
|
48
|
-
while (
|
49
|
-
(a[a_start] == b[b_start]))
|
47
|
+
# Collect any common elements at the beginning...
|
48
|
+
while (a_start <= a_finish) and (b_start <= b_finish) and (a[a_start] == b[b_start])
|
50
49
|
vector[a_start] = b_start
|
51
50
|
a_start += 1
|
52
51
|
b_start += 1
|
53
52
|
end
|
54
|
-
b_start = a_start
|
55
53
|
|
56
54
|
# Now the end...
|
57
|
-
while (
|
58
|
-
(a[a_finish] == b[b_finish]))
|
55
|
+
while (a_start <= a_finish) and (b_start <= b_finish) and (a[a_finish] == b[b_finish])
|
59
56
|
vector[a_finish] = b_finish
|
60
57
|
a_finish -= 1
|
61
58
|
b_finish -= 1
|
62
59
|
end
|
63
60
|
|
64
61
|
# Now, compute the equivalence classes of positions of elements.
|
62
|
+
# An explanation for how this works: https://codeforces.com/topic/92191
|
65
63
|
b_matches = position_hash(b, b_start..b_finish)
|
66
64
|
|
67
65
|
thresh = []
|
68
66
|
links = []
|
69
67
|
string = a.kind_of?(String)
|
70
68
|
|
71
|
-
(a_start
|
69
|
+
(a_start..a_finish).each do |i|
|
72
70
|
ai = string ? a[i, 1] : a[i]
|
73
71
|
bm = b_matches[ai]
|
74
72
|
k = nil
|
75
73
|
bm.reverse_each do |j|
|
74
|
+
# Although the threshold check is not mandatory for this to work,
|
75
|
+
# it may have an optimization purpose
|
76
|
+
# An attempt to remove it: https://github.com/halostatue/diff-lcs/pull/72
|
77
|
+
# Why it is reintroduced: https://github.com/halostatue/diff-lcs/issues/78
|
76
78
|
if k and (thresh[k] > j) and (thresh[k - 1] < j)
|
77
79
|
thresh[k] = j
|
78
80
|
else
|
79
81
|
k = replace_next_larger(thresh, j, k)
|
80
82
|
end
|
81
|
-
links[k] = [
|
83
|
+
links[k] = [k.positive? ? links[k - 1] : nil, i, j] unless k.nil?
|
82
84
|
end
|
83
85
|
end
|
84
86
|
|
85
87
|
unless thresh.empty?
|
86
88
|
link = links[thresh.size - 1]
|
87
|
-
|
89
|
+
until link.nil?
|
88
90
|
vector[link[1]] = link[2]
|
89
91
|
link = link[0]
|
90
92
|
end
|
@@ -93,14 +95,15 @@ class << Diff::LCS::Internals
|
|
93
95
|
vector
|
94
96
|
end
|
95
97
|
|
96
|
-
# This method will analyze the provided patchset to provide a
|
97
|
-
#
|
98
|
-
#
|
99
|
-
#
|
98
|
+
# This method will analyze the provided patchset to provide a single-pass
|
99
|
+
# normalization (conversion of the array form of Diff::LCS::Change objects to
|
100
|
+
# the object form of same) and detection of whether the patchset represents
|
101
|
+
# changes to be made.
|
100
102
|
def analyze_patchset(patchset, depth = 0)
|
101
|
-
|
103
|
+
fail 'Patchset too complex' if depth > 1
|
102
104
|
|
103
105
|
has_changes = false
|
106
|
+
new_patchset = []
|
104
107
|
|
105
108
|
# Format:
|
106
109
|
# [ # patchset
|
@@ -110,29 +113,28 @@ class << Diff::LCS::Internals
|
|
110
113
|
# ]
|
111
114
|
# ]
|
112
115
|
|
113
|
-
patchset
|
116
|
+
patchset.each do |hunk|
|
114
117
|
case hunk
|
115
118
|
when Diff::LCS::Change
|
116
119
|
has_changes ||= !hunk.unchanged?
|
117
|
-
hunk
|
120
|
+
new_patchset << hunk
|
118
121
|
when Array
|
119
|
-
# Detect if the 'hunk' is actually an array-format
|
120
|
-
# Change object.
|
122
|
+
# Detect if the 'hunk' is actually an array-format change object.
|
121
123
|
if Diff::LCS::Change.valid_action? hunk[0]
|
122
124
|
hunk = Diff::LCS::Change.from_a(hunk)
|
123
125
|
has_changes ||= !hunk.unchanged?
|
124
|
-
hunk
|
126
|
+
new_patchset << hunk
|
125
127
|
else
|
126
128
|
with_changes, hunk = analyze_patchset(hunk, depth + 1)
|
127
129
|
has_changes ||= with_changes
|
128
|
-
hunk
|
130
|
+
new_patchset.concat(hunk)
|
129
131
|
end
|
130
132
|
else
|
131
|
-
|
133
|
+
fail ArgumentError, "Cannot normalise a hunk of class #{hunk.class}."
|
132
134
|
end
|
133
135
|
end
|
134
136
|
|
135
|
-
[
|
137
|
+
[has_changes, new_patchset]
|
136
138
|
end
|
137
139
|
|
138
140
|
# Examine the patchset and the source to see in which direction the
|
@@ -173,13 +175,11 @@ class << Diff::LCS::Internals
|
|
173
175
|
when '!'
|
174
176
|
if le == change.old_element
|
175
177
|
left_match += 1
|
178
|
+
elsif re == change.new_element
|
179
|
+
right_match += 1
|
176
180
|
else
|
177
|
-
|
178
|
-
|
179
|
-
else
|
180
|
-
left_miss += 1
|
181
|
-
right_miss += 1
|
182
|
-
end
|
181
|
+
left_miss += 1
|
182
|
+
right_miss += 1
|
183
183
|
end
|
184
184
|
end
|
185
185
|
when Diff::LCS::Change
|
@@ -209,16 +209,16 @@ class << Diff::LCS::Internals
|
|
209
209
|
end
|
210
210
|
end
|
211
211
|
|
212
|
-
break if
|
212
|
+
break if !limit.nil? && (count > limit)
|
213
213
|
end
|
214
214
|
|
215
|
-
no_left =
|
216
|
-
no_right =
|
215
|
+
no_left = left_match.zero? && left_miss.positive?
|
216
|
+
no_right = right_match.zero? && right_miss.positive?
|
217
217
|
|
218
|
-
case [
|
219
|
-
when [
|
218
|
+
case [no_left, no_right]
|
219
|
+
when [false, true]
|
220
220
|
:patch
|
221
|
-
when [
|
221
|
+
when [true, false]
|
222
222
|
:unpatch
|
223
223
|
else
|
224
224
|
case left_match <=> right_match
|
@@ -235,7 +235,8 @@ class << Diff::LCS::Internals
|
|
235
235
|
:patch
|
236
236
|
end
|
237
237
|
else
|
238
|
-
|
238
|
+
fail "The provided patchset does not appear to apply to the provided \
|
239
|
+
enumerable as either source or destination value."
|
239
240
|
end
|
240
241
|
end
|
241
242
|
end
|
@@ -256,16 +257,16 @@ class << Diff::LCS::Internals
|
|
256
257
|
end
|
257
258
|
|
258
259
|
# Binary search for the insertion point
|
259
|
-
last_index ||= enum.size
|
260
|
+
last_index ||= enum.size - 1
|
260
261
|
first_index = 0
|
261
|
-
while
|
262
|
+
while first_index <= last_index
|
262
263
|
i = (first_index + last_index) >> 1
|
263
264
|
|
264
265
|
found = enum[i]
|
265
266
|
|
266
|
-
if value == found
|
267
|
-
|
268
|
-
|
267
|
+
return nil if value == found
|
268
|
+
|
269
|
+
if value > found
|
269
270
|
first_index = i + 1
|
270
271
|
else
|
271
272
|
last_index = i - 1
|
@@ -275,7 +276,7 @@ class << Diff::LCS::Internals
|
|
275
276
|
# The insertion point is in first_index; overwrite the next larger
|
276
277
|
# value.
|
277
278
|
enum[first_index] = value
|
278
|
-
|
279
|
+
first_index
|
279
280
|
end
|
280
281
|
private :replace_next_larger
|
281
282
|
|