1#!/bin/sh
   2test_description='Merge-recursive merging renames'
   4. ./test-lib.sh
   5modify () {
   7        sed -e "$1" <"$2" >"$2.x" &&
   8        mv "$2.x" "$2"
   9}
  10test_expect_success setup \
  12'
  13cat >A <<\EOF &&
  14a aaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaa
  15b bbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbb
  16c cccccccccccccccccccccccccccccccccccccccccccccccc
  17d dddddddddddddddddddddddddddddddddddddddddddddddd
  18e eeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeee
  19f ffffffffffffffffffffffffffffffffffffffffffffffff
  20g gggggggggggggggggggggggggggggggggggggggggggggggg
  21h hhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhh
  22i iiiiiiiiiiiiiiiiiiiiiiiiiiiiiiiiiiiiiiiiiiiiiiii
  23j jjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjj
  24k kkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkk
  25l llllllllllllllllllllllllllllllllllllllllllllllll
  26m mmmmmmmmmmmmmmmmmmmmmmmmmmmmmmmmmmmmmmmmmmmmmmmm
  27n nnnnnnnnnnnnnnnnnnnnnnnnnnnnnnnnnnnnnnnnnnnnnnnn
  28o oooooooooooooooooooooooooooooooooooooooooooooooo
  29EOF
  30cat >M <<\EOF &&
  32A AAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAA
  33B BBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBB
  34C CCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCC
  35D DDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDD
  36E EEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEE
  37F FFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFF
  38G GGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGGG
  39H HHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHH
  40I IIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIII
  41J JJJJJJJJJJJJJJJJJJJJJJJJJJJJJJJJJJJJJJJJJJJJJJJJ
  42K KKKKKKKKKKKKKKKKKKKKKKKKKKKKKKKKKKKKKKKKKKKKKKKK
  43L LLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLL
  44M MMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMM
  45N NNNNNNNNNNNNNNNNNNNNNNNNNNNNNNNNNNNNNNNNNNNNNNNN
  46O OOOOOOOOOOOOOOOOOOOOOOOOOOOOOOOOOOOOOOOOOOOOOOOO
  47EOF
  48git add A M &&
  50git commit -m "initial has A and M" &&
  51git branch white &&
  52git branch red &&
  53git branch blue &&
  54git branch yellow &&
  55git branch change &&
  56git branch change+rename &&
  57sed -e "/^g /s/.*/g : master changes a line/" <A >A+ &&
  59mv A+ A &&
  60git commit -a -m "master updates A" &&
  61git checkout yellow &&
  63rm -f M &&
  64git commit -a -m "yellow removes M" &&
  65git checkout white &&
  67sed -e "/^g /s/.*/g : white changes a line/" <A >B &&
  68sed -e "/^G /s/.*/G : colored branch changes a line/" <M >N &&
  69rm -f A M &&
  70git update-index --add --remove A B M N &&
  71git commit -m "white renames A->B, M->N" &&
  72git checkout red &&
  74sed -e "/^g /s/.*/g : red changes a line/" <A >B &&
  75sed -e "/^G /s/.*/G : colored branch changes a line/" <M >N &&
  76rm -f A M &&
  77git update-index --add --remove A B M N &&
  78git commit -m "red renames A->B, M->N" &&
  79git checkout blue &&
  81sed -e "/^g /s/.*/g : blue changes a line/" <A >C &&
  82sed -e "/^G /s/.*/G : colored branch changes a line/" <M >N &&
  83rm -f A M &&
  84git update-index --add --remove A C M N &&
  85git commit -m "blue renames A->C, M->N" &&
  86git checkout change &&
  88sed -e "/^g /s/.*/g : changed line/" <A >A+ &&
  89mv A+ A &&
  90git commit -q -a -m "changed" &&
  91git checkout change+rename &&
  93sed -e "/^g /s/.*/g : changed line/" <A >B &&
  94rm A &&
  95git update-index --add B &&
  96git commit -q -a -m "changed and renamed" &&
  97git checkout master'
  99test_expect_success 'pull renaming branch into unrenaming one' \
 101'
 102        git show-branch &&
 103        test_expect_code 1 git pull . white &&
 104        git ls-files -s &&
 105        git ls-files -u B >b.stages &&
 106        test_line_count = 3 b.stages &&
 107        git ls-files -s N >n.stages &&
 108        test_line_count = 1 n.stages &&
 109        sed -ne "/^g/{
 110        p
 111        q
 112        }" B | grep master &&
 113        git diff --exit-code white N
 114'
 115test_expect_success 'pull renaming branch into another renaming one' \
 117'
 118        rm -f B &&
 119        git reset --hard &&
 120        git checkout red &&
 121        test_expect_code 1 git pull . white &&
 122        git ls-files -u B >b.stages &&
 123        test_line_count = 3 b.stages &&
 124        git ls-files -s N >n.stages &&
 125        test_line_count = 1 n.stages &&
 126        sed -ne "/^g/{
 127        p
 128        q
 129        }" B | grep red &&
 130        git diff --exit-code white N
 131'
 132test_expect_success 'pull unrenaming branch into renaming one' \
 134'
 135        git reset --hard &&
 136        git show-branch &&
 137        test_expect_code 1 git pull . master &&
 138        git ls-files -u B >b.stages &&
 139        test_line_count = 3 b.stages &&
 140        git ls-files -s N >n.stages &&
 141        test_line_count = 1 n.stages &&
 142        sed -ne "/^g/{
 143        p
 144        q
 145        }" B | grep red &&
 146        git diff --exit-code white N
 147'
 148test_expect_success 'pull conflicting renames' \
 150'
 151        git reset --hard &&
 152        git show-branch &&
 153        test_expect_code 1 git pull . blue &&
 154        git ls-files -u A >a.stages &&
 155        test_line_count = 1 a.stages &&
 156        git ls-files -u B >b.stages &&
 157        test_line_count = 1 b.stages &&
 158        git ls-files -u C >c.stages &&
 159        test_line_count = 1 c.stages &&
 160        git ls-files -s N >n.stages &&
 161        test_line_count = 1 n.stages &&
 162        sed -ne "/^g/{
 163        p
 164        q
 165        }" B | grep red &&
 166        git diff --exit-code white N
 167'
 168test_expect_success 'interference with untracked working tree file' '
 170        git reset --hard &&
 171        git show-branch &&
 172        echo >A this file should not matter &&
 173        test_expect_code 1 git pull . white &&
 174        test_path_is_file A
 175'
 176test_expect_success 'interference with untracked working tree file' '
 178        git reset --hard &&
 179        git checkout white &&
 180        git show-branch &&
 181        rm -f A &&
 182        echo >A this file should not matter &&
 183        test_expect_code 1 git pull . red &&
 184        test_path_is_file A
 185'
 186test_expect_success 'interference with untracked working tree file' '
 188        git reset --hard &&
 189        rm -f A M &&
 190        git checkout -f master &&
 191        git tag -f anchor &&
 192        git show-branch &&
 193        git pull . yellow &&
 194        test_path_is_missing M &&
 195        git reset --hard anchor
 196'
 197test_expect_success 'updated working tree file should prevent the merge' '
 199        git reset --hard &&
 200        rm -f A M &&
 201        git checkout -f master &&
 202        git tag -f anchor &&
 203        git show-branch &&
 204        echo >>M one line addition &&
 205        cat M >M.saved &&
 206        test_expect_code 128 git pull . yellow &&
 207        test_cmp M M.saved &&
 208        rm -f M.saved
 209'
 210test_expect_success 'updated working tree file should prevent the merge' '
 212        git reset --hard &&
 213        rm -f A M &&
 214        git checkout -f master &&
 215        git tag -f anchor &&
 216        git show-branch &&
 217        echo >>M one line addition &&
 218        cat M >M.saved &&
 219        git update-index M &&
 220        test_expect_code 128 git pull . yellow &&
 221        test_cmp M M.saved &&
 222        rm -f M.saved
 223'
 224test_expect_success 'interference with untracked working tree file' '
 226        git reset --hard &&
 227        rm -f A M &&
 228        git checkout -f yellow &&
 229        git tag -f anchor &&
 230        git show-branch &&
 231        echo >M this file should not matter &&
 232        git pull . master &&
 233        test_path_is_file M &&
 234        ! {
 235                git ls-files -s |
 236                grep M
 237        } &&
 238        git reset --hard anchor
 239'
 240test_expect_success 'merge of identical changes in a renamed file' '
 242        rm -f A M N &&
 243        git reset --hard &&
 244        git checkout change+rename &&
 245        GIT_MERGE_VERBOSITY=3 git merge change | grep "^Skipped B" &&
 246        git reset --hard HEAD^ &&
 247        git checkout change &&
 248        GIT_MERGE_VERBOSITY=3 git merge change+rename | grep "^Skipped B"
 249'
 250test_expect_success 'setup for rename + d/f conflicts' '
 252        git reset --hard &&
 253        git checkout --orphan dir-in-way &&
 254        git rm -rf . &&
 255        mkdir sub &&
 257        mkdir dir &&
 258        printf "1\n2\n3\n4\n5\n6\n7\n8\n9\n10\n" >sub/file &&
 259        echo foo >dir/file-in-the-way &&
 260        git add -A &&
 261        git commit -m "Common commmit" &&
 262        echo 11 >>sub/file &&
 264        echo more >>dir/file-in-the-way &&
 265        git add -u &&
 266        git commit -m "Commit to merge, with dir in the way" &&
 267        git checkout -b dir-not-in-way &&
 269        git reset --soft HEAD^ &&
 270        git rm -rf dir &&
 271        git commit -m "Commit to merge, with dir removed" -- dir sub/file &&
 272        git checkout -b renamed-file-has-no-conflicts dir-in-way~1 &&
 274        git rm -rf dir &&
 275        git rm sub/file &&
 276        printf "1\n2\n3\n4\n5555\n6\n7\n8\n9\n10\n" >dir &&
 277        git add dir &&
 278        git commit -m "Independent change" &&
 279        git checkout -b renamed-file-has-conflicts dir-in-way~1 &&
 281        git rm -rf dir &&
 282        git mv sub/file dir &&
 283        echo 12 >>dir &&
 284        git add dir &&
 285        git commit -m "Conflicting change"
 286'
 287printf "1\n2\n3\n4\n5555\n6\n7\n8\n9\n10\n11\n" >expected
 289test_expect_success 'Rename+D/F conflict; renamed file merges + dir not in way' '
 291        git reset --hard &&
 292        git checkout -q renamed-file-has-no-conflicts^0 &&
 293        git merge --strategy=recursive dir-not-in-way &&
 294        git diff --quiet &&
 295        test -f dir &&
 296        test_cmp expected dir
 297'
 298test_expect_success 'Rename+D/F conflict; renamed file merges but dir in way' '
 300        git reset --hard &&
 301        rm -rf dir~* &&
 302        git checkout -q renamed-file-has-no-conflicts^0 &&
 303        test_must_fail git merge --strategy=recursive dir-in-way >output &&
 304        grep "CONFLICT (delete/modify): dir/file-in-the-way" output &&
 306        grep "Auto-merging dir" output &&
 307        grep "Adding as dir~HEAD instead" output &&
 308        test 2 -eq "$(git ls-files -u | wc -l)" &&
 310        test 2 -eq "$(git ls-files -u dir/file-in-the-way | wc -l)" &&
 311        test_must_fail git diff --quiet &&
 313        test_must_fail git diff --cached --quiet &&
 314        test -f dir/file-in-the-way &&
 316        test -f dir~HEAD &&
 317        test_cmp expected dir~HEAD
 318'
 319test_expect_success 'Same as previous, but merged other way' '
 321        git reset --hard &&
 322        rm -rf dir~* &&
 323        git checkout -q dir-in-way^0 &&
 324        test_must_fail git merge --strategy=recursive renamed-file-has-no-conflicts >output 2>errors &&
 325        ! grep "error: refusing to lose untracked file at" errors &&
 327        grep "CONFLICT (delete/modify): dir/file-in-the-way" output &&
 328        grep "Auto-merging dir" output &&
 329        grep "Adding as dir~renamed-file-has-no-conflicts instead" output &&
 330        test 2 -eq "$(git ls-files -u | wc -l)" &&
 332        test 2 -eq "$(git ls-files -u dir/file-in-the-way | wc -l)" &&
 333        test_must_fail git diff --quiet &&
 335        test_must_fail git diff --cached --quiet &&
 336        test -f dir/file-in-the-way &&
 338        test -f dir~renamed-file-has-no-conflicts &&
 339        test_cmp expected dir~renamed-file-has-no-conflicts
 340'
 341cat >expected <<\EOF &&
 3431
 3442
 3453
 3464
 3475
 3486
 3497
 3508
 3519
 35210
 353<<<<<<< HEAD
 35412
 355=======
 35611
 357>>>>>>> dir-not-in-way
 358EOF
 359test_expect_success 'Rename+D/F conflict; renamed file cannot merge, dir not in way' '
 361        git reset --hard &&
 362        rm -rf dir~* &&
 363        git checkout -q renamed-file-has-conflicts^0 &&
 364        test_must_fail git merge --strategy=recursive dir-not-in-way &&
 365        test 3 -eq "$(git ls-files -u | wc -l)" &&
 367        test 3 -eq "$(git ls-files -u dir | wc -l)" &&
 368        test_must_fail git diff --quiet &&
 370        test_must_fail git diff --cached --quiet &&
 371        test -f dir &&
 373        test_cmp expected dir
 374'
 375test_expect_success 'Rename+D/F conflict; renamed file cannot merge and dir in the way' '
 377        modify s/dir-not-in-way/dir-in-way/ expected &&
 378        git reset --hard &&
 380        rm -rf dir~* &&
 381        git checkout -q renamed-file-has-conflicts^0 &&
 382        test_must_fail git merge --strategy=recursive dir-in-way &&
 383        test 5 -eq "$(git ls-files -u | wc -l)" &&
 385        test 3 -eq "$(git ls-files -u dir | grep -v file-in-the-way | wc -l)" &&
 386        test 2 -eq "$(git ls-files -u dir/file-in-the-way | wc -l)" &&
 387        test_must_fail git diff --quiet &&
 389        test_must_fail git diff --cached --quiet &&
 390        test -f dir/file-in-the-way &&
 392        test -f dir~HEAD &&
 393        test_cmp expected dir~HEAD
 394'
 395cat >expected <<\EOF &&
 3971
 3982
 3993
 4004
 4015
 4026
 4037
 4048
 4059
 40610
 407<<<<<<< HEAD
 40811
 409=======
 41012
 411>>>>>>> renamed-file-has-conflicts
 412EOF
 413test_expect_success 'Same as previous, but merged other way' '
 415        git reset --hard &&
 416        rm -rf dir~* &&
 417        git checkout -q dir-in-way^0 &&
 418        test_must_fail git merge --strategy=recursive renamed-file-has-conflicts &&
 419        test 5 -eq "$(git ls-files -u | wc -l)" &&
 421        test 3 -eq "$(git ls-files -u dir | grep -v file-in-the-way | wc -l)" &&
 422        test 2 -eq "$(git ls-files -u dir/file-in-the-way | wc -l)" &&
 423        test_must_fail git diff --quiet &&
 425        test_must_fail git diff --cached --quiet &&
 426        test -f dir/file-in-the-way &&
 428        test -f dir~renamed-file-has-conflicts &&
 429        test_cmp expected dir~renamed-file-has-conflicts
 430'
 431test_expect_success 'setup both rename source and destination involved in D/F conflict' '
 433        git reset --hard &&
 434        git checkout --orphan rename-dest &&
 435        git rm -rf . &&
 436        git clean -fdqx &&
 437        mkdir one &&
 439        echo stuff >one/file &&
 440        git add -A &&
 441        git commit -m "Common commmit" &&
 442        git mv one/file destdir &&
 444        git commit -m "Renamed to destdir" &&
 445        git checkout -b source-conflict HEAD~1 &&
 447        git rm -rf one &&
 448        mkdir destdir &&
 449        touch one destdir/foo &&
 450        git add -A &&
 451        git commit -m "Conflicts in the way"
 452'
 453test_expect_success 'both rename source and destination involved in D/F conflict' '
 455        git reset --hard &&
 456        rm -rf dir~* &&
 457        git checkout -q rename-dest^0 &&
 458        test_must_fail git merge --strategy=recursive source-conflict &&
 459        test 1 -eq "$(git ls-files -u | wc -l)" &&
 461        test_must_fail git diff --quiet &&
 463        test -f destdir/foo &&
 465        test -f one &&
 466        test -f destdir~HEAD &&
 467        test "stuff" = "$(cat destdir~HEAD)"
 468'
 469test_expect_success 'setup pair rename to parent of other (D/F conflicts)' '
 471        git reset --hard &&
 472        git checkout --orphan rename-two &&
 473        git rm -rf . &&
 474        git clean -fdqx &&
 475        mkdir one &&
 477        mkdir two &&
 478        echo stuff >one/file &&
 479        echo other >two/file &&
 480        git add -A &&
 481        git commit -m "Common commmit" &&
 482        git rm -rf one &&
 484        git mv two/file one &&
 485        git commit -m "Rename two/file -> one" &&
 486        git checkout -b rename-one HEAD~1 &&
 488        git rm -rf two &&
 489        git mv one/file two &&
 490        rm -r one &&
 491        git commit -m "Rename one/file -> two"
 492'
 493test_expect_success 'pair rename to parent of other (D/F conflicts) w/ untracked dir' '
 495        git checkout -q rename-one^0 &&
 496        mkdir one &&
 497        test_must_fail git merge --strategy=recursive rename-two &&
 498        test 2 -eq "$(git ls-files -u | wc -l)" &&
 500        test 1 -eq "$(git ls-files -u one | wc -l)" &&
 501        test 1 -eq "$(git ls-files -u two | wc -l)" &&
 502        test_must_fail git diff --quiet &&
 504        test 4 -eq $(find . | grep -v .git | wc -l) &&
 506        test -d one &&
 508        test -f one~rename-two &&
 509        test -f two &&
 510        test "other" = $(cat one~rename-two) &&
 511        test "stuff" = $(cat two)
 512'
 513test_expect_success 'pair rename to parent of other (D/F conflicts) w/ clean start' '
 515        git reset --hard &&
 516        git clean -fdqx &&
 517        test_must_fail git merge --strategy=recursive rename-two &&
 518        test 2 -eq "$(git ls-files -u | wc -l)" &&
 520        test 1 -eq "$(git ls-files -u one | wc -l)" &&
 521        test 1 -eq "$(git ls-files -u two | wc -l)" &&
 522        test_must_fail git diff --quiet &&
 524        test 3 -eq $(find . | grep -v .git | wc -l) &&
 526        test -f one &&
 528        test -f two &&
 529        test "other" = $(cat one) &&
 530        test "stuff" = $(cat two)
 531'
 532test_expect_success 'setup rename of one file to two, with directories in the way' '
 534        git reset --hard &&
 535        git checkout --orphan first-rename &&
 536        git rm -rf . &&
 537        git clean -fdqx &&
 538        echo stuff >original &&
 540        git add -A &&
 541        git commit -m "Common commmit" &&
 542        mkdir two &&
 544        >two/file &&
 545        git add two/file &&
 546        git mv original one &&
 547        git commit -m "Put two/file in the way, rename to one" &&
 548        git checkout -b second-rename HEAD~1 &&
 550        mkdir one &&
 551        >one/file &&
 552        git add one/file &&
 553        git mv original two &&
 554        git commit -m "Put one/file in the way, rename to two"
 555'
 556test_expect_success 'check handling of differently renamed file with D/F conflicts' '
 558        git checkout -q first-rename^0 &&
 559        test_must_fail git merge --strategy=recursive second-rename &&
 560        test 5 -eq "$(git ls-files -s | wc -l)" &&
 562        test 3 -eq "$(git ls-files -u | wc -l)" &&
 563        test 1 -eq "$(git ls-files -u one | wc -l)" &&
 564        test 1 -eq "$(git ls-files -u two | wc -l)" &&
 565        test 1 -eq "$(git ls-files -u original | wc -l)" &&
 566        test 2 -eq "$(git ls-files -o | wc -l)" &&
 567        test -f one/file &&
 569        test -f two/file &&
 570        test -f one~HEAD &&
 571        test -f two~second-rename &&
 572        ! test -f original
 573'
 574test_expect_success 'setup rename one file to two; directories moving out of the way' '
 576        git reset --hard &&
 577        git checkout --orphan first-rename-redo &&
 578        git rm -rf . &&
 579        git clean -fdqx &&
 580        echo stuff >original &&
 582        mkdir one two &&
 583        touch one/file two/file &&
 584        git add -A &&
 585        git commit -m "Common commmit" &&
 586        git rm -rf one &&
 588        git mv original one &&
 589        git commit -m "Rename to one" &&
 590        git checkout -b second-rename-redo HEAD~1 &&
 592        git rm -rf two &&
 593        git mv original two &&
 594        git commit -m "Rename to two"
 595'
 596test_expect_success 'check handling of differently renamed file with D/F conflicts' '
 598        git checkout -q first-rename-redo^0 &&
 599        test_must_fail git merge --strategy=recursive second-rename-redo &&
 600        test 3 -eq "$(git ls-files -u | wc -l)" &&
 602        test 1 -eq "$(git ls-files -u one | wc -l)" &&
 603        test 1 -eq "$(git ls-files -u two | wc -l)" &&
 604        test 1 -eq "$(git ls-files -u original | wc -l)" &&
 605        test 0 -eq "$(git ls-files -o | wc -l)" &&
 606        test -f one &&
 608        test -f two &&
 609        ! test -f original
 610'
 611test_done