|  | #!/bin/sh | 
|  |  | 
|  | test_description=check-ignore | 
|  |  | 
|  | . ./test-lib.sh | 
|  |  | 
|  | init_vars () { | 
|  | global_excludes="global-excludes" | 
|  | } | 
|  |  | 
|  | enable_global_excludes () { | 
|  | init_vars && | 
|  | git config core.excludesfile "$global_excludes" | 
|  | } | 
|  |  | 
|  | expect_in () { | 
|  | dest="$HOME/expected-$1" text="$2" | 
|  | if test -z "$text" | 
|  | then | 
|  | >"$dest" # avoid newline | 
|  | else | 
|  | echo "$text" >"$dest" | 
|  | fi | 
|  | } | 
|  |  | 
|  | expect () { | 
|  | expect_in stdout "$1" | 
|  | } | 
|  |  | 
|  | expect_from_stdin () { | 
|  | cat >"$HOME/expected-stdout" | 
|  | } | 
|  |  | 
|  | test_stderr () { | 
|  | expected="$1" | 
|  | expect_in stderr "$1" && | 
|  | test_i18ncmp "$HOME/expected-stderr" "$HOME/stderr" | 
|  | } | 
|  |  | 
|  | broken_c_unquote () { | 
|  | "$PERL_PATH" -pe 's/^"//; s/\\//; s/"$//; tr/\n/\0/' "$@" | 
|  | } | 
|  |  | 
|  | broken_c_unquote_verbose () { | 
|  | "$PERL_PATH" -pe 's/	"/	/; s/\\//; s/"$//; tr/:\t\n/\0/' "$@" | 
|  | } | 
|  |  | 
|  | stderr_contains () { | 
|  | regexp="$1" | 
|  | if test_i18ngrep "$regexp" "$HOME/stderr" | 
|  | then | 
|  | return 0 | 
|  | else | 
|  | echo "didn't find /$regexp/ in $HOME/stderr" | 
|  | cat "$HOME/stderr" | 
|  | return 1 | 
|  | fi | 
|  | } | 
|  |  | 
|  | stderr_empty_on_success () { | 
|  | expect_code="$1" | 
|  | if test $expect_code = 0 | 
|  | then | 
|  | test_stderr "" | 
|  | else | 
|  | # If we expect failure then stderr might or might not be empty | 
|  | # due to --quiet - the caller can check its contents | 
|  | return 0 | 
|  | fi | 
|  | } | 
|  |  | 
|  | test_check_ignore () { | 
|  | args="$1" expect_code="${2:-0}" global_args="$3" | 
|  |  | 
|  | init_vars && | 
|  | rm -f "$HOME/stdout" "$HOME/stderr" "$HOME/cmd" && | 
|  | echo git $global_args check-ignore $quiet_opt $verbose_opt $non_matching_opt $no_index_opt $args \ | 
|  | >"$HOME/cmd" && | 
|  | echo "$expect_code" >"$HOME/expected-exit-code" && | 
|  | test_expect_code "$expect_code" \ | 
|  | git $global_args check-ignore $quiet_opt $verbose_opt $non_matching_opt $no_index_opt $args \ | 
|  | >"$HOME/stdout" 2>"$HOME/stderr" && | 
|  | test_cmp "$HOME/expected-stdout" "$HOME/stdout" && | 
|  | stderr_empty_on_success "$expect_code" | 
|  | } | 
|  |  | 
|  | # Runs the same code with 4 different levels of output verbosity: | 
|  | # | 
|  | #   1. with -q / --quiet | 
|  | #   2. with default verbosity | 
|  | #   3. with -v / --verbose | 
|  | #   4. with -v / --verbose, *and* -n / --non-matching | 
|  | # | 
|  | # expecting success each time.  Takes advantage of the fact that | 
|  | # check-ignore --verbose output is the same as normal output except | 
|  | # for the extra first column. | 
|  | # | 
|  | # A parameter is used to determine if the tests are run with the | 
|  | # normal case (using the index), or with the --no-index option. | 
|  | # | 
|  | # Arguments: | 
|  | #   - (optional) prereqs for this test, e.g. 'SYMLINKS' | 
|  | #   - test name | 
|  | #   - output to expect from the fourth verbosity mode (the output | 
|  | #     from the other verbosity modes is automatically inferred | 
|  | #     from this value) | 
|  | #   - code to run (should invoke test_check_ignore) | 
|  | #   - index option: --index or --no-index | 
|  | test_expect_success_multiple () { | 
|  | prereq= | 
|  | if test $# -eq 5 | 
|  | then | 
|  | prereq=$1 | 
|  | shift | 
|  | fi | 
|  | if test "$4" = "--index" | 
|  | then | 
|  | no_index_opt= | 
|  | else | 
|  | no_index_opt=$4 | 
|  | fi | 
|  | testname="$1" expect_all="$2" code="$3" | 
|  |  | 
|  | expect_verbose=$( echo "$expect_all" | grep -v '^::	' ) | 
|  | expect=$( echo "$expect_verbose" | sed -e 's/.*	//' ) | 
|  |  | 
|  | test_expect_success $prereq "$testname${no_index_opt:+ with $no_index_opt}" ' | 
|  | expect "$expect" && | 
|  | eval "$code" | 
|  | ' | 
|  |  | 
|  | # --quiet is only valid when a single pattern is passed | 
|  | if test $( echo "$expect_all" | wc -l ) = 1 | 
|  | then | 
|  | for quiet_opt in '-q' '--quiet' | 
|  | do | 
|  | opts="${no_index_opt:+$no_index_opt }$quiet_opt" | 
|  | test_expect_success $prereq "$testname${opts:+ with $opts}" " | 
|  | expect '' && | 
|  | $code | 
|  | " | 
|  | done | 
|  | quiet_opt= | 
|  | fi | 
|  |  | 
|  | for verbose_opt in '-v' '--verbose' | 
|  | do | 
|  | for non_matching_opt in '' '-n' '--non-matching' | 
|  | do | 
|  | if test -n "$non_matching_opt" | 
|  | then | 
|  | my_expect="$expect_all" | 
|  | else | 
|  | my_expect="$expect_verbose" | 
|  | fi | 
|  |  | 
|  | test_code=" | 
|  | expect '$my_expect' && | 
|  | $code | 
|  | " | 
|  | opts="${no_index_opt:+$no_index_opt }$verbose_opt${non_matching_opt:+ $non_matching_opt}" | 
|  | test_expect_success $prereq "$testname${opts:+ with $opts}" "$test_code" | 
|  | done | 
|  | done | 
|  | verbose_opt= | 
|  | non_matching_opt= | 
|  | no_index_opt= | 
|  | } | 
|  |  | 
|  | test_expect_success_multi () { | 
|  | test_expect_success_multiple "$@" "--index" | 
|  | } | 
|  |  | 
|  | test_expect_success_no_index_multi () { | 
|  | test_expect_success_multiple "$@" "--no-index" | 
|  | } | 
|  |  | 
|  | test_expect_success 'setup' ' | 
|  | init_vars && | 
|  | mkdir -p a/b/ignored-dir a/submodule b && | 
|  | if test_have_prereq SYMLINKS | 
|  | then | 
|  | ln -s b a/symlink | 
|  | fi && | 
|  | ( | 
|  | cd a/submodule && | 
|  | git init && | 
|  | echo a >a && | 
|  | git add a && | 
|  | git commit -m"commit in submodule" | 
|  | ) && | 
|  | git add a/submodule && | 
|  | cat <<-\EOF >.gitignore && | 
|  | one | 
|  | ignored-* | 
|  | top-level-dir/ | 
|  | EOF | 
|  | for dir in . a | 
|  | do | 
|  | : >$dir/not-ignored && | 
|  | : >$dir/ignored-and-untracked && | 
|  | : >$dir/ignored-but-in-index | 
|  | done && | 
|  | git add -f ignored-but-in-index a/ignored-but-in-index && | 
|  | cat <<-\EOF >a/.gitignore && | 
|  | two* | 
|  | *three | 
|  | EOF | 
|  | cat <<-\EOF >a/b/.gitignore && | 
|  | four | 
|  | five | 
|  | # this comment should affect the line numbers | 
|  | six | 
|  | ignored-dir/ | 
|  | # and so should this blank line: | 
|  |  | 
|  | !on* | 
|  | !two | 
|  | EOF | 
|  | echo "seven" >a/b/ignored-dir/.gitignore && | 
|  | test -n "$HOME" && | 
|  | cat <<-\EOF >"$global_excludes" && | 
|  | globalone | 
|  | !globaltwo | 
|  | globalthree | 
|  | EOF | 
|  | cat <<-\EOF >>.git/info/exclude | 
|  | per-repo | 
|  | EOF | 
|  | ' | 
|  |  | 
|  | ############################################################################ | 
|  | # | 
|  | # test invalid inputs | 
|  |  | 
|  | test_expect_success_multi '. corner-case' '::	.' ' | 
|  | test_check_ignore . 1 | 
|  | ' | 
|  |  | 
|  | test_expect_success_multi 'empty command line' '' ' | 
|  | test_check_ignore "" 128 && | 
|  | stderr_contains "fatal: no path specified" | 
|  | ' | 
|  |  | 
|  | test_expect_success_multi '--stdin with empty STDIN' '' ' | 
|  | test_check_ignore "--stdin" 1 </dev/null && | 
|  | test_stderr "" | 
|  | ' | 
|  |  | 
|  | test_expect_success '-q with multiple args' ' | 
|  | expect "" && | 
|  | test_check_ignore "-q one two" 128 && | 
|  | stderr_contains "fatal: --quiet is only valid with a single pathname" | 
|  | ' | 
|  |  | 
|  | test_expect_success '--quiet with multiple args' ' | 
|  | expect "" && | 
|  | test_check_ignore "--quiet one two" 128 && | 
|  | stderr_contains "fatal: --quiet is only valid with a single pathname" | 
|  | ' | 
|  |  | 
|  | for verbose_opt in '-v' '--verbose' | 
|  | do | 
|  | for quiet_opt in '-q' '--quiet' | 
|  | do | 
|  | test_expect_success "$quiet_opt $verbose_opt" " | 
|  | expect '' && | 
|  | test_check_ignore '$quiet_opt $verbose_opt foo' 128 && | 
|  | stderr_contains 'fatal: cannot have both --quiet and --verbose' | 
|  | " | 
|  | done | 
|  | done | 
|  |  | 
|  | test_expect_success '--quiet with multiple args' ' | 
|  | expect "" && | 
|  | test_check_ignore "--quiet one two" 128 && | 
|  | stderr_contains "fatal: --quiet is only valid with a single pathname" | 
|  | ' | 
|  |  | 
|  | test_expect_success_multi 'erroneous use of --' '' ' | 
|  | test_check_ignore "--" 128 && | 
|  | stderr_contains "fatal: no path specified" | 
|  | ' | 
|  |  | 
|  | test_expect_success_multi '--stdin with superfluous arg' '' ' | 
|  | test_check_ignore "--stdin foo" 128 && | 
|  | stderr_contains "fatal: cannot specify pathnames with --stdin" | 
|  | ' | 
|  |  | 
|  | test_expect_success_multi '--stdin -z with superfluous arg' '' ' | 
|  | test_check_ignore "--stdin -z foo" 128 && | 
|  | stderr_contains "fatal: cannot specify pathnames with --stdin" | 
|  | ' | 
|  |  | 
|  | test_expect_success_multi '-z without --stdin' '' ' | 
|  | test_check_ignore "-z" 128 && | 
|  | stderr_contains "fatal: -z only makes sense with --stdin" | 
|  | ' | 
|  |  | 
|  | test_expect_success_multi '-z without --stdin and superfluous arg' '' ' | 
|  | test_check_ignore "-z foo" 128 && | 
|  | stderr_contains "fatal: -z only makes sense with --stdin" | 
|  | ' | 
|  |  | 
|  | test_expect_success_multi 'needs work tree' '' ' | 
|  | ( | 
|  | cd .git && | 
|  | test_check_ignore "foo" 128 | 
|  | ) && | 
|  | stderr_contains "fatal: This operation must be run in a work tree" | 
|  | ' | 
|  |  | 
|  | ############################################################################ | 
|  | # | 
|  | # test standard ignores | 
|  |  | 
|  | # First make sure that the presence of a file in the working tree | 
|  | # does not impact results, but that the presence of a file in the | 
|  | # index does unless the --no-index option is used. | 
|  |  | 
|  | for subdir in '' 'a/' | 
|  | do | 
|  | if test -z "$subdir" | 
|  | then | 
|  | where="at top-level" | 
|  | else | 
|  | where="in subdir $subdir" | 
|  | fi | 
|  |  | 
|  | test_expect_success_multi "non-existent file $where not ignored" \ | 
|  | "::	${subdir}non-existent" \ | 
|  | "test_check_ignore '${subdir}non-existent' 1" | 
|  |  | 
|  | test_expect_success_no_index_multi "non-existent file $where not ignored" \ | 
|  | "::	${subdir}non-existent" \ | 
|  | "test_check_ignore '${subdir}non-existent' 1" | 
|  |  | 
|  | test_expect_success_multi "non-existent file $where ignored" \ | 
|  | ".gitignore:1:one	${subdir}one" \ | 
|  | "test_check_ignore '${subdir}one'" | 
|  |  | 
|  | test_expect_success_no_index_multi "non-existent file $where ignored" \ | 
|  | ".gitignore:1:one	${subdir}one" \ | 
|  | "test_check_ignore '${subdir}one'" | 
|  |  | 
|  | test_expect_success_multi "existing untracked file $where not ignored" \ | 
|  | "::	${subdir}not-ignored" \ | 
|  | "test_check_ignore '${subdir}not-ignored' 1" | 
|  |  | 
|  | test_expect_success_no_index_multi "existing untracked file $where not ignored" \ | 
|  | "::	${subdir}not-ignored" \ | 
|  | "test_check_ignore '${subdir}not-ignored' 1" | 
|  |  | 
|  | test_expect_success_multi "existing tracked file $where not ignored" \ | 
|  | "::	${subdir}ignored-but-in-index" \ | 
|  | "test_check_ignore '${subdir}ignored-but-in-index' 1" | 
|  |  | 
|  | test_expect_success_no_index_multi "existing tracked file $where shown as ignored" \ | 
|  | ".gitignore:2:ignored-*	${subdir}ignored-but-in-index" \ | 
|  | "test_check_ignore '${subdir}ignored-but-in-index'" | 
|  |  | 
|  | test_expect_success_multi "existing untracked file $where ignored" \ | 
|  | ".gitignore:2:ignored-*	${subdir}ignored-and-untracked" \ | 
|  | "test_check_ignore '${subdir}ignored-and-untracked'" | 
|  |  | 
|  | test_expect_success_no_index_multi "existing untracked file $where ignored" \ | 
|  | ".gitignore:2:ignored-*	${subdir}ignored-and-untracked" \ | 
|  | "test_check_ignore '${subdir}ignored-and-untracked'" | 
|  |  | 
|  | test_expect_success_multi "mix of file types $where" \ | 
|  | "::	${subdir}non-existent | 
|  | .gitignore:1:one	${subdir}one | 
|  | ::	${subdir}not-ignored | 
|  | ::	${subdir}ignored-but-in-index | 
|  | .gitignore:2:ignored-*	${subdir}ignored-and-untracked" \ | 
|  | "test_check_ignore ' | 
|  | ${subdir}non-existent | 
|  | ${subdir}one | 
|  | ${subdir}not-ignored | 
|  | ${subdir}ignored-but-in-index | 
|  | ${subdir}ignored-and-untracked' | 
|  | " | 
|  |  | 
|  | test_expect_success_no_index_multi "mix of file types $where" \ | 
|  | "::	${subdir}non-existent | 
|  | .gitignore:1:one	${subdir}one | 
|  | ::	${subdir}not-ignored | 
|  | .gitignore:2:ignored-*	${subdir}ignored-but-in-index | 
|  | .gitignore:2:ignored-*	${subdir}ignored-and-untracked" \ | 
|  | "test_check_ignore ' | 
|  | ${subdir}non-existent | 
|  | ${subdir}one | 
|  | ${subdir}not-ignored | 
|  | ${subdir}ignored-but-in-index | 
|  | ${subdir}ignored-and-untracked' | 
|  | " | 
|  | done | 
|  |  | 
|  | # Having established the above, from now on we mostly test against | 
|  | # files which do not exist in the working tree or index. | 
|  |  | 
|  | test_expect_success 'sub-directory local ignore' ' | 
|  | expect "a/3-three" && | 
|  | test_check_ignore "a/3-three a/three-not-this-one" | 
|  | ' | 
|  |  | 
|  | test_expect_success 'sub-directory local ignore with --verbose'  ' | 
|  | expect "a/.gitignore:2:*three	a/3-three" && | 
|  | test_check_ignore "--verbose a/3-three a/three-not-this-one" | 
|  | ' | 
|  |  | 
|  | test_expect_success 'local ignore inside a sub-directory' ' | 
|  | expect "3-three" && | 
|  | ( | 
|  | cd a && | 
|  | test_check_ignore "3-three three-not-this-one" | 
|  | ) | 
|  | ' | 
|  | test_expect_success 'local ignore inside a sub-directory with --verbose' ' | 
|  | expect "a/.gitignore:2:*three	3-three" && | 
|  | ( | 
|  | cd a && | 
|  | test_check_ignore "--verbose 3-three three-not-this-one" | 
|  | ) | 
|  | ' | 
|  |  | 
|  | test_expect_success_multi 'nested include' \ | 
|  | 'a/b/.gitignore:8:!on*	a/b/one' ' | 
|  | test_check_ignore "a/b/one" | 
|  | ' | 
|  |  | 
|  | ############################################################################ | 
|  | # | 
|  | # test ignored sub-directories | 
|  |  | 
|  | test_expect_success_multi 'ignored sub-directory' \ | 
|  | 'a/b/.gitignore:5:ignored-dir/	a/b/ignored-dir' ' | 
|  | test_check_ignore "a/b/ignored-dir" | 
|  | ' | 
|  |  | 
|  | test_expect_success 'multiple files inside ignored sub-directory' ' | 
|  | expect_from_stdin <<-\EOF && | 
|  | a/b/ignored-dir/foo | 
|  | a/b/ignored-dir/twoooo | 
|  | a/b/ignored-dir/seven | 
|  | EOF | 
|  | test_check_ignore "a/b/ignored-dir/foo a/b/ignored-dir/twoooo a/b/ignored-dir/seven" | 
|  | ' | 
|  |  | 
|  | test_expect_success 'multiple files inside ignored sub-directory with -v' ' | 
|  | expect_from_stdin <<-\EOF && | 
|  | a/b/.gitignore:5:ignored-dir/	a/b/ignored-dir/foo | 
|  | a/b/.gitignore:5:ignored-dir/	a/b/ignored-dir/twoooo | 
|  | a/b/.gitignore:5:ignored-dir/	a/b/ignored-dir/seven | 
|  | EOF | 
|  | test_check_ignore "-v a/b/ignored-dir/foo a/b/ignored-dir/twoooo a/b/ignored-dir/seven" | 
|  | ' | 
|  |  | 
|  | test_expect_success 'cd to ignored sub-directory' ' | 
|  | expect_from_stdin <<-\EOF && | 
|  | foo | 
|  | twoooo | 
|  | ../one | 
|  | seven | 
|  | ../../one | 
|  | EOF | 
|  | ( | 
|  | cd a/b/ignored-dir && | 
|  | test_check_ignore "foo twoooo ../one seven ../../one" | 
|  | ) | 
|  | ' | 
|  |  | 
|  | test_expect_success 'cd to ignored sub-directory with -v' ' | 
|  | expect_from_stdin <<-\EOF && | 
|  | a/b/.gitignore:5:ignored-dir/	foo | 
|  | a/b/.gitignore:5:ignored-dir/	twoooo | 
|  | a/b/.gitignore:8:!on*	../one | 
|  | a/b/.gitignore:5:ignored-dir/	seven | 
|  | .gitignore:1:one	../../one | 
|  | EOF | 
|  | ( | 
|  | cd a/b/ignored-dir && | 
|  | test_check_ignore "-v foo twoooo ../one seven ../../one" | 
|  | ) | 
|  | ' | 
|  |  | 
|  | ############################################################################ | 
|  | # | 
|  | # test handling of symlinks | 
|  |  | 
|  | test_expect_success_multi SYMLINKS 'symlink' '::	a/symlink' ' | 
|  | test_check_ignore "a/symlink" 1 | 
|  | ' | 
|  |  | 
|  | test_expect_success_multi SYMLINKS 'beyond a symlink' '' ' | 
|  | test_check_ignore "a/symlink/foo" 128 && | 
|  | test_stderr "fatal: pathspec '\''a/symlink/foo'\'' is beyond a symbolic link" | 
|  | ' | 
|  |  | 
|  | test_expect_success_multi SYMLINKS 'beyond a symlink from subdirectory' '' ' | 
|  | ( | 
|  | cd a && | 
|  | test_check_ignore "symlink/foo" 128 | 
|  | ) && | 
|  | test_stderr "fatal: pathspec '\''symlink/foo'\'' is beyond a symbolic link" | 
|  | ' | 
|  |  | 
|  | ############################################################################ | 
|  | # | 
|  | # test handling of submodules | 
|  |  | 
|  | test_expect_success_multi 'submodule' '' ' | 
|  | test_check_ignore "a/submodule/one" 128 && | 
|  | test_stderr "fatal: Pathspec '\''a/submodule/one'\'' is in submodule '\''a/submodule'\''" | 
|  | ' | 
|  |  | 
|  | test_expect_success_multi 'submodule from subdirectory' '' ' | 
|  | ( | 
|  | cd a && | 
|  | test_check_ignore "submodule/one" 128 | 
|  | ) && | 
|  | test_stderr "fatal: Pathspec '\''submodule/one'\'' is in submodule '\''a/submodule'\''" | 
|  | ' | 
|  |  | 
|  | ############################################################################ | 
|  | # | 
|  | # test handling of global ignore files | 
|  |  | 
|  | test_expect_success 'global ignore not yet enabled' ' | 
|  | expect_from_stdin <<-\EOF && | 
|  | .git/info/exclude:7:per-repo	per-repo | 
|  | a/.gitignore:2:*three	a/globalthree | 
|  | .git/info/exclude:7:per-repo	a/per-repo | 
|  | EOF | 
|  | test_check_ignore "-v globalone per-repo a/globalthree a/per-repo not-ignored a/globaltwo" | 
|  | ' | 
|  |  | 
|  | test_expect_success 'global ignore' ' | 
|  | enable_global_excludes && | 
|  | expect_from_stdin <<-\EOF && | 
|  | globalone | 
|  | per-repo | 
|  | globalthree | 
|  | a/globalthree | 
|  | a/per-repo | 
|  | globaltwo | 
|  | EOF | 
|  | test_check_ignore "globalone per-repo globalthree a/globalthree a/per-repo not-ignored globaltwo" | 
|  | ' | 
|  |  | 
|  | test_expect_success 'global ignore with -v' ' | 
|  | enable_global_excludes && | 
|  | expect_from_stdin <<-EOF && | 
|  | $global_excludes:1:globalone	globalone | 
|  | .git/info/exclude:7:per-repo	per-repo | 
|  | $global_excludes:3:globalthree	globalthree | 
|  | a/.gitignore:2:*three	a/globalthree | 
|  | .git/info/exclude:7:per-repo	a/per-repo | 
|  | $global_excludes:2:!globaltwo	globaltwo | 
|  | EOF | 
|  | test_check_ignore "-v globalone per-repo globalthree a/globalthree a/per-repo not-ignored globaltwo" | 
|  | ' | 
|  |  | 
|  | ############################################################################ | 
|  | # | 
|  | # test --stdin | 
|  |  | 
|  | cat <<-\EOF >stdin | 
|  | one | 
|  | not-ignored | 
|  | a/one | 
|  | a/not-ignored | 
|  | a/b/on | 
|  | a/b/one | 
|  | a/b/one one | 
|  | "a/b/one two" | 
|  | "a/b/one\"three" | 
|  | a/b/not-ignored | 
|  | a/b/two | 
|  | a/b/twooo | 
|  | globaltwo | 
|  | a/globaltwo | 
|  | a/b/globaltwo | 
|  | b/globaltwo | 
|  | EOF | 
|  | cat <<-\EOF >expected-default | 
|  | one | 
|  | a/one | 
|  | a/b/on | 
|  | a/b/one | 
|  | a/b/one one | 
|  | a/b/one two | 
|  | "a/b/one\"three" | 
|  | a/b/two | 
|  | a/b/twooo | 
|  | globaltwo | 
|  | a/globaltwo | 
|  | a/b/globaltwo | 
|  | b/globaltwo | 
|  | EOF | 
|  | cat <<-EOF >expected-verbose | 
|  | .gitignore:1:one	one | 
|  | .gitignore:1:one	a/one | 
|  | a/b/.gitignore:8:!on*	a/b/on | 
|  | a/b/.gitignore:8:!on*	a/b/one | 
|  | a/b/.gitignore:8:!on*	a/b/one one | 
|  | a/b/.gitignore:8:!on*	a/b/one two | 
|  | a/b/.gitignore:8:!on*	"a/b/one\\"three" | 
|  | a/b/.gitignore:9:!two	a/b/two | 
|  | a/.gitignore:1:two*	a/b/twooo | 
|  | $global_excludes:2:!globaltwo	globaltwo | 
|  | $global_excludes:2:!globaltwo	a/globaltwo | 
|  | $global_excludes:2:!globaltwo	a/b/globaltwo | 
|  | $global_excludes:2:!globaltwo	b/globaltwo | 
|  | EOF | 
|  |  | 
|  | broken_c_unquote stdin >stdin0 | 
|  |  | 
|  | broken_c_unquote expected-default >expected-default0 | 
|  |  | 
|  | broken_c_unquote_verbose expected-verbose >expected-verbose0 | 
|  |  | 
|  | test_expect_success '--stdin' ' | 
|  | expect_from_stdin <expected-default && | 
|  | test_check_ignore "--stdin" <stdin | 
|  | ' | 
|  |  | 
|  | test_expect_success '--stdin -q' ' | 
|  | expect "" && | 
|  | test_check_ignore "-q --stdin" <stdin | 
|  | ' | 
|  |  | 
|  | test_expect_success '--stdin -v' ' | 
|  | expect_from_stdin <expected-verbose && | 
|  | test_check_ignore "-v --stdin" <stdin | 
|  | ' | 
|  |  | 
|  | for opts in '--stdin -z' '-z --stdin' | 
|  | do | 
|  | test_expect_success "$opts" " | 
|  | expect_from_stdin <expected-default0 && | 
|  | test_check_ignore '$opts' <stdin0 | 
|  | " | 
|  |  | 
|  | test_expect_success "$opts -q" " | 
|  | expect "" && | 
|  | test_check_ignore '-q $opts' <stdin0 | 
|  | " | 
|  |  | 
|  | test_expect_success "$opts -v" " | 
|  | expect_from_stdin <expected-verbose0 && | 
|  | test_check_ignore '-v $opts' <stdin0 | 
|  | " | 
|  | done | 
|  |  | 
|  | cat <<-\EOF >stdin | 
|  | ../one | 
|  | ../not-ignored | 
|  | one | 
|  | not-ignored | 
|  | b/on | 
|  | b/one | 
|  | b/one one | 
|  | "b/one two" | 
|  | "b/one\"three" | 
|  | b/two | 
|  | b/not-ignored | 
|  | b/twooo | 
|  | ../globaltwo | 
|  | globaltwo | 
|  | b/globaltwo | 
|  | ../b/globaltwo | 
|  | c/not-ignored | 
|  | EOF | 
|  | # N.B. we deliberately end STDIN with a non-matching pattern in order | 
|  | # to test that the exit code indicates that one or more of the | 
|  | # provided paths is ignored - in other words, that it represents an | 
|  | # aggregation of all the results, not just the final result. | 
|  |  | 
|  | cat <<-EOF >expected-all | 
|  | .gitignore:1:one	../one | 
|  | ::	../not-ignored | 
|  | .gitignore:1:one	one | 
|  | ::	not-ignored | 
|  | a/b/.gitignore:8:!on*	b/on | 
|  | a/b/.gitignore:8:!on*	b/one | 
|  | a/b/.gitignore:8:!on*	b/one one | 
|  | a/b/.gitignore:8:!on*	b/one two | 
|  | a/b/.gitignore:8:!on*	"b/one\\"three" | 
|  | a/b/.gitignore:9:!two	b/two | 
|  | ::	b/not-ignored | 
|  | a/.gitignore:1:two*	b/twooo | 
|  | $global_excludes:2:!globaltwo	../globaltwo | 
|  | $global_excludes:2:!globaltwo	globaltwo | 
|  | $global_excludes:2:!globaltwo	b/globaltwo | 
|  | $global_excludes:2:!globaltwo	../b/globaltwo | 
|  | ::	c/not-ignored | 
|  | EOF | 
|  | grep -v '^::	' expected-all >expected-verbose | 
|  | sed -e 's/.*	//' expected-verbose >expected-default | 
|  |  | 
|  | broken_c_unquote stdin >stdin0 | 
|  |  | 
|  | broken_c_unquote expected-default >expected-default0 | 
|  |  | 
|  | broken_c_unquote_verbose expected-verbose >expected-verbose0 | 
|  |  | 
|  | test_expect_success '--stdin from subdirectory' ' | 
|  | expect_from_stdin <expected-default && | 
|  | ( | 
|  | cd a && | 
|  | test_check_ignore "--stdin" <../stdin | 
|  | ) | 
|  | ' | 
|  |  | 
|  | test_expect_success '--stdin from subdirectory with -v' ' | 
|  | expect_from_stdin <expected-verbose && | 
|  | ( | 
|  | cd a && | 
|  | test_check_ignore "--stdin -v" <../stdin | 
|  | ) | 
|  | ' | 
|  |  | 
|  | test_expect_success '--stdin from subdirectory with -v -n' ' | 
|  | expect_from_stdin <expected-all && | 
|  | ( | 
|  | cd a && | 
|  | test_check_ignore "--stdin -v -n" <../stdin | 
|  | ) | 
|  | ' | 
|  |  | 
|  | for opts in '--stdin -z' '-z --stdin' | 
|  | do | 
|  | test_expect_success "$opts from subdirectory" ' | 
|  | expect_from_stdin <expected-default0 && | 
|  | ( | 
|  | cd a && | 
|  | test_check_ignore "'"$opts"'" <../stdin0 | 
|  | ) | 
|  | ' | 
|  |  | 
|  | test_expect_success "$opts from subdirectory with -v" ' | 
|  | expect_from_stdin <expected-verbose0 && | 
|  | ( | 
|  | cd a && | 
|  | test_check_ignore "'"$opts"' -v" <../stdin0 | 
|  | ) | 
|  | ' | 
|  | done | 
|  |  | 
|  | test_expect_success PIPE 'streaming support for --stdin' ' | 
|  | mkfifo in out && | 
|  | (git check-ignore -n -v --stdin <in >out &) && | 
|  |  | 
|  | # We cannot just "echo >in" because check-ignore would get EOF | 
|  | # after echo exited; instead we open the descriptor in our | 
|  | # shell, and then echo to the fd. We make sure to close it at | 
|  | # the end, so that the subprocess does get EOF and dies | 
|  | # properly. | 
|  | # | 
|  | # Similarly, we must keep "out" open so that check-ignore does | 
|  | # not ever get SIGPIPE trying to write to us. Not only would that | 
|  | # produce incorrect results, but then there would be no writer on the | 
|  | # other end of the pipe, and we would potentially block forever trying | 
|  | # to open it. | 
|  | exec 9>in && | 
|  | exec 8<out && | 
|  | test_when_finished "exec 9>&-" && | 
|  | test_when_finished "exec 8<&-" && | 
|  | echo >&9 one && | 
|  | read response <&8 && | 
|  | echo "$response" | grep "^\.gitignore:1:one	one" && | 
|  | echo >&9 two && | 
|  | read response <&8 && | 
|  | echo "$response" | grep "^::	two" | 
|  | ' | 
|  |  | 
|  | ############################################################################ | 
|  | # | 
|  | # test whitespace handling | 
|  |  | 
|  | test_expect_success 'trailing whitespace is ignored' ' | 
|  | mkdir whitespace && | 
|  | >whitespace/trailing && | 
|  | >whitespace/untracked && | 
|  | echo "whitespace/trailing   " >ignore && | 
|  | cat >expect <<EOF && | 
|  | whitespace/untracked | 
|  | EOF | 
|  | : >err.expect && | 
|  | git ls-files -o -X ignore whitespace >actual 2>err && | 
|  | test_cmp expect actual && | 
|  | test_cmp err.expect err | 
|  | ' | 
|  |  | 
|  | test_expect_success !MINGW 'quoting allows trailing whitespace' ' | 
|  | rm -rf whitespace && | 
|  | mkdir whitespace && | 
|  | >"whitespace/trailing  " && | 
|  | >whitespace/untracked && | 
|  | echo "whitespace/trailing\\ \\ " >ignore && | 
|  | echo whitespace/untracked >expect && | 
|  | : >err.expect && | 
|  | git ls-files -o -X ignore whitespace >actual 2>err && | 
|  | test_cmp expect actual && | 
|  | test_cmp err.expect err | 
|  | ' | 
|  |  | 
|  | test_expect_success !MINGW,!CYGWIN 'correct handling of backslashes' ' | 
|  | rm -rf whitespace && | 
|  | mkdir whitespace && | 
|  | >"whitespace/trailing 1  " && | 
|  | >"whitespace/trailing 2 \\\\" && | 
|  | >"whitespace/trailing 3 \\\\" && | 
|  | >"whitespace/trailing 4   \\ " && | 
|  | >"whitespace/trailing 5 \\ \\ " && | 
|  | >"whitespace/trailing 6 \\a\\" && | 
|  | >whitespace/untracked && | 
|  | sed -e "s/Z$//" >ignore <<-\EOF && | 
|  | whitespace/trailing 1 \    Z | 
|  | whitespace/trailing 2 \\\\Z | 
|  | whitespace/trailing 3 \\\\ Z | 
|  | whitespace/trailing 4   \\\    Z | 
|  | whitespace/trailing 5 \\ \\\   Z | 
|  | whitespace/trailing 6 \\a\\Z | 
|  | EOF | 
|  | echo whitespace/untracked >expect && | 
|  | >err.expect && | 
|  | git ls-files -o -X ignore whitespace >actual 2>err && | 
|  | test_cmp expect actual && | 
|  | test_cmp err.expect err | 
|  | ' | 
|  |  | 
|  | test_expect_success 'info/exclude trumps core.excludesfile' ' | 
|  | echo >>global-excludes usually-ignored && | 
|  | echo >>.git/info/exclude "!usually-ignored" && | 
|  | >usually-ignored && | 
|  | echo "?? usually-ignored" >expect && | 
|  |  | 
|  | git status --porcelain usually-ignored >actual && | 
|  | test_cmp expect actual | 
|  | ' | 
|  |  | 
|  | test_done |