xdiff: fix merging of appended hunk with -W
[git] / t / t7810-grep.sh
1 #!/bin/sh
2 #
3 # Copyright (c) 2006 Junio C Hamano
4 #
5
6 test_description='git grep various.
7 '
8
9 . ./test-lib.sh
10
11 cat >hello.c <<EOF
12 #include <assert.h>
13 #include <stdio.h>
14
15 int main(int argc, const char **argv)
16 {
17         printf("Hello world.\n");
18         return 0;
19         /* char ?? */
20 }
21 EOF
22
23 test_expect_success setup '
24         {
25                 echo foo mmap bar
26                 echo foo_mmap bar
27                 echo foo_mmap bar mmap
28                 echo foo mmap bar_mmap
29                 echo foo_mmap bar mmap baz
30         } >file &&
31         {
32                 echo Hello world
33                 echo HeLLo world
34                 echo Hello_world
35                 echo HeLLo_world
36         } >hello_world &&
37         {
38                 echo "a+b*c"
39                 echo "a+bc"
40                 echo "abc"
41         } >ab &&
42         echo vvv >v &&
43         echo ww w >w &&
44         echo x x xx x >x &&
45         echo y yy >y &&
46         echo zzz > z &&
47         mkdir t &&
48         echo test >t/t &&
49         echo vvv >t/v &&
50         mkdir t/a &&
51         echo vvv >t/a/v &&
52         {
53                 echo "line without leading space1"
54                 echo " line with leading space1"
55                 echo " line with leading space2"
56                 echo " line with leading space3"
57                 echo "line without leading space2"
58         } >space &&
59         git add . &&
60         test_tick &&
61         git commit -m initial
62 '
63
64 test_expect_success 'grep should not segfault with a bad input' '
65         test_must_fail git grep "("
66 '
67
68 for H in HEAD ''
69 do
70         case "$H" in
71         HEAD)   HC='HEAD:' L='HEAD' ;;
72         '')     HC= L='in working tree' ;;
73         esac
74
75         test_expect_success "grep -w $L" '
76                 {
77                         echo ${HC}file:1:foo mmap bar
78                         echo ${HC}file:3:foo_mmap bar mmap
79                         echo ${HC}file:4:foo mmap bar_mmap
80                         echo ${HC}file:5:foo_mmap bar mmap baz
81                 } >expected &&
82                 git -c grep.linenumber=false grep -n -w -e mmap $H >actual &&
83                 test_cmp expected actual
84         '
85
86         test_expect_success "grep -w $L" '
87                 {
88                         echo ${HC}file:1:foo mmap bar
89                         echo ${HC}file:3:foo_mmap bar mmap
90                         echo ${HC}file:4:foo mmap bar_mmap
91                         echo ${HC}file:5:foo_mmap bar mmap baz
92                 } >expected &&
93                 git -c grep.linenumber=true grep -w -e mmap $H >actual &&
94                 test_cmp expected actual
95         '
96
97         test_expect_success "grep -w $L" '
98                 {
99                         echo ${HC}file:foo mmap bar
100                         echo ${HC}file:foo_mmap bar mmap
101                         echo ${HC}file:foo mmap bar_mmap
102                         echo ${HC}file:foo_mmap bar mmap baz
103                 } >expected &&
104                 git -c grep.linenumber=true grep --no-line-number -w -e mmap $H >actual &&
105                 test_cmp expected actual
106         '
107
108         test_expect_success "grep -w $L (w)" '
109                 : >expected &&
110                 test_must_fail git grep -n -w -e "^w" $H >actual &&
111                 test_cmp expected actual
112         '
113
114         test_expect_success "grep -w $L (x)" '
115                 {
116                         echo ${HC}x:1:x x xx x
117                 } >expected &&
118                 git grep -n -w -e "x xx* x" $H >actual &&
119                 test_cmp expected actual
120         '
121
122         test_expect_success "grep -w $L (y-1)" '
123                 {
124                         echo ${HC}y:1:y yy
125                 } >expected &&
126                 git grep -n -w -e "^y" $H >actual &&
127                 test_cmp expected actual
128         '
129
130         test_expect_success "grep -w $L (y-2)" '
131                 : >expected &&
132                 if git grep -n -w -e "^y y" $H >actual
133                 then
134                         echo should not have matched
135                         cat actual
136                         false
137                 else
138                         test_cmp expected actual
139                 fi
140         '
141
142         test_expect_success "grep -w $L (z)" '
143                 : >expected &&
144                 if git grep -n -w -e "^z" $H >actual
145                 then
146                         echo should not have matched
147                         cat actual
148                         false
149                 else
150                         test_cmp expected actual
151                 fi
152         '
153
154         test_expect_success "grep $L (t-1)" '
155                 echo "${HC}t/t:1:test" >expected &&
156                 git grep -n -e test $H >actual &&
157                 test_cmp expected actual
158         '
159
160         test_expect_success "grep $L (t-2)" '
161                 echo "${HC}t:1:test" >expected &&
162                 (
163                         cd t &&
164                         git grep -n -e test $H
165                 ) >actual &&
166                 test_cmp expected actual
167         '
168
169         test_expect_success "grep $L (t-3)" '
170                 echo "${HC}t/t:1:test" >expected &&
171                 (
172                         cd t &&
173                         git grep --full-name -n -e test $H
174                 ) >actual &&
175                 test_cmp expected actual
176         '
177
178         test_expect_success "grep -c $L (no /dev/null)" '
179                 ! git grep -c test $H | grep /dev/null
180         '
181
182         test_expect_success "grep --max-depth -1 $L" '
183                 {
184                         echo ${HC}t/a/v:1:vvv
185                         echo ${HC}t/v:1:vvv
186                         echo ${HC}v:1:vvv
187                 } >expected &&
188                 git grep --max-depth -1 -n -e vvv $H >actual &&
189                 test_cmp expected actual
190         '
191
192         test_expect_success "grep --max-depth 0 $L" '
193                 {
194                         echo ${HC}v:1:vvv
195                 } >expected &&
196                 git grep --max-depth 0 -n -e vvv $H >actual &&
197                 test_cmp expected actual
198         '
199
200         test_expect_success "grep --max-depth 0 -- '*' $L" '
201                 {
202                         echo ${HC}t/a/v:1:vvv
203                         echo ${HC}t/v:1:vvv
204                         echo ${HC}v:1:vvv
205                 } >expected &&
206                 git grep --max-depth 0 -n -e vvv $H -- "*" >actual &&
207                 test_cmp expected actual
208         '
209
210         test_expect_success "grep --max-depth 1 $L" '
211                 {
212                         echo ${HC}t/v:1:vvv
213                         echo ${HC}v:1:vvv
214                 } >expected &&
215                 git grep --max-depth 1 -n -e vvv $H >actual &&
216                 test_cmp expected actual
217         '
218
219         test_expect_success "grep --max-depth 0 -- t $L" '
220                 {
221                         echo ${HC}t/v:1:vvv
222                 } >expected &&
223                 git grep --max-depth 0 -n -e vvv $H -- t >actual &&
224                 test_cmp expected actual
225         '
226
227         test_expect_success "grep --max-depth 0 -- . t $L" '
228                 {
229                         echo ${HC}t/v:1:vvv
230                         echo ${HC}v:1:vvv
231                 } >expected &&
232                 git grep --max-depth 0 -n -e vvv $H -- . t >actual &&
233                 test_cmp expected actual
234         '
235
236         test_expect_success "grep --max-depth 0 -- t . $L" '
237                 {
238                         echo ${HC}t/v:1:vvv
239                         echo ${HC}v:1:vvv
240                 } >expected &&
241                 git grep --max-depth 0 -n -e vvv $H -- t . >actual &&
242                 test_cmp expected actual
243         '
244         test_expect_success "grep $L with grep.extendedRegexp=false" '
245                 echo "${HC}ab:a+bc" >expected &&
246                 git -c grep.extendedRegexp=false grep "a+b*c" $H ab >actual &&
247                 test_cmp expected actual
248         '
249
250         test_expect_success "grep $L with grep.extendedRegexp=true" '
251                 echo "${HC}ab:abc" >expected &&
252                 git -c grep.extendedRegexp=true grep "a+b*c" $H ab >actual &&
253                 test_cmp expected actual
254         '
255
256         test_expect_success "grep $L with grep.patterntype=basic" '
257                 echo "${HC}ab:a+bc" >expected &&
258                 git -c grep.patterntype=basic grep "a+b*c" $H ab >actual &&
259                 test_cmp expected actual
260         '
261
262         test_expect_success "grep $L with grep.patterntype=extended" '
263                 echo "${HC}ab:abc" >expected &&
264                 git -c grep.patterntype=extended grep "a+b*c" $H ab >actual &&
265                 test_cmp expected actual
266         '
267
268         test_expect_success "grep $L with grep.patterntype=fixed" '
269                 echo "${HC}ab:a+b*c" >expected &&
270                 git -c grep.patterntype=fixed grep "a+b*c" $H ab >actual &&
271                 test_cmp expected actual
272         '
273
274         test_expect_success LIBPCRE "grep $L with grep.patterntype=perl" '
275                 echo "${HC}ab:a+b*c" >expected &&
276                 git -c grep.patterntype=perl grep "a\x{2b}b\x{2a}c" $H ab >actual &&
277                 test_cmp expected actual
278         '
279
280         test_expect_success "grep $L with grep.patternType=default and grep.extendedRegexp=true" '
281                 echo "${HC}ab:abc" >expected &&
282                 git \
283                         -c grep.patternType=default \
284                         -c grep.extendedRegexp=true \
285                         grep "a+b*c" $H ab >actual &&
286                 test_cmp expected actual
287         '
288
289         test_expect_success "grep $L with grep.extendedRegexp=true and grep.patternType=default" '
290                 echo "${HC}ab:abc" >expected &&
291                 git \
292                         -c grep.extendedRegexp=true \
293                         -c grep.patternType=default \
294                         grep "a+b*c" $H ab >actual &&
295                 test_cmp expected actual
296         '
297
298         test_expect_success "grep $L with grep.patternType=extended and grep.extendedRegexp=false" '
299                 echo "${HC}ab:abc" >expected &&
300                 git \
301                         -c grep.patternType=extended \
302                         -c grep.extendedRegexp=false \
303                         grep "a+b*c" $H ab >actual &&
304                 test_cmp expected actual
305         '
306
307         test_expect_success "grep $L with grep.patternType=basic and grep.extendedRegexp=true" '
308                 echo "${HC}ab:a+bc" >expected &&
309                 git \
310                         -c grep.patternType=basic \
311                         -c grep.extendedRegexp=true \
312                         grep "a+b*c" $H ab >actual &&
313                 test_cmp expected actual
314         '
315
316         test_expect_success "grep $L with grep.extendedRegexp=false and grep.patternType=extended" '
317                 echo "${HC}ab:abc" >expected &&
318                 git \
319                         -c grep.extendedRegexp=false \
320                         -c grep.patternType=extended \
321                         grep "a+b*c" $H ab >actual &&
322                 test_cmp expected actual
323         '
324
325         test_expect_success "grep $L with grep.extendedRegexp=true and grep.patternType=basic" '
326                 echo "${HC}ab:a+bc" >expected &&
327                 git \
328                         -c grep.extendedRegexp=true \
329                         -c grep.patternType=basic \
330                         grep "a+b*c" $H ab >actual &&
331                 test_cmp expected actual
332         '
333
334         test_expect_success "grep --count $L" '
335                 echo ${HC}ab:3 >expected &&
336                 git grep --count -e b $H -- ab >actual &&
337                 test_cmp expected actual
338         '
339
340         test_expect_success "grep --count -h $L" '
341                 echo 3 >expected &&
342                 git grep --count -h -e b $H -- ab >actual &&
343                 test_cmp expected actual
344         '
345 done
346
347 cat >expected <<EOF
348 file
349 EOF
350 test_expect_success 'grep -l -C' '
351         git grep -l -C1 foo >actual &&
352         test_cmp expected actual
353 '
354
355 cat >expected <<EOF
356 file:5
357 EOF
358 test_expect_success 'grep -l -C' '
359         git grep -c -C1 foo >actual &&
360         test_cmp expected actual
361 '
362
363 test_expect_success 'grep -L -C' '
364         git ls-files >expected &&
365         git grep -L -C1 nonexistent_string >actual &&
366         test_cmp expected actual
367 '
368
369 cat >expected <<EOF
370 file:foo mmap bar_mmap
371 EOF
372
373 test_expect_success 'grep -e A --and -e B' '
374         git grep -e "foo mmap" --and -e bar_mmap >actual &&
375         test_cmp expected actual
376 '
377
378 cat >expected <<EOF
379 file:foo_mmap bar mmap
380 file:foo_mmap bar mmap baz
381 EOF
382
383
384 test_expect_success 'grep ( -e A --or -e B ) --and -e B' '
385         git grep \( -e foo_ --or -e baz \) \
386                 --and -e " mmap" >actual &&
387         test_cmp expected actual
388 '
389
390 cat >expected <<EOF
391 file:foo mmap bar
392 EOF
393
394 test_expect_success 'grep -e A --and --not -e B' '
395         git grep -e "foo mmap" --and --not -e bar_mmap >actual &&
396         test_cmp expected actual
397 '
398
399 test_expect_success 'grep should ignore GREP_OPTIONS' '
400         GREP_OPTIONS=-v git grep " mmap bar\$" >actual &&
401         test_cmp expected actual
402 '
403
404 test_expect_success 'grep -f, non-existent file' '
405         test_must_fail git grep -f patterns
406 '
407
408 cat >expected <<EOF
409 file:foo mmap bar
410 file:foo_mmap bar
411 file:foo_mmap bar mmap
412 file:foo mmap bar_mmap
413 file:foo_mmap bar mmap baz
414 EOF
415
416 cat >pattern <<EOF
417 mmap
418 EOF
419
420 test_expect_success 'grep -f, one pattern' '
421         git grep -f pattern >actual &&
422         test_cmp expected actual
423 '
424
425 cat >expected <<EOF
426 file:foo mmap bar
427 file:foo_mmap bar
428 file:foo_mmap bar mmap
429 file:foo mmap bar_mmap
430 file:foo_mmap bar mmap baz
431 t/a/v:vvv
432 t/v:vvv
433 v:vvv
434 EOF
435
436 cat >patterns <<EOF
437 mmap
438 vvv
439 EOF
440
441 test_expect_success 'grep -f, multiple patterns' '
442         git grep -f patterns >actual &&
443         test_cmp expected actual
444 '
445
446 test_expect_success 'grep, multiple patterns' '
447         git grep "$(cat patterns)" >actual &&
448         test_cmp expected actual
449 '
450
451 cat >expected <<EOF
452 file:foo mmap bar
453 file:foo_mmap bar
454 file:foo_mmap bar mmap
455 file:foo mmap bar_mmap
456 file:foo_mmap bar mmap baz
457 t/a/v:vvv
458 t/v:vvv
459 v:vvv
460 EOF
461
462 cat >patterns <<EOF
463
464 mmap
465
466 vvv
467
468 EOF
469
470 test_expect_success 'grep -f, ignore empty lines' '
471         git grep -f patterns >actual &&
472         test_cmp expected actual
473 '
474
475 test_expect_success 'grep -f, ignore empty lines, read patterns from stdin' '
476         git grep -f - <patterns >actual &&
477         test_cmp expected actual
478 '
479
480 cat >expected <<EOF
481 y:y yy
482 --
483 z:zzz
484 EOF
485
486 test_expect_success 'grep -q, silently report matches' '
487         >empty &&
488         git grep -q mmap >actual &&
489         test_cmp empty actual &&
490         test_must_fail git grep -q qfwfq >actual &&
491         test_cmp empty actual
492 '
493
494 test_expect_success 'grep -C1 hunk mark between files' '
495         git grep -C1 "^[yz]" >actual &&
496         test_cmp expected actual
497 '
498
499 test_expect_success 'log grep setup' '
500         echo a >>file &&
501         test_tick &&
502         GIT_AUTHOR_NAME="With * Asterisk" \
503         GIT_AUTHOR_EMAIL="xyzzy@frotz.com" \
504         git commit -a -m "second" &&
505
506         echo a >>file &&
507         test_tick &&
508         git commit -a -m "third" &&
509
510         echo a >>file &&
511         test_tick &&
512         GIT_AUTHOR_NAME="Night Fall" \
513         GIT_AUTHOR_EMAIL="nitfol@frobozz.com" \
514         git commit -a -m "fourth"
515 '
516
517 test_expect_success 'log grep (1)' '
518         git log --author=author --pretty=tformat:%s >actual &&
519         {
520                 echo third && echo initial
521         } >expect &&
522         test_cmp expect actual
523 '
524
525 test_expect_success 'log grep (2)' '
526         git log --author=" * " -F --pretty=tformat:%s >actual &&
527         {
528                 echo second
529         } >expect &&
530         test_cmp expect actual
531 '
532
533 test_expect_success 'log grep (3)' '
534         git log --author="^A U" --pretty=tformat:%s >actual &&
535         {
536                 echo third && echo initial
537         } >expect &&
538         test_cmp expect actual
539 '
540
541 test_expect_success 'log grep (4)' '
542         git log --author="frotz\.com>$" --pretty=tformat:%s >actual &&
543         {
544                 echo second
545         } >expect &&
546         test_cmp expect actual
547 '
548
549 test_expect_success 'log grep (5)' '
550         git log --author=Thor -F --pretty=tformat:%s >actual &&
551         {
552                 echo third && echo initial
553         } >expect &&
554         test_cmp expect actual
555 '
556
557 test_expect_success 'log grep (6)' '
558         git log --author=-0700  --pretty=tformat:%s >actual &&
559         >expect &&
560         test_cmp expect actual
561 '
562
563 test_expect_success 'log grep (7)' '
564         git log -g --grep-reflog="commit: third" --pretty=tformat:%s >actual &&
565         echo third >expect &&
566         test_cmp expect actual
567 '
568
569 test_expect_success 'log grep (8)' '
570         git log -g --grep-reflog="commit: third" --grep-reflog="commit: second" --pretty=tformat:%s >actual &&
571         {
572                 echo third && echo second
573         } >expect &&
574         test_cmp expect actual
575 '
576
577 test_expect_success 'log grep (9)' '
578         git log -g --grep-reflog="commit: third" --author="Thor" --pretty=tformat:%s >actual &&
579         echo third >expect &&
580         test_cmp expect actual
581 '
582
583 test_expect_success 'log grep (9)' '
584         git log -g --grep-reflog="commit: third" --author="non-existant" --pretty=tformat:%s >actual &&
585         : >expect &&
586         test_cmp expect actual
587 '
588
589 test_expect_success 'log --grep-reflog can only be used under -g' '
590         test_must_fail git log --grep-reflog="commit: third"
591 '
592
593 test_expect_success 'log with multiple --grep uses union' '
594         git log --grep=i --grep=r --format=%s >actual &&
595         {
596                 echo fourth && echo third && echo initial
597         } >expect &&
598         test_cmp expect actual
599 '
600
601 test_expect_success 'log --all-match with multiple --grep uses intersection' '
602         git log --all-match --grep=i --grep=r --format=%s >actual &&
603         {
604                 echo third
605         } >expect &&
606         test_cmp expect actual
607 '
608
609 test_expect_success 'log with multiple --author uses union' '
610         git log --author="Thor" --author="Aster" --format=%s >actual &&
611         {
612             echo third && echo second && echo initial
613         } >expect &&
614         test_cmp expect actual
615 '
616
617 test_expect_success 'log --all-match with multiple --author still uses union' '
618         git log --all-match --author="Thor" --author="Aster" --format=%s >actual &&
619         {
620             echo third && echo second && echo initial
621         } >expect &&
622         test_cmp expect actual
623 '
624
625 test_expect_success 'log --grep --author uses intersection' '
626         # grep matches only third and fourth
627         # author matches only initial and third
628         git log --author="A U Thor" --grep=r --format=%s >actual &&
629         {
630                 echo third
631         } >expect &&
632         test_cmp expect actual
633 '
634
635 test_expect_success 'log --grep --grep --author takes union of greps and intersects with author' '
636         # grep matches initial and second but not third
637         # author matches only initial and third
638         git log --author="A U Thor" --grep=s --grep=l --format=%s >actual &&
639         {
640                 echo initial
641         } >expect &&
642         test_cmp expect actual
643 '
644
645 test_expect_success 'log ---all-match -grep --author --author still takes union of authors and intersects with grep' '
646         # grep matches only initial and third
647         # author matches all but second
648         git log --all-match --author="Thor" --author="Night" --grep=i --format=%s >actual &&
649         {
650             echo third && echo initial
651         } >expect &&
652         test_cmp expect actual
653 '
654
655 test_expect_success 'log --grep --author --author takes union of authors and intersects with grep' '
656         # grep matches only initial and third
657         # author matches all but second
658         git log --author="Thor" --author="Night" --grep=i --format=%s >actual &&
659         {
660             echo third && echo initial
661         } >expect &&
662         test_cmp expect actual
663 '
664
665 test_expect_success 'log --all-match --grep --grep --author takes intersection' '
666         # grep matches only third
667         # author matches only initial and third
668         git log --all-match --author="A U Thor" --grep=i --grep=r --format=%s >actual &&
669         {
670                 echo third
671         } >expect &&
672         test_cmp expect actual
673 '
674
675 test_expect_success 'log --author does not search in timestamp' '
676         : >expect &&
677         git log --author="$GIT_AUTHOR_DATE" >actual &&
678         test_cmp expect actual
679 '
680
681 test_expect_success 'log --committer does not search in timestamp' '
682         : >expect &&
683         git log --committer="$GIT_COMMITTER_DATE" >actual &&
684         test_cmp expect actual
685 '
686
687 test_expect_success 'grep with CE_VALID file' '
688         git update-index --assume-unchanged t/t &&
689         rm t/t &&
690         test "$(git grep test)" = "t/t:test" &&
691         git update-index --no-assume-unchanged t/t &&
692         git checkout t/t
693 '
694
695 cat >expected <<EOF
696 hello.c=#include <stdio.h>
697 hello.c:        return 0;
698 EOF
699
700 test_expect_success 'grep -p with userdiff' '
701         git config diff.custom.funcname "^#" &&
702         echo "hello.c diff=custom" >.gitattributes &&
703         git grep -p return >actual &&
704         test_cmp expected actual
705 '
706
707 cat >expected <<EOF
708 hello.c=int main(int argc, const char **argv)
709 hello.c:        return 0;
710 EOF
711
712 test_expect_success 'grep -p' '
713         rm -f .gitattributes &&
714         git grep -p return >actual &&
715         test_cmp expected actual
716 '
717
718 cat >expected <<EOF
719 hello.c-#include <stdio.h>
720 hello.c-
721 hello.c=int main(int argc, const char **argv)
722 hello.c-{
723 hello.c-        printf("Hello world.\n");
724 hello.c:        return 0;
725 EOF
726
727 test_expect_success 'grep -p -B5' '
728         git grep -p -B5 return >actual &&
729         test_cmp expected actual
730 '
731
732 cat >expected <<EOF
733 hello.c=int main(int argc, const char **argv)
734 hello.c-{
735 hello.c-        printf("Hello world.\n");
736 hello.c:        return 0;
737 hello.c-        /* char ?? */
738 hello.c-}
739 EOF
740
741 test_expect_success 'grep -W' '
742         git grep -W return >actual &&
743         test_cmp expected actual
744 '
745
746 cat >expected <<EOF
747 hello.c-#include <assert.h>
748 hello.c:#include <stdio.h>
749 EOF
750
751 test_expect_success 'grep -W shows no trailing empty lines' '
752         git grep -W stdio >actual &&
753         test_cmp expected actual
754 '
755
756 cat >expected <<EOF
757 hello.c=        printf("Hello world.\n");
758 hello.c:        return 0;
759 hello.c-        /* char ?? */
760 EOF
761
762 test_expect_success 'grep -W with userdiff' '
763         test_when_finished "rm -f .gitattributes" &&
764         git config diff.custom.xfuncname "(printf.*|})$" &&
765         echo "hello.c diff=custom" >.gitattributes &&
766         git grep -W return >actual &&
767         test_cmp expected actual
768 '
769
770 test_expect_success 'grep from a subdirectory to search wider area (1)' '
771         mkdir -p s &&
772         (
773                 cd s && git grep "x x x" ..
774         )
775 '
776
777 test_expect_success 'grep from a subdirectory to search wider area (2)' '
778         mkdir -p s &&
779         (
780                 cd s || exit 1
781                 ( git grep xxyyzz .. >out ; echo $? >status )
782                 ! test -s out &&
783                 test 1 = $(cat status)
784         )
785 '
786
787 cat >expected <<EOF
788 hello.c:int main(int argc, const char **argv)
789 EOF
790
791 test_expect_success 'grep -Fi' '
792         git grep -Fi "CHAR *" >actual &&
793         test_cmp expected actual
794 '
795
796 test_expect_success 'outside of git repository' '
797         rm -fr non &&
798         mkdir -p non/git/sub &&
799         echo hello >non/git/file1 &&
800         echo world >non/git/sub/file2 &&
801         {
802                 echo file1:hello &&
803                 echo sub/file2:world
804         } >non/expect.full &&
805         echo file2:world >non/expect.sub &&
806         (
807                 GIT_CEILING_DIRECTORIES="$(pwd)/non" &&
808                 export GIT_CEILING_DIRECTORIES &&
809                 cd non/git &&
810                 test_must_fail git grep o &&
811                 git grep --no-index o >../actual.full &&
812                 test_cmp ../expect.full ../actual.full &&
813                 cd sub &&
814                 test_must_fail git grep o &&
815                 git grep --no-index o >../../actual.sub &&
816                 test_cmp ../../expect.sub ../../actual.sub
817         ) &&
818
819         echo ".*o*" >non/git/.gitignore &&
820         (
821                 GIT_CEILING_DIRECTORIES="$(pwd)/non" &&
822                 export GIT_CEILING_DIRECTORIES &&
823                 cd non/git &&
824                 test_must_fail git grep o &&
825                 git grep --no-index --exclude-standard o >../actual.full &&
826                 test_cmp ../expect.full ../actual.full &&
827
828                 {
829                         echo ".gitignore:.*o*" &&
830                         cat ../expect.full
831                 } >../expect.with.ignored &&
832                 git grep --no-index --no-exclude o >../actual.full &&
833                 test_cmp ../expect.with.ignored ../actual.full
834         )
835 '
836
837 test_expect_success 'outside of git repository with fallbackToNoIndex' '
838         rm -fr non &&
839         mkdir -p non/git/sub &&
840         echo hello >non/git/file1 &&
841         echo world >non/git/sub/file2 &&
842         cat <<-\EOF >non/expect.full &&
843         file1:hello
844         sub/file2:world
845         EOF
846         echo file2:world >non/expect.sub &&
847         (
848                 GIT_CEILING_DIRECTORIES="$(pwd)/non" &&
849                 export GIT_CEILING_DIRECTORIES &&
850                 cd non/git &&
851                 test_must_fail git -c grep.fallbackToNoIndex=false grep o &&
852                 git -c grep.fallbackToNoIndex=true grep o >../actual.full &&
853                 test_cmp ../expect.full ../actual.full &&
854                 cd sub &&
855                 test_must_fail git -c grep.fallbackToNoIndex=false grep o &&
856                 git -c grep.fallbackToNoIndex=true grep o >../../actual.sub &&
857                 test_cmp ../../expect.sub ../../actual.sub
858         ) &&
859
860         echo ".*o*" >non/git/.gitignore &&
861         (
862                 GIT_CEILING_DIRECTORIES="$(pwd)/non" &&
863                 export GIT_CEILING_DIRECTORIES &&
864                 cd non/git &&
865                 test_must_fail git -c grep.fallbackToNoIndex=false grep o &&
866                 git -c grep.fallbackToNoIndex=true grep --exclude-standard o >../actual.full &&
867                 test_cmp ../expect.full ../actual.full &&
868
869                 {
870                         echo ".gitignore:.*o*" &&
871                         cat ../expect.full
872                 } >../expect.with.ignored &&
873                 git -c grep.fallbackToNoIndex grep --no-exclude o >../actual.full &&
874                 test_cmp ../expect.with.ignored ../actual.full
875         )
876 '
877
878 test_expect_success 'inside git repository but with --no-index' '
879         rm -fr is &&
880         mkdir -p is/git/sub &&
881         echo hello >is/git/file1 &&
882         echo world >is/git/sub/file2 &&
883         echo ".*o*" >is/git/.gitignore &&
884         {
885                 echo file1:hello &&
886                 echo sub/file2:world
887         } >is/expect.unignored &&
888         {
889                 echo ".gitignore:.*o*" &&
890                 cat is/expect.unignored
891         } >is/expect.full &&
892         : >is/expect.empty &&
893         echo file2:world >is/expect.sub &&
894         (
895                 cd is/git &&
896                 git init &&
897                 test_must_fail git grep o >../actual.full &&
898                 test_cmp ../expect.empty ../actual.full &&
899
900                 git grep --untracked o >../actual.unignored &&
901                 test_cmp ../expect.unignored ../actual.unignored &&
902
903                 git grep --no-index o >../actual.full &&
904                 test_cmp ../expect.full ../actual.full &&
905
906                 git grep --no-index --exclude-standard o >../actual.unignored &&
907                 test_cmp ../expect.unignored ../actual.unignored &&
908
909                 cd sub &&
910                 test_must_fail git grep o >../../actual.sub &&
911                 test_cmp ../../expect.empty ../../actual.sub &&
912
913                 git grep --no-index o >../../actual.sub &&
914                 test_cmp ../../expect.sub ../../actual.sub &&
915
916                 git grep --untracked o >../../actual.sub &&
917                 test_cmp ../../expect.sub ../../actual.sub
918         )
919 '
920
921 test_expect_success 'setup double-dash tests' '
922 cat >double-dash <<EOF &&
923 --
924 ->
925 other
926 EOF
927 git add double-dash
928 '
929
930 cat >expected <<EOF
931 double-dash:->
932 EOF
933 test_expect_success 'grep -- pattern' '
934         git grep -- "->" >actual &&
935         test_cmp expected actual
936 '
937 test_expect_success 'grep -- pattern -- pathspec' '
938         git grep -- "->" -- double-dash >actual &&
939         test_cmp expected actual
940 '
941 test_expect_success 'grep -e pattern -- path' '
942         git grep -e "->" -- double-dash >actual &&
943         test_cmp expected actual
944 '
945
946 cat >expected <<EOF
947 double-dash:--
948 EOF
949 test_expect_success 'grep -e -- -- path' '
950         git grep -e -- -- double-dash >actual &&
951         test_cmp expected actual
952 '
953
954 cat >expected <<EOF
955 hello.c:int main(int argc, const char **argv)
956 hello.c:        printf("Hello world.\n");
957 EOF
958
959 test_expect_success LIBPCRE 'grep --perl-regexp pattern' '
960         git grep --perl-regexp "\p{Ps}.*?\p{Pe}" hello.c >actual &&
961         test_cmp expected actual
962 '
963
964 test_expect_success LIBPCRE 'grep -P pattern' '
965         git grep -P "\p{Ps}.*?\p{Pe}" hello.c >actual &&
966         test_cmp expected actual
967 '
968
969 test_expect_success 'grep pattern with grep.extendedRegexp=true' '
970         >empty &&
971         test_must_fail git -c grep.extendedregexp=true \
972                 grep "\p{Ps}.*?\p{Pe}" hello.c >actual &&
973         test_cmp empty actual
974 '
975
976 test_expect_success LIBPCRE 'grep -P pattern with grep.extendedRegexp=true' '
977         git -c grep.extendedregexp=true \
978                 grep -P "\p{Ps}.*?\p{Pe}" hello.c >actual &&
979         test_cmp expected actual
980 '
981
982 test_expect_success LIBPCRE 'grep -P -v pattern' '
983         {
984                 echo "ab:a+b*c"
985                 echo "ab:a+bc"
986         } >expected &&
987         git grep -P -v "abc" ab >actual &&
988         test_cmp expected actual
989 '
990
991 test_expect_success LIBPCRE 'grep -P -i pattern' '
992         cat >expected <<-EOF &&
993         hello.c:        printf("Hello world.\n");
994         EOF
995         git grep -P -i "PRINTF\([^\d]+\)" hello.c >actual &&
996         test_cmp expected actual
997 '
998
999 test_expect_success LIBPCRE 'grep -P -w pattern' '
1000         {
1001                 echo "hello_world:Hello world"
1002                 echo "hello_world:HeLLo world"
1003         } >expected &&
1004         git grep -P -w "He((?i)ll)o" hello_world >actual &&
1005         test_cmp expected actual
1006 '
1007
1008 test_expect_success 'grep -G invalidpattern properly dies ' '
1009         test_must_fail git grep -G "a["
1010 '
1011
1012 test_expect_success 'grep invalidpattern properly dies with grep.patternType=basic' '
1013         test_must_fail git -c grep.patterntype=basic grep "a["
1014 '
1015
1016 test_expect_success 'grep -E invalidpattern properly dies ' '
1017         test_must_fail git grep -E "a["
1018 '
1019
1020 test_expect_success 'grep invalidpattern properly dies with grep.patternType=extended' '
1021         test_must_fail git -c grep.patterntype=extended grep "a["
1022 '
1023
1024 test_expect_success LIBPCRE 'grep -P invalidpattern properly dies ' '
1025         test_must_fail git grep -P "a["
1026 '
1027
1028 test_expect_success LIBPCRE 'grep invalidpattern properly dies with grep.patternType=perl' '
1029         test_must_fail git -c grep.patterntype=perl grep "a["
1030 '
1031
1032 test_expect_success 'grep -G -E -F pattern' '
1033         echo "ab:a+b*c" >expected &&
1034         git grep -G -E -F "a+b*c" ab >actual &&
1035         test_cmp expected actual
1036 '
1037
1038 test_expect_success 'grep pattern with grep.patternType=basic, =extended, =fixed' '
1039         echo "ab:a+b*c" >expected &&
1040         git \
1041                 -c grep.patterntype=basic \
1042                 -c grep.patterntype=extended \
1043                 -c grep.patterntype=fixed \
1044                 grep "a+b*c" ab >actual &&
1045         test_cmp expected actual
1046 '
1047
1048 test_expect_success 'grep -E -F -G pattern' '
1049         echo "ab:a+bc" >expected &&
1050         git grep -E -F -G "a+b*c" ab >actual &&
1051         test_cmp expected actual
1052 '
1053
1054 test_expect_success 'grep pattern with grep.patternType=extended, =fixed, =basic' '
1055         echo "ab:a+bc" >expected &&
1056         git \
1057                 -c grep.patterntype=extended \
1058                 -c grep.patterntype=fixed \
1059                 -c grep.patterntype=basic \
1060                 grep "a+b*c" ab >actual &&
1061         test_cmp expected actual
1062 '
1063
1064 test_expect_success 'grep -F -G -E pattern' '
1065         echo "ab:abc" >expected &&
1066         git grep -F -G -E "a+b*c" ab >actual &&
1067         test_cmp expected actual
1068 '
1069
1070 test_expect_success 'grep pattern with grep.patternType=fixed, =basic, =extended' '
1071         echo "ab:abc" >expected &&
1072         git \
1073                 -c grep.patterntype=fixed \
1074                 -c grep.patterntype=basic \
1075                 -c grep.patterntype=extended \
1076                 grep "a+b*c" ab >actual &&
1077         test_cmp expected actual
1078 '
1079
1080 test_expect_success 'grep -G -F -P -E pattern' '
1081         >empty &&
1082         test_must_fail git grep -G -F -P -E "a\x{2b}b\x{2a}c" ab >actual &&
1083         test_cmp empty actual
1084 '
1085
1086 test_expect_success 'grep pattern with grep.patternType=fixed, =basic, =perl, =extended' '
1087         >empty &&
1088         test_must_fail git \
1089                 -c grep.patterntype=fixed \
1090                 -c grep.patterntype=basic \
1091                 -c grep.patterntype=perl \
1092                 -c grep.patterntype=extended \
1093                 grep "a\x{2b}b\x{2a}c" ab >actual &&
1094         test_cmp empty actual
1095 '
1096
1097 test_expect_success LIBPCRE 'grep -G -F -E -P pattern' '
1098         echo "ab:a+b*c" >expected &&
1099         git grep -G -F -E -P "a\x{2b}b\x{2a}c" ab >actual &&
1100         test_cmp expected actual
1101 '
1102
1103 test_expect_success LIBPCRE 'grep pattern with grep.patternType=fixed, =basic, =extended, =perl' '
1104         echo "ab:a+b*c" >expected &&
1105         git \
1106                 -c grep.patterntype=fixed \
1107                 -c grep.patterntype=basic \
1108                 -c grep.patterntype=extended \
1109                 -c grep.patterntype=perl \
1110                 grep "a\x{2b}b\x{2a}c" ab >actual &&
1111         test_cmp expected actual
1112 '
1113
1114 test_expect_success LIBPCRE 'grep -P pattern with grep.patternType=fixed' '
1115         echo "ab:a+b*c" >expected &&
1116         git \
1117                 -c grep.patterntype=fixed \
1118                 grep -P "a\x{2b}b\x{2a}c" ab >actual &&
1119         test_cmp expected actual
1120 '
1121
1122 test_expect_success 'grep -F pattern with grep.patternType=basic' '
1123         echo "ab:a+b*c" >expected &&
1124         git \
1125                 -c grep.patterntype=basic \
1126                 grep -F "*c" ab >actual &&
1127         test_cmp expected actual
1128 '
1129
1130 test_expect_success 'grep -G pattern with grep.patternType=fixed' '
1131         {
1132                 echo "ab:a+b*c"
1133                 echo "ab:a+bc"
1134         } >expected &&
1135         git \
1136                 -c grep.patterntype=fixed \
1137                 grep -G "a+b" ab >actual &&
1138         test_cmp expected actual
1139 '
1140
1141 test_expect_success 'grep -E pattern with grep.patternType=fixed' '
1142         {
1143                 echo "ab:a+b*c"
1144                 echo "ab:a+bc"
1145                 echo "ab:abc"
1146         } >expected &&
1147         git \
1148                 -c grep.patterntype=fixed \
1149                 grep -E "a+" ab >actual &&
1150         test_cmp expected actual
1151 '
1152
1153 cat >expected <<EOF
1154 hello.c<RED>:<RESET>int main(int argc, const char **argv)
1155 hello.c<RED>-<RESET>{
1156 <RED>--<RESET>
1157 hello.c<RED>:<RESET>    /* char ?? */
1158 hello.c<RED>-<RESET>}
1159 <RED>--<RESET>
1160 hello_world<RED>:<RESET>Hello_world
1161 hello_world<RED>-<RESET>HeLLo_world
1162 EOF
1163
1164 test_expect_success 'grep --color, separator' '
1165         test_config color.grep.context          normal &&
1166         test_config color.grep.filename         normal &&
1167         test_config color.grep.function         normal &&
1168         test_config color.grep.linenumber       normal &&
1169         test_config color.grep.match            normal &&
1170         test_config color.grep.selected         normal &&
1171         test_config color.grep.separator        red &&
1172
1173         git grep --color=always -A1 -e char -e lo_w hello.c hello_world |
1174         test_decode_color >actual &&
1175         test_cmp expected actual
1176 '
1177
1178 cat >expected <<EOF
1179 hello.c:int main(int argc, const char **argv)
1180 hello.c:        /* char ?? */
1181
1182 hello_world:Hello_world
1183 EOF
1184
1185 test_expect_success 'grep --break' '
1186         git grep --break -e char -e lo_w hello.c hello_world >actual &&
1187         test_cmp expected actual
1188 '
1189
1190 cat >expected <<EOF
1191 hello.c:int main(int argc, const char **argv)
1192 hello.c-{
1193 --
1194 hello.c:        /* char ?? */
1195 hello.c-}
1196
1197 hello_world:Hello_world
1198 hello_world-HeLLo_world
1199 EOF
1200
1201 test_expect_success 'grep --break with context' '
1202         git grep --break -A1 -e char -e lo_w hello.c hello_world >actual &&
1203         test_cmp expected actual
1204 '
1205
1206 cat >expected <<EOF
1207 hello.c
1208 int main(int argc, const char **argv)
1209         /* char ?? */
1210 hello_world
1211 Hello_world
1212 EOF
1213
1214 test_expect_success 'grep --heading' '
1215         git grep --heading -e char -e lo_w hello.c hello_world >actual &&
1216         test_cmp expected actual
1217 '
1218
1219 cat >expected <<EOF
1220 <BOLD;GREEN>hello.c<RESET>
1221 4:int main(int argc, const <BLACK;BYELLOW>char<RESET> **argv)
1222 8:      /* <BLACK;BYELLOW>char<RESET> ?? */
1223
1224 <BOLD;GREEN>hello_world<RESET>
1225 3:Hel<BLACK;BYELLOW>lo_w<RESET>orld
1226 EOF
1227
1228 test_expect_success 'mimic ack-grep --group' '
1229         test_config color.grep.context          normal &&
1230         test_config color.grep.filename         "bold green" &&
1231         test_config color.grep.function         normal &&
1232         test_config color.grep.linenumber       normal &&
1233         test_config color.grep.match            "black yellow" &&
1234         test_config color.grep.selected         normal &&
1235         test_config color.grep.separator        normal &&
1236
1237         git grep --break --heading -n --color \
1238                 -e char -e lo_w hello.c hello_world |
1239         test_decode_color >actual &&
1240         test_cmp expected actual
1241 '
1242
1243 cat >expected <<EOF
1244 space: line with leading space1
1245 space: line with leading space2
1246 space: line with leading space3
1247 EOF
1248
1249 test_expect_success LIBPCRE 'grep -E "^ "' '
1250         git grep -E "^ " space >actual &&
1251         test_cmp expected actual
1252 '
1253
1254 test_expect_success LIBPCRE 'grep -P "^ "' '
1255         git grep -P "^ " space >actual &&
1256         test_cmp expected actual
1257 '
1258
1259 cat >expected <<EOF
1260 space-line without leading space1
1261 space: line <RED>with <RESET>leading space1
1262 space: line <RED>with <RESET>leading <RED>space2<RESET>
1263 space: line <RED>with <RESET>leading space3
1264 space:line without leading <RED>space2<RESET>
1265 EOF
1266
1267 test_expect_success 'grep --color -e A -e B with context' '
1268         test_config color.grep.context          normal &&
1269         test_config color.grep.filename         normal &&
1270         test_config color.grep.function         normal &&
1271         test_config color.grep.linenumber       normal &&
1272         test_config color.grep.matchContext     normal &&
1273         test_config color.grep.matchSelected    red &&
1274         test_config color.grep.selected         normal &&
1275         test_config color.grep.separator        normal &&
1276
1277         git grep --color=always -C2 -e "with " -e space2  space |
1278         test_decode_color >actual &&
1279         test_cmp expected actual
1280 '
1281
1282 cat >expected <<EOF
1283 space-line without leading space1
1284 space- line with leading space1
1285 space: line <RED>with <RESET>leading <RED>space2<RESET>
1286 space- line with leading space3
1287 space-line without leading space2
1288 EOF
1289
1290 test_expect_success 'grep --color -e A --and -e B with context' '
1291         test_config color.grep.context          normal &&
1292         test_config color.grep.filename         normal &&
1293         test_config color.grep.function         normal &&
1294         test_config color.grep.linenumber       normal &&
1295         test_config color.grep.matchContext     normal &&
1296         test_config color.grep.matchSelected    red &&
1297         test_config color.grep.selected         normal &&
1298         test_config color.grep.separator        normal &&
1299
1300         git grep --color=always -C2 -e "with " --and -e space2  space |
1301         test_decode_color >actual &&
1302         test_cmp expected actual
1303 '
1304
1305 cat >expected <<EOF
1306 space-line without leading space1
1307 space: line <RED>with <RESET>leading space1
1308 space- line with leading space2
1309 space: line <RED>with <RESET>leading space3
1310 space-line without leading space2
1311 EOF
1312
1313 test_expect_success 'grep --color -e A --and --not -e B with context' '
1314         test_config color.grep.context          normal &&
1315         test_config color.grep.filename         normal &&
1316         test_config color.grep.function         normal &&
1317         test_config color.grep.linenumber       normal &&
1318         test_config color.grep.matchContext     normal &&
1319         test_config color.grep.matchSelected    red &&
1320         test_config color.grep.selected         normal &&
1321         test_config color.grep.separator        normal &&
1322
1323         git grep --color=always -C2 -e "with " --and --not -e space2  space |
1324         test_decode_color >actual &&
1325         test_cmp expected actual
1326 '
1327
1328 cat >expected <<EOF
1329 hello.c-
1330 hello.c=int main(int argc, const char **argv)
1331 hello.c-{
1332 hello.c:        pr<RED>int<RESET>f("<RED>Hello<RESET> world.\n");
1333 hello.c-        return 0;
1334 hello.c-        /* char ?? */
1335 hello.c-}
1336 EOF
1337
1338 test_expect_success 'grep --color -e A --and -e B -p with context' '
1339         test_config color.grep.context          normal &&
1340         test_config color.grep.filename         normal &&
1341         test_config color.grep.function         normal &&
1342         test_config color.grep.linenumber       normal &&
1343         test_config color.grep.matchContext     normal &&
1344         test_config color.grep.matchSelected    red &&
1345         test_config color.grep.selected         normal &&
1346         test_config color.grep.separator        normal &&
1347
1348         git grep --color=always -p -C3 -e int --and -e Hello --no-index hello.c |
1349         test_decode_color >actual &&
1350         test_cmp expected actual
1351 '
1352
1353 test_done