2010-06-23 21:28:58 +02:00
|
|
|
#!/bin/sh
|
|
|
|
|
2019-12-27 14:47:10 +01:00
|
|
|
test_description='checkout'
|
2010-06-23 21:28:58 +02:00
|
|
|
|
|
|
|
. ./test-lib.sh
|
|
|
|
|
2020-01-07 05:53:02 +01:00
|
|
|
# Arguments: [!] <branch> <sha> [<checkout options>]
|
2010-06-23 21:28:58 +02:00
|
|
|
#
|
|
|
|
# Runs "git checkout" to switch to <branch>, testing that
|
|
|
|
#
|
|
|
|
# 1) we are on the specified branch, <branch>;
|
|
|
|
# 2) HEAD is <sha>; if <sha> is not specified, the old HEAD is used.
|
|
|
|
#
|
|
|
|
# If <checkout options> is not specified, "git checkout" is run with -b.
|
2020-01-07 05:53:02 +01:00
|
|
|
#
|
|
|
|
# If the first argument is `!`, "git checkout" is expected to fail when
|
|
|
|
# it is run.
|
2020-01-07 05:52:59 +01:00
|
|
|
do_checkout () {
|
2020-01-07 05:53:02 +01:00
|
|
|
should_fail= &&
|
|
|
|
if test "x$1" = "x!"
|
|
|
|
then
|
|
|
|
should_fail=yes &&
|
|
|
|
shift
|
|
|
|
fi &&
|
2010-06-23 21:28:58 +02:00
|
|
|
exp_branch=$1 &&
|
|
|
|
exp_ref="refs/heads/$exp_branch" &&
|
|
|
|
|
|
|
|
# if <sha> is not specified, use HEAD.
|
|
|
|
exp_sha=${2:-$(git rev-parse --verify HEAD)} &&
|
|
|
|
|
|
|
|
# default options for git checkout: -b
|
2020-01-07 05:53:00 +01:00
|
|
|
if test -z "$3"
|
|
|
|
then
|
2010-06-23 21:28:58 +02:00
|
|
|
opts="-b"
|
|
|
|
else
|
|
|
|
opts="$3"
|
|
|
|
fi
|
|
|
|
|
2020-01-07 05:53:02 +01:00
|
|
|
if test -n "$should_fail"
|
|
|
|
then
|
|
|
|
test_must_fail git checkout $opts $exp_branch $exp_sha
|
|
|
|
else
|
|
|
|
git checkout $opts $exp_branch $exp_sha &&
|
2020-01-07 05:53:03 +01:00
|
|
|
echo "$exp_ref" >ref.expect &&
|
|
|
|
git rev-parse --symbolic-full-name HEAD >ref.actual &&
|
|
|
|
test_cmp ref.expect ref.actual &&
|
|
|
|
echo "$exp_sha" >sha.expect &&
|
|
|
|
git rev-parse --verify HEAD >sha.actual &&
|
|
|
|
test_cmp sha.expect sha.actual
|
2020-01-07 05:53:02 +01:00
|
|
|
fi
|
2010-06-23 21:28:58 +02:00
|
|
|
}
|
|
|
|
|
2020-01-07 05:52:59 +01:00
|
|
|
test_dirty_unmergeable () {
|
2020-01-26 21:23:05 +01:00
|
|
|
test_expect_code 1 git diff --exit-code
|
|
|
|
}
|
|
|
|
|
|
|
|
test_dirty_unmergeable_discards_changes () {
|
|
|
|
git diff --exit-code
|
2010-06-23 21:28:58 +02:00
|
|
|
}
|
|
|
|
|
2020-01-07 05:52:59 +01:00
|
|
|
setup_dirty_unmergeable () {
|
2010-06-23 21:28:58 +02:00
|
|
|
echo >>file1 change2
|
|
|
|
}
|
|
|
|
|
2020-01-07 05:52:59 +01:00
|
|
|
test_dirty_mergeable () {
|
2020-01-26 21:23:05 +01:00
|
|
|
test_expect_code 1 git diff --cached --exit-code
|
|
|
|
}
|
|
|
|
|
|
|
|
test_dirty_mergeable_discards_changes () {
|
|
|
|
git diff --cached --exit-code
|
2010-06-23 21:28:58 +02:00
|
|
|
}
|
|
|
|
|
2020-01-07 05:52:59 +01:00
|
|
|
setup_dirty_mergeable () {
|
2010-06-23 21:28:58 +02:00
|
|
|
echo >file2 file2 &&
|
|
|
|
git add file2
|
|
|
|
}
|
|
|
|
|
|
|
|
test_expect_success 'setup' '
|
|
|
|
test_commit initial file1 &&
|
|
|
|
HEAD1=$(git rev-parse --verify HEAD) &&
|
|
|
|
|
|
|
|
test_commit change1 file1 &&
|
|
|
|
HEAD2=$(git rev-parse --verify HEAD) &&
|
|
|
|
|
|
|
|
git branch -m branch1
|
|
|
|
'
|
|
|
|
|
|
|
|
test_expect_success 'checkout -b to a new branch, set to HEAD' '
|
2019-04-27 14:02:19 +02:00
|
|
|
test_when_finished "
|
|
|
|
git checkout branch1 &&
|
|
|
|
test_might_fail git branch -D branch2" &&
|
2010-06-23 21:28:58 +02:00
|
|
|
do_checkout branch2
|
|
|
|
'
|
|
|
|
|
2019-04-27 14:02:22 +02:00
|
|
|
test_expect_success 'checkout -b to a merge base' '
|
|
|
|
test_when_finished "
|
|
|
|
git checkout branch1 &&
|
|
|
|
test_might_fail git branch -D branch2" &&
|
|
|
|
git checkout -b branch2 branch1...
|
|
|
|
'
|
|
|
|
|
2010-06-23 21:28:58 +02:00
|
|
|
test_expect_success 'checkout -b to a new branch, set to an explicit ref' '
|
2019-04-27 14:02:19 +02:00
|
|
|
test_when_finished "
|
|
|
|
git checkout branch1 &&
|
|
|
|
test_might_fail git branch -D branch2" &&
|
2010-06-23 21:28:58 +02:00
|
|
|
do_checkout branch2 $HEAD1
|
|
|
|
'
|
|
|
|
|
|
|
|
test_expect_success 'checkout -b to a new branch with unmergeable changes fails' '
|
|
|
|
setup_dirty_unmergeable &&
|
2020-01-07 05:53:02 +01:00
|
|
|
do_checkout ! branch2 $HEAD1 &&
|
2010-06-23 21:28:58 +02:00
|
|
|
test_dirty_unmergeable
|
|
|
|
'
|
|
|
|
|
|
|
|
test_expect_success 'checkout -f -b to a new branch with unmergeable changes discards changes' '
|
2019-04-27 14:02:19 +02:00
|
|
|
test_when_finished "
|
|
|
|
git checkout branch1 &&
|
|
|
|
test_might_fail git branch -D branch2" &&
|
|
|
|
|
2010-06-23 21:28:58 +02:00
|
|
|
# still dirty and on branch1
|
|
|
|
do_checkout branch2 $HEAD1 "-f -b" &&
|
2020-01-26 21:23:05 +01:00
|
|
|
test_dirty_unmergeable_discards_changes
|
2010-06-23 21:28:58 +02:00
|
|
|
'
|
|
|
|
|
|
|
|
test_expect_success 'checkout -b to a new branch preserves mergeable changes' '
|
2019-04-27 14:02:19 +02:00
|
|
|
test_when_finished "
|
|
|
|
git reset --hard &&
|
|
|
|
git checkout branch1 &&
|
|
|
|
test_might_fail git branch -D branch2" &&
|
2010-06-23 21:28:58 +02:00
|
|
|
|
|
|
|
setup_dirty_mergeable &&
|
|
|
|
do_checkout branch2 $HEAD1 &&
|
|
|
|
test_dirty_mergeable
|
|
|
|
'
|
|
|
|
|
|
|
|
test_expect_success 'checkout -f -b to a new branch with mergeable changes discards changes' '
|
2019-04-27 14:02:19 +02:00
|
|
|
test_when_finished git reset --hard HEAD &&
|
2010-06-23 21:28:58 +02:00
|
|
|
setup_dirty_mergeable &&
|
|
|
|
do_checkout branch2 $HEAD1 "-f -b" &&
|
2020-01-26 21:23:05 +01:00
|
|
|
test_dirty_mergeable_discards_changes
|
2010-06-23 21:28:58 +02:00
|
|
|
'
|
|
|
|
|
|
|
|
test_expect_success 'checkout -b to an existing branch fails' '
|
2019-04-27 14:02:19 +02:00
|
|
|
test_when_finished git reset --hard HEAD &&
|
2020-01-07 05:53:02 +01:00
|
|
|
do_checkout ! branch2 $HEAD2
|
2010-06-23 21:28:58 +02:00
|
|
|
'
|
|
|
|
|
2011-08-20 23:49:49 +02:00
|
|
|
test_expect_success 'checkout -b to @{-1} fails with the right branch name' '
|
|
|
|
git checkout branch1 &&
|
|
|
|
git checkout branch2 &&
|
|
|
|
echo >expect "fatal: A branch named '\''branch1'\'' already exists." &&
|
|
|
|
test_must_fail git checkout -b @{-1} 2>actual &&
|
2016-06-17 22:21:07 +02:00
|
|
|
test_i18ncmp expect actual
|
2011-08-20 23:49:49 +02:00
|
|
|
'
|
|
|
|
|
2010-06-23 21:29:00 +02:00
|
|
|
test_expect_success 'checkout -B to an existing branch resets branch to HEAD' '
|
|
|
|
git checkout branch1 &&
|
|
|
|
|
|
|
|
do_checkout branch2 "" -B
|
|
|
|
'
|
|
|
|
|
2019-04-27 14:02:22 +02:00
|
|
|
test_expect_success 'checkout -B to a merge base' '
|
|
|
|
git checkout branch1 &&
|
|
|
|
|
|
|
|
git checkout -B branch2 branch1...
|
|
|
|
'
|
|
|
|
|
2010-08-09 18:52:26 +02:00
|
|
|
test_expect_success 'checkout -B to an existing branch from detached HEAD resets branch to HEAD' '
|
2020-01-07 05:53:03 +01:00
|
|
|
head=$(git rev-parse --verify HEAD) &&
|
|
|
|
git checkout "$head" &&
|
2010-08-09 18:52:26 +02:00
|
|
|
|
|
|
|
do_checkout branch2 "" -B
|
|
|
|
'
|
|
|
|
|
2010-06-23 21:29:00 +02:00
|
|
|
test_expect_success 'checkout -B to an existing branch with an explicit ref resets branch to that ref' '
|
|
|
|
git checkout branch1 &&
|
|
|
|
|
|
|
|
do_checkout branch2 $HEAD1 -B
|
|
|
|
'
|
|
|
|
|
|
|
|
test_expect_success 'checkout -B to an existing branch with unmergeable changes fails' '
|
|
|
|
git checkout branch1 &&
|
|
|
|
|
|
|
|
setup_dirty_unmergeable &&
|
2020-01-07 05:53:02 +01:00
|
|
|
do_checkout ! branch2 $HEAD1 -B &&
|
2010-06-23 21:29:00 +02:00
|
|
|
test_dirty_unmergeable
|
|
|
|
'
|
|
|
|
|
|
|
|
test_expect_success 'checkout -f -B to an existing branch with unmergeable changes discards changes' '
|
|
|
|
# still dirty and on branch1
|
|
|
|
do_checkout branch2 $HEAD1 "-f -B" &&
|
2020-01-26 21:23:05 +01:00
|
|
|
test_dirty_unmergeable_discards_changes
|
2010-06-23 21:29:00 +02:00
|
|
|
'
|
|
|
|
|
|
|
|
test_expect_success 'checkout -B to an existing branch preserves mergeable changes' '
|
2019-04-27 14:02:19 +02:00
|
|
|
test_when_finished git reset --hard &&
|
2010-06-23 21:29:00 +02:00
|
|
|
git checkout branch1 &&
|
|
|
|
|
|
|
|
setup_dirty_mergeable &&
|
|
|
|
do_checkout branch2 $HEAD1 -B &&
|
|
|
|
test_dirty_mergeable
|
|
|
|
'
|
|
|
|
|
|
|
|
test_expect_success 'checkout -f -B to an existing branch with mergeable changes discards changes' '
|
|
|
|
git checkout branch1 &&
|
|
|
|
|
|
|
|
setup_dirty_mergeable &&
|
|
|
|
do_checkout branch2 $HEAD1 "-f -B" &&
|
2020-01-26 21:23:05 +01:00
|
|
|
test_dirty_mergeable_discards_changes
|
2010-06-23 21:29:00 +02:00
|
|
|
'
|
|
|
|
|
2011-06-06 07:17:04 +02:00
|
|
|
test_expect_success 'checkout -b <describe>' '
|
|
|
|
git tag -f -m "First commit" initial initial &&
|
|
|
|
git checkout -f change1 &&
|
|
|
|
name=$(git describe) &&
|
|
|
|
git checkout -b $name &&
|
|
|
|
git diff --exit-code change1 &&
|
|
|
|
echo "refs/heads/$name" >expect &&
|
|
|
|
git symbolic-ref HEAD >actual &&
|
|
|
|
test_cmp expect actual
|
|
|
|
'
|
|
|
|
|
2011-11-26 09:54:55 +01:00
|
|
|
test_expect_success 'checkout -B to the current branch works' '
|
2011-08-20 23:49:48 +02:00
|
|
|
git checkout branch1 &&
|
2011-11-26 09:54:55 +01:00
|
|
|
git checkout -B branch1-scratch &&
|
|
|
|
|
2011-08-20 23:49:48 +02:00
|
|
|
setup_dirty_mergeable &&
|
2011-11-26 09:54:55 +01:00
|
|
|
git checkout -B branch1-scratch initial &&
|
|
|
|
test_dirty_mergeable
|
2011-08-20 23:49:48 +02:00
|
|
|
'
|
|
|
|
|
2019-01-23 21:02:01 +01:00
|
|
|
test_expect_success 'checkout -b after clone --no-checkout does a checkout of HEAD' '
|
2019-01-23 21:02:00 +01:00
|
|
|
git init src &&
|
|
|
|
test_commit -C src a &&
|
|
|
|
rev="$(git -C src rev-parse HEAD)" &&
|
|
|
|
git clone --no-checkout src dest &&
|
|
|
|
git -C dest checkout "$rev" -b branch &&
|
|
|
|
test_path_is_file dest/a.t
|
|
|
|
'
|
|
|
|
|
2010-06-23 21:28:58 +02:00
|
|
|
test_done
|