249 lines
		
	
	
	
		
			6.1 KiB
		
	
	
	
		
			Bash
		
	
	
		
			Executable file
		
	
	
	
	
			
		
		
	
	
			249 lines
		
	
	
	
		
			6.1 KiB
		
	
	
	
		
			Bash
		
	
	
		
			Executable file
		
	
	
	
	
| #!/bin/sh
 | |
| 
 | |
| test_description='git grep in binary files'
 | |
| 
 | |
| . ./test-lib.sh
 | |
| 
 | |
| nul_match () {
 | |
| 	matches=$1
 | |
| 	flags=$2
 | |
| 	pattern=$3
 | |
| 	pattern_human=$(echo "$pattern" | sed 's/Q/<NUL>/g')
 | |
| 
 | |
| 	if test "$matches" = 1
 | |
| 	then
 | |
| 		test_expect_success "git grep -f f $flags '$pattern_human' a" "
 | |
| 			printf '$pattern' | q_to_nul >f &&
 | |
| 			git grep -f f $flags a
 | |
| 		"
 | |
| 	elif test "$matches" = 0
 | |
| 	then
 | |
| 		test_expect_success "git grep -f f $flags '$pattern_human' a" "
 | |
| 			printf '$pattern' | q_to_nul >f &&
 | |
| 			test_must_fail git grep -f f $flags a
 | |
| 		"
 | |
| 	elif test "$matches" = T1
 | |
| 	then
 | |
| 		test_expect_failure "git grep -f f $flags '$pattern_human' a" "
 | |
| 			printf '$pattern' | q_to_nul >f &&
 | |
| 			git grep -f f $flags a
 | |
| 		"
 | |
| 	elif test "$matches" = T0
 | |
| 	then
 | |
| 		test_expect_failure "git grep -f f $flags '$pattern_human' a" "
 | |
| 			printf '$pattern' | q_to_nul >f &&
 | |
| 			test_must_fail git grep -f f $flags a
 | |
| 		"
 | |
| 	else
 | |
| 		test_expect_success "PANIC: Test framework error. Unknown matches value $matches" 'false'
 | |
| 	fi
 | |
| }
 | |
| 
 | |
| test_expect_success 'setup' "
 | |
| 	echo 'binaryQfileQm[*]cQ*æQð' | q_to_nul >a &&
 | |
| 	git add a &&
 | |
| 	git commit -m.
 | |
| "
 | |
| 
 | |
| test_expect_success 'git grep ina a' '
 | |
| 	echo Binary file a matches >expect &&
 | |
| 	git grep ina a >actual &&
 | |
| 	test_cmp expect actual
 | |
| '
 | |
| 
 | |
| test_expect_success 'git grep -ah ina a' '
 | |
| 	git grep -ah ina a >actual &&
 | |
| 	test_cmp a actual
 | |
| '
 | |
| 
 | |
| test_expect_success 'git grep -I ina a' '
 | |
| 	test_must_fail git grep -I ina a >actual &&
 | |
| 	test_must_be_empty actual
 | |
| '
 | |
| 
 | |
| test_expect_success 'git grep -c ina a' '
 | |
| 	echo a:1 >expect &&
 | |
| 	git grep -c ina a >actual &&
 | |
| 	test_cmp expect actual
 | |
| '
 | |
| 
 | |
| test_expect_success 'git grep -l ina a' '
 | |
| 	echo a >expect &&
 | |
| 	git grep -l ina a >actual &&
 | |
| 	test_cmp expect actual
 | |
| '
 | |
| 
 | |
| test_expect_success 'git grep -L bar a' '
 | |
| 	echo a >expect &&
 | |
| 	git grep -L bar a >actual &&
 | |
| 	test_cmp expect actual
 | |
| '
 | |
| 
 | |
| test_expect_success 'git grep -q ina a' '
 | |
| 	git grep -q ina a >actual &&
 | |
| 	test_must_be_empty actual
 | |
| '
 | |
| 
 | |
| test_expect_success 'git grep -F ile a' '
 | |
| 	git grep -F ile a
 | |
| '
 | |
| 
 | |
| test_expect_success 'git grep -Fi iLE a' '
 | |
| 	git grep -Fi iLE a
 | |
| '
 | |
| 
 | |
| # This test actually passes on platforms where regexec() supports the
 | |
| # flag REG_STARTEND.
 | |
| test_expect_success 'git grep ile a' '
 | |
| 	git grep ile a
 | |
| '
 | |
| 
 | |
| test_expect_failure 'git grep .fi a' '
 | |
| 	git grep .fi a
 | |
| '
 | |
| 
 | |
| nul_match 1 '-F' 'yQf'
 | |
| nul_match 0 '-F' 'yQx'
 | |
| nul_match 1 '-Fi' 'YQf'
 | |
| nul_match 0 '-Fi' 'YQx'
 | |
| nul_match 1 '' 'yQf'
 | |
| nul_match 0 '' 'yQx'
 | |
| nul_match 1 '' 'æQð'
 | |
| nul_match 1 '-F' 'eQm[*]c'
 | |
| nul_match 1 '-Fi' 'EQM[*]C'
 | |
| 
 | |
| # Regex patterns that would match but shouldn't with -F
 | |
| nul_match 0 '-F' 'yQ[f]'
 | |
| nul_match 0 '-F' '[y]Qf'
 | |
| nul_match 0 '-Fi' 'YQ[F]'
 | |
| nul_match 0 '-Fi' '[Y]QF'
 | |
| nul_match 0 '-F' 'æQ[ð]'
 | |
| nul_match 0 '-F' '[æ]Qð'
 | |
| nul_match 0 '-Fi' 'ÆQ[Ð]'
 | |
| nul_match 0 '-Fi' '[Æ]QÐ'
 | |
| 
 | |
| # kwset is disabled on -i & non-ASCII. No way to match non-ASCII \0
 | |
| # patterns case-insensitively.
 | |
| nul_match T1 '-i' 'ÆQÐ'
 | |
| 
 | |
| # \0 implicitly disables regexes. This is an undocumented internal
 | |
| # limitation.
 | |
| nul_match T1 '' 'yQ[f]'
 | |
| nul_match T1 '' '[y]Qf'
 | |
| nul_match T1 '-i' 'YQ[F]'
 | |
| nul_match T1 '-i' '[Y]Qf'
 | |
| nul_match T1 '' 'æQ[ð]'
 | |
| nul_match T1 '' '[æ]Qð'
 | |
| nul_match T1 '-i' 'ÆQ[Ð]'
 | |
| 
 | |
| # ... because of \0 implicitly disabling regexes regexes that
 | |
| # should/shouldn't match don't do the right thing.
 | |
| nul_match T1 '' 'eQm.*cQ'
 | |
| nul_match T1 '-i' 'EQM.*cQ'
 | |
| nul_match T0 '' 'eQm[*]c'
 | |
| nul_match T0 '-i' 'EQM[*]C'
 | |
| 
 | |
| # Due to the REG_STARTEND extension when kwset() is disabled on -i &
 | |
| # non-ASCII the string will be matched in its entirety, but the
 | |
| # pattern will be cut off at the first \0.
 | |
| nul_match 0 '-i' 'NOMATCHQð'
 | |
| nul_match T0 '-i' '[Æ]QNOMATCH'
 | |
| nul_match T0 '-i' '[æ]QNOMATCH'
 | |
| # Matches, but for the wrong reasons, just stops at [æ]
 | |
| nul_match 1 '-i' '[Æ]Qð'
 | |
| nul_match 1 '-i' '[æ]Qð'
 | |
| 
 | |
| # Ensure that the matcher doesn't regress to something that stops at
 | |
| # \0
 | |
| nul_match 0 '-F' 'yQ[f]'
 | |
| nul_match 0 '-Fi' 'YQ[F]'
 | |
| nul_match 0 '' 'yQNOMATCH'
 | |
| nul_match 0 '' 'QNOMATCH'
 | |
| nul_match 0 '-i' 'YQNOMATCH'
 | |
| nul_match 0 '-i' 'QNOMATCH'
 | |
| nul_match 0 '-F' 'æQ[ð]'
 | |
| nul_match 0 '-Fi' 'ÆQ[Ð]'
 | |
| nul_match 0 '' 'yQNÓMATCH'
 | |
| nul_match 0 '' 'QNÓMATCH'
 | |
| nul_match 0 '-i' 'YQNÓMATCH'
 | |
| nul_match 0 '-i' 'QNÓMATCH'
 | |
| 
 | |
| test_expect_success 'grep respects binary diff attribute' '
 | |
| 	echo text >t &&
 | |
| 	git add t &&
 | |
| 	echo t:text >expect &&
 | |
| 	git grep text t >actual &&
 | |
| 	test_cmp expect actual &&
 | |
| 	echo "t -diff" >.gitattributes &&
 | |
| 	echo "Binary file t matches" >expect &&
 | |
| 	git grep text t >actual &&
 | |
| 	test_cmp expect actual
 | |
| '
 | |
| 
 | |
| test_expect_success 'grep --cached respects binary diff attribute' '
 | |
| 	git grep --cached text t >actual &&
 | |
| 	test_cmp expect actual
 | |
| '
 | |
| 
 | |
| test_expect_success 'grep --cached respects binary diff attribute (2)' '
 | |
| 	git add .gitattributes &&
 | |
| 	rm .gitattributes &&
 | |
| 	git grep --cached text t >actual &&
 | |
| 	test_when_finished "git rm --cached .gitattributes" &&
 | |
| 	test_when_finished "git checkout .gitattributes" &&
 | |
| 	test_cmp expect actual
 | |
| '
 | |
| 
 | |
| test_expect_success 'grep revision respects binary diff attribute' '
 | |
| 	git commit -m new &&
 | |
| 	echo "Binary file HEAD:t matches" >expect &&
 | |
| 	git grep text HEAD -- t >actual &&
 | |
| 	test_when_finished "git reset HEAD^" &&
 | |
| 	test_cmp expect actual
 | |
| '
 | |
| 
 | |
| test_expect_success 'grep respects not-binary diff attribute' '
 | |
| 	echo binQary | q_to_nul >b &&
 | |
| 	git add b &&
 | |
| 	echo "Binary file b matches" >expect &&
 | |
| 	git grep bin b >actual &&
 | |
| 	test_cmp expect actual &&
 | |
| 	echo "b diff" >.gitattributes &&
 | |
| 	echo "b:binQary" >expect &&
 | |
| 	git grep bin b >actual.raw &&
 | |
| 	nul_to_q <actual.raw >actual &&
 | |
| 	test_cmp expect actual
 | |
| '
 | |
| 
 | |
| cat >nul_to_q_textconv <<'EOF'
 | |
| #!/bin/sh
 | |
| "$PERL_PATH" -pe 'y/\000/Q/' < "$1"
 | |
| EOF
 | |
| chmod +x nul_to_q_textconv
 | |
| 
 | |
| test_expect_success 'setup textconv filters' '
 | |
| 	echo a diff=foo >.gitattributes &&
 | |
| 	git config diff.foo.textconv "\"$(pwd)\""/nul_to_q_textconv
 | |
| '
 | |
| 
 | |
| test_expect_success 'grep does not honor textconv' '
 | |
| 	test_must_fail git grep Qfile
 | |
| '
 | |
| 
 | |
| test_expect_success 'grep --textconv honors textconv' '
 | |
| 	echo "a:binaryQfileQm[*]cQ*æQð" >expect &&
 | |
| 	git grep --textconv Qfile >actual &&
 | |
| 	test_cmp expect actual
 | |
| '
 | |
| 
 | |
| test_expect_success 'grep --no-textconv does not honor textconv' '
 | |
| 	test_must_fail git grep --no-textconv Qfile
 | |
| '
 | |
| 
 | |
| test_expect_success 'grep --textconv blob honors textconv' '
 | |
| 	echo "HEAD:a:binaryQfileQm[*]cQ*æQð" >expect &&
 | |
| 	git grep --textconv Qfile HEAD:a >actual &&
 | |
| 	test_cmp expect actual
 | |
| '
 | |
| 
 | |
| test_done
 |