mirror of
https://github.com/git/git.git
synced 2024-12-12 11:24:02 +08:00
3dc0b7f0dc
1f2e05f0b7
("wildmatch: fix exponential behavior", 2023-03-20)
introduced a new test with a background process. Backgrounding
necessarily gives a result of 0, so that a seemingly broken && chain is
not really broken.
Adjust t3070 slightly so that our chain lint test recognizes the
construct for what it is and does not raise a false positive.
Signed-off-by: Michael J Gruber <git@grubix.eu>
Signed-off-by: Junio C Hamano <gitster@pobox.com>
446 lines
13 KiB
Bash
Executable File
446 lines
13 KiB
Bash
Executable File
#!/bin/sh
|
|
|
|
test_description='wildmatch tests'
|
|
|
|
TEST_PASSES_SANITIZE_LEAK=true
|
|
. ./test-lib.sh
|
|
|
|
should_create_test_file() {
|
|
file=$1
|
|
|
|
case $file in
|
|
# `touch .` will succeed but obviously not do what we intend
|
|
# here.
|
|
".")
|
|
return 1
|
|
;;
|
|
# We cannot create a file with an empty filename.
|
|
"")
|
|
return 1
|
|
;;
|
|
# The tests that are testing that e.g. foo//bar is matched by
|
|
# foo/*/bar can't be tested on filesystems since there's no
|
|
# way we're getting a double slash.
|
|
*//*)
|
|
return 1
|
|
;;
|
|
# When testing the difference between foo/bar and foo/bar/ we
|
|
# can't test the latter.
|
|
*/)
|
|
return 1
|
|
;;
|
|
# On Windows, \ in paths is silently converted to /, which
|
|
# would result in the "touch" below working, but the test
|
|
# itself failing. See 6fd1106aa4 ("t3700: Skip a test with
|
|
# backslashes in pathspec", 2009-03-13) for prior art and
|
|
# details.
|
|
*\\*)
|
|
if ! test_have_prereq BSLASHPSPEC
|
|
then
|
|
return 1
|
|
fi
|
|
# NOTE: The ;;& bash extension is not portable, so
|
|
# this test needs to be at the end of the pattern
|
|
# list.
|
|
#
|
|
# If we want to add more conditional returns we either
|
|
# need a new case statement, or turn this whole thing
|
|
# into a series of "if" tests.
|
|
;;
|
|
esac
|
|
|
|
|
|
# On Windows proper (i.e. not Cygwin) many file names which
|
|
# under Cygwin would be emulated don't work.
|
|
if test_have_prereq MINGW
|
|
then
|
|
case $file in
|
|
" ")
|
|
# Files called " " are forbidden on Windows
|
|
return 1
|
|
;;
|
|
*\<*|*\>*|*:*|*\"*|*\|*|*\?*|*\**)
|
|
# Files with various special characters aren't
|
|
# allowed on Windows. Sourced from
|
|
# https://stackoverflow.com/a/31976060
|
|
return 1
|
|
;;
|
|
esac
|
|
fi
|
|
|
|
return 0
|
|
}
|
|
|
|
match_with_function() {
|
|
text=$1
|
|
pattern=$2
|
|
match_expect=$3
|
|
match_function=$4
|
|
|
|
if test "$match_expect" = 1
|
|
then
|
|
test_expect_success "$match_function: match '$text' '$pattern'" "
|
|
test-tool wildmatch $match_function '$text' '$pattern'
|
|
"
|
|
elif test "$match_expect" = 0
|
|
then
|
|
test_expect_success "$match_function: no match '$text' '$pattern'" "
|
|
test_must_fail test-tool wildmatch $match_function '$text' '$pattern'
|
|
"
|
|
else
|
|
test_expect_success "PANIC: Test framework error. Unknown matches value $match_expect" 'false'
|
|
fi
|
|
|
|
}
|
|
|
|
match_with_ls_files() {
|
|
text=$1
|
|
pattern=$2
|
|
match_expect=$3
|
|
match_function=$4
|
|
ls_files_args=$5
|
|
|
|
match_stdout_stderr_cmp="
|
|
tr -d '\0' <actual.raw >actual &&
|
|
test_must_be_empty actual.err &&
|
|
test_cmp expect actual"
|
|
|
|
if test "$match_expect" = 'E'
|
|
then
|
|
if test -e .git/created_test_file
|
|
then
|
|
test_expect_success EXPENSIVE_ON_WINDOWS "$match_function (via ls-files): match dies on '$pattern' '$text'" "
|
|
printf '%s' '$text' >expect &&
|
|
test_must_fail git$ls_files_args ls-files -z -- '$pattern'
|
|
"
|
|
else
|
|
test_expect_failure EXPENSIVE_ON_WINDOWS "$match_function (via ls-files): match skip '$pattern' '$text'" 'false'
|
|
fi
|
|
elif test "$match_expect" = 1
|
|
then
|
|
if test -e .git/created_test_file
|
|
then
|
|
test_expect_success EXPENSIVE_ON_WINDOWS "$match_function (via ls-files): match '$pattern' '$text'" "
|
|
printf '%s' '$text' >expect &&
|
|
git$ls_files_args ls-files -z -- '$pattern' >actual.raw 2>actual.err &&
|
|
$match_stdout_stderr_cmp
|
|
"
|
|
else
|
|
test_expect_failure EXPENSIVE_ON_WINDOWS "$match_function (via ls-files): match skip '$pattern' '$text'" 'false'
|
|
fi
|
|
elif test "$match_expect" = 0
|
|
then
|
|
if test -e .git/created_test_file
|
|
then
|
|
test_expect_success EXPENSIVE_ON_WINDOWS "$match_function (via ls-files): no match '$pattern' '$text'" "
|
|
>expect &&
|
|
git$ls_files_args ls-files -z -- '$pattern' >actual.raw 2>actual.err &&
|
|
$match_stdout_stderr_cmp
|
|
"
|
|
else
|
|
test_expect_failure EXPENSIVE_ON_WINDOWS "$match_function (via ls-files): no match skip '$pattern' '$text'" 'false'
|
|
fi
|
|
else
|
|
test_expect_success "PANIC: Test framework error. Unknown matches value $match_expect" 'false'
|
|
fi
|
|
}
|
|
|
|
match() {
|
|
if test "$#" = 6
|
|
then
|
|
# When test-tool wildmatch and git ls-files produce the same
|
|
# result.
|
|
match_glob=$1
|
|
match_file_glob=$match_glob
|
|
match_iglob=$2
|
|
match_file_iglob=$match_iglob
|
|
match_pathmatch=$3
|
|
match_file_pathmatch=$match_pathmatch
|
|
match_pathmatchi=$4
|
|
match_file_pathmatchi=$match_pathmatchi
|
|
text=$5
|
|
pattern=$6
|
|
elif test "$#" = 10
|
|
then
|
|
match_glob=$1
|
|
match_iglob=$2
|
|
match_pathmatch=$3
|
|
match_pathmatchi=$4
|
|
match_file_glob=$5
|
|
match_file_iglob=$6
|
|
match_file_pathmatch=$7
|
|
match_file_pathmatchi=$8
|
|
text=$9
|
|
pattern=${10}
|
|
fi
|
|
|
|
test_expect_success EXPENSIVE_ON_WINDOWS 'cleanup after previous file test' '
|
|
if test -e .git/created_test_file
|
|
then
|
|
git reset &&
|
|
git clean -df
|
|
fi
|
|
'
|
|
|
|
printf '%s' "$text" >.git/expected_test_file
|
|
|
|
test_expect_success EXPENSIVE_ON_WINDOWS "setup match file test for $text" '
|
|
file=$(cat .git/expected_test_file) &&
|
|
if should_create_test_file "$file"
|
|
then
|
|
dirs=${file%/*} &&
|
|
if test "$file" != "$dirs"
|
|
then
|
|
mkdir -p -- "$dirs" &&
|
|
touch -- "./$text"
|
|
else
|
|
touch -- "./$file"
|
|
fi &&
|
|
git add -A &&
|
|
printf "%s" "$file" >.git/created_test_file
|
|
elif test -e .git/created_test_file
|
|
then
|
|
rm .git/created_test_file
|
|
fi
|
|
'
|
|
|
|
# $1: Case sensitive glob match: test-tool wildmatch & ls-files
|
|
match_with_function "$text" "$pattern" $match_glob "wildmatch"
|
|
match_with_ls_files "$text" "$pattern" $match_file_glob "wildmatch" " --glob-pathspecs"
|
|
|
|
# $2: Case insensitive glob match: test-tool wildmatch & ls-files
|
|
match_with_function "$text" "$pattern" $match_iglob "iwildmatch"
|
|
match_with_ls_files "$text" "$pattern" $match_file_iglob "iwildmatch" " --glob-pathspecs --icase-pathspecs"
|
|
|
|
# $3: Case sensitive path match: test-tool wildmatch & ls-files
|
|
match_with_function "$text" "$pattern" $match_pathmatch "pathmatch"
|
|
match_with_ls_files "$text" "$pattern" $match_file_pathmatch "pathmatch" ""
|
|
|
|
# $4: Case insensitive path match: test-tool wildmatch & ls-files
|
|
match_with_function "$text" "$pattern" $match_pathmatchi "ipathmatch"
|
|
match_with_ls_files "$text" "$pattern" $match_file_pathmatchi "ipathmatch" " --icase-pathspecs"
|
|
}
|
|
|
|
# Basic wildmatch features
|
|
match 1 1 1 1 foo foo
|
|
match 0 0 0 0 foo bar
|
|
match 1 1 1 1 '' ""
|
|
match 1 1 1 1 foo '???'
|
|
match 0 0 0 0 foo '??'
|
|
match 1 1 1 1 foo '*'
|
|
match 1 1 1 1 foo 'f*'
|
|
match 0 0 0 0 foo '*f'
|
|
match 1 1 1 1 foo '*foo*'
|
|
match 1 1 1 1 foobar '*ob*a*r*'
|
|
match 1 1 1 1 aaaaaaabababab '*ab'
|
|
match 1 1 1 1 'foo*' 'foo\*'
|
|
match 0 0 0 0 foobar 'foo\*bar'
|
|
match 1 1 1 1 'f\oo' 'f\\oo'
|
|
match 1 1 1 1 ball '*[al]?'
|
|
match 0 0 0 0 ten '[ten]'
|
|
match 1 1 1 1 ten '**[!te]'
|
|
match 0 0 0 0 ten '**[!ten]'
|
|
match 1 1 1 1 ten 't[a-g]n'
|
|
match 0 0 0 0 ten 't[!a-g]n'
|
|
match 1 1 1 1 ton 't[!a-g]n'
|
|
match 1 1 1 1 ton 't[^a-g]n'
|
|
match 1 1 1 1 'a]b' 'a[]]b'
|
|
match 1 1 1 1 a-b 'a[]-]b'
|
|
match 1 1 1 1 'a]b' 'a[]-]b'
|
|
match 0 0 0 0 aab 'a[]-]b'
|
|
match 1 1 1 1 aab 'a[]a-]b'
|
|
match 1 1 1 1 ']' ']'
|
|
|
|
# Extended slash-matching features
|
|
match 0 0 1 1 'foo/baz/bar' 'foo*bar'
|
|
match 0 0 1 1 'foo/baz/bar' 'foo**bar'
|
|
match 1 1 1 1 'foobazbar' 'foo**bar'
|
|
match 1 1 1 1 'foo/baz/bar' 'foo/**/bar'
|
|
match 1 1 0 0 'foo/baz/bar' 'foo/**/**/bar'
|
|
match 1 1 1 1 'foo/b/a/z/bar' 'foo/**/bar'
|
|
match 1 1 1 1 'foo/b/a/z/bar' 'foo/**/**/bar'
|
|
match 1 1 0 0 'foo/bar' 'foo/**/bar'
|
|
match 1 1 0 0 'foo/bar' 'foo/**/**/bar'
|
|
match 0 0 1 1 'foo/bar' 'foo?bar'
|
|
match 0 0 1 1 'foo/bar' 'foo[/]bar'
|
|
match 0 0 1 1 'foo/bar' 'foo[^a-z]bar'
|
|
match 0 0 1 1 'foo/bar' 'f[^eiu][^eiu][^eiu][^eiu][^eiu]r'
|
|
match 1 1 1 1 'foo-bar' 'f[^eiu][^eiu][^eiu][^eiu][^eiu]r'
|
|
match 1 1 0 0 'foo' '**/foo'
|
|
match 1 1 1 1 'XXX/foo' '**/foo'
|
|
match 1 1 1 1 'bar/baz/foo' '**/foo'
|
|
match 0 0 1 1 'bar/baz/foo' '*/foo'
|
|
match 0 0 1 1 'foo/bar/baz' '**/bar*'
|
|
match 1 1 1 1 'deep/foo/bar/baz' '**/bar/*'
|
|
match 0 0 1 1 'deep/foo/bar/baz/' '**/bar/*'
|
|
match 1 1 1 1 'deep/foo/bar/baz/' '**/bar/**'
|
|
match 0 0 0 0 'deep/foo/bar' '**/bar/*'
|
|
match 1 1 1 1 'deep/foo/bar/' '**/bar/**'
|
|
match 0 0 1 1 'foo/bar/baz' '**/bar**'
|
|
match 1 1 1 1 'foo/bar/baz/x' '*/bar/**'
|
|
match 0 0 1 1 'deep/foo/bar/baz/x' '*/bar/**'
|
|
match 1 1 1 1 'deep/foo/bar/baz/x' '**/bar/*/*'
|
|
|
|
# Various additional tests
|
|
match 0 0 0 0 'acrt' 'a[c-c]st'
|
|
match 1 1 1 1 'acrt' 'a[c-c]rt'
|
|
match 0 0 0 0 ']' '[!]-]'
|
|
match 1 1 1 1 'a' '[!]-]'
|
|
match 0 0 0 0 '' '\'
|
|
match 0 0 0 0 \
|
|
1 1 1 1 '\' '\'
|
|
match 0 0 0 0 'XXX/\' '*/\'
|
|
match 1 1 1 1 'XXX/\' '*/\\'
|
|
match 1 1 1 1 'foo' 'foo'
|
|
match 1 1 1 1 '@foo' '@foo'
|
|
match 0 0 0 0 'foo' '@foo'
|
|
match 1 1 1 1 '[ab]' '\[ab]'
|
|
match 1 1 1 1 '[ab]' '[[]ab]'
|
|
match 1 1 1 1 '[ab]' '[[:]ab]'
|
|
match 0 0 0 0 '[ab]' '[[::]ab]'
|
|
match 1 1 1 1 '[ab]' '[[:digit]ab]'
|
|
match 1 1 1 1 '[ab]' '[\[:]ab]'
|
|
match 1 1 1 1 '?a?b' '\??\?b'
|
|
match 1 1 1 1 'abc' '\a\b\c'
|
|
match 0 0 0 0 \
|
|
E E E E 'foo' ''
|
|
match 1 1 1 1 'foo/bar/baz/to' '**/t[o]'
|
|
|
|
# Character class tests
|
|
match 1 1 1 1 'a1B' '[[:alpha:]][[:digit:]][[:upper:]]'
|
|
match 0 1 0 1 'a' '[[:digit:][:upper:][:space:]]'
|
|
match 1 1 1 1 'A' '[[:digit:][:upper:][:space:]]'
|
|
match 1 1 1 1 '1' '[[:digit:][:upper:][:space:]]'
|
|
match 0 0 0 0 '1' '[[:digit:][:upper:][:spaci:]]'
|
|
match 1 1 1 1 ' ' '[[:digit:][:upper:][:space:]]'
|
|
match 0 0 0 0 '.' '[[:digit:][:upper:][:space:]]'
|
|
match 1 1 1 1 '.' '[[:digit:][:punct:][:space:]]'
|
|
match 1 1 1 1 '5' '[[:xdigit:]]'
|
|
match 1 1 1 1 'f' '[[:xdigit:]]'
|
|
match 1 1 1 1 'D' '[[:xdigit:]]'
|
|
match 1 1 1 1 '_' '[[:alnum:][:alpha:][:blank:][:cntrl:][:digit:][:graph:][:lower:][:print:][:punct:][:space:][:upper:][:xdigit:]]'
|
|
match 1 1 1 1 '.' '[^[:alnum:][:alpha:][:blank:][:cntrl:][:digit:][:lower:][:space:][:upper:][:xdigit:]]'
|
|
match 1 1 1 1 '5' '[a-c[:digit:]x-z]'
|
|
match 1 1 1 1 'b' '[a-c[:digit:]x-z]'
|
|
match 1 1 1 1 'y' '[a-c[:digit:]x-z]'
|
|
match 0 0 0 0 'q' '[a-c[:digit:]x-z]'
|
|
|
|
# Additional tests, including some malformed wildmatch patterns
|
|
match 1 1 1 1 ']' '[\\-^]'
|
|
match 0 0 0 0 '[' '[\\-^]'
|
|
match 1 1 1 1 '-' '[\-_]'
|
|
match 1 1 1 1 ']' '[\]]'
|
|
match 0 0 0 0 '\]' '[\]]'
|
|
match 0 0 0 0 '\' '[\]]'
|
|
match 0 0 0 0 'ab' 'a[]b'
|
|
match 0 0 0 0 \
|
|
1 1 1 1 'a[]b' 'a[]b'
|
|
match 0 0 0 0 \
|
|
1 1 1 1 'ab[' 'ab['
|
|
match 0 0 0 0 'ab' '[!'
|
|
match 0 0 0 0 'ab' '[-'
|
|
match 1 1 1 1 '-' '[-]'
|
|
match 0 0 0 0 '-' '[a-'
|
|
match 0 0 0 0 '-' '[!a-'
|
|
match 1 1 1 1 '-' '[--A]'
|
|
match 1 1 1 1 '5' '[--A]'
|
|
match 1 1 1 1 ' ' '[ --]'
|
|
match 1 1 1 1 '$' '[ --]'
|
|
match 1 1 1 1 '-' '[ --]'
|
|
match 0 0 0 0 '0' '[ --]'
|
|
match 1 1 1 1 '-' '[---]'
|
|
match 1 1 1 1 '-' '[------]'
|
|
match 0 0 0 0 'j' '[a-e-n]'
|
|
match 1 1 1 1 '-' '[a-e-n]'
|
|
match 1 1 1 1 'a' '[!------]'
|
|
match 0 0 0 0 '[' '[]-a]'
|
|
match 1 1 1 1 '^' '[]-a]'
|
|
match 0 0 0 0 '^' '[!]-a]'
|
|
match 1 1 1 1 '[' '[!]-a]'
|
|
match 1 1 1 1 '^' '[a^bc]'
|
|
match 1 1 1 1 '-b]' '[a-]b]'
|
|
match 0 0 0 0 '\' '[\]'
|
|
match 1 1 1 1 '\' '[\\]'
|
|
match 0 0 0 0 '\' '[!\\]'
|
|
match 1 1 1 1 'G' '[A-\\]'
|
|
match 0 0 0 0 'aaabbb' 'b*a'
|
|
match 0 0 0 0 'aabcaa' '*ba*'
|
|
match 1 1 1 1 ',' '[,]'
|
|
match 1 1 1 1 ',' '[\\,]'
|
|
match 1 1 1 1 '\' '[\\,]'
|
|
match 1 1 1 1 '-' '[,-.]'
|
|
match 0 0 0 0 '+' '[,-.]'
|
|
match 0 0 0 0 '-.]' '[,-.]'
|
|
match 1 1 1 1 '2' '[\1-\3]'
|
|
match 1 1 1 1 '3' '[\1-\3]'
|
|
match 0 0 0 0 '4' '[\1-\3]'
|
|
match 1 1 1 1 '\' '[[-\]]'
|
|
match 1 1 1 1 '[' '[[-\]]'
|
|
match 1 1 1 1 ']' '[[-\]]'
|
|
match 0 0 0 0 '-' '[[-\]]'
|
|
|
|
# Test recursion
|
|
match 1 1 1 1 '-adobe-courier-bold-o-normal--12-120-75-75-m-70-iso8859-1' '-*-*-*-*-*-*-12-*-*-*-m-*-*-*'
|
|
match 0 0 0 0 '-adobe-courier-bold-o-normal--12-120-75-75-X-70-iso8859-1' '-*-*-*-*-*-*-12-*-*-*-m-*-*-*'
|
|
match 0 0 0 0 '-adobe-courier-bold-o-normal--12-120-75-75-/-70-iso8859-1' '-*-*-*-*-*-*-12-*-*-*-m-*-*-*'
|
|
match 1 1 1 1 'XXX/adobe/courier/bold/o/normal//12/120/75/75/m/70/iso8859/1' 'XXX/*/*/*/*/*/*/12/*/*/*/m/*/*/*'
|
|
match 0 0 0 0 'XXX/adobe/courier/bold/o/normal//12/120/75/75/X/70/iso8859/1' 'XXX/*/*/*/*/*/*/12/*/*/*/m/*/*/*'
|
|
match 1 1 1 1 'abcd/abcdefg/abcdefghijk/abcdefghijklmnop.txt' '**/*a*b*g*n*t'
|
|
match 0 0 0 0 'abcd/abcdefg/abcdefghijk/abcdefghijklmnop.txtz' '**/*a*b*g*n*t'
|
|
match 0 0 0 0 foo '*/*/*'
|
|
match 0 0 0 0 foo/bar '*/*/*'
|
|
match 1 1 1 1 foo/bba/arr '*/*/*'
|
|
match 0 0 1 1 foo/bb/aa/rr '*/*/*'
|
|
match 1 1 1 1 foo/bb/aa/rr '**/**/**'
|
|
match 1 1 1 1 abcXdefXghi '*X*i'
|
|
match 0 0 1 1 ab/cXd/efXg/hi '*X*i'
|
|
match 1 1 1 1 ab/cXd/efXg/hi '*/*X*/*/*i'
|
|
match 1 1 1 1 ab/cXd/efXg/hi '**/*X*/**/*i'
|
|
|
|
# Extra pathmatch tests
|
|
match 0 0 0 0 foo fo
|
|
match 1 1 1 1 foo/bar foo/bar
|
|
match 1 1 1 1 foo/bar 'foo/*'
|
|
match 0 0 1 1 foo/bba/arr 'foo/*'
|
|
match 1 1 1 1 foo/bba/arr 'foo/**'
|
|
match 0 0 1 1 foo/bba/arr 'foo*'
|
|
match 0 0 1 1 \
|
|
1 1 1 1 foo/bba/arr 'foo**'
|
|
match 0 0 1 1 foo/bba/arr 'foo/*arr'
|
|
match 0 0 1 1 foo/bba/arr 'foo/**arr'
|
|
match 0 0 0 0 foo/bba/arr 'foo/*z'
|
|
match 0 0 0 0 foo/bba/arr 'foo/**z'
|
|
match 0 0 1 1 foo/bar 'foo?bar'
|
|
match 0 0 1 1 foo/bar 'foo[/]bar'
|
|
match 0 0 1 1 foo/bar 'foo[^a-z]bar'
|
|
match 0 0 1 1 ab/cXd/efXg/hi '*Xg*i'
|
|
|
|
# Extra case-sensitivity tests
|
|
match 0 1 0 1 'a' '[A-Z]'
|
|
match 1 1 1 1 'A' '[A-Z]'
|
|
match 0 1 0 1 'A' '[a-z]'
|
|
match 1 1 1 1 'a' '[a-z]'
|
|
match 0 1 0 1 'a' '[[:upper:]]'
|
|
match 1 1 1 1 'A' '[[:upper:]]'
|
|
match 0 1 0 1 'A' '[[:lower:]]'
|
|
match 1 1 1 1 'a' '[[:lower:]]'
|
|
match 0 1 0 1 'A' '[B-Za]'
|
|
match 1 1 1 1 'a' '[B-Za]'
|
|
match 0 1 0 1 'A' '[B-a]'
|
|
match 1 1 1 1 'a' '[B-a]'
|
|
match 0 1 0 1 'z' '[Z-y]'
|
|
match 1 1 1 1 'Z' '[Z-y]'
|
|
|
|
test_expect_success 'matching does not exhibit exponential behavior' '
|
|
{
|
|
test-tool wildmatch wildmatch \
|
|
aaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaab \
|
|
"*a*a*a*a*a*a*a*a*a*a*a*a*a*a*a*a" &
|
|
pid=$!
|
|
} &&
|
|
sleep 2 &&
|
|
! kill $!
|
|
'
|
|
|
|
test_done
|