#!/bin/sh test_description='compare & swap push force/delete safety' GIT_TEST_DEFAULT_INITIAL_BRANCH_NAME=main export GIT_TEST_DEFAULT_INITIAL_BRANCH_NAME . ./test-lib.sh setup_srcdst_basic () { rm -fr src dst && git clone --no-local . src && git clone --no-local src dst && ( cd src && git checkout HEAD^0 ) } # For tests with "--force-if-includes". setup_src_dup_dst () { rm -fr src dup dst && git init --bare dst && git clone --no-local dst src && git clone --no-local dst dup ( cd src && test_commit A && test_commit B && test_commit C && git push origin ) && ( cd dup && git fetch && git merge origin/main && git switch -c branch main~2 && test_commit D && test_commit E && git push origin --all ) && ( cd src && git switch main && git fetch --all && git branch branch --track origin/branch && git rebase origin/main ) && ( cd dup && git switch main && test_commit F && test_commit G && git switch branch && test_commit H && git push origin --all ) } test_expect_success setup ' # create template repository test_commit A && test_commit B && test_commit C ' test_expect_success 'push to update (protected)' ' setup_srcdst_basic && ( cd dst && test_commit D && test_must_fail git push --force-with-lease=main:main origin main 2>err && grep "stale info" err ) && git ls-remote . refs/heads/main >expect && git ls-remote src refs/heads/main >actual && test_cmp expect actual ' test_expect_success 'push to update (protected, forced)' ' setup_srcdst_basic && ( cd dst && test_commit D && git push --force --force-with-lease=main:main origin main 2>err && grep "forced update" err ) && git ls-remote dst refs/heads/main >expect && git ls-remote src refs/heads/main >actual && test_cmp expect actual ' test_expect_success 'push to update (protected, tracking)' ' setup_srcdst_basic && ( cd src && git checkout main && test_commit D && git checkout HEAD^0 ) && git ls-remote src refs/heads/main >expect && ( cd dst && test_commit E && git ls-remote . refs/remotes/origin/main >expect && test_must_fail git push --force-with-lease=main origin main && git ls-remote . refs/remotes/origin/main >actual && test_cmp expect actual ) && git ls-remote src refs/heads/main >actual && test_cmp expect actual ' test_expect_success 'push to update (protected, tracking, forced)' ' setup_srcdst_basic && ( cd src && git checkout main && test_commit D && git checkout HEAD^0 ) && ( cd dst && test_commit E && git ls-remote . refs/remotes/origin/main >expect && git push --force --force-with-lease=main origin main ) && git ls-remote dst refs/heads/main >expect && git ls-remote src refs/heads/main >actual && test_cmp expect actual ' test_expect_success 'push to update (allowed)' ' setup_srcdst_basic && ( cd dst && test_commit D && git push --force-with-lease=main:main^ origin main ) && git ls-remote dst refs/heads/main >expect && git ls-remote src refs/heads/main >actual && test_cmp expect actual ' test_expect_success 'push to update (allowed, tracking)' ' setup_srcdst_basic && ( cd dst && test_commit D && git push --force-with-lease=main origin main 2>err && ! grep "forced update" err ) && git ls-remote dst refs/heads/main >expect && git ls-remote src refs/heads/main >actual && test_cmp expect actual ' test_expect_success 'push to update (allowed even though no-ff)' ' setup_srcdst_basic && ( cd dst && git reset --hard HEAD^ && test_commit D && git push --force-with-lease=main origin main 2>err && grep "forced update" err ) && git ls-remote dst refs/heads/main >expect && git ls-remote src refs/heads/main >actual && test_cmp expect actual ' test_expect_success 'push to delete (protected)' ' setup_srcdst_basic && git ls-remote src refs/heads/main >expect && ( cd dst && test_must_fail git push --force-with-lease=main:main^ origin :main ) && git ls-remote src refs/heads/main >actual && test_cmp expect actual ' test_expect_success 'push to delete (protected, forced)' ' setup_srcdst_basic && ( cd dst && git push --force --force-with-lease=main:main^ origin :main ) && git ls-remote src refs/heads/main >actual && test_must_be_empty actual ' test_expect_success 'push to delete (allowed)' ' setup_srcdst_basic && ( cd dst && git push --force-with-lease=main origin :main 2>err && grep deleted err ) && git ls-remote src refs/heads/main >actual && test_must_be_empty actual ' test_expect_success 'cover everything with default force-with-lease (protected)' ' setup_srcdst_basic && ( cd src && git branch nain main^ ) && git ls-remote src refs/heads/\* >expect && ( cd dst && test_must_fail git push --force-with-lease origin main main:nain ) && git ls-remote src refs/heads/\* >actual && test_cmp expect actual ' test_expect_success 'cover everything with default force-with-lease (allowed)' ' setup_srcdst_basic && ( cd src && git branch nain main^ ) && ( cd dst && git fetch && git push --force-with-lease origin main main:nain ) && git ls-remote dst refs/heads/main | sed -e "s/main/nain/" >expect && git ls-remote src refs/heads/nain >actual && test_cmp expect actual ' test_expect_success 'new branch covered by force-with-lease' ' setup_srcdst_basic && ( cd dst && git branch branch main && git push --force-with-lease=branch origin branch ) && git ls-remote dst refs/heads/branch >expect && git ls-remote src refs/heads/branch >actual && test_cmp expect actual ' test_expect_success 'new branch covered by force-with-lease (explicit)' ' setup_srcdst_basic && ( cd dst && git branch branch main && git push --force-with-lease=branch: origin branch ) && git ls-remote dst refs/heads/branch >expect && git ls-remote src refs/heads/branch >actual && test_cmp expect actual ' test_expect_success 'new branch already exists' ' setup_srcdst_basic && ( cd src && git checkout -b branch main && test_commit F ) && ( cd dst && git branch branch main && test_must_fail git push --force-with-lease=branch: origin branch ) ' test_expect_success 'background updates of REMOTE can be mitigated with a non-updated REMOTE-push' ' rm -rf src dst && git init --bare src.bare && test_when_finished "rm -rf src.bare" && git clone --no-local src.bare dst && test_when_finished "rm -rf dst" && ( cd dst && test_commit G && git remote add origin-push ../src.bare && git push origin-push main:main ) && git clone --no-local src.bare dst2 && test_when_finished "rm -rf dst2" && ( cd dst2 && test_commit H && git push ) && ( cd dst && test_commit I && git fetch origin && test_must_fail git push --force-with-lease origin-push && git fetch origin-push && git push --force-with-lease origin-push ) ' test_expect_success 'background updates to remote can be mitigated with "--force-if-includes"' ' setup_src_dup_dst && test_when_finished "rm -fr dst src dup" && git ls-remote dst refs/heads/main >expect.main && git ls-remote dst refs/heads/branch >expect.branch && ( cd src && git switch branch && test_commit I && git switch main && test_commit J && git fetch --all && test_must_fail git push --force-with-lease --force-if-includes --all ) && git ls-remote dst refs/heads/main >actual.main && git ls-remote dst refs/heads/branch >actual.branch && test_cmp expect.main actual.main && test_cmp expect.branch actual.branch ' test_expect_success 'background updates to remote can be mitigated with "push.useForceIfIncludes"' ' setup_src_dup_dst && test_when_finished "rm -fr dst src dup" && git ls-remote dst refs/heads/main >expect.main && ( cd src && git switch branch && test_commit I && git switch main && test_commit J && git fetch --all && git config --local push.useForceIfIncludes true && test_must_fail git push --force-with-lease=main origin main ) && git ls-remote dst refs/heads/main >actual.main && test_cmp expect.main actual.main ' test_expect_success '"--force-if-includes" should be disabled for --force-with-lease="<refname>:<expect>"' ' setup_src_dup_dst && test_when_finished "rm -fr dst src dup" && git ls-remote dst refs/heads/main >expect.main && ( cd src && git switch branch && test_commit I && git switch main && test_commit J && remote_head="$(git rev-parse refs/remotes/origin/main)" && git fetch --all && test_must_fail git push --force-if-includes --force-with-lease="main:$remote_head" 2>err && grep "stale info" err ) && git ls-remote dst refs/heads/main >actual.main && test_cmp expect.main actual.main ' test_expect_success '"--force-if-includes" should allow forced update after a rebase ("pull --rebase")' ' setup_src_dup_dst && test_when_finished "rm -fr dst src dup" && ( cd src && git switch branch && test_commit I && git switch main && test_commit J && git pull --rebase origin main && git push --force-if-includes --force-with-lease="main" ) ' test_expect_success '"--force-if-includes" should allow forced update after a rebase ("pull --rebase", local rebase)' ' setup_src_dup_dst && test_when_finished "rm -fr dst src dup" && ( cd src && git switch branch && test_commit I && git switch main && test_commit J && git pull --rebase origin main && git rebase --onto HEAD~4 HEAD~1 && git push --force-if-includes --force-with-lease="main" ) ' test_expect_success '"--force-if-includes" should allow deletes' ' setup_src_dup_dst && test_when_finished "rm -fr dst src dup" && ( cd src && git switch branch && git pull --rebase origin branch && git push --force-if-includes --force-with-lease="branch" origin :branch ) ' test_done