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