|  | #!/bin/sh | 
|  |  | 
|  | test_description='pushing to a repository using the atomic push option' | 
|  |  | 
|  | . ./test-lib.sh | 
|  |  | 
|  | mk_repo_pair () { | 
|  | rm -rf workbench upstream && | 
|  | test_create_repo upstream && | 
|  | test_create_repo workbench && | 
|  | ( | 
|  | cd upstream && | 
|  | git config receive.denyCurrentBranch warn | 
|  | ) && | 
|  | ( | 
|  | cd workbench && | 
|  | git remote add up ../upstream | 
|  | ) | 
|  | } | 
|  |  | 
|  | # Compare the ref ($1) in upstream with a ref value from workbench ($2) | 
|  | # i.e. test_refs second HEAD@{2} | 
|  | test_refs () { | 
|  | test $# = 2 && | 
|  | git -C upstream rev-parse --verify "$1" >expect && | 
|  | git -C workbench rev-parse --verify "$2" >actual && | 
|  | test_cmp expect actual | 
|  | } | 
|  |  | 
|  | fmt_status_report () { | 
|  | sed -n \ | 
|  | -e "/^To / { s/   */ /g; p; }" \ | 
|  | -e "/^ ! / { s/   */ /g; p; }" | 
|  | } | 
|  |  | 
|  | test_expect_success 'atomic push works for a single branch' ' | 
|  | mk_repo_pair && | 
|  | ( | 
|  | cd workbench && | 
|  | test_commit one && | 
|  | git push --mirror up && | 
|  | test_commit two && | 
|  | git push --atomic up master | 
|  | ) && | 
|  | test_refs master master | 
|  | ' | 
|  |  | 
|  | test_expect_success 'atomic push works for two branches' ' | 
|  | mk_repo_pair && | 
|  | ( | 
|  | cd workbench && | 
|  | test_commit one && | 
|  | git branch second && | 
|  | git push --mirror up && | 
|  | test_commit two && | 
|  | git checkout second && | 
|  | test_commit three && | 
|  | git push --atomic up master second | 
|  | ) && | 
|  | test_refs master master && | 
|  | test_refs second second | 
|  | ' | 
|  |  | 
|  | test_expect_success 'atomic push works in combination with --mirror' ' | 
|  | mk_repo_pair && | 
|  | ( | 
|  | cd workbench && | 
|  | test_commit one && | 
|  | git checkout -b second && | 
|  | test_commit two && | 
|  | git push --atomic --mirror up | 
|  | ) && | 
|  | test_refs master master && | 
|  | test_refs second second | 
|  | ' | 
|  |  | 
|  | test_expect_success 'atomic push works in combination with --force' ' | 
|  | mk_repo_pair && | 
|  | ( | 
|  | cd workbench && | 
|  | test_commit one && | 
|  | git branch second master && | 
|  | test_commit two_a && | 
|  | git checkout second && | 
|  | test_commit two_b && | 
|  | test_commit three_b && | 
|  | test_commit four && | 
|  | git push --mirror up && | 
|  | # The actual test is below | 
|  | git checkout master && | 
|  | test_commit three_a && | 
|  | git checkout second && | 
|  | git reset --hard HEAD^ && | 
|  | git push --force --atomic up master second | 
|  | ) && | 
|  | test_refs master master && | 
|  | test_refs second second | 
|  | ' | 
|  |  | 
|  | # set up two branches where master can be pushed but second can not | 
|  | # (non-fast-forward). Since second can not be pushed the whole operation | 
|  | # will fail and leave master untouched. | 
|  | test_expect_success 'atomic push fails if one branch fails' ' | 
|  | mk_repo_pair && | 
|  | ( | 
|  | cd workbench && | 
|  | test_commit one && | 
|  | git checkout -b second master && | 
|  | test_commit two && | 
|  | test_commit three && | 
|  | test_commit four && | 
|  | git push --mirror up && | 
|  | git reset --hard HEAD~2 && | 
|  | test_commit five && | 
|  | git checkout master && | 
|  | test_commit six && | 
|  | test_must_fail git push --atomic --all up | 
|  | ) && | 
|  | test_refs master HEAD@{7} && | 
|  | test_refs second HEAD@{4} | 
|  | ' | 
|  |  | 
|  | test_expect_success 'atomic push fails if one tag fails remotely' ' | 
|  | # prepare the repo | 
|  | mk_repo_pair && | 
|  | ( | 
|  | cd workbench && | 
|  | test_commit one && | 
|  | git checkout -b second master && | 
|  | test_commit two && | 
|  | git push --mirror up | 
|  | ) && | 
|  | # a third party modifies the server side: | 
|  | ( | 
|  | cd upstream && | 
|  | git checkout second && | 
|  | git tag test_tag second | 
|  | ) && | 
|  | # see if we can now push both branches. | 
|  | ( | 
|  | cd workbench && | 
|  | git checkout master && | 
|  | test_commit three && | 
|  | git checkout second && | 
|  | test_commit four && | 
|  | git tag test_tag && | 
|  | test_must_fail git push --tags --atomic up master second | 
|  | ) && | 
|  | test_refs master HEAD@{3} && | 
|  | test_refs second HEAD@{1} | 
|  | ' | 
|  |  | 
|  | test_expect_success 'atomic push obeys update hook preventing a branch to be pushed' ' | 
|  | mk_repo_pair && | 
|  | ( | 
|  | cd workbench && | 
|  | test_commit one && | 
|  | git checkout -b second master && | 
|  | test_commit two && | 
|  | git push --mirror up | 
|  | ) && | 
|  | ( | 
|  | cd upstream && | 
|  | HOOKDIR="$(git rev-parse --git-dir)/hooks" && | 
|  | HOOK="$HOOKDIR/update" && | 
|  | mkdir -p "$HOOKDIR" && | 
|  | write_script "$HOOK" <<-\EOF | 
|  | # only allow update to master from now on | 
|  | test "$1" = "refs/heads/master" | 
|  | EOF | 
|  | ) && | 
|  | ( | 
|  | cd workbench && | 
|  | git checkout master && | 
|  | test_commit three && | 
|  | git checkout second && | 
|  | test_commit four && | 
|  | test_must_fail git push --atomic up master second | 
|  | ) && | 
|  | test_refs master HEAD@{3} && | 
|  | test_refs second HEAD@{1} | 
|  | ' | 
|  |  | 
|  | test_expect_success 'atomic push is not advertised if configured' ' | 
|  | mk_repo_pair && | 
|  | ( | 
|  | cd upstream && | 
|  | git config receive.advertiseatomic 0 | 
|  | ) && | 
|  | ( | 
|  | cd workbench && | 
|  | test_commit one && | 
|  | git push --mirror up && | 
|  | test_commit two && | 
|  | test_must_fail git push --atomic up master | 
|  | ) && | 
|  | test_refs master HEAD@{1} | 
|  | ' | 
|  |  | 
|  | # References in upstream : master(1) one(1) foo(1) | 
|  | # References in workbench: master(2)        foo(1) two(2) bar(2) | 
|  | # Atomic push            : master(2)               two(2) bar(2) | 
|  | test_expect_success 'atomic push reports (reject by update hook)' ' | 
|  | mk_repo_pair && | 
|  | ( | 
|  | cd workbench && | 
|  | test_commit one && | 
|  | git branch foo && | 
|  | git push up master one foo && | 
|  | git tag -d one | 
|  | ) && | 
|  | ( | 
|  | mkdir -p upstream/.git/hooks && | 
|  | cat >upstream/.git/hooks/update <<-EOF && | 
|  | #!/bin/sh | 
|  |  | 
|  | if test "\$1" = "refs/heads/bar" | 
|  | then | 
|  | echo >&2 "Pusing to branch bar is prohibited" | 
|  | exit 1 | 
|  | fi | 
|  | EOF | 
|  | chmod a+x upstream/.git/hooks/update | 
|  | ) && | 
|  | ( | 
|  | cd workbench && | 
|  | test_commit two && | 
|  | git branch bar | 
|  | ) && | 
|  | test_must_fail git -C workbench \ | 
|  | push --atomic up master two bar >out 2>&1 && | 
|  | fmt_status_report <out >actual && | 
|  | cat >expect <<-EOF && | 
|  | To ../upstream | 
|  | ! [remote rejected] master -> master (atomic push failure) | 
|  | ! [remote rejected] two -> two (atomic push failure) | 
|  | ! [remote rejected] bar -> bar (hook declined) | 
|  | EOF | 
|  | test_cmp expect actual | 
|  | ' | 
|  |  | 
|  | # References in upstream : master(1) one(1) foo(1) | 
|  | # References in workbench: master(2)        foo(1) two(2) bar(2) | 
|  | test_expect_success 'atomic push reports (mirror, but reject by update hook)' ' | 
|  | ( | 
|  | cd workbench && | 
|  | git remote remove up && | 
|  | git remote add up ../upstream | 
|  | ) && | 
|  | test_must_fail git -C workbench \ | 
|  | push --atomic --mirror up >out 2>&1 && | 
|  | fmt_status_report <out >actual && | 
|  | cat >expect <<-EOF && | 
|  | To ../upstream | 
|  | ! [remote rejected] master -> master (atomic push failure) | 
|  | ! [remote rejected] one (atomic push failure) | 
|  | ! [remote rejected] bar -> bar (hook declined) | 
|  | ! [remote rejected] two -> two (atomic push failure) | 
|  | EOF | 
|  | test_cmp expect actual | 
|  | ' | 
|  |  | 
|  | # References in upstream : master(2) one(1) foo(1) | 
|  | # References in workbench: master(1)        foo(1) two(2) bar(2) | 
|  | test_expect_success 'atomic push reports (reject by non-ff)' ' | 
|  | rm upstream/.git/hooks/update && | 
|  | ( | 
|  | cd workbench && | 
|  | git push up master && | 
|  | git reset --hard HEAD^ | 
|  | ) && | 
|  | test_must_fail git -C workbench \ | 
|  | push --atomic up master foo bar >out 2>&1 && | 
|  | fmt_status_report <out >actual && | 
|  | cat >expect <<-EOF && | 
|  | To ../upstream | 
|  | ! [rejected] master -> master (non-fast-forward) | 
|  | ! [rejected] bar -> bar (atomic push failed) | 
|  | EOF | 
|  | test_cmp expect actual | 
|  | ' | 
|  |  | 
|  | test_done |