#!/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 merge, pull, add and split subcommands of git subtree. ' TEST_DIRECTORY=$(pwd)/../../../t export TEST_DIRECTORY . ../../../t/test-lib.sh 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~ } last_commit_message() { git log --pretty=format:%s -1 } test_expect_success 'init subproj' ' test_create_repo "sub proj" ' # To the subproject! cd ./"sub proj" test_expect_success 'add sub1' ' create sub1 && git commit -m "sub1" && git branch sub1 && git branch -m master subproj ' # Save this hash for testing later. subdir_hash=$(git rev-parse HEAD) test_expect_success 'add sub2' ' create sub2 && git commit -m "sub2" && git branch sub2 ' test_expect_success 'add sub3' ' create sub3 && git commit -m "sub3" && git branch sub3 ' # Back to mainline cd .. test_expect_success 'enable log.date=relative to catch errors' ' git config log.date relative ' test_expect_success 'add main4' ' create main4 && git commit -m "main4" && git branch -m master mainline && git branch subdir ' test_expect_success 'fetch subproj history' ' git fetch ./"sub proj" sub1 && git branch sub1 FETCH_HEAD ' test_expect_success 'no subtree exists in main tree' ' test_must_fail git subtree merge --prefix="sub dir" sub1 ' test_expect_success 'no pull from non-existant subtree' ' test_must_fail git subtree pull --prefix="sub dir" ./"sub proj" sub1 ' test_expect_success 'check if --message works for add' ' git subtree add --prefix="sub dir" --message="Added subproject" sub1 && check_equal ''"$(last_commit_message)"'' "Added subproject" && undo ' test_expect_success 'check if --message works as -m and --prefix as -P' ' git subtree add -P "sub dir" -m "Added subproject using git subtree" sub1 && check_equal ''"$(last_commit_message)"'' "Added subproject using git subtree" && undo ' test_expect_success 'check if --message works with squash too' ' git subtree add -P "sub dir" -m "Added subproject with squash" --squash sub1 && check_equal ''"$(last_commit_message)"'' "Added subproject with squash" && undo ' test_expect_success 'add subproj to mainline' ' git subtree add --prefix="sub dir"/ FETCH_HEAD && check_equal ''"$(last_commit_message)"'' "Add '"'sub dir/'"' from commit '"'"'''"$(git rev-parse sub1)"'''"'"'" ' # this shouldn't actually do anything, since FETCH_HEAD is already a parent test_expect_success 'merge fetched subproj' ' git merge -m "merge -s -ours" -s ours FETCH_HEAD ' test_expect_success 'add main-sub5' ' create "sub dir/main-sub5" && git commit -m "main-sub5" ' test_expect_success 'add main6' ' create main6 && git commit -m "main6 boring" ' test_expect_success 'add main-sub7' ' create "sub dir/main-sub7" && git commit -m "main-sub7" ' test_expect_success 'fetch new subproj history' ' git fetch ./"sub proj" sub2 && git branch sub2 FETCH_HEAD ' test_expect_success 'check if --message works for merge' ' git subtree merge --prefix="sub dir" -m "Merged changes from subproject" sub2 && check_equal ''"$(last_commit_message)"'' "Merged changes from subproject" && undo ' test_expect_success 'check if --message for merge works with squash too' ' git subtree merge --prefix "sub dir" -m "Merged changes from subproject using squash" --squash sub2 && check_equal ''"$(last_commit_message)"'' "Merged changes from subproject using squash" && undo ' test_expect_success 'merge new subproj history into subdir' ' git subtree merge --prefix="sub dir" FETCH_HEAD && git branch pre-split && check_equal ''"$(last_commit_message)"'' "Merge commit '"'"'"$(git rev-parse sub2)"'"'"' into mainline" && undo ' test_expect_success 'Check that prefix argument is required for split' ' 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 && rm -f expected actual ' test_expect_success 'Check that the exists for a split' ' 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 # rm -f expected actual ' test_expect_success 'check if --message works for split+rejoin' ' spl1=''"$(git subtree split --annotate='"'*'"' --prefix "sub dir" --onto FETCH_HEAD --message "Split & rejoin" --rejoin)"'' && git branch spl1 "$spl1" && check_equal ''"$(last_commit_message)"'' "Split & rejoin" && undo ' test_expect_success 'check split with --branch' ' spl1=$(git subtree split --annotate='"'*'"' --prefix "sub dir" --onto FETCH_HEAD --message "Split & rejoin" --rejoin) && undo && git subtree split --annotate='"'*'"' --prefix "sub dir" --onto FETCH_HEAD --branch splitbr1 && check_equal ''"$(git rev-parse splitbr1)"'' "$spl1" ' test_expect_success 'check hash of split' ' spl1=$(git subtree split --prefix "sub dir") && git subtree split --prefix "sub dir" --branch splitbr1test && check_equal ''"$(git rev-parse splitbr1test)"'' "$spl1" && new_hash=$(git rev-parse splitbr1test~2) && check_equal ''"$new_hash"'' "$subdir_hash" ' test_expect_success 'check split with --branch for an existing branch' ' spl1=''"$(git subtree split --annotate='"'*'"' --prefix "sub dir" --onto FETCH_HEAD --message "Split & rejoin" --rejoin)"'' && undo && git branch splitbr2 sub1 && git subtree split --annotate='"'*'"' --prefix "sub dir" --onto FETCH_HEAD --branch splitbr2 && check_equal ''"$(git rev-parse splitbr2)"'' "$spl1" ' test_expect_success 'check split with --branch for an incompatible branch' ' test_must_fail git subtree split --prefix "sub dir" --onto FETCH_HEAD --branch subdir ' test_expect_success 'check split+rejoin' ' spl1=''"$(git subtree split --annotate='"'*'"' --prefix "sub dir" --onto FETCH_HEAD --message "Split & rejoin" --rejoin)"'' && undo && git subtree split --annotate='"'*'"' --prefix "sub dir" --onto FETCH_HEAD --rejoin && check_equal ''"$(last_commit_message)"'' "Split '"'"'sub dir/'"'"' into commit '"'"'"$spl1"'"'"'" ' test_expect_success 'add main-sub8' ' create "sub dir/main-sub8" && git commit -m "main-sub8" ' # To the subproject! cd ./"sub proj" test_expect_success 'merge split into subproj' ' git fetch .. spl1 && git branch spl1 FETCH_HEAD && git merge FETCH_HEAD ' test_expect_success 'add sub9' ' create sub9 && git commit -m "sub9" ' # Back to mainline cd .. test_expect_success 'split for sub8' ' split2=''"$(git subtree split --annotate='"'*'"' --prefix "sub dir/" --rejoin)"'' && git branch split2 "$split2" ' test_expect_success 'add main-sub10' ' create "sub dir/main-sub10" && git commit -m "main-sub10" ' test_expect_success 'split for sub10' ' spl3=''"$(git subtree split --annotate='"'*'"' --prefix "sub dir" --rejoin)"'' && git branch spl3 "$spl3" ' # To the subproject! cd ./"sub proj" test_expect_success 'merge split into subproj' ' git fetch .. spl3 && git branch spl3 FETCH_HEAD && git merge FETCH_HEAD && git branch subproj-merge-spl3 ' chkm="main4 main6" chkms="main-sub10 main-sub5 main-sub7 main-sub8" chkms_sub=$(cat <&2 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_expect_success 'verify one file change per commit' ' x= && list=''"$(git log --pretty=format:'"'commit: %H'"' | joincommits)"'' && # test_debug "echo HERE" && # test_debug "echo ''"$list"''" && (git log --pretty=format:'"'commit: %H'"' | joincommits | ( while read commit a b; do test_debug "echo Verifying commit "''"$commit"'' test_debug "echo a: "''"$a"'' test_debug "echo b: "''"$b"'' check_equal "$b" "" x=1 done check_equal "$x" 1 )) ' # test push cd ../.. mkdir test-push cd test-push test_expect_success 'init main' ' test_create_repo main ' test_expect_success 'init sub' ' test_create_repo "sub project" ' cd ./"sub project" test_expect_success 'add subproject' ' create "sub project" && git commit -m "Sub project: 1" && git branch sub-branch-1 ' cd ../main test_expect_success 'make first commit and add subproject' ' create "main-1" && git commit -m "main: 1" && git subtree add "../sub project" --prefix "sub dir" --message "Added subproject" sub-branch-1 && check_equal "$(last_commit_message)" "Added subproject" ' test_expect_success 'make second commit to a subproject file and push it into a sub project' ' create "sub dir/sub1" && git commit -m "Sub project: 2" && git subtree push "../sub project" --prefix "sub dir" sub-branch-1 ' cd ../"sub project" test_expect_success 'Test second commit is pushed' ' git checkout sub-branch-1 && check_equal "$(last_commit_message)" "Sub project: 2" ' test_done