| #!/bin/sh |
| |
| test_description=check-ignore |
| |
| TEST_PASSES_SANITIZE_LEAK=true |
| TEST_CREATE_REPO_NO_TEMPLATE=1 |
| . ./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_cmp "$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_grep "$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 || return 1 |
| 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 |
| mkdir .git/info && |
| 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 'nested include of negated pattern' ' |
| expect "" && |
| test_check_ignore "a/b/one" 1 |
| ' |
| |
| test_expect_success 'nested include of negated pattern with -q' ' |
| expect "" && |
| test_check_ignore "-q a/b/one" 1 |
| ' |
| |
| test_expect_success 'nested include of negated pattern with -v' ' |
| expect "a/b/.gitignore:8:!on* a/b/one" && |
| test_check_ignore "-v a/b/one" 0 |
| ' |
| |
| test_expect_success 'nested include of negated pattern with -v -n' ' |
| expect "a/b/.gitignore:8:!on* a/b/one" && |
| test_check_ignore "-v -n a/b/one" 0 |
| ' |
| |
| ############################################################################ |
| # |
| # 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 |
| 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:1:per-repo per-repo |
| a/.gitignore:2:*three a/globalthree |
| .git/info/exclude:1: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 |
| 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:1:per-repo per-repo |
| $global_excludes:3:globalthree globalthree |
| a/.gitignore:2:*three a/globalthree |
| .git/info/exclude:1: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/twooo |
| 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 |
| cat <<-EOF >expected-default |
| ../one |
| one |
| b/twooo |
| EOF |
| grep -v '^:: ' expected-all >expected-verbose |
| |
| 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_expect_success 'existing file and directory' ' |
| test_when_finished "rm one" && |
| test_when_finished "rmdir top-level-dir" && |
| >one && |
| mkdir top-level-dir && |
| git check-ignore one top-level-dir >actual && |
| grep one actual && |
| grep top-level-dir actual |
| ' |
| |
| test_expect_success 'existing directory and file' ' |
| test_when_finished "rm one" && |
| test_when_finished "rmdir top-level-dir" && |
| >one && |
| mkdir top-level-dir && |
| git check-ignore top-level-dir one >actual && |
| grep one actual && |
| grep top-level-dir actual |
| ' |
| |
| test_expect_success 'exact prefix matching (with root)' ' |
| test_when_finished rm -r a && |
| mkdir -p a/git a/git-foo && |
| touch a/git/foo a/git-foo/bar && |
| echo /git/ >a/.gitignore && |
| git check-ignore a/git a/git/foo a/git-foo a/git-foo/bar >actual && |
| cat >expect <<-\EOF && |
| a/git |
| a/git/foo |
| EOF |
| test_cmp expect actual |
| ' |
| |
| test_expect_success 'exact prefix matching (without root)' ' |
| test_when_finished rm -r a && |
| mkdir -p a/git a/git-foo && |
| touch a/git/foo a/git-foo/bar && |
| echo git/ >a/.gitignore && |
| git check-ignore a/git a/git/foo a/git-foo a/git-foo/bar >actual && |
| cat >expect <<-\EOF && |
| a/git |
| a/git/foo |
| EOF |
| test_cmp expect actual |
| ' |
| |
| test_expect_success 'directories and ** matches' ' |
| cat >.gitignore <<-\EOF && |
| data/** |
| !data/**/ |
| !data/**/*.txt |
| EOF |
| git check-ignore file \ |
| data/file data/data1/file1 data/data1/file1.txt \ |
| data/data2/file2 data/data2/file2.txt >actual && |
| cat >expect <<-\EOF && |
| data/file |
| data/data1/file1 |
| data/data2/file2 |
| EOF |
| test_cmp expect actual |
| ' |
| |
| ############################################################################ |
| # |
| # 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 |
| git ls-files -o -X ignore whitespace >actual 2>err && |
| test_cmp expect actual && |
| test_must_be_empty 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 && |
| git ls-files -o -X ignore whitespace >actual 2>err && |
| test_cmp expect actual && |
| test_must_be_empty 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 && |
| git ls-files -o -X ignore whitespace >actual 2>err && |
| test_cmp expect actual && |
| test_must_be_empty 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_expect_success SYMLINKS 'set up ignore file for symlink tests' ' |
| echo "*" >ignore && |
| rm -f .gitignore .git/info/exclude |
| ' |
| |
| test_expect_success SYMLINKS 'symlinks respected in core.excludesFile' ' |
| test_when_finished "rm symlink" && |
| ln -s ignore symlink && |
| test_config core.excludesFile "$(pwd)/symlink" && |
| echo file >expect && |
| git check-ignore file >actual 2>err && |
| test_cmp expect actual && |
| test_must_be_empty err |
| ' |
| |
| test_expect_success SYMLINKS 'symlinks respected in info/exclude' ' |
| test_when_finished "rm .git/info/exclude" && |
| ln -s ../../ignore .git/info/exclude && |
| echo file >expect && |
| git check-ignore file >actual 2>err && |
| test_cmp expect actual && |
| test_must_be_empty err |
| ' |
| |
| test_expect_success SYMLINKS 'symlinks not respected in-tree' ' |
| test_when_finished "rm .gitignore" && |
| ln -s ignore .gitignore && |
| mkdir subdir && |
| ln -s ignore subdir/.gitignore && |
| test_must_fail git check-ignore subdir/file >actual 2>err && |
| test_must_be_empty actual && |
| test_grep "unable to access.*gitignore" err |
| ' |
| |
| test_expect_success EXPENSIVE 'large exclude file ignored in tree' ' |
| test_when_finished "rm .gitignore" && |
| dd if=/dev/zero of=.gitignore bs=101M count=1 && |
| git ls-files -o --exclude-standard 2>err && |
| echo "warning: ignoring excessively large pattern file: .gitignore" >expect && |
| test_cmp expect err |
| ' |
| |
| test_done |