#!/bin/sh # # Copyright (c) 2012 Avery Pennaraum # Copyright (c) 2015 Alexey Shumkin # test_description='Basic porcelain support for subtrees This test verifies the basic operation of the add, pull, merge and split subcommands of git subtree. ' TEST_DIRECTORY=$(pwd)/../../../t export TEST_DIRECTORY . ../../../t/test-lib.sh subtree_test_create_repo() { test_create_repo "$1" ( cd $1 git config log.date relative ) } create() { echo "$1" >"$1" git add "$1" } check_equal() { test_debug 'echo' test_debug "echo \"check a:\" \"{$1}\"" test_debug "echo \" b:\" \"{$2}\"" if [ "$1" = "$2" ]; then return 0 else return 1 fi } undo() { git reset --hard HEAD~ } # Make sure no patch changes more than one file. # The original set of commits changed only one file each. # A multi-file change would imply that we pruned commits # too aggressively. join_commits() { commit= all= while read x y; do if [ -z "$x" ]; then continue elif [ "$x" = "commit:" ]; then if [ -n "$commit" ]; then echo "$commit $all" all= fi commit="$y" else all="$all $y" fi done echo "$commit $all" } test_create_commit() ( repo=$1 commit=$2 cd "$repo" mkdir -p $(dirname "$commit") \ || error "Could not create directory for commit" echo "$commit" >"$commit" git add "$commit" || error "Could not add commit" git commit -m "$commit" || error "Could not commit" ) last_commit_message() { git log --pretty=format:%s -1 } subtree_test_count=0 next_test() { subtree_test_count=$(($subtree_test_count+1)) } # # Tests for 'git subtree add' # next_test test_expect_success 'no merge from non-existent subtree' ' subtree_test_create_repo "$subtree_test_count" && subtree_test_create_repo "$subtree_test_count/sub proj" && test_create_commit "$subtree_test_count" main1 && test_create_commit "$subtree_test_count/sub proj" sub1 && ( cd "$subtree_test_count" && git fetch ./"sub proj" master && test_must_fail git subtree merge --prefix="sub dir" FETCH_HEAD ) ' next_test test_expect_success 'no pull from non-existent subtree' ' subtree_test_create_repo "$subtree_test_count" && subtree_test_create_repo "$subtree_test_count/sub proj" && test_create_commit "$subtree_test_count" main1 && test_create_commit "$subtree_test_count/sub proj" sub1 && ( cd "$subtree_test_count" && git fetch ./"sub proj" master && test_must_fail git subtree pull --prefix="sub dir" ./"sub proj" master )' next_test test_expect_success 'add subproj as subtree into sub dir/ with --prefix' ' subtree_test_create_repo "$subtree_test_count" && subtree_test_create_repo "$subtree_test_count/sub proj" && test_create_commit "$subtree_test_count" main1 && test_create_commit "$subtree_test_count/sub proj" sub1 && ( cd "$subtree_test_count" && git fetch ./"sub proj" master && git subtree add --prefix="sub dir" FETCH_HEAD && check_equal "$(last_commit_message)" "Add '\''sub dir/'\'' from commit '\''$(git rev-parse FETCH_HEAD)'\''" ) ' next_test test_expect_success 'add subproj as subtree into sub dir/ with --prefix and --message' ' subtree_test_create_repo "$subtree_test_count" && subtree_test_create_repo "$subtree_test_count/sub proj" && test_create_commit "$subtree_test_count" main1 && test_create_commit "$subtree_test_count/sub proj" sub1 && ( cd "$subtree_test_count" && git fetch ./"sub proj" master && git subtree add --prefix="sub dir" --message="Added subproject" FETCH_HEAD && check_equal "$(last_commit_message)" "Added subproject" ) ' next_test test_expect_success 'add subproj as subtree into sub dir/ with --prefix as -P and --message as -m' ' subtree_test_create_repo "$subtree_test_count" && subtree_test_create_repo "$subtree_test_count/sub proj" && test_create_commit "$subtree_test_count" main1 && test_create_commit "$subtree_test_count/sub proj" sub1 && ( cd "$subtree_test_count" && git fetch ./"sub proj" master && git subtree add -P "sub dir" -m "Added subproject" FETCH_HEAD && check_equal "$(last_commit_message)" "Added subproject" ) ' next_test test_expect_success 'add subproj as subtree into sub dir/ with --squash and --prefix and --message' ' subtree_test_create_repo "$subtree_test_count" && subtree_test_create_repo "$subtree_test_count/sub proj" && test_create_commit "$subtree_test_count" main1 && test_create_commit "$subtree_test_count/sub proj" sub1 && ( cd "$subtree_test_count" && git fetch ./"sub proj" master && git subtree add --prefix="sub dir" --message="Added subproject with squash" --squash FETCH_HEAD && check_equal "$(last_commit_message)" "Added subproject with squash" ) ' # # Tests for 'git subtree merge' # next_test test_expect_success 'merge new subproj history into sub dir/ with --prefix' ' subtree_test_create_repo "$subtree_test_count" && subtree_test_create_repo "$subtree_test_count/sub proj" && test_create_commit "$subtree_test_count" main1 && test_create_commit "$subtree_test_count/sub proj" sub1 && ( cd "$subtree_test_count" && git fetch ./"sub proj" master && git subtree add --prefix="sub dir" FETCH_HEAD ) && test_create_commit "$subtree_test_count/sub proj" sub2 && ( cd "$subtree_test_count" && git fetch ./"sub proj" master && git subtree merge --prefix="sub dir" FETCH_HEAD && check_equal "$(last_commit_message)" "Merge commit '\''$(git rev-parse FETCH_HEAD)'\''" ) ' next_test test_expect_success 'merge new subproj history into sub dir/ with --prefix and --message' ' subtree_test_create_repo "$subtree_test_count" && subtree_test_create_repo "$subtree_test_count/sub proj" && test_create_commit "$subtree_test_count" main1 && test_create_commit "$subtree_test_count/sub proj" sub1 && ( cd "$subtree_test_count" && git fetch ./"sub proj" master && git subtree add --prefix="sub dir" FETCH_HEAD ) && test_create_commit "$subtree_test_count/sub proj" sub2 && ( cd "$subtree_test_count" && git fetch ./"sub proj" master && git subtree merge --prefix="sub dir" --message="Merged changes from subproject" FETCH_HEAD && check_equal "$(last_commit_message)" "Merged changes from subproject" ) ' next_test test_expect_success 'merge new subproj history into sub dir/ with --squash and --prefix and --message' ' subtree_test_create_repo "$subtree_test_count/sub proj" && subtree_test_create_repo "$subtree_test_count" && test_create_commit "$subtree_test_count" main1 && test_create_commit "$subtree_test_count/sub proj" sub1 && ( cd "$subtree_test_count" && git fetch ./"sub proj" master && git subtree add --prefix="sub dir" FETCH_HEAD ) && test_create_commit "$subtree_test_count/sub proj" sub2 && ( cd "$subtree_test_count" && git fetch ./"sub proj" master && git subtree merge --prefix="sub dir" --message="Merged changes from subproject using squash" --squash FETCH_HEAD && check_equal "$(last_commit_message)" "Merged changes from subproject using squash" ) ' next_test test_expect_success 'merge the added subproj again, should do nothing' ' subtree_test_create_repo "$subtree_test_count" && subtree_test_create_repo "$subtree_test_count/sub proj" && test_create_commit "$subtree_test_count" main1 && test_create_commit "$subtree_test_count/sub proj" sub1 && ( cd "$subtree_test_count" && git fetch ./"sub proj" master && git subtree add --prefix="sub dir" FETCH_HEAD && # this shouldn not actually do anything, since FETCH_HEAD # is already a parent result=$(git merge -s ours -m "merge -s -ours" FETCH_HEAD) && check_equal "${result}" "Already up-to-date." ) ' next_test test_expect_success 'merge new subproj history into subdir/ with a slash appended to the argument of --prefix' ' test_create_repo "$test_count" && test_create_repo "$test_count/subproj" && test_create_commit "$test_count" main1 && test_create_commit "$test_count/subproj" sub1 && ( cd "$test_count" && git fetch ./subproj master && git subtree add --prefix=subdir/ FETCH_HEAD ) && test_create_commit "$test_count/subproj" sub2 && ( cd "$test_count" && git fetch ./subproj master && git subtree merge --prefix=subdir/ FETCH_HEAD && check_equal "$(last_commit_message)" "Merge commit '\''$(git rev-parse FETCH_HEAD)'\''" ) ' # # Tests for 'git subtree split' # next_test test_expect_success 'split requires option --prefix' ' subtree_test_create_repo "$subtree_test_count" && subtree_test_create_repo "$subtree_test_count/sub proj" && test_create_commit "$subtree_test_count" main1 && test_create_commit "$subtree_test_count/sub proj" sub1 && ( cd "$subtree_test_count" && git fetch ./"sub proj" master && git subtree add --prefix="sub dir" FETCH_HEAD && echo "You must provide the --prefix option." > expected && test_must_fail git subtree split > actual 2>&1 && test_debug "printf '"expected: "'" && test_debug "cat expected" && test_debug "printf '"actual: "'" && test_debug "cat actual" && test_cmp expected actual ) ' next_test test_expect_success 'split requires path given by option --prefix must exist' ' subtree_test_create_repo "$subtree_test_count" && subtree_test_create_repo "$subtree_test_count/sub proj" && test_create_commit "$subtree_test_count" main1 && test_create_commit "$subtree_test_count/sub proj" sub1 && ( cd "$subtree_test_count" && git fetch ./"sub proj" master && git subtree add --prefix="sub dir" FETCH_HEAD && echo "'\''non-existent-directory'\'' does not exist; use '\''git subtree add'\''" > expected && test_must_fail git subtree split --prefix=non-existent-directory > actual 2>&1 && test_debug "printf '"expected: "'" && test_debug "cat expected" && test_debug "printf '"actual: "'" && test_debug "cat actual" && test_cmp expected actual ) ' next_test test_expect_success 'split sub dir/ with --rejoin' ' subtree_test_create_repo "$subtree_test_count" && subtree_test_create_repo "$subtree_test_count/sub proj" && test_create_commit "$subtree_test_count" main1 && test_create_commit "$subtree_test_count/sub proj" sub1 && ( cd "$subtree_test_count" && git fetch ./"sub proj" master && git subtree add --prefix="sub dir" FETCH_HEAD ) && test_create_commit "$subtree_test_count" "sub dir"/main-sub1 && test_create_commit "$subtree_test_count" main2 && test_create_commit "$subtree_test_count/sub proj" sub2 && test_create_commit "$subtree_test_count" "sub dir"/main-sub2 && ( cd "$subtree_test_count" && git fetch ./"sub proj" master && git subtree merge --prefix="sub dir" FETCH_HEAD && split_hash=$(git subtree split --prefix="sub dir" --annotate="*") && git subtree split --prefix="sub dir" --annotate="*" --rejoin && check_equal "$(last_commit_message)" "Split '\''sub dir/'\'' into commit '\''$split_hash'\''" ) ' next_test test_expect_success 'split sub dir/ with --rejoin and --message' ' subtree_test_create_repo "$subtree_test_count" && subtree_test_create_repo "$subtree_test_count/sub proj" && test_create_commit "$subtree_test_count" main1 && test_create_commit "$subtree_test_count/sub proj" sub1 && ( cd "$subtree_test_count" && git fetch ./"sub proj" master && git subtree add --prefix="sub dir" FETCH_HEAD ) && test_create_commit "$subtree_test_count" "sub dir"/main-sub1 && test_create_commit "$subtree_test_count" main2 && test_create_commit "$subtree_test_count/sub proj" sub2 && test_create_commit "$subtree_test_count" "sub dir"/main-sub2 && ( cd "$subtree_test_count" && git fetch ./"sub proj" master && git subtree merge --prefix="sub dir" FETCH_HEAD && git subtree split --prefix="sub dir" --message="Split & rejoin" --annotate="*" --rejoin && check_equal "$(last_commit_message)" "Split & rejoin" ) ' next_test test_expect_success 'split "sub dir"/ with --branch' ' subtree_test_create_repo "$subtree_test_count" && subtree_test_create_repo "$subtree_test_count/sub proj" && test_create_commit "$subtree_test_count" main1 && test_create_commit "$subtree_test_count/sub proj" sub1 && ( cd "$subtree_test_count" && git fetch ./"sub proj" master && git subtree add --prefix="sub dir" FETCH_HEAD ) && test_create_commit "$subtree_test_count" "sub dir"/main-sub1 && test_create_commit "$subtree_test_count" main2 && test_create_commit "$subtree_test_count/sub proj" sub2 && test_create_commit "$subtree_test_count" "sub dir"/main-sub2 && ( cd "$subtree_test_count" && git fetch ./"sub proj" master && git subtree merge --prefix="sub dir" FETCH_HEAD && split_hash=$(git subtree split --prefix="sub dir" --annotate="*") && git subtree split --prefix="sub dir" --annotate="*" --branch subproj-br && check_equal "$(git rev-parse subproj-br)" "$split_hash" ) ' next_test test_expect_success 'check hash of split' ' subtree_test_create_repo "$subtree_test_count" && subtree_test_create_repo "$subtree_test_count/sub proj" && test_create_commit "$subtree_test_count" main1 && test_create_commit "$subtree_test_count/sub proj" sub1 && ( cd "$subtree_test_count" && git fetch ./"sub proj" master && git subtree add --prefix="sub dir" FETCH_HEAD ) && test_create_commit "$subtree_test_count" "sub dir"/main-sub1 && test_create_commit "$subtree_test_count" main2 && test_create_commit "$subtree_test_count/sub proj" sub2 && test_create_commit "$subtree_test_count" "sub dir"/main-sub2 && ( cd "$subtree_test_count" && git fetch ./"sub proj" master && git subtree merge --prefix="sub dir" FETCH_HEAD && split_hash=$(git subtree split --prefix="sub dir" --annotate="*") && git subtree split --prefix="sub dir" --annotate="*" --branch subproj-br && check_equal "$(git rev-parse subproj-br)" "$split_hash" && # Check hash of split new_hash=$(git rev-parse subproj-br^2) && ( cd ./"sub proj" && subdir_hash=$(git rev-parse HEAD) && check_equal ''"$new_hash"'' "$subdir_hash" ) ) ' next_test test_expect_success 'split "sub dir"/ with --branch for an existing branch' ' subtree_test_create_repo "$subtree_test_count" && subtree_test_create_repo "$subtree_test_count/sub proj" && test_create_commit "$subtree_test_count" main1 && test_create_commit "$subtree_test_count/sub proj" sub1 && ( cd "$subtree_test_count" && git fetch ./"sub proj" master && git branch subproj-br FETCH_HEAD && git subtree add --prefix="sub dir" FETCH_HEAD ) && test_create_commit "$subtree_test_count" "sub dir"/main-sub1 && test_create_commit "$subtree_test_count" main2 && test_create_commit "$subtree_test_count/sub proj" sub2 && test_create_commit "$subtree_test_count" "sub dir"/main-sub2 && ( cd "$subtree_test_count" && git fetch ./"sub proj" master && git subtree merge --prefix="sub dir" FETCH_HEAD && split_hash=$(git subtree split --prefix="sub dir" --annotate="*") && git subtree split --prefix="sub dir" --annotate="*" --branch subproj-br && check_equal "$(git rev-parse subproj-br)" "$split_hash" ) ' next_test test_expect_success 'split "sub dir"/ with --branch for an incompatible branch' ' subtree_test_create_repo "$subtree_test_count" && subtree_test_create_repo "$subtree_test_count/sub proj" && test_create_commit "$subtree_test_count" main1 && test_create_commit "$subtree_test_count/sub proj" sub1 && ( cd "$subtree_test_count" && git branch init HEAD && git fetch ./"sub proj" master && git subtree add --prefix="sub dir" FETCH_HEAD ) && test_create_commit "$subtree_test_count" "sub dir"/main-sub1 && test_create_commit "$subtree_test_count" main2 && test_create_commit "$subtree_test_count/sub proj" sub2 && test_create_commit "$subtree_test_count" "sub dir"/main-sub2 && ( cd "$subtree_test_count" && git fetch ./"sub proj" master && git subtree merge --prefix="sub dir" FETCH_HEAD && test_must_fail git subtree split --prefix="sub dir" --branch init ) ' # # Validity checking # next_test test_expect_success 'make sure exactly the right set of files ends up in the subproj' ' subtree_test_create_repo "$subtree_test_count" && subtree_test_create_repo "$subtree_test_count/sub proj" && test_create_commit "$subtree_test_count" main1 && test_create_commit "$subtree_test_count/sub proj" sub1 && ( cd "$subtree_test_count" && git fetch ./"sub proj" master && git subtree add --prefix="sub dir" FETCH_HEAD ) && test_create_commit "$subtree_test_count" "sub dir"/main-sub1 && test_create_commit "$subtree_test_count" main2 && test_create_commit "$subtree_test_count/sub proj" sub2 && test_create_commit "$subtree_test_count" "sub dir"/main-sub2 && ( cd "$subtree_test_count" && git fetch ./"sub proj" master && git subtree merge --prefix="sub dir" FETCH_HEAD && git subtree split --prefix="sub dir" --annotate="*" --branch subproj-br --rejoin ) && test_create_commit "$subtree_test_count/sub proj" sub3 && test_create_commit "$subtree_test_count" "sub dir"/main-sub3 && ( cd "$subtree_test_count/sub proj" && git fetch .. subproj-br && git merge FETCH_HEAD ) && test_create_commit "$subtree_test_count/sub proj" sub4 && ( cd "$subtree_test_count" && git subtree split --prefix="sub dir" --annotate="*" --branch subproj-br --rejoin ) && test_create_commit "$subtree_test_count" "sub dir"/main-sub4 && ( cd "$subtree_test_count" && git subtree split --prefix="sub dir" --annotate="*" --branch subproj-br --rejoin ) && ( cd "$subtree_test_count/sub proj" && git fetch .. subproj-br && git merge FETCH_HEAD && chks="sub1 sub2 sub3 sub4" && chks_sub=$(cat <