t / t6022-merge-rename.shon commit refs.c: flatten get_ref_store() a bit (126c9e0)
   1#!/bin/sh
   2
   3test_description='Merge-recursive merging renames'
   4. ./test-lib.sh
   5
   6modify () {
   7        sed -e "$1" <"$2" >"$2.x" &&
   8        mv "$2.x" "$2"
   9}
  10
  11test_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
  30
  31cat >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
  48
  49git 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 &&
  57
  58sed -e "/^g /s/.*/g : master changes a line/" <A >A+ &&
  59mv A+ A &&
  60git commit -a -m "master updates A" &&
  61
  62git checkout yellow &&
  63rm -f M &&
  64git commit -a -m "yellow removes M" &&
  65
  66git 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" &&
  72
  73git 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" &&
  79
  80git 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" &&
  86
  87git checkout change &&
  88sed -e "/^g /s/.*/g : changed line/" <A >A+ &&
  89mv A+ A &&
  90git commit -q -a -m "changed" &&
  91
  92git 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" &&
  97
  98git checkout master'
  99
 100test_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'
 115
 116test_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'
 132
 133test_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'
 148
 149test_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'
 168
 169test_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'
 176
 177test_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'
 186
 187test_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'
 197
 198test_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'
 210
 211test_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'
 224
 225test_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'
 240
 241test_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 | test_i18ngrep "^Skipped B" &&
 246        git reset --hard HEAD^ &&
 247        git checkout change &&
 248        GIT_MERGE_VERBOSITY=3 git merge change+rename | test_i18ngrep "^Skipped B"
 249'
 250
 251test_expect_success 'setup for rename + d/f conflicts' '
 252        git reset --hard &&
 253        git checkout --orphan dir-in-way &&
 254        git rm -rf . &&
 255        git clean -fdqx &&
 256
 257        mkdir sub &&
 258        mkdir dir &&
 259        printf "1\n2\n3\n4\n5\n6\n7\n8\n9\n10\n" >sub/file &&
 260        echo foo >dir/file-in-the-way &&
 261        git add -A &&
 262        git commit -m "Common commit" &&
 263
 264        echo 11 >>sub/file &&
 265        echo more >>dir/file-in-the-way &&
 266        git add -u &&
 267        git commit -m "Commit to merge, with dir in the way" &&
 268
 269        git checkout -b dir-not-in-way &&
 270        git reset --soft HEAD^ &&
 271        git rm -rf dir &&
 272        git commit -m "Commit to merge, with dir removed" -- dir sub/file &&
 273
 274        git checkout -b renamed-file-has-no-conflicts dir-in-way~1 &&
 275        git rm -rf dir &&
 276        git rm sub/file &&
 277        printf "1\n2\n3\n4\n5555\n6\n7\n8\n9\n10\n" >dir &&
 278        git add dir &&
 279        git commit -m "Independent change" &&
 280
 281        git checkout -b renamed-file-has-conflicts dir-in-way~1 &&
 282        git rm -rf dir &&
 283        git mv sub/file dir &&
 284        echo 12 >>dir &&
 285        git add dir &&
 286        git commit -m "Conflicting change"
 287'
 288
 289printf "1\n2\n3\n4\n5555\n6\n7\n8\n9\n10\n11\n" >expected
 290
 291test_expect_success 'Rename+D/F conflict; renamed file merges + dir not in way' '
 292        git reset --hard &&
 293        git checkout -q renamed-file-has-no-conflicts^0 &&
 294        git merge --strategy=recursive dir-not-in-way &&
 295        git diff --quiet &&
 296        test -f dir &&
 297        test_cmp expected dir
 298'
 299
 300test_expect_success 'Rename+D/F conflict; renamed file merges but dir in way' '
 301        git reset --hard &&
 302        rm -rf dir~* &&
 303        git checkout -q renamed-file-has-no-conflicts^0 &&
 304        test_must_fail git merge --strategy=recursive dir-in-way >output &&
 305
 306        test_i18ngrep "CONFLICT (modify/delete): dir/file-in-the-way" output &&
 307        test_i18ngrep "Auto-merging dir" output &&
 308        test_i18ngrep "Adding as dir~HEAD instead" output &&
 309
 310        test 3 -eq "$(git ls-files -u | wc -l)" &&
 311        test 2 -eq "$(git ls-files -u dir/file-in-the-way | wc -l)" &&
 312
 313        test_must_fail git diff --quiet &&
 314        test_must_fail git diff --cached --quiet &&
 315
 316        test -f dir/file-in-the-way &&
 317        test -f dir~HEAD &&
 318        test_cmp expected dir~HEAD
 319'
 320
 321test_expect_success 'Same as previous, but merged other way' '
 322        git reset --hard &&
 323        rm -rf dir~* &&
 324        git checkout -q dir-in-way^0 &&
 325        test_must_fail git merge --strategy=recursive renamed-file-has-no-conflicts >output 2>errors &&
 326
 327        ! grep "error: refusing to lose untracked file at" errors &&
 328        test_i18ngrep "CONFLICT (modify/delete): dir/file-in-the-way" output &&
 329        test_i18ngrep "Auto-merging dir" output &&
 330        test_i18ngrep "Adding as dir~renamed-file-has-no-conflicts instead" output &&
 331
 332        test 3 -eq "$(git ls-files -u | wc -l)" &&
 333        test 2 -eq "$(git ls-files -u dir/file-in-the-way | wc -l)" &&
 334
 335        test_must_fail git diff --quiet &&
 336        test_must_fail git diff --cached --quiet &&
 337
 338        test -f dir/file-in-the-way &&
 339        test -f dir~renamed-file-has-no-conflicts &&
 340        test_cmp expected dir~renamed-file-has-no-conflicts
 341'
 342
 343cat >expected <<\EOF &&
 3441
 3452
 3463
 3474
 3485
 3496
 3507
 3518
 3529
 35310
 354<<<<<<< HEAD:dir
 35512
 356=======
 35711
 358>>>>>>> dir-not-in-way:sub/file
 359EOF
 360
 361test_expect_success 'Rename+D/F conflict; renamed file cannot merge, dir not in way' '
 362        git reset --hard &&
 363        rm -rf dir~* &&
 364        git checkout -q renamed-file-has-conflicts^0 &&
 365        test_must_fail git merge --strategy=recursive dir-not-in-way &&
 366
 367        test 3 -eq "$(git ls-files -u | wc -l)" &&
 368        test 3 -eq "$(git ls-files -u dir | wc -l)" &&
 369
 370        test_must_fail git diff --quiet &&
 371        test_must_fail git diff --cached --quiet &&
 372
 373        test -f dir &&
 374        test_cmp expected dir
 375'
 376
 377test_expect_success 'Rename+D/F conflict; renamed file cannot merge and dir in the way' '
 378        modify s/dir-not-in-way/dir-in-way/ expected &&
 379
 380        git reset --hard &&
 381        rm -rf dir~* &&
 382        git checkout -q renamed-file-has-conflicts^0 &&
 383        test_must_fail git merge --strategy=recursive dir-in-way &&
 384
 385        test 5 -eq "$(git ls-files -u | wc -l)" &&
 386        test 3 -eq "$(git ls-files -u dir | grep -v file-in-the-way | wc -l)" &&
 387        test 2 -eq "$(git ls-files -u dir/file-in-the-way | wc -l)" &&
 388
 389        test_must_fail git diff --quiet &&
 390        test_must_fail git diff --cached --quiet &&
 391
 392        test -f dir/file-in-the-way &&
 393        test -f dir~HEAD &&
 394        test_cmp expected dir~HEAD
 395'
 396
 397cat >expected <<\EOF &&
 3981
 3992
 4003
 4014
 4025
 4036
 4047
 4058
 4069
 40710
 408<<<<<<< HEAD:sub/file
 40911
 410=======
 41112
 412>>>>>>> renamed-file-has-conflicts:dir
 413EOF
 414
 415test_expect_success 'Same as previous, but merged other way' '
 416        git reset --hard &&
 417        rm -rf dir~* &&
 418        git checkout -q dir-in-way^0 &&
 419        test_must_fail git merge --strategy=recursive renamed-file-has-conflicts &&
 420
 421        test 5 -eq "$(git ls-files -u | wc -l)" &&
 422        test 3 -eq "$(git ls-files -u dir | grep -v file-in-the-way | wc -l)" &&
 423        test 2 -eq "$(git ls-files -u dir/file-in-the-way | wc -l)" &&
 424
 425        test_must_fail git diff --quiet &&
 426        test_must_fail git diff --cached --quiet &&
 427
 428        test -f dir/file-in-the-way &&
 429        test -f dir~renamed-file-has-conflicts &&
 430        test_cmp expected dir~renamed-file-has-conflicts
 431'
 432
 433test_expect_success 'setup both rename source and destination involved in D/F conflict' '
 434        git reset --hard &&
 435        git checkout --orphan rename-dest &&
 436        git rm -rf . &&
 437        git clean -fdqx &&
 438
 439        mkdir one &&
 440        echo stuff >one/file &&
 441        git add -A &&
 442        git commit -m "Common commit" &&
 443
 444        git mv one/file destdir &&
 445        git commit -m "Renamed to destdir" &&
 446
 447        git checkout -b source-conflict HEAD~1 &&
 448        git rm -rf one &&
 449        mkdir destdir &&
 450        touch one destdir/foo &&
 451        git add -A &&
 452        git commit -m "Conflicts in the way"
 453'
 454
 455test_expect_success 'both rename source and destination involved in D/F conflict' '
 456        git reset --hard &&
 457        rm -rf dir~* &&
 458        git checkout -q rename-dest^0 &&
 459        test_must_fail git merge --strategy=recursive source-conflict &&
 460
 461        test 1 -eq "$(git ls-files -u | wc -l)" &&
 462
 463        test_must_fail git diff --quiet &&
 464
 465        test -f destdir/foo &&
 466        test -f one &&
 467        test -f destdir~HEAD &&
 468        test "stuff" = "$(cat destdir~HEAD)"
 469'
 470
 471test_expect_success 'setup pair rename to parent of other (D/F conflicts)' '
 472        git reset --hard &&
 473        git checkout --orphan rename-two &&
 474        git rm -rf . &&
 475        git clean -fdqx &&
 476
 477        mkdir one &&
 478        mkdir two &&
 479        echo stuff >one/file &&
 480        echo other >two/file &&
 481        git add -A &&
 482        git commit -m "Common commit" &&
 483
 484        git rm -rf one &&
 485        git mv two/file one &&
 486        git commit -m "Rename two/file -> one" &&
 487
 488        git checkout -b rename-one HEAD~1 &&
 489        git rm -rf two &&
 490        git mv one/file two &&
 491        rm -r one &&
 492        git commit -m "Rename one/file -> two"
 493'
 494
 495test_expect_success 'pair rename to parent of other (D/F conflicts) w/ untracked dir' '
 496        git checkout -q rename-one^0 &&
 497        mkdir one &&
 498        test_must_fail git merge --strategy=recursive rename-two &&
 499
 500        test 2 -eq "$(git ls-files -u | wc -l)" &&
 501        test 1 -eq "$(git ls-files -u one | wc -l)" &&
 502        test 1 -eq "$(git ls-files -u two | wc -l)" &&
 503
 504        test_must_fail git diff --quiet &&
 505
 506        test 4 -eq $(find . | grep -v .git | wc -l) &&
 507
 508        test -d one &&
 509        test -f one~rename-two &&
 510        test -f two &&
 511        test "other" = $(cat one~rename-two) &&
 512        test "stuff" = $(cat two)
 513'
 514
 515test_expect_success 'pair rename to parent of other (D/F conflicts) w/ clean start' '
 516        git reset --hard &&
 517        git clean -fdqx &&
 518        test_must_fail git merge --strategy=recursive rename-two &&
 519
 520        test 2 -eq "$(git ls-files -u | wc -l)" &&
 521        test 1 -eq "$(git ls-files -u one | wc -l)" &&
 522        test 1 -eq "$(git ls-files -u two | wc -l)" &&
 523
 524        test_must_fail git diff --quiet &&
 525
 526        test 3 -eq $(find . | grep -v .git | wc -l) &&
 527
 528        test -f one &&
 529        test -f two &&
 530        test "other" = $(cat one) &&
 531        test "stuff" = $(cat two)
 532'
 533
 534test_expect_success 'setup rename of one file to two, with directories in the way' '
 535        git reset --hard &&
 536        git checkout --orphan first-rename &&
 537        git rm -rf . &&
 538        git clean -fdqx &&
 539
 540        echo stuff >original &&
 541        git add -A &&
 542        git commit -m "Common commit" &&
 543
 544        mkdir two &&
 545        >two/file &&
 546        git add two/file &&
 547        git mv original one &&
 548        git commit -m "Put two/file in the way, rename to one" &&
 549
 550        git checkout -b second-rename HEAD~1 &&
 551        mkdir one &&
 552        >one/file &&
 553        git add one/file &&
 554        git mv original two &&
 555        git commit -m "Put one/file in the way, rename to two"
 556'
 557
 558test_expect_success 'check handling of differently renamed file with D/F conflicts' '
 559        git checkout -q first-rename^0 &&
 560        test_must_fail git merge --strategy=recursive second-rename &&
 561
 562        test 5 -eq "$(git ls-files -s | wc -l)" &&
 563        test 3 -eq "$(git ls-files -u | wc -l)" &&
 564        test 1 -eq "$(git ls-files -u one | wc -l)" &&
 565        test 1 -eq "$(git ls-files -u two | wc -l)" &&
 566        test 1 -eq "$(git ls-files -u original | wc -l)" &&
 567        test 2 -eq "$(git ls-files -o | wc -l)" &&
 568
 569        test -f one/file &&
 570        test -f two/file &&
 571        test -f one~HEAD &&
 572        test -f two~second-rename &&
 573        ! test -f original
 574'
 575
 576test_expect_success 'setup rename one file to two; directories moving out of the way' '
 577        git reset --hard &&
 578        git checkout --orphan first-rename-redo &&
 579        git rm -rf . &&
 580        git clean -fdqx &&
 581
 582        echo stuff >original &&
 583        mkdir one two &&
 584        touch one/file two/file &&
 585        git add -A &&
 586        git commit -m "Common commit" &&
 587
 588        git rm -rf one &&
 589        git mv original one &&
 590        git commit -m "Rename to one" &&
 591
 592        git checkout -b second-rename-redo HEAD~1 &&
 593        git rm -rf two &&
 594        git mv original two &&
 595        git commit -m "Rename to two"
 596'
 597
 598test_expect_success 'check handling of differently renamed file with D/F conflicts' '
 599        git checkout -q first-rename-redo^0 &&
 600        test_must_fail git merge --strategy=recursive second-rename-redo &&
 601
 602        test 3 -eq "$(git ls-files -u | wc -l)" &&
 603        test 1 -eq "$(git ls-files -u one | wc -l)" &&
 604        test 1 -eq "$(git ls-files -u two | wc -l)" &&
 605        test 1 -eq "$(git ls-files -u original | wc -l)" &&
 606        test 0 -eq "$(git ls-files -o | wc -l)" &&
 607
 608        test -f one &&
 609        test -f two &&
 610        ! test -f original
 611'
 612
 613test_expect_success 'setup avoid unnecessary update, normal rename' '
 614        git reset --hard &&
 615        git checkout --orphan avoid-unnecessary-update-1 &&
 616        git rm -rf . &&
 617        git clean -fdqx &&
 618
 619        printf "1\n2\n3\n4\n5\n6\n7\n8\n9\n10\n" >original &&
 620        git add -A &&
 621        git commit -m "Common commit" &&
 622
 623        git mv original rename &&
 624        echo 11 >>rename &&
 625        git add -u &&
 626        git commit -m "Renamed and modified" &&
 627
 628        git checkout -b merge-branch-1 HEAD~1 &&
 629        echo "random content" >random-file &&
 630        git add -A &&
 631        git commit -m "Random, unrelated changes"
 632'
 633
 634test_expect_success 'avoid unnecessary update, normal rename' '
 635        git checkout -q avoid-unnecessary-update-1^0 &&
 636        test-chmtime =1000000000 rename &&
 637        test-chmtime -v +0 rename >expect &&
 638        git merge merge-branch-1 &&
 639        test-chmtime -v +0 rename >actual &&
 640        test_cmp expect actual # "rename" should have stayed intact
 641'
 642
 643test_expect_success 'setup to test avoiding unnecessary update, with D/F conflict' '
 644        git reset --hard &&
 645        git checkout --orphan avoid-unnecessary-update-2 &&
 646        git rm -rf . &&
 647        git clean -fdqx &&
 648
 649        mkdir df &&
 650        printf "1\n2\n3\n4\n5\n6\n7\n8\n9\n10\n" >df/file &&
 651        git add -A &&
 652        git commit -m "Common commit" &&
 653
 654        git mv df/file temp &&
 655        rm -rf df &&
 656        git mv temp df &&
 657        echo 11 >>df &&
 658        git add -u &&
 659        git commit -m "Renamed and modified" &&
 660
 661        git checkout -b merge-branch-2 HEAD~1 &&
 662        >unrelated-change &&
 663        git add unrelated-change &&
 664        git commit -m "Only unrelated changes"
 665'
 666
 667test_expect_success 'avoid unnecessary update, with D/F conflict' '
 668        git checkout -q avoid-unnecessary-update-2^0 &&
 669        test-chmtime =1000000000 df &&
 670        test-chmtime -v +0 df >expect &&
 671        git merge merge-branch-2 &&
 672        test-chmtime -v +0 df >actual &&
 673        test_cmp expect actual # "df" should have stayed intact
 674'
 675
 676test_expect_success 'setup avoid unnecessary update, dir->(file,nothing)' '
 677        git rm -rf . &&
 678        git clean -fdqx &&
 679        rm -rf .git &&
 680        git init &&
 681
 682        >irrelevant &&
 683        mkdir df &&
 684        >df/file &&
 685        git add -A &&
 686        git commit -mA &&
 687
 688        git checkout -b side &&
 689        git rm -rf df &&
 690        git commit -mB &&
 691
 692        git checkout master &&
 693        git rm -rf df &&
 694        echo bla >df &&
 695        git add -A &&
 696        git commit -m "Add a newfile"
 697'
 698
 699test_expect_success 'avoid unnecessary update, dir->(file,nothing)' '
 700        git checkout -q master^0 &&
 701        test-chmtime =1000000000 df &&
 702        test-chmtime -v +0 df >expect &&
 703        git merge side &&
 704        test-chmtime -v +0 df >actual &&
 705        test_cmp expect actual # "df" should have stayed intact
 706'
 707
 708test_expect_success 'setup avoid unnecessary update, modify/delete' '
 709        git rm -rf . &&
 710        git clean -fdqx &&
 711        rm -rf .git &&
 712        git init &&
 713
 714        >irrelevant &&
 715        >file &&
 716        git add -A &&
 717        git commit -mA &&
 718
 719        git checkout -b side &&
 720        git rm -f file &&
 721        git commit -m "Delete file" &&
 722
 723        git checkout master &&
 724        echo bla >file &&
 725        git add -A &&
 726        git commit -m "Modify file"
 727'
 728
 729test_expect_success 'avoid unnecessary update, modify/delete' '
 730        git checkout -q master^0 &&
 731        test-chmtime =1000000000 file &&
 732        test-chmtime -v +0 file >expect &&
 733        test_must_fail git merge side &&
 734        test-chmtime -v +0 file >actual &&
 735        test_cmp expect actual # "file" should have stayed intact
 736'
 737
 738test_expect_success 'setup avoid unnecessary update, rename/add-dest' '
 739        git rm -rf . &&
 740        git clean -fdqx &&
 741        rm -rf .git &&
 742        git init &&
 743
 744        printf "1\n2\n3\n4\n5\n6\n7\n8\n" >file &&
 745        git add -A &&
 746        git commit -mA &&
 747
 748        git checkout -b side &&
 749        cp file newfile &&
 750        git add -A &&
 751        git commit -m "Add file copy" &&
 752
 753        git checkout master &&
 754        git mv file newfile &&
 755        git commit -m "Rename file"
 756'
 757
 758test_expect_success 'avoid unnecessary update, rename/add-dest' '
 759        git checkout -q master^0 &&
 760        test-chmtime =1000000000 newfile &&
 761        test-chmtime -v +0 newfile >expect &&
 762        git merge side &&
 763        test-chmtime -v +0 newfile >actual &&
 764        test_cmp expect actual # "file" should have stayed intact
 765'
 766
 767test_expect_success 'setup merge of rename + small change' '
 768        git reset --hard &&
 769        git checkout --orphan rename-plus-small-change &&
 770        git rm -rf . &&
 771        git clean -fdqx &&
 772
 773        echo ORIGINAL >file &&
 774        git add file &&
 775
 776        test_tick &&
 777        git commit -m Initial &&
 778        git checkout -b rename_branch &&
 779        git mv file renamed_file &&
 780        git commit -m Rename &&
 781        git checkout rename-plus-small-change &&
 782        echo NEW-VERSION >file &&
 783        git commit -a -m Reformat
 784'
 785
 786test_expect_success 'merge rename + small change' '
 787        git merge rename_branch &&
 788
 789        test 1 -eq $(git ls-files -s | wc -l) &&
 790        test 0 -eq $(git ls-files -o | wc -l) &&
 791        test $(git rev-parse HEAD:renamed_file) = $(git rev-parse HEAD~1:file)
 792'
 793
 794test_expect_success 'setup for use of extended merge markers' '
 795        git rm -rf . &&
 796        git clean -fdqx &&
 797        rm -rf .git &&
 798        git init &&
 799
 800        printf "1\n2\n3\n4\n5\n6\n7\n8\n" >original_file &&
 801        git add original_file &&
 802        git commit -mA &&
 803
 804        git checkout -b rename &&
 805        echo 9 >>original_file &&
 806        git add original_file &&
 807        git mv original_file renamed_file &&
 808        git commit -mB &&
 809
 810        git checkout master &&
 811        echo 8.5 >>original_file &&
 812        git add original_file &&
 813        git commit -mC
 814'
 815
 816cat >expected <<\EOF &&
 8171
 8182
 8193
 8204
 8215
 8226
 8237
 8248
 825<<<<<<< HEAD:renamed_file
 8269
 827=======
 8288.5
 829>>>>>>> master^0:original_file
 830EOF
 831
 832test_expect_success 'merge master into rename has correct extended markers' '
 833        git checkout rename^0 &&
 834        test_must_fail git merge -s recursive master^0 &&
 835        test_cmp expected renamed_file
 836'
 837
 838cat >expected <<\EOF &&
 8391
 8402
 8413
 8424
 8435
 8446
 8457
 8468
 847<<<<<<< HEAD:original_file
 8488.5
 849=======
 8509
 851>>>>>>> rename^0:renamed_file
 852EOF
 853
 854test_expect_success 'merge rename into master has correct extended markers' '
 855        git reset --hard &&
 856        git checkout master^0 &&
 857        test_must_fail git merge -s recursive rename^0 &&
 858        test_cmp expected renamed_file
 859'
 860
 861test_expect_success 'setup spurious "refusing to lose untracked" message' '
 862        git rm -rf . &&
 863        git clean -fdqx &&
 864        rm -rf .git &&
 865        git init &&
 866
 867        > irrelevant_file &&
 868        printf "1\n2\n3\n4\n5\n6\n7\n8\n" >original_file &&
 869        git add irrelevant_file original_file &&
 870        git commit -mA &&
 871
 872        git checkout -b rename &&
 873        git mv original_file renamed_file &&
 874        git commit -mB &&
 875
 876        git checkout master &&
 877        git rm original_file &&
 878        git commit -mC
 879'
 880
 881test_expect_success 'no spurious "refusing to lose untracked" message' '
 882        git checkout master^0 &&
 883        test_must_fail git merge rename^0 2>errors.txt &&
 884        ! grep "refusing to lose untracked file" errors.txt
 885'
 886
 887test_expect_success 'do not follow renames for empty files' '
 888        git checkout -f -b empty-base &&
 889        >empty1 &&
 890        git add empty1 &&
 891        git commit -m base &&
 892        echo content >empty1 &&
 893        git add empty1 &&
 894        git commit -m fill &&
 895        git checkout -b empty-topic HEAD^ &&
 896        git mv empty1 empty2 &&
 897        git commit -m rename &&
 898        test_must_fail git merge empty-base &&
 899        >expect &&
 900        test_cmp expect empty2
 901'
 902
 903test_done