| #!/bin/sh |
| |
| test_description='test config file include directives' |
| TEST_PASSES_SANITIZE_LEAK=true |
| . ./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_expect_success 'onbranch with unborn branch' ' |
| test_when_finished "rm -rf repo" && |
| git init repo && |
| ( |
| cd repo && |
| git config set includeIf.onbranch:"*".path config.inc && |
| git config set -f .git/config.inc foo.bar baz && |
| git config get foo.bar |
| ) |
| ' |
| |
| test_expect_success 'onbranch with detached HEAD' ' |
| test_when_finished "rm -rf repo" && |
| git init repo && |
| ( |
| cd repo && |
| git config set "includeIf.onbranch:*.path" config.inc && |
| git config set -f .git/config.inc foo.bar baz && |
| test_commit initial && |
| git switch --detach HEAD && |
| test_must_fail git config get foo.bar |
| ) |
| ' |
| |
| test_expect_success 'onbranch without repository' ' |
| test_when_finished "rm -f .gitconfig config.inc" && |
| git config set -f .gitconfig "includeIf.onbranch:**.path" config.inc && |
| git config set -f config.inc foo.bar baz && |
| git config get foo.bar && |
| test_must_fail nongit git config get foo.bar |
| ' |
| |
| test_expect_success 'onbranch without repository but explicit nonexistent Git directory' ' |
| test_when_finished "rm -f .gitconfig config.inc" && |
| git config set -f .gitconfig "includeIf.onbranch:**.path" config.inc && |
| git config set -f config.inc foo.bar baz && |
| git config get foo.bar && |
| test_must_fail nongit git --git-dir=nonexistent config get foo.bar |
| ' |
| |
| test_done |