1#!/bin/sh
2
3test_description='git blame ignore fuzzy heuristic'
4. ./test-lib.sh
5
6pick_author='s/^[0-9a-f^]* *(\([^ ]*\) .*/\1/'
7
8# Each test is composed of 4 variables:
9# titleN - the test name
10# aN - the initial content
11# bN - the final content
12# expectedN - the line numbers from aN that we expect git blame
13# on bN to identify, or "Final" if bN itself should
14# be identified as the origin of that line.
15
16# We start at test 2 because setup will show as test 1
17title2="Regression test for partially overlapping search ranges"
18cat <<EOF >a2
191
202
213
22abcdef
235
246
257
26ijkl
279
2810
2911
30pqrs
3113
3214
3315
34wxyz
3517
3618
3719
38EOF
39cat <<EOF >b2
40abcde
41ijk
42pqr
43wxy
44EOF
45cat <<EOF >expected2
464
478
4812
4916
50EOF
51
52title3="Combine 3 lines into 2"
53cat <<EOF >a3
54if ((maxgrow==0) ||
55 ( single_line_field && (field->dcols < maxgrow)) ||
56 (!single_line_field && (field->drows < maxgrow)))
57EOF
58cat <<EOF >b3
59if ((maxgrow == 0) || (single_line_field && (field->dcols < maxgrow)) ||
60 (!single_line_field && (field->drows < maxgrow))) {
61EOF
62cat <<EOF >expected3
632
643
65EOF
66
67title4="Add curly brackets"
68cat <<EOF >a4
69 if (rows) *rows = field->rows;
70 if (cols) *cols = field->cols;
71 if (frow) *frow = field->frow;
72 if (fcol) *fcol = field->fcol;
73EOF
74cat <<EOF >b4
75 if (rows) {
76 *rows = field->rows;
77 }
78 if (cols) {
79 *cols = field->cols;
80 }
81 if (frow) {
82 *frow = field->frow;
83 }
84 if (fcol) {
85 *fcol = field->fcol;
86 }
87EOF
88cat <<EOF >expected4
891
901
91Final
922
932
94Final
953
963
97Final
984
994
100Final
101EOF
102
103
104title5="Combine many lines and change case"
105cat <<EOF >a5
106for(row=0,pBuffer=field->buf;
107 row<height;
108 row++,pBuffer+=width )
109{
110 if ((len = (int)( After_End_Of_Data( pBuffer, width ) - pBuffer )) > 0)
111 {
112 wmove( win, row, 0 );
113 waddnstr( win, pBuffer, len );
114EOF
115cat <<EOF >b5
116for (Row = 0, PBuffer = field->buf; Row < Height; Row++, PBuffer += Width) {
117 if ((Len = (int)(afterEndOfData(PBuffer, Width) - PBuffer)) > 0) {
118 wmove(win, Row, 0);
119 waddnstr(win, PBuffer, Len);
120EOF
121cat <<EOF >expected5
1221
1235
1247
1258
126EOF
127
128title6="Rename and combine lines"
129cat <<EOF >a6
130bool need_visual_update = ((form != (FORM *)0) &&
131 (form->status & _POSTED) &&
132 (form->current==field));
133
134if (need_visual_update)
135 Synchronize_Buffer(form);
136
137if (single_line_field)
138{
139 growth = field->cols * amount;
140 if (field->maxgrow)
141 growth = Minimum(field->maxgrow - field->dcols,growth);
142 field->dcols += growth;
143 if (field->dcols == field->maxgrow)
144EOF
145cat <<EOF >b6
146bool NeedVisualUpdate = ((Form != (FORM *)0) && (Form->status & _POSTED) &&
147 (Form->current == field));
148
149if (NeedVisualUpdate) {
150 synchronizeBuffer(Form);
151}
152
153if (SingleLineField) {
154 Growth = field->cols * amount;
155 if (field->maxgrow) {
156 Growth = Minimum(field->maxgrow - field->dcols, Growth);
157 }
158 field->dcols += Growth;
159 if (field->dcols == field->maxgrow) {
160EOF
161cat <<EOF >expected6
1621
1633
1644
1655
1666
167Final
1687
1698
17010
17111
17212
173Final
17413
17514
176EOF
177
178# Both lines match identically so position must be used to tie-break.
179title7="Same line twice"
180cat <<EOF >a7
181abc
182abc
183EOF
184cat <<EOF >b7
185abcd
186abcd
187EOF
188cat <<EOF >expected7
1891
1902
191EOF
192
193title8="Enforce line order"
194cat <<EOF >a8
195abcdef
196ghijkl
197ab
198EOF
199cat <<EOF >b8
200ghijk
201abcd
202EOF
203cat <<EOF >expected8
2042
2053
206EOF
207
208title9="Expand lines and rename variables"
209cat <<EOF >a9
210int myFunction(int ArgumentOne, Thing *ArgTwo, Blah XuglyBug) {
211 Squiggle FabulousResult = squargle(ArgumentOne, *ArgTwo,
212 XuglyBug) + EwwwGlobalWithAReallyLongNameYepTooLong;
213 return FabulousResult * 42;
214}
215EOF
216cat <<EOF >b9
217int myFunction(int argument_one, Thing *arg_asdfgh,
218 Blah xugly_bug) {
219 Squiggle fabulous_result = squargle(argument_one,
220 *arg_asdfgh, xugly_bug)
221 + g_ewww_global_with_a_really_long_name_yep_too_long;
222 return fabulous_result * 42;
223}
224EOF
225cat <<EOF >expected9
2261
2271
2282
2293
2303
2314
2325
233EOF
234
235title10="Two close matches versus one less close match"
236cat <<EOF >a10
237abcdef
238abcdef
239ghijkl
240EOF
241cat <<EOF >b10
242gh
243abcdefx
244EOF
245cat <<EOF >expected10
246Final
2472
248EOF
249
250# The first line of b matches best with the last line of a, but the overall
251# match is better if we match it with the the first line of a.
252title11="Piggy in the middle"
253cat <<EOF >a11
254abcdefg
255ijklmn
256abcdefgh
257EOF
258cat <<EOF >b11
259abcdefghx
260ijklm
261EOF
262cat <<EOF >expected11
2631
2642
265EOF
266
267title12="No trailing newline"
268printf "abc\ndef" >a12
269printf "abx\nstu" >b12
270cat <<EOF >expected12
2711
272Final
273EOF
274
275title13="Reorder includes"
276cat <<EOF >a13
277#include "c.h"
278#include "b.h"
279#include "a.h"
280#include "e.h"
281#include "d.h"
282EOF
283cat <<EOF >b13
284#include "a.h"
285#include "b.h"
286#include "c.h"
287#include "d.h"
288#include "e.h"
289EOF
290cat <<EOF >expected13
2913
2922
2931
2945
2954
296EOF
297
298last_test=13
299
300test_expect_success setup '
301 { for i in $(test_seq 2 $last_test)
302 do
303 # Append each line in a separate commit to make it easy to
304 # check which original line the blame output relates to.
305
306 line_count=0 &&
307 { while IFS= read line
308 do
309 line_count=$((line_count+1)) &&
310 echo "$line" >>"$i" &&
311 git add "$i" &&
312 test_tick &&
313 GIT_AUTHOR_NAME="$line_count" git commit -m "$line_count"
314 done } <"a$i"
315 done } &&
316
317 { for i in $(test_seq 2 $last_test)
318 do
319 # Overwrite the files with the final content.
320 cp b$i $i &&
321 git add $i
322 done } &&
323 test_tick &&
324
325 # Commit the final content all at once so it can all be
326 # referred to with the same commit ID.
327 GIT_AUTHOR_NAME=Final git commit -m Final &&
328
329 IGNOREME=$(git rev-parse HEAD)
330'
331
332for i in $(test_seq 2 $last_test); do
333 eval title="\$title$i"
334 test_expect_success "$title" \
335 "git blame -M9 --ignore-rev $IGNOREME $i >output &&
336 sed -e \"$pick_author\" output >actual &&
337 test_cmp expected$i actual"
338done
339
340# This invoked a null pointer dereference when the chunk callback was called
341# with a zero length parent chunk and there were no more suspects.
342test_expect_success 'Diff chunks with no suspects' '
343 test_write_lines xy1 A B C xy1 >file &&
344 git add file &&
345 test_tick &&
346 GIT_AUTHOR_NAME=1 git commit -m 1 &&
347
348 test_write_lines xy2 A B xy2 C xy2 >file &&
349 git add file &&
350 test_tick &&
351 GIT_AUTHOR_NAME=2 git commit -m 2 &&
352 REV_2=$(git rev-parse HEAD) &&
353
354 test_write_lines xy3 A >file &&
355 git add file &&
356 test_tick &&
357 GIT_AUTHOR_NAME=3 git commit -m 3 &&
358 REV_3=$(git rev-parse HEAD) &&
359
360 test_write_lines 1 1 >expected &&
361
362 git blame --ignore-rev $REV_2 --ignore-rev $REV_3 file >output &&
363 sed -e "$pick_author" output >actual &&
364
365 test_cmp expected actual
366 '
367
368test_expect_success 'position matching' '
369 test_write_lines abc def >file2 &&
370 git add file2 &&
371 test_tick &&
372 GIT_AUTHOR_NAME=1 git commit -m 1 &&
373
374 test_write_lines abc def abc def >file2 &&
375 git add file2 &&
376 test_tick &&
377 GIT_AUTHOR_NAME=2 git commit -m 2 &&
378
379 test_write_lines abcx defx abcx defx >file2 &&
380 git add file2 &&
381 test_tick &&
382 GIT_AUTHOR_NAME=3 git commit -m 3 &&
383 REV_3=$(git rev-parse HEAD) &&
384
385 test_write_lines abcy defy abcx defx >file2 &&
386 git add file2 &&
387 test_tick &&
388 GIT_AUTHOR_NAME=4 git commit -m 4 &&
389 REV_4=$(git rev-parse HEAD) &&
390
391 test_write_lines 1 1 2 2 >expected &&
392
393 git blame --ignore-rev $REV_3 --ignore-rev $REV_4 file2 >output &&
394 sed -e "$pick_author" output >actual &&
395
396 test_cmp expected actual
397 '
398
399# This fails if each blame entry is processed independently instead of
400# processing each diff change in full.
401test_expect_success 'preserve order' '
402 test_write_lines bcde >file3 &&
403 git add file3 &&
404 test_tick &&
405 GIT_AUTHOR_NAME=1 git commit -m 1 &&
406
407 test_write_lines bcde fghij >file3 &&
408 git add file3 &&
409 test_tick &&
410 GIT_AUTHOR_NAME=2 git commit -m 2 &&
411
412 test_write_lines bcde fghij abcd >file3 &&
413 git add file3 &&
414 test_tick &&
415 GIT_AUTHOR_NAME=3 git commit -m 3 &&
416
417 test_write_lines abcdx fghijx bcdex >file3 &&
418 git add file3 &&
419 test_tick &&
420 GIT_AUTHOR_NAME=4 git commit -m 4 &&
421 REV_4=$(git rev-parse HEAD) &&
422
423 test_write_lines abcdx fghijy bcdex >file3 &&
424 git add file3 &&
425 test_tick &&
426 GIT_AUTHOR_NAME=5 git commit -m 5 &&
427 REV_5=$(git rev-parse HEAD) &&
428
429 test_write_lines 1 2 3 >expected &&
430
431 git blame --ignore-rev $REV_4 --ignore-rev $REV_5 file3 >output &&
432 sed -e "$pick_author" output >actual &&
433
434 test_cmp expected actual
435 '
436
437test_done