|  | #!/bin/sh | 
|  |  | 
|  | test_description='wildmatch tests' | 
|  |  | 
|  | . ./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_done |