|  | #!/bin/sh | 
|  |  | 
|  | test_description='test config file include directives' | 
|  | . ./test-lib.sh | 
|  |  | 
|  | # Force setup_explicit_git_dir() to run until the end. This is needed | 
|  | # by some tests to make sure real_path() is called on $GIT_DIR. The | 
|  | # caller needs to make sure git commands are run from a subdirectory | 
|  | # though or real_path() will not be called. | 
|  | force_setup_explicit_git_dir() { | 
|  | GIT_DIR="$(pwd)/.git" | 
|  | GIT_WORK_TREE="$(pwd)" | 
|  | export GIT_DIR GIT_WORK_TREE | 
|  | } | 
|  |  | 
|  | test_expect_success 'include file by absolute path' ' | 
|  | echo "[test]one = 1" >one && | 
|  | echo "[include]path = \"$(pwd)/one\"" >.gitconfig && | 
|  | echo 1 >expect && | 
|  | git config test.one >actual && | 
|  | test_cmp expect actual | 
|  | ' | 
|  |  | 
|  | test_expect_success 'include file by relative path' ' | 
|  | echo "[test]one = 1" >one && | 
|  | echo "[include]path = one" >.gitconfig && | 
|  | echo 1 >expect && | 
|  | git config test.one >actual && | 
|  | test_cmp expect actual | 
|  | ' | 
|  |  | 
|  | test_expect_success 'chained relative paths' ' | 
|  | mkdir subdir && | 
|  | echo "[test]three = 3" >subdir/three && | 
|  | echo "[include]path = three" >subdir/two && | 
|  | echo "[include]path = subdir/two" >.gitconfig && | 
|  | echo 3 >expect && | 
|  | git config test.three >actual && | 
|  | test_cmp expect actual | 
|  | ' | 
|  |  | 
|  | test_expect_success 'include paths get tilde-expansion' ' | 
|  | echo "[test]one = 1" >one && | 
|  | echo "[include]path = ~/one" >.gitconfig && | 
|  | echo 1 >expect && | 
|  | git config test.one >actual && | 
|  | test_cmp expect actual | 
|  | ' | 
|  |  | 
|  | test_expect_success 'include options can still be examined' ' | 
|  | echo "[test]one = 1" >one && | 
|  | echo "[include]path = one" >.gitconfig && | 
|  | echo one >expect && | 
|  | git config include.path >actual && | 
|  | test_cmp expect actual | 
|  | ' | 
|  |  | 
|  | test_expect_success 'listing includes option and expansion' ' | 
|  | echo "[test]one = 1" >one && | 
|  | echo "[include]path = one" >.gitconfig && | 
|  | cat >expect <<-\EOF && | 
|  | include.path=one | 
|  | test.one=1 | 
|  | EOF | 
|  | git config --list >actual.full && | 
|  | grep -v -e ^core -e ^extensions actual.full >actual && | 
|  | test_cmp expect actual | 
|  | ' | 
|  |  | 
|  | test_expect_success 'single file lookup does not expand includes by default' ' | 
|  | echo "[test]one = 1" >one && | 
|  | echo "[include]path = one" >.gitconfig && | 
|  | test_must_fail git config -f .gitconfig test.one && | 
|  | test_must_fail git config --global test.one && | 
|  | echo 1 >expect && | 
|  | git config --includes -f .gitconfig test.one >actual && | 
|  | test_cmp expect actual | 
|  | ' | 
|  |  | 
|  | test_expect_success 'single file list does not expand includes by default' ' | 
|  | echo "[test]one = 1" >one && | 
|  | echo "[include]path = one" >.gitconfig && | 
|  | echo "include.path=one" >expect && | 
|  | git config -f .gitconfig --list >actual && | 
|  | test_cmp expect actual | 
|  | ' | 
|  |  | 
|  | test_expect_success 'writing config file does not expand includes' ' | 
|  | echo "[test]one = 1" >one && | 
|  | echo "[include]path = one" >.gitconfig && | 
|  | git config test.two 2 && | 
|  | echo 2 >expect && | 
|  | git config --no-includes test.two >actual && | 
|  | test_cmp expect actual && | 
|  | test_must_fail git config --no-includes test.one | 
|  | ' | 
|  |  | 
|  | test_expect_success 'config modification does not affect includes' ' | 
|  | echo "[test]one = 1" >one && | 
|  | echo "[include]path = one" >.gitconfig && | 
|  | git config test.one 2 && | 
|  | echo 1 >expect && | 
|  | git config -f one test.one >actual && | 
|  | test_cmp expect actual && | 
|  | cat >expect <<-\EOF && | 
|  | 1 | 
|  | 2 | 
|  | EOF | 
|  | git config --get-all test.one >actual && | 
|  | test_cmp expect actual | 
|  | ' | 
|  |  | 
|  | test_expect_success 'missing include files are ignored' ' | 
|  | cat >.gitconfig <<-\EOF && | 
|  | [include]path = non-existent | 
|  | [test]value = yes | 
|  | EOF | 
|  | echo yes >expect && | 
|  | git config test.value >actual && | 
|  | test_cmp expect actual | 
|  | ' | 
|  |  | 
|  | test_expect_success 'absolute includes from command line work' ' | 
|  | echo "[test]one = 1" >one && | 
|  | echo 1 >expect && | 
|  | git -c include.path="$(pwd)/one" config test.one >actual && | 
|  | test_cmp expect actual | 
|  | ' | 
|  |  | 
|  | test_expect_success 'relative includes from command line fail' ' | 
|  | echo "[test]one = 1" >one && | 
|  | test_must_fail git -c include.path=one config test.one | 
|  | ' | 
|  |  | 
|  | test_expect_success 'absolute includes from blobs work' ' | 
|  | echo "[test]one = 1" >one && | 
|  | echo "[include]path=$(pwd)/one" >blob && | 
|  | blob=$(git hash-object -w blob) && | 
|  | echo 1 >expect && | 
|  | git config --blob=$blob test.one >actual && | 
|  | test_cmp expect actual | 
|  | ' | 
|  |  | 
|  | test_expect_success 'relative includes from blobs fail' ' | 
|  | echo "[test]one = 1" >one && | 
|  | echo "[include]path=one" >blob && | 
|  | blob=$(git hash-object -w blob) && | 
|  | test_must_fail git config --blob=$blob test.one | 
|  | ' | 
|  |  | 
|  | test_expect_success 'absolute includes from stdin work' ' | 
|  | echo "[test]one = 1" >one && | 
|  | echo 1 >expect && | 
|  | echo "[include]path=\"$(pwd)/one\"" | | 
|  | git config --file - test.one >actual && | 
|  | test_cmp expect actual | 
|  | ' | 
|  |  | 
|  | test_expect_success 'relative includes from stdin line fail' ' | 
|  | echo "[test]one = 1" >one && | 
|  | echo "[include]path=one" | | 
|  | test_must_fail git config --file - test.one | 
|  | ' | 
|  |  | 
|  | test_expect_success 'conditional include, both unanchored' ' | 
|  | git init foo && | 
|  | ( | 
|  | cd foo && | 
|  | echo "[includeIf \"gitdir:foo/\"]path=bar" >>.git/config && | 
|  | echo "[test]one=1" >.git/bar && | 
|  | echo 1 >expect && | 
|  | git config test.one >actual && | 
|  | test_cmp expect actual | 
|  | ) | 
|  | ' | 
|  |  | 
|  | test_expect_success 'conditional include, $HOME expansion' ' | 
|  | ( | 
|  | cd foo && | 
|  | echo "[includeIf \"gitdir:~/foo/\"]path=bar2" >>.git/config && | 
|  | echo "[test]two=2" >.git/bar2 && | 
|  | echo 2 >expect && | 
|  | git config test.two >actual && | 
|  | test_cmp expect actual | 
|  | ) | 
|  | ' | 
|  |  | 
|  | test_expect_success 'conditional include, full pattern' ' | 
|  | ( | 
|  | cd foo && | 
|  | echo "[includeIf \"gitdir:**/foo/**\"]path=bar3" >>.git/config && | 
|  | echo "[test]three=3" >.git/bar3 && | 
|  | echo 3 >expect && | 
|  | git config test.three >actual && | 
|  | test_cmp expect actual | 
|  | ) | 
|  | ' | 
|  |  | 
|  | test_expect_success 'conditional include, relative path' ' | 
|  | echo "[includeIf \"gitdir:./foo/.git\"]path=bar4" >>.gitconfig && | 
|  | echo "[test]four=4" >bar4 && | 
|  | ( | 
|  | cd foo && | 
|  | echo 4 >expect && | 
|  | git config test.four >actual && | 
|  | test_cmp expect actual | 
|  | ) | 
|  | ' | 
|  |  | 
|  | test_expect_success 'conditional include, both unanchored, icase' ' | 
|  | ( | 
|  | cd foo && | 
|  | echo "[includeIf \"gitdir/i:FOO/\"]path=bar5" >>.git/config && | 
|  | echo "[test]five=5" >.git/bar5 && | 
|  | echo 5 >expect && | 
|  | git config test.five >actual && | 
|  | test_cmp expect actual | 
|  | ) | 
|  | ' | 
|  |  | 
|  | test_expect_success 'conditional include, early config reading' ' | 
|  | ( | 
|  | cd foo && | 
|  | echo "[includeIf \"gitdir:foo/\"]path=bar6" >>.git/config && | 
|  | echo "[test]six=6" >.git/bar6 && | 
|  | echo 6 >expect && | 
|  | test-tool config read_early_config test.six >actual && | 
|  | test_cmp expect actual | 
|  | ) | 
|  | ' | 
|  |  | 
|  | test_expect_success 'conditional include with /**/' ' | 
|  | REPO=foo/bar/repo && | 
|  | git init $REPO && | 
|  | cat >>$REPO/.git/config <<-\EOF && | 
|  | [includeIf "gitdir:**/foo/**/bar/**"] | 
|  | path=bar7 | 
|  | EOF | 
|  | echo "[test]seven=7" >$REPO/.git/bar7 && | 
|  | echo 7 >expect && | 
|  | git -C $REPO config test.seven >actual && | 
|  | test_cmp expect actual | 
|  | ' | 
|  |  | 
|  | test_expect_success SYMLINKS 'conditional include, set up symlinked $HOME' ' | 
|  | mkdir real-home && | 
|  | ln -s real-home home && | 
|  | ( | 
|  | HOME="$TRASH_DIRECTORY/home" && | 
|  | export HOME && | 
|  | cd "$HOME" && | 
|  |  | 
|  | git init foo && | 
|  | cd foo && | 
|  | mkdir sub | 
|  | ) | 
|  | ' | 
|  |  | 
|  | test_expect_success SYMLINKS 'conditional include, $HOME expansion with symlinks' ' | 
|  | ( | 
|  | HOME="$TRASH_DIRECTORY/home" && | 
|  | export HOME && | 
|  | cd "$HOME"/foo && | 
|  |  | 
|  | echo "[includeIf \"gitdir:~/foo/\"]path=bar2" >>.git/config && | 
|  | echo "[test]two=2" >.git/bar2 && | 
|  | echo 2 >expect && | 
|  | force_setup_explicit_git_dir && | 
|  | git -C sub config test.two >actual && | 
|  | test_cmp expect actual | 
|  | ) | 
|  | ' | 
|  |  | 
|  | test_expect_success SYMLINKS 'conditional include, relative path with symlinks' ' | 
|  | echo "[includeIf \"gitdir:./foo/.git\"]path=bar4" >home/.gitconfig && | 
|  | echo "[test]four=4" >home/bar4 && | 
|  | ( | 
|  | HOME="$TRASH_DIRECTORY/home" && | 
|  | export HOME && | 
|  | cd "$HOME"/foo && | 
|  |  | 
|  | echo 4 >expect && | 
|  | force_setup_explicit_git_dir && | 
|  | git -C sub config test.four >actual && | 
|  | test_cmp expect actual | 
|  | ) | 
|  | ' | 
|  |  | 
|  | test_expect_success SYMLINKS 'conditional include, gitdir matching symlink' ' | 
|  | ln -s foo bar && | 
|  | ( | 
|  | cd bar && | 
|  | echo "[includeIf \"gitdir:bar/\"]path=bar7" >>.git/config && | 
|  | echo "[test]seven=7" >.git/bar7 && | 
|  | echo 7 >expect && | 
|  | git config test.seven >actual && | 
|  | test_cmp expect actual | 
|  | ) | 
|  | ' | 
|  |  | 
|  | test_expect_success SYMLINKS 'conditional include, gitdir matching symlink, icase' ' | 
|  | ( | 
|  | cd bar && | 
|  | echo "[includeIf \"gitdir/i:BAR/\"]path=bar8" >>.git/config && | 
|  | echo "[test]eight=8" >.git/bar8 && | 
|  | echo 8 >expect && | 
|  | git config test.eight >actual && | 
|  | test_cmp expect actual | 
|  | ) | 
|  | ' | 
|  |  | 
|  | test_expect_success 'conditional include, onbranch' ' | 
|  | echo "[includeIf \"onbranch:foo-branch\"]path=bar9" >>.git/config && | 
|  | echo "[test]nine=9" >.git/bar9 && | 
|  | git checkout -b main && | 
|  | test_must_fail git config test.nine && | 
|  | git checkout -b foo-branch && | 
|  | echo 9 >expect && | 
|  | git config test.nine >actual && | 
|  | test_cmp expect actual | 
|  | ' | 
|  |  | 
|  | test_expect_success 'conditional include, onbranch, wildcard' ' | 
|  | echo "[includeIf \"onbranch:?oo-*/**\"]path=bar10" >>.git/config && | 
|  | echo "[test]ten=10" >.git/bar10 && | 
|  | git checkout -b not-foo-branch/a && | 
|  | test_must_fail git config test.ten && | 
|  |  | 
|  | echo 10 >expect && | 
|  | git checkout -b foo-branch/a/b/c && | 
|  | git config test.ten >actual && | 
|  | test_cmp expect actual && | 
|  |  | 
|  | git checkout -b moo-bar/a && | 
|  | git config test.ten >actual && | 
|  | test_cmp expect actual | 
|  | ' | 
|  |  | 
|  | test_expect_success 'conditional include, onbranch, implicit /** for /' ' | 
|  | echo "[includeIf \"onbranch:foo-dir/\"]path=bar11" >>.git/config && | 
|  | echo "[test]eleven=11" >.git/bar11 && | 
|  | git checkout -b not-foo-dir/a && | 
|  | test_must_fail git config test.eleven && | 
|  |  | 
|  | echo 11 >expect && | 
|  | git checkout -b foo-dir/a/b/c && | 
|  | git config test.eleven >actual && | 
|  | test_cmp expect actual | 
|  | ' | 
|  |  | 
|  | test_expect_success 'include cycles are detected' ' | 
|  | git init --bare cycle && | 
|  | git -C cycle config include.path cycle && | 
|  | git config -f cycle/cycle include.path config && | 
|  | test_must_fail git -C cycle config --get-all test.value 2>stderr && | 
|  | grep "exceeded maximum include depth" stderr | 
|  | ' | 
|  |  | 
|  | test_done |