You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
445 lines
13 KiB
445 lines
13 KiB
#!/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
|
|
|