| #!/bin/sh |
| |
| test_description='pre-commit and pre-merge-commit hooks' |
| |
| GIT_TEST_DEFAULT_INITIAL_BRANCH_NAME=main |
| export GIT_TEST_DEFAULT_INITIAL_BRANCH_NAME |
| |
| TEST_PASSES_SANITIZE_LEAK=true |
| . ./test-lib.sh |
| |
| test_expect_success 'root commit' ' |
| echo "root" >file && |
| git add file && |
| git commit -m "zeroth" && |
| git checkout -b side && |
| echo "foo" >foo && |
| git add foo && |
| git commit -m "make it non-ff" && |
| git branch side-orig side && |
| git checkout main |
| ' |
| |
| test_expect_success 'setup conflicting branches' ' |
| test_when_finished "git checkout main" && |
| git checkout -b conflicting-a main && |
| echo a >conflicting && |
| git add conflicting && |
| git commit -m conflicting-a && |
| git checkout -b conflicting-b main && |
| echo b >conflicting && |
| git add conflicting && |
| git commit -m conflicting-b |
| ' |
| |
| test_expect_success 'with no hook' ' |
| test_when_finished "rm -f actual_hooks" && |
| echo "foo" >file && |
| git add file && |
| git commit -m "first" && |
| test_path_is_missing actual_hooks |
| ' |
| |
| test_expect_success 'with no hook (merge)' ' |
| test_when_finished "rm -f actual_hooks" && |
| git branch -f side side-orig && |
| git checkout side && |
| git merge -m "merge main" main && |
| git checkout main && |
| test_path_is_missing actual_hooks |
| ' |
| |
| test_expect_success '--no-verify with no hook' ' |
| test_when_finished "rm -f actual_hooks" && |
| echo "bar" >file && |
| git add file && |
| git commit --no-verify -m "bar" && |
| test_path_is_missing actual_hooks |
| ' |
| |
| test_expect_success '--no-verify with no hook (merge)' ' |
| test_when_finished "rm -f actual_hooks" && |
| git branch -f side side-orig && |
| git checkout side && |
| git merge --no-verify -m "merge main" main && |
| git checkout main && |
| test_path_is_missing actual_hooks |
| ' |
| |
| setup_success_hook () { |
| test_when_finished "rm -f actual_hooks expected_hooks" && |
| echo "$1" >expected_hooks && |
| test_hook "$1" <<-EOF |
| echo $1 >>actual_hooks |
| EOF |
| } |
| |
| test_expect_success 'with succeeding hook' ' |
| setup_success_hook "pre-commit" && |
| echo "more" >>file && |
| git add file && |
| git commit -m "more" && |
| test_cmp expected_hooks actual_hooks |
| ' |
| |
| test_expect_success 'with succeeding hook (merge)' ' |
| setup_success_hook "pre-merge-commit" && |
| git checkout side && |
| git merge -m "merge main" main && |
| git checkout main && |
| test_cmp expected_hooks actual_hooks |
| ' |
| |
| test_expect_success 'automatic merge fails; both hooks are available' ' |
| setup_success_hook "pre-commit" && |
| setup_success_hook "pre-merge-commit" && |
| |
| git checkout conflicting-a && |
| test_must_fail git merge -m "merge conflicting-b" conflicting-b && |
| test_path_is_missing actual_hooks && |
| |
| echo "pre-commit" >expected_hooks && |
| echo a+b >conflicting && |
| git add conflicting && |
| git commit -m "resolve conflict" && |
| test_cmp expected_hooks actual_hooks |
| ' |
| |
| test_expect_success '--no-verify with succeeding hook' ' |
| setup_success_hook "pre-commit" && |
| echo "even more" >>file && |
| git add file && |
| git commit --no-verify -m "even more" && |
| test_path_is_missing actual_hooks |
| ' |
| |
| test_expect_success '--no-verify with succeeding hook (merge)' ' |
| setup_success_hook "pre-merge-commit" && |
| git branch -f side side-orig && |
| git checkout side && |
| git merge --no-verify -m "merge main" main && |
| git checkout main && |
| test_path_is_missing actual_hooks |
| ' |
| |
| setup_failing_hook () { |
| test_when_finished "rm -f actual_hooks" && |
| test_hook "$1" <<-EOF |
| echo $1-failing-hook >>actual_hooks |
| exit 1 |
| EOF |
| } |
| |
| test_expect_success 'with failing hook' ' |
| setup_failing_hook "pre-commit" && |
| test_when_finished "rm -f expected_hooks" && |
| echo "pre-commit-failing-hook" >expected_hooks && |
| |
| echo "another" >>file && |
| git add file && |
| test_must_fail git commit -m "another" && |
| test_cmp expected_hooks actual_hooks |
| ' |
| |
| test_expect_success '--no-verify with failing hook' ' |
| setup_failing_hook "pre-commit" && |
| echo "stuff" >>file && |
| git add file && |
| git commit --no-verify -m "stuff" && |
| test_path_is_missing actual_hooks |
| ' |
| |
| test_expect_success 'with failing hook (merge)' ' |
| setup_failing_hook "pre-merge-commit" && |
| echo "pre-merge-commit-failing-hook" >expected_hooks && |
| git checkout side && |
| test_must_fail git merge -m "merge main" main && |
| git checkout main && |
| test_cmp expected_hooks actual_hooks |
| ' |
| |
| test_expect_success '--no-verify with failing hook (merge)' ' |
| setup_failing_hook "pre-merge-commit" && |
| |
| git branch -f side side-orig && |
| git checkout side && |
| git merge --no-verify -m "merge main" main && |
| git checkout main && |
| test_path_is_missing actual_hooks |
| ' |
| |
| setup_non_exec_hook () { |
| test_when_finished "rm -f actual_hooks" && |
| test_hook "$1" <<-\EOF && |
| echo non-exec >>actual_hooks |
| exit 1 |
| EOF |
| test_hook --disable "$1" |
| } |
| |
| |
| test_expect_success POSIXPERM 'with non-executable hook' ' |
| setup_non_exec_hook "pre-commit" && |
| echo "content" >>file && |
| git add file && |
| git commit -m "content" && |
| test_path_is_missing actual_hooks |
| ' |
| |
| test_expect_success POSIXPERM '--no-verify with non-executable hook' ' |
| setup_non_exec_hook "pre-commit" && |
| echo "more content" >>file && |
| git add file && |
| git commit --no-verify -m "more content" && |
| test_path_is_missing actual_hooks |
| ' |
| |
| test_expect_success POSIXPERM 'with non-executable hook (merge)' ' |
| setup_non_exec_hook "pre-merge" && |
| git branch -f side side-orig && |
| git checkout side && |
| git merge -m "merge main" main && |
| git checkout main && |
| test_path_is_missing actual_hooks |
| ' |
| |
| test_expect_success POSIXPERM '--no-verify with non-executable hook (merge)' ' |
| setup_non_exec_hook "pre-merge" && |
| git branch -f side side-orig && |
| git checkout side && |
| git merge --no-verify -m "merge main" main && |
| git checkout main && |
| test_path_is_missing actual_hooks |
| ' |
| |
| setup_require_prefix_hook () { |
| test_when_finished "rm -f expected_hooks" && |
| echo require-prefix >expected_hooks && |
| test_hook pre-commit <<-\EOF |
| echo require-prefix >>actual_hooks |
| test $GIT_PREFIX = "success/" |
| EOF |
| } |
| |
| test_expect_success 'with hook requiring GIT_PREFIX' ' |
| test_when_finished "rm -rf actual_hooks success" && |
| setup_require_prefix_hook && |
| echo "more content" >>file && |
| git add file && |
| mkdir success && |
| ( |
| cd success && |
| git commit -m "hook requires GIT_PREFIX = success/" |
| ) && |
| test_cmp expected_hooks actual_hooks |
| ' |
| |
| test_expect_success 'with failing hook requiring GIT_PREFIX' ' |
| test_when_finished "rm -rf actual_hooks fail" && |
| setup_require_prefix_hook && |
| echo "more content" >>file && |
| git add file && |
| mkdir fail && |
| ( |
| cd fail && |
| test_must_fail git commit -m "hook must fail" |
| ) && |
| git checkout -- file && |
| test_cmp expected_hooks actual_hooks |
| ' |
| |
| setup_require_author_hook () { |
| test_when_finished "rm -f expected_hooks actual_hooks" && |
| echo check-author >expected_hooks && |
| test_hook pre-commit <<-\EOF |
| echo check-author >>actual_hooks |
| test "$GIT_AUTHOR_NAME" = "New Author" && |
| test "$GIT_AUTHOR_EMAIL" = "newauthor@example.com" |
| EOF |
| } |
| |
| |
| test_expect_success 'check the author in hook' ' |
| setup_require_author_hook && |
| cat >expected_hooks <<-EOF && |
| check-author |
| check-author |
| check-author |
| EOF |
| test_must_fail git commit --allow-empty -m "by a.u.thor" && |
| ( |
| GIT_AUTHOR_NAME="New Author" && |
| GIT_AUTHOR_EMAIL="newauthor@example.com" && |
| export GIT_AUTHOR_NAME GIT_AUTHOR_EMAIL && |
| git commit --allow-empty -m "by new.author via env" && |
| git show -s |
| ) && |
| git commit --author="New Author <newauthor@example.com>" \ |
| --allow-empty -m "by new.author via command line" && |
| git show -s && |
| test_cmp expected_hooks actual_hooks |
| ' |
| |
| test_done |