1#!/bin/sh
   2#
   3# Copyright (c) 2007 Shawn Pearce
   4#
   5test_description='test git fast-import utility'
   7. ./test-lib.sh
   8. "$TEST_DIRECTORY"/diff-lib.sh ;# test-lib chdir's into trash
   9verify_packs () {
  11        for p in .git/objects/pack/*.pack
  12        do
  13                git verify-pack "$@" "$p" || return
  14        done
  15}
  16file2_data='file2
  18second line of EOF'
  19file3_data='EOF
  21in 3rd file
  22 END'
  23file4_data=abcd
  25file4_len=4
  26file5_data='an inline file.
  28  we should see it later.'
  29file6_data='#!/bin/sh
  31echo "$@"'
  32###
  34### series A
  35###
  36test_expect_success 'empty stream succeeds' '
  38        git config fastimport.unpackLimit 0 &&
  39        git fast-import </dev/null
  40'
  41test_expect_success 'truncated stream complains' '
  43        echo "tag foo" | test_must_fail git fast-import
  44'
  45test_expect_success 'A: create pack from stdin' '
  47        test_tick &&
  48        cat >input <<-INPUT_END &&
  49        blob
  50        mark :2
  51        data <<EOF
  52        $file2_data
  53        EOF
  54        blob
  56        mark :3
  57        data <<END
  58        $file3_data
  59        END
  60        blob
  62        mark :4
  63        data $file4_len
  64        $file4_data
  65        commit refs/heads/master
  66        mark :5
  67        committer $GIT_COMMITTER_NAME <$GIT_COMMITTER_EMAIL> $GIT_COMMITTER_DATE
  68        data <<COMMIT
  69        initial
  70        COMMIT
  71        M 644 :2 file2
  73        M 644 :3 file3
  74        M 755 :4 file4
  75        tag series-A
  77        from :5
  78        data <<EOF
  79        An annotated tag without a tagger
  80        EOF
  81        tag series-A-blob
  83        from :3
  84        data <<EOF
  85        An annotated tag that annotates a blob.
  86        EOF
  87        INPUT_END
  89        git fast-import --export-marks=marks.out <input &&
  90        git whatchanged master
  91'
  92test_expect_success 'A: verify pack' '
  94        verify_packs
  95'
  96test_expect_success 'A: verify commit' '
  98        cat >expect <<-EOF &&
  99        author $GIT_COMMITTER_NAME <$GIT_COMMITTER_EMAIL> $GIT_COMMITTER_DATE
 100        committer $GIT_COMMITTER_NAME <$GIT_COMMITTER_EMAIL> $GIT_COMMITTER_DATE
 101        initial
 103        EOF
 104        git cat-file commit master | sed 1d >actual &&
 105        test_cmp expect actual
 106'
 107test_expect_success 'A: verify tree' '
 109        cat >expect <<-EOF &&
 110        100644 blob file2
 111        100644 blob file3
 112        100755 blob file4
 113        EOF
 114        git cat-file -p master^{tree} | sed "s/ [0-9a-f]*       / /" >actual &&
 115        test_cmp expect actual
 116'
 117test_expect_success 'A: verify file2' '
 119        echo "$file2_data" >expect &&
 120        git cat-file blob master:file2 >actual &&
 121        test_cmp expect actual
 122'
 123test_expect_success 'A: verify file3' '
 125        echo "$file3_data" >expect &&
 126        git cat-file blob master:file3 >actual &&
 127        test_cmp expect actual
 128'
 129test_expect_success 'A: verify file4' '
 131        printf "$file4_data" >expect &&
 132        git cat-file blob master:file4 >actual &&
 133        test_cmp expect actual
 134'
 135test_expect_success 'A: verify tag/series-A' '
 137        cat >expect <<-EOF &&
 138        object $(git rev-parse refs/heads/master)
 139        type commit
 140        tag series-A
 141        An annotated tag without a tagger
 143        EOF
 144        git cat-file tag tags/series-A >actual &&
 145        test_cmp expect actual
 146'
 147test_expect_success 'A: verify tag/series-A-blob' '
 149        cat >expect <<-EOF &&
 150        object $(git rev-parse refs/heads/master:file3)
 151        type blob
 152        tag series-A-blob
 153        An annotated tag that annotates a blob.
 155        EOF
 156        git cat-file tag tags/series-A-blob >actual &&
 157        test_cmp expect actual
 158'
 159test_expect_success 'A: verify marks output' '
 161        cat >expect <<-EOF &&
 162        :2 $(git rev-parse --verify master:file2)
 163        :3 $(git rev-parse --verify master:file3)
 164        :4 $(git rev-parse --verify master:file4)
 165        :5 $(git rev-parse --verify master^0)
 166        EOF
 167        test_cmp expect marks.out
 168'
 169test_expect_success 'A: verify marks import' '
 171        git fast-import \
 172                --import-marks=marks.out \
 173                --export-marks=marks.new \
 174                </dev/null &&
 175        test_cmp expect marks.new
 176'
 177test_expect_success 'A: tag blob by sha1' '
 179        test_tick &&
 180        new_blob=$(echo testing | git hash-object --stdin) &&
 181        cat >input <<-INPUT_END &&
 182        tag series-A-blob-2
 183        from $(git rev-parse refs/heads/master:file3)
 184        data <<EOF
 185        Tag blob by sha1.
 186        EOF
 187        blob
 189        mark :6
 190        data <<EOF
 191        testing
 192        EOF
 193        commit refs/heads/new_blob
 195        committer  <> 0 +0000
 196        data 0
 197        M 644 :6 new_blob
 198        #pretend we got sha1 from fast-import
 199        ls "new_blob"
 200        tag series-A-blob-3
 202        from $new_blob
 203        data <<EOF
 204        Tag new_blob.
 205        EOF
 206        INPUT_END
 207        cat >expect <<-EOF &&
 209        object $(git rev-parse refs/heads/master:file3)
 210        type blob
 211        tag series-A-blob-2
 212        Tag blob by sha1.
 214        object $new_blob
 215        type blob
 216        tag series-A-blob-3
 217        Tag new_blob.
 219        EOF
 220        git fast-import <input &&
 222        git cat-file tag tags/series-A-blob-2 >actual &&
 223        git cat-file tag tags/series-A-blob-3 >>actual &&
 224        test_cmp expect actual
 225'
 226test_expect_success 'A: verify marks import does not crash' '
 228        test_tick &&
 229        cat >input <<-INPUT_END &&
 230        commit refs/heads/verify--import-marks
 231        committer $GIT_COMMITTER_NAME <$GIT_COMMITTER_EMAIL> $GIT_COMMITTER_DATE
 232        data <<COMMIT
 233        recreate from :5
 234        COMMIT
 235        from :5
 237        M 755 :2 copy-of-file2
 238        INPUT_END
 240        git fast-import --import-marks=marks.out <input &&
 242        git whatchanged verify--import-marks
 243'
 244test_expect_success 'A: verify pack' '
 246        verify_packs
 247'
 248test_expect_success 'A: verify diff' '
 250        cat >expect <<-EOF &&
 251        :000000 100755 0000000000000000000000000000000000000000 7123f7f44e39be127c5eb701e5968176ee9d78b1 A      copy-of-file2
 252        EOF
 253        git diff-tree -M -r master verify--import-marks >actual &&
 254        compare_diff_raw expect actual &&
 255        test $(git rev-parse --verify master:file2) \
 256            = $(git rev-parse --verify verify--import-marks:copy-of-file2)
 257'
 258test_expect_success 'A: export marks with large values' '
 260        test_tick &&
 261        mt=$(git hash-object --stdin < /dev/null) &&
 262        >input.blob &&
 263        >marks.exp &&
 264        >tree.exp &&
 265        cat >input.commit <<-EOF &&
 267        commit refs/heads/verify--dump-marks
 268        committer $GIT_COMMITTER_NAME <$GIT_COMMITTER_EMAIL> $GIT_COMMITTER_DATE
 269        data <<COMMIT
 270        test the sparse array dumping routines with exponentially growing marks
 271        COMMIT
 272        EOF
 273        i=0 l=4 m=6 n=7 &&
 275        while test "$i" -lt 27
 276        do
 277                cat >>input.blob <<-EOF &&
 278                blob
 279                mark :$l
 280                data 0
 281                blob
 282                mark :$m
 283                data 0
 284                blob
 285                mark :$n
 286                data 0
 287                EOF
 288                echo "M 100644 :$l l$i" >>input.commit &&
 289                echo "M 100644 :$m m$i" >>input.commit &&
 290                echo "M 100644 :$n n$i" >>input.commit &&
 291                echo ":$l $mt" >>marks.exp &&
 293                echo ":$m $mt" >>marks.exp &&
 294                echo ":$n $mt" >>marks.exp &&
 295                printf "100644 blob $mt\tl$i\n" >>tree.exp &&
 297                printf "100644 blob $mt\tm$i\n" >>tree.exp &&
 298                printf "100644 blob $mt\tn$i\n" >>tree.exp &&
 299                l=$(($l + $l)) &&
 301                m=$(($m + $m)) &&
 302                n=$(($l + $n)) &&
 303                i=$((1 + $i)) || return 1
 305        done &&
 306        sort tree.exp > tree.exp_s &&
 308        cat input.blob input.commit | git fast-import --export-marks=marks.large &&
 310        git ls-tree refs/heads/verify--dump-marks >tree.out &&
 311        test_cmp tree.exp_s tree.out &&
 312        test_cmp marks.exp marks.large
 313'
 314###
 316### series B
 317###
 318test_expect_success 'B: fail on invalid blob sha1' '
 320        test_tick &&
 321        cat >input <<-INPUT_END &&
 322        commit refs/heads/branch
 323        mark :1
 324        committer $GIT_COMMITTER_NAME <$GIT_COMMITTER_EMAIL> $GIT_COMMITTER_DATE
 325        data <<COMMIT
 326        corrupt
 327        COMMIT
 328        from refs/heads/master
 330        M 755 0000000000000000000000000000000000000001 zero1
 331        INPUT_END
 333        test_when_finished "rm -f .git/objects/pack_* .git/objects/index_*" &&
 335        test_must_fail git fast-import <input
 336'
 337test_expect_success 'B: accept branch name "TEMP_TAG"' '
 339        cat >input <<-INPUT_END &&
 340        commit TEMP_TAG
 341        committer $GIT_COMMITTER_NAME <$GIT_COMMITTER_EMAIL> $GIT_COMMITTER_DATE
 342        data <<COMMIT
 343        tag base
 344        COMMIT
 345        from refs/heads/master
 347        INPUT_END
 349        test_when_finished "rm -f .git/TEMP_TAG
 351                git gc
 352                git prune" &&
 353        git fast-import <input &&
 354        test -f .git/TEMP_TAG &&
 355        test $(git rev-parse master) = $(git rev-parse TEMP_TAG^)
 356'
 357test_expect_success 'B: accept empty committer' '
 359        cat >input <<-INPUT_END &&
 360        commit refs/heads/empty-committer-1
 361        committer  <> $GIT_COMMITTER_DATE
 362        data <<COMMIT
 363        empty commit
 364        COMMIT
 365        INPUT_END
 366        test_when_finished "git update-ref -d refs/heads/empty-committer-1
 368                git gc
 369                git prune" &&
 370        git fast-import <input &&
 371        out=$(git fsck) &&
 372        echo "$out" &&
 373        test -z "$out"
 374'
 375test_expect_success 'B: accept and fixup committer with no name' '
 377        cat >input <<-INPUT_END &&
 378        commit refs/heads/empty-committer-2
 379        committer <a@b.com> $GIT_COMMITTER_DATE
 380        data <<COMMIT
 381        empty commit
 382        COMMIT
 383        INPUT_END
 384        test_when_finished "git update-ref -d refs/heads/empty-committer-2
 386                git gc
 387                git prune" &&
 388        git fast-import <input &&
 389        out=$(git fsck) &&
 390        echo "$out" &&
 391        test -z "$out"
 392'
 393test_expect_success 'B: fail on invalid committer (1)' '
 395        cat >input <<-INPUT_END &&
 396        commit refs/heads/invalid-committer
 397        committer Name email> $GIT_COMMITTER_DATE
 398        data <<COMMIT
 399        empty commit
 400        COMMIT
 401        INPUT_END
 402        test_when_finished "git update-ref -d refs/heads/invalid-committer" &&
 404        test_must_fail git fast-import <input
 405'
 406test_expect_success 'B: fail on invalid committer (2)' '
 408        cat >input <<-INPUT_END &&
 409        commit refs/heads/invalid-committer
 410        committer Name <e<mail> $GIT_COMMITTER_DATE
 411        data <<COMMIT
 412        empty commit
 413        COMMIT
 414        INPUT_END
 415        test_when_finished "git update-ref -d refs/heads/invalid-committer" &&
 417        test_must_fail git fast-import <input
 418'
 419test_expect_success 'B: fail on invalid committer (3)' '
 421        cat >input <<-INPUT_END &&
 422        commit refs/heads/invalid-committer
 423        committer Name <email>> $GIT_COMMITTER_DATE
 424        data <<COMMIT
 425        empty commit
 426        COMMIT
 427        INPUT_END
 428        test_when_finished "git update-ref -d refs/heads/invalid-committer" &&
 430        test_must_fail git fast-import <input
 431'
 432test_expect_success 'B: fail on invalid committer (4)' '
 434        cat >input <<-INPUT_END &&
 435        commit refs/heads/invalid-committer
 436        committer Name <email $GIT_COMMITTER_DATE
 437        data <<COMMIT
 438        empty commit
 439        COMMIT
 440        INPUT_END
 441        test_when_finished "git update-ref -d refs/heads/invalid-committer" &&
 443        test_must_fail git fast-import <input
 444'
 445test_expect_success 'B: fail on invalid committer (5)' '
 447        cat >input <<-INPUT_END &&
 448        commit refs/heads/invalid-committer
 449        committer Name<email> $GIT_COMMITTER_DATE
 450        data <<COMMIT
 451        empty commit
 452        COMMIT
 453        INPUT_END
 454        test_when_finished "git update-ref -d refs/heads/invalid-committer" &&
 456        test_must_fail git fast-import <input
 457'
 458###
 460### series C
 461###
 462test_expect_success 'C: incremental import create pack from stdin' '
 464        newf=$(echo hi newf | git hash-object -w --stdin) &&
 465        oldf=$(git rev-parse --verify master:file2) &&
 466        test_tick &&
 467        cat >input <<-INPUT_END &&
 468        commit refs/heads/branch
 469        committer $GIT_COMMITTER_NAME <$GIT_COMMITTER_EMAIL> $GIT_COMMITTER_DATE
 470        data <<COMMIT
 471        second
 472        COMMIT
 473        from refs/heads/master
 475        M 644 $oldf file2/oldf
 476        M 755 $newf file2/newf
 477        D file3
 478        INPUT_END
 480        git fast-import <input &&
 482        git whatchanged branch
 483'
 484test_expect_success 'C: verify pack' '
 486        verify_packs
 487'
 488test_expect_success 'C: validate reuse existing blob' '
 490        test $newf = $(git rev-parse --verify branch:file2/newf) &&
 491        test $oldf = $(git rev-parse --verify branch:file2/oldf)
 492'
 493test_expect_success 'C: verify commit' '
 495        cat >expect <<-EOF &&
 496        parent $(git rev-parse --verify master^0)
 497        author $GIT_COMMITTER_NAME <$GIT_COMMITTER_EMAIL> $GIT_COMMITTER_DATE
 498        committer $GIT_COMMITTER_NAME <$GIT_COMMITTER_EMAIL> $GIT_COMMITTER_DATE
 499        second
 501        EOF
 502        git cat-file commit branch | sed 1d >actual &&
 504        test_cmp expect actual
 505'
 506test_expect_success 'C: validate rename result' '
 508        cat >expect <<-EOF &&
 509        :000000 100755 0000000000000000000000000000000000000000 f1fb5da718392694d0076d677d6d0e364c79b0bc A      file2/newf
 510        :100644 100644 7123f7f44e39be127c5eb701e5968176ee9d78b1 7123f7f44e39be127c5eb701e5968176ee9d78b1 R100   file2   file2/oldf
 511        :100644 000000 0d92e9f3374ae2947c23aa477cbc68ce598135f1 0000000000000000000000000000000000000000 D      file3
 512        EOF
 513        git diff-tree -M -r master branch >actual &&
 514        compare_diff_raw expect actual
 515'
 516###
 518### series D
 519###
 520test_expect_success 'D: inline data in commit' '
 522        test_tick &&
 523        cat >input <<-INPUT_END &&
 524        commit refs/heads/branch
 525        committer $GIT_COMMITTER_NAME <$GIT_COMMITTER_EMAIL> $GIT_COMMITTER_DATE
 526        data <<COMMIT
 527        third
 528        COMMIT
 529        from refs/heads/branch^0
 531        M 644 inline newdir/interesting
 532        data <<EOF
 533        $file5_data
 534        EOF
 535        M 755 inline newdir/exec.sh
 537        data <<EOF
 538        $file6_data
 539        EOF
 540        INPUT_END
 542        git fast-import <input &&
 544        git whatchanged branch
 545'
 546test_expect_success 'D: verify pack' '
 548        verify_packs
 549'
 550test_expect_success 'D: validate new files added' '
 552        cat >expect <<-EOF &&
 553        :000000 100755 0000000000000000000000000000000000000000 e74b7d465e52746be2b4bae983670711e6e66657 A      newdir/exec.sh
 554        :000000 100644 0000000000000000000000000000000000000000 fcf778cda181eaa1cbc9e9ce3a2e15ee9f9fe791 A      newdir/interesting
 555        EOF
 556        git diff-tree -M -r branch^ branch >actual &&
 557        compare_diff_raw expect actual
 558'
 559test_expect_success 'D: verify file5' '
 561        echo "$file5_data" >expect &&
 562        git cat-file blob branch:newdir/interesting >actual &&
 563        test_cmp expect actual
 564'
 565test_expect_success 'D: verify file6' '
 567        echo "$file6_data" >expect &&
 568        git cat-file blob branch:newdir/exec.sh >actual &&
 569        test_cmp expect actual
 570'
 571###
 573### series E
 574###
 575test_expect_success 'E: rfc2822 date, --date-format=raw' '
 577        cat >input <<-INPUT_END &&
 578        commit refs/heads/branch
 579        author $GIT_AUTHOR_NAME <$GIT_AUTHOR_EMAIL> Tue Feb 6 11:22:18 2007 -0500
 580        committer $GIT_COMMITTER_NAME <$GIT_COMMITTER_EMAIL> Tue Feb 6 12:35:02 2007 -0500
 581        data <<COMMIT
 582        RFC 2822 type date
 583        COMMIT
 584        from refs/heads/branch^0
 586        INPUT_END
 588        test_must_fail git fast-import --date-format=raw <input
 590'
 591test_expect_success 'E: rfc2822 date, --date-format=rfc2822' '
 592        git fast-import --date-format=rfc2822 <input
 593'
 594test_expect_success 'E: verify pack' '
 596        verify_packs
 597'
 598test_expect_success 'E: verify commit' '
 600        cat >expect <<-EOF &&
 601        author $GIT_AUTHOR_NAME <$GIT_AUTHOR_EMAIL> 1170778938 -0500
 602        committer $GIT_COMMITTER_NAME <$GIT_COMMITTER_EMAIL> 1170783302 -0500
 603        RFC 2822 type date
 605        EOF
 606        git cat-file commit branch | sed 1,2d >actual &&
 607        test_cmp expect actual
 608'
 609###
 611### series F
 612###
 613test_expect_success 'F: non-fast-forward update skips' '
 615        old_branch=$(git rev-parse --verify branch^0) &&
 616        test_tick &&
 617        cat >input <<-INPUT_END &&
 618        commit refs/heads/branch
 619        committer $GIT_COMMITTER_NAME <$GIT_COMMITTER_EMAIL> $GIT_COMMITTER_DATE
 620        data <<COMMIT
 621        losing things already?
 622        COMMIT
 623        from refs/heads/branch~1
 625        reset refs/heads/other
 627        from refs/heads/branch
 628        INPUT_END
 630        test_must_fail git fast-import <input &&
 632        # branch must remain unaffected
 633        test $old_branch = $(git rev-parse --verify branch^0)
 634'
 635test_expect_success 'F: verify pack' '
 637        verify_packs
 638'
 639test_expect_success 'F: verify other commit' '
 641        cat >expect <<-EOF &&
 642        tree $(git rev-parse branch~1^{tree})
 643        parent $(git rev-parse branch~1)
 644        author $GIT_COMMITTER_NAME <$GIT_COMMITTER_EMAIL> $GIT_COMMITTER_DATE
 645        committer $GIT_COMMITTER_NAME <$GIT_COMMITTER_EMAIL> $GIT_COMMITTER_DATE
 646        losing things already?
 648        EOF
 649        git cat-file commit other >actual &&
 650        test_cmp expect actual
 651'
 652###
 654### series G
 655###
 656test_expect_success 'G: non-fast-forward update forced' '
 658        old_branch=$(git rev-parse --verify branch^0) &&
 659        test_tick &&
 660        cat >input <<-INPUT_END &&
 661        commit refs/heads/branch
 662        committer $GIT_COMMITTER_NAME <$GIT_COMMITTER_EMAIL> $GIT_COMMITTER_DATE
 663        data <<COMMIT
 664        losing things already?
 665        COMMIT
 666        from refs/heads/branch~1
 668        INPUT_END
 670        git fast-import --force <input
 671'
 672test_expect_success 'G: verify pack' '
 674        verify_packs
 675'
 676test_expect_success 'G: branch changed, but logged' '
 678        test $old_branch != $(git rev-parse --verify branch^0) &&
 679        test $old_branch = $(git rev-parse --verify branch@{1})
 680'
 681###
 683### series H
 684###
 685test_expect_success 'H: deletall, add 1' '
 687        test_tick &&
 688        cat >input <<-INPUT_END &&
 689        commit refs/heads/H
 690        committer $GIT_COMMITTER_NAME <$GIT_COMMITTER_EMAIL> $GIT_COMMITTER_DATE
 691        data <<COMMIT
 692        third
 693        COMMIT
 694        from refs/heads/branch^0
 696        M 644 inline i-will-die
 697        data <<EOF
 698        this file will never exist.
 699        EOF
 700        deleteall
 702        M 644 inline h/e/l/lo
 703        data <<EOF
 704        $file5_data
 705        EOF
 706        INPUT_END
 708        git fast-import <input &&
 709        git whatchanged H
 710'
 711test_expect_success 'H: verify pack' '
 713        verify_packs
 714'
 715test_expect_success 'H: validate old files removed, new files added' '
 717        cat >expect <<-EOF &&
 718        :100755 000000 f1fb5da718392694d0076d677d6d0e364c79b0bc 0000000000000000000000000000000000000000 D      file2/newf
 719        :100644 000000 7123f7f44e39be127c5eb701e5968176ee9d78b1 0000000000000000000000000000000000000000 D      file2/oldf
 720        :100755 000000 85df50785d62d3b05ab03d9cbf7e4a0b49449730 0000000000000000000000000000000000000000 D      file4
 721        :100644 100644 fcf778cda181eaa1cbc9e9ce3a2e15ee9f9fe791 fcf778cda181eaa1cbc9e9ce3a2e15ee9f9fe791 R100   newdir/interesting      h/e/l/lo
 722        :100755 000000 e74b7d465e52746be2b4bae983670711e6e66657 0000000000000000000000000000000000000000 D      newdir/exec.sh
 723        EOF
 724        git diff-tree -M -r H^ H >actual &&
 725        compare_diff_raw expect actual
 726'
 727test_expect_success 'H: verify file' '
 729        echo "$file5_data" >expect &&
 730        git cat-file blob H:h/e/l/lo >actual &&
 731        test_cmp expect actual
 732'
 733###
 735### series I
 736###
 737test_expect_success 'I: export-pack-edges' '
 739        cat >input <<-INPUT_END &&
 740        commit refs/heads/export-boundary
 741        committer $GIT_COMMITTER_NAME <$GIT_COMMITTER_EMAIL> $GIT_COMMITTER_DATE
 742        data <<COMMIT
 743        we have a border.  its only 40 characters wide.
 744        COMMIT
 745        from refs/heads/branch
 747        INPUT_END
 749        git fast-import --export-pack-edges=edges.list <input
 750'
 751test_expect_success 'I: verify edge list' '
 753        cat >expect <<-EOF &&
 754        .git/objects/pack/pack-.pack: $(git rev-parse --verify export-boundary)
 755        EOF
 756        sed -e s/pack-.*pack/pack-.pack/ edges.list >actual &&
 757        test_cmp expect actual
 758'
 759###
 761### series J
 762###
 763test_expect_success 'J: reset existing branch creates empty commit' '
 765        cat >input <<-INPUT_END &&
 766        commit refs/heads/J
 767        committer $GIT_COMMITTER_NAME <$GIT_COMMITTER_EMAIL> $GIT_COMMITTER_DATE
 768        data <<COMMIT
 769        create J
 770        COMMIT
 771        from refs/heads/branch
 773        reset refs/heads/J
 775        commit refs/heads/J
 777        committer $GIT_COMMITTER_NAME <$GIT_COMMITTER_EMAIL> $GIT_COMMITTER_DATE
 778        data <<COMMIT
 779        initialize J
 780        COMMIT
 781        INPUT_END
 783        git fast-import <input
 784'
 785test_expect_success 'J: branch has 1 commit, empty tree' '
 786        test 1 = $(git rev-list J | wc -l) &&
 787        test 0 = $(git ls-tree J | wc -l)
 788'
 789test_expect_success 'J: tag must fail on empty branch' '
 791        cat >input <<-INPUT_END &&
 792        reset refs/heads/J2
 793        tag wrong_tag
 795        from refs/heads/J2
 796        data <<EOF
 797        Tag branch that was reset.
 798        EOF
 799        INPUT_END
 800        test_must_fail git fast-import <input
 801'
 802###
 804### series K
 805###
 806test_expect_success 'K: reinit branch with from' '
 808        cat >input <<-INPUT_END &&
 809        commit refs/heads/K
 810        committer $GIT_COMMITTER_NAME <$GIT_COMMITTER_EMAIL> $GIT_COMMITTER_DATE
 811        data <<COMMIT
 812        create K
 813        COMMIT
 814        from refs/heads/branch
 816        commit refs/heads/K
 818        committer $GIT_COMMITTER_NAME <$GIT_COMMITTER_EMAIL> $GIT_COMMITTER_DATE
 819        data <<COMMIT
 820        redo K
 821        COMMIT
 822        from refs/heads/branch^1
 824        INPUT_END
 826        git fast-import <input
 827'
 828test_expect_success 'K: verify K^1 = branch^1' '
 829        test $(git rev-parse --verify branch^1) \
 830                = $(git rev-parse --verify K^1)
 831'
 832###
 834### series L
 835###
 836test_expect_success 'L: verify internal tree sorting' '
 838        cat >input <<-INPUT_END &&
 839        blob
 840        mark :1
 841        data <<EOF
 842        some data
 843        EOF
 844        blob
 846        mark :2
 847        data <<EOF
 848        other data
 849        EOF
 850        commit refs/heads/L
 852        committer $GIT_COMMITTER_NAME <$GIT_COMMITTER_EMAIL> $GIT_COMMITTER_DATE
 853        data <<COMMIT
 854        create L
 855        COMMIT
 856        M 644 :1 b.
 858        M 644 :1 b/other
 859        M 644 :1 ba
 860        commit refs/heads/L
 862        committer $GIT_COMMITTER_NAME <$GIT_COMMITTER_EMAIL> $GIT_COMMITTER_DATE
 863        data <<COMMIT
 864        update L
 865        COMMIT
 866        M 644 :2 b.
 868        M 644 :2 b/other
 869        M 644 :2 ba
 870        INPUT_END
 871        cat >expect <<-EXPECT_END &&
 873        :100644 100644 4268632... 55d3a52... M  b.
 874        :040000 040000 0ae5cac... 443c768... M  b
 875        :100644 100644 4268632... 55d3a52... M  ba
 876        EXPECT_END
 877        git fast-import <input &&
 879        GIT_PRINT_SHA1_ELLIPSIS="yes" git diff-tree --abbrev --raw L^ L >output &&
 880        test_cmp expect output
 881'
 882test_expect_success 'L: nested tree copy does not corrupt deltas' '
 884        cat >input <<-INPUT_END &&
 885        blob
 886        mark :1
 887        data <<EOF
 888        the data
 889        EOF
 890        commit refs/heads/L2
 892        committer C O Mitter <committer@example.com> 1112912473 -0700
 893        data <<COMMIT
 894        init L2
 895        COMMIT
 896        M 644 :1 a/b/c
 897        M 644 :1 a/b/d
 898        M 644 :1 a/e/f
 899        commit refs/heads/L2
 901        committer C O Mitter <committer@example.com> 1112912473 -0700
 902        data <<COMMIT
 903        update L2
 904        COMMIT
 905        C a g
 906        C a/e g/b
 907        M 644 :1 g/b/h
 908        INPUT_END
 909        cat >expect <<-\EOF &&
 911        g/b/f
 912        g/b/h
 913        EOF
 914        test_when_finished "git update-ref -d refs/heads/L2" &&
 916        git fast-import <input &&
 917        git ls-tree L2 g/b/ >tmp &&
 918        cat tmp | cut -f 2 >actual &&
 919        test_cmp expect actual &&
 920        git fsck $(git rev-parse L2)
 921'
 922###
 924### series M
 925###
 926test_expect_success 'M: rename file in same subdirectory' '
 928        test_tick &&
 929        cat >input <<-INPUT_END &&
 930        commit refs/heads/M1
 931        committer $GIT_COMMITTER_NAME <$GIT_COMMITTER_EMAIL> $GIT_COMMITTER_DATE
 932        data <<COMMIT
 933        file rename
 934        COMMIT
 935        from refs/heads/branch^0
 937        R file2/newf file2/n.e.w.f
 938        INPUT_END
 940        cat >expect <<-EOF &&
 942        :100755 100755 f1fb5da718392694d0076d677d6d0e364c79b0bc f1fb5da718392694d0076d677d6d0e364c79b0bc R100   file2/newf      file2/n.e.w.f
 943        EOF
 944        git fast-import <input &&
 945        git diff-tree -M -r M1^ M1 >actual &&
 946        compare_diff_raw expect actual
 947'
 948test_expect_success 'M: rename file to new subdirectory' '
 950        cat >input <<-INPUT_END &&
 951        commit refs/heads/M2
 952        committer $GIT_COMMITTER_NAME <$GIT_COMMITTER_EMAIL> $GIT_COMMITTER_DATE
 953        data <<COMMIT
 954        file rename
 955        COMMIT
 956        from refs/heads/branch^0
 958        R file2/newf i/am/new/to/you
 959        INPUT_END
 961        cat >expect <<-EOF &&
 963        :100755 100755 f1fb5da718392694d0076d677d6d0e364c79b0bc f1fb5da718392694d0076d677d6d0e364c79b0bc R100   file2/newf      i/am/new/to/you
 964        EOF
 965        git fast-import <input &&
 966        git diff-tree -M -r M2^ M2 >actual &&
 967        compare_diff_raw expect actual
 968'
 969test_expect_success 'M: rename subdirectory to new subdirectory' '
 971        cat >input <<-INPUT_END &&
 972        commit refs/heads/M3
 973        committer $GIT_COMMITTER_NAME <$GIT_COMMITTER_EMAIL> $GIT_COMMITTER_DATE
 974        data <<COMMIT
 975        file rename
 976        COMMIT
 977        from refs/heads/M2^0
 979        R i other/sub
 980        INPUT_END
 982        cat >expect <<-EOF &&
 984        :100755 100755 f1fb5da718392694d0076d677d6d0e364c79b0bc f1fb5da718392694d0076d677d6d0e364c79b0bc R100   i/am/new/to/you other/sub/am/new/to/you
 985        EOF
 986        git fast-import <input &&
 987        git diff-tree -M -r M3^ M3 >actual &&
 988        compare_diff_raw expect actual
 989'
 990test_expect_success 'M: rename root to subdirectory' '
 992        cat >input <<-INPUT_END &&
 993        commit refs/heads/M4
 994        committer $GIT_COMMITTER_NAME <$GIT_COMMITTER_EMAIL> $GIT_COMMITTER_DATE
 995        data <<COMMIT
 996        rename root
 997        COMMIT
 998        from refs/heads/M2^0
1000        R "" sub
1001        INPUT_END
1003        cat >expect <<-EOF &&
1005        :100644 100644 7123f7f44e39be127c5eb701e5968176ee9d78b1 7123f7f44e39be127c5eb701e5968176ee9d78b1 R100   file2/oldf      sub/file2/oldf
1006        :100755 100755 85df50785d62d3b05ab03d9cbf7e4a0b49449730 85df50785d62d3b05ab03d9cbf7e4a0b49449730 R100   file4   sub/file4
1007        :100755 100755 f1fb5da718392694d0076d677d6d0e364c79b0bc f1fb5da718392694d0076d677d6d0e364c79b0bc R100   i/am/new/to/you sub/i/am/new/to/you
1008        :100755 100755 e74b7d465e52746be2b4bae983670711e6e66657 e74b7d465e52746be2b4bae983670711e6e66657 R100   newdir/exec.sh  sub/newdir/exec.sh
1009        :100644 100644 fcf778cda181eaa1cbc9e9ce3a2e15ee9f9fe791 fcf778cda181eaa1cbc9e9ce3a2e15ee9f9fe791 R100   newdir/interesting      sub/newdir/interesting
1010        EOF
1011        git fast-import <input &&
1012        git diff-tree -M -r M4^ M4 >actual &&
1013        cat actual &&
1014        compare_diff_raw expect actual
1015'
1016###
1018### series N
1019###
1020test_expect_success 'N: copy file in same subdirectory' '
1022        test_tick &&
1023        cat >input <<-INPUT_END &&
1024        commit refs/heads/N1
1025        committer $GIT_COMMITTER_NAME <$GIT_COMMITTER_EMAIL> $GIT_COMMITTER_DATE
1026        data <<COMMIT
1027        file copy
1028        COMMIT
1029        from refs/heads/branch^0
1031        C file2/newf file2/n.e.w.f
1032        INPUT_END
1034        cat >expect <<-EOF &&
1036        :100755 100755 f1fb5da718392694d0076d677d6d0e364c79b0bc f1fb5da718392694d0076d677d6d0e364c79b0bc C100   file2/newf      file2/n.e.w.f
1037        EOF
1038        git fast-import <input &&
1039        git diff-tree -C --find-copies-harder -r N1^ N1 >actual &&
1040        compare_diff_raw expect actual
1041'
1042test_expect_success 'N: copy then modify subdirectory' '
1044        cat >input <<-INPUT_END &&
1045        commit refs/heads/N2
1046        committer $GIT_COMMITTER_NAME <$GIT_COMMITTER_EMAIL> $GIT_COMMITTER_DATE
1047        data <<COMMIT
1048        clean directory copy
1049        COMMIT
1050        from refs/heads/branch^0
1052        C file2 file3
1053        commit refs/heads/N2
1055        committer $GIT_COMMITTER_NAME <$GIT_COMMITTER_EMAIL> $GIT_COMMITTER_DATE
1056        data <<COMMIT
1057        modify directory copy
1058        COMMIT
1059        M 644 inline file3/file5
1061        data <<EOF
1062        $file5_data
1063        EOF
1064        INPUT_END
1066        cat >expect <<-EOF &&
1068        :100644 100644 fcf778cda181eaa1cbc9e9ce3a2e15ee9f9fe791 fcf778cda181eaa1cbc9e9ce3a2e15ee9f9fe791 C100   newdir/interesting      file3/file5
1069        :100755 100755 f1fb5da718392694d0076d677d6d0e364c79b0bc f1fb5da718392694d0076d677d6d0e364c79b0bc C100   file2/newf      file3/newf
1070        :100644 100644 7123f7f44e39be127c5eb701e5968176ee9d78b1 7123f7f44e39be127c5eb701e5968176ee9d78b1 C100   file2/oldf      file3/oldf
1071        EOF
1072        git fast-import <input &&
1073        git diff-tree -C --find-copies-harder -r N2^^ N2 >actual &&
1074        compare_diff_raw expect actual
1075'
1076test_expect_success 'N: copy dirty subdirectory' '
1078        cat >input <<-INPUT_END &&
1079        commit refs/heads/N3
1080        committer $GIT_COMMITTER_NAME <$GIT_COMMITTER_EMAIL> $GIT_COMMITTER_DATE
1081        data <<COMMIT
1082        dirty directory copy
1083        COMMIT
1084        from refs/heads/branch^0
1086        M 644 inline file2/file5
1087        data <<EOF
1088        $file5_data
1089        EOF
1090        C file2 file3
1092        D file2/file5
1093        INPUT_END
1095        git fast-import <input &&
1097        test $(git rev-parse N2^{tree}) = $(git rev-parse N3^{tree})
1098'
1099test_expect_success 'N: copy directory by id' '
1101        cat >expect <<-\EOF &&
1102        :100755 100755 f1fb5da718392694d0076d677d6d0e364c79b0bc f1fb5da718392694d0076d677d6d0e364c79b0bc C100   file2/newf      file3/newf
1103        :100644 100644 7123f7f44e39be127c5eb701e5968176ee9d78b1 7123f7f44e39be127c5eb701e5968176ee9d78b1 C100   file2/oldf      file3/oldf
1104        EOF
1105        subdir=$(git rev-parse refs/heads/branch^0:file2) &&
1106        cat >input <<-INPUT_END &&
1107        commit refs/heads/N4
1108        committer $GIT_COMMITTER_NAME <$GIT_COMMITTER_EMAIL> $GIT_COMMITTER_DATE
1109        data <<COMMIT
1110        copy by tree hash
1111        COMMIT
1112        from refs/heads/branch^0
1114        M 040000 $subdir file3
1115        INPUT_END
1116        git fast-import <input &&
1117        git diff-tree -C --find-copies-harder -r N4^ N4 >actual &&
1118        compare_diff_raw expect actual
1119'
1120test_expect_success PIPE 'N: read and copy directory' '
1122        cat >expect <<-\EOF &&
1123        :100755 100755 f1fb5da718392694d0076d677d6d0e364c79b0bc f1fb5da718392694d0076d677d6d0e364c79b0bc C100   file2/newf      file3/newf
1124        :100644 100644 7123f7f44e39be127c5eb701e5968176ee9d78b1 7123f7f44e39be127c5eb701e5968176ee9d78b1 C100   file2/oldf      file3/oldf
1125        EOF
1126        git update-ref -d refs/heads/N4 &&
1127        rm -f backflow &&
1128        mkfifo backflow &&
1129        (
1130                exec <backflow &&
1131                cat <<-EOF &&
1132                commit refs/heads/N4
1133                committer $GIT_COMMITTER_NAME <$GIT_COMMITTER_EMAIL> $GIT_COMMITTER_DATE
1134                data <<COMMIT
1135                copy by tree hash, part 2
1136                COMMIT
1137                from refs/heads/branch^0
1139                ls "file2"
1140                EOF
1141                read mode type tree filename &&
1142                echo "M 040000 $tree file3"
1143        ) |
1144        git fast-import --cat-blob-fd=3 3>backflow &&
1145        git diff-tree -C --find-copies-harder -r N4^ N4 >actual &&
1146        compare_diff_raw expect actual
1147'
1148test_expect_success PIPE 'N: empty directory reads as missing' '
1150        cat <<-\EOF >expect &&
1151        OBJNAME
1152        :000000 100644 OBJNAME OBJNAME A        unrelated
1153        EOF
1154        echo "missing src" >expect.response &&
1155        git update-ref -d refs/heads/read-empty &&
1156        rm -f backflow &&
1157        mkfifo backflow &&
1158        (
1159                exec <backflow &&
1160                cat <<-EOF &&
1161                commit refs/heads/read-empty
1162                committer $GIT_COMMITTER_NAME <$GIT_COMMITTER_EMAIL> $GIT_COMMITTER_DATE
1163                data <<COMMIT
1164                read "empty" (missing) directory
1165                COMMIT
1166                M 100644 inline src/greeting
1168                data <<BLOB
1169                hello
1170                BLOB
1171                C src/greeting dst1/non-greeting
1172                C src/greeting unrelated
1173                # leave behind "empty" src directory
1174                D src/greeting
1175                ls "src"
1176                EOF
1177                read -r line &&
1178                printf "%s\n" "$line" >response &&
1179                cat <<-\EOF
1180                D dst1
1181                D dst2
1182                EOF
1183        ) |
1184        git fast-import --cat-blob-fd=3 3>backflow &&
1185        test_cmp expect.response response &&
1186        git rev-list read-empty |
1187        git diff-tree -r --root --stdin |
1188        sed "s/$OID_REGEX/OBJNAME/g" >actual &&
1189        test_cmp expect actual
1190'
1191test_expect_success 'N: copy root directory by tree hash' '
1193        cat >expect <<-\EOF &&
1194        :100755 000000 f1fb5da718392694d0076d677d6d0e364c79b0bc 0000000000000000000000000000000000000000 D      file3/newf
1195        :100644 000000 7123f7f44e39be127c5eb701e5968176ee9d78b1 0000000000000000000000000000000000000000 D      file3/oldf
1196        EOF
1197        root=$(git rev-parse refs/heads/branch^0^{tree}) &&
1198        cat >input <<-INPUT_END &&
1199        commit refs/heads/N6
1200        committer $GIT_COMMITTER_NAME <$GIT_COMMITTER_EMAIL> $GIT_COMMITTER_DATE
1201        data <<COMMIT
1202        copy root directory by tree hash
1203        COMMIT
1204        from refs/heads/branch^0
1206        M 040000 $root ""
1207        INPUT_END
1208        git fast-import <input &&
1209        git diff-tree -C --find-copies-harder -r N4 N6 >actual &&
1210        compare_diff_raw expect actual
1211'
1212test_expect_success 'N: copy root by path' '
1214        cat >expect <<-\EOF &&
1215        :100755 100755 f1fb5da718392694d0076d677d6d0e364c79b0bc f1fb5da718392694d0076d677d6d0e364c79b0bc C100   file2/newf      oldroot/file2/newf
1216        :100644 100644 7123f7f44e39be127c5eb701e5968176ee9d78b1 7123f7f44e39be127c5eb701e5968176ee9d78b1 C100   file2/oldf      oldroot/file2/oldf
1217        :100755 100755 85df50785d62d3b05ab03d9cbf7e4a0b49449730 85df50785d62d3b05ab03d9cbf7e4a0b49449730 C100   file4   oldroot/file4
1218        :100755 100755 e74b7d465e52746be2b4bae983670711e6e66657 e74b7d465e52746be2b4bae983670711e6e66657 C100   newdir/exec.sh  oldroot/newdir/exec.sh
1219        :100644 100644 fcf778cda181eaa1cbc9e9ce3a2e15ee9f9fe791 fcf778cda181eaa1cbc9e9ce3a2e15ee9f9fe791 C100   newdir/interesting      oldroot/newdir/interesting
1220        EOF
1221        cat >input <<-INPUT_END &&
1222        commit refs/heads/N-copy-root-path
1223        committer $GIT_COMMITTER_NAME <$GIT_COMMITTER_EMAIL> $GIT_COMMITTER_DATE
1224        data <<COMMIT
1225        copy root directory by (empty) path
1226        COMMIT
1227        from refs/heads/branch^0
1229        C "" oldroot
1230        INPUT_END
1231        git fast-import <input &&
1232        git diff-tree -C --find-copies-harder -r branch N-copy-root-path >actual &&
1233        compare_diff_raw expect actual
1234'
1235test_expect_success 'N: delete directory by copying' '
1237        cat >expect <<-\EOF &&
1238        OBJID
1239        :100644 000000 OBJID OBJID D    foo/bar/qux
1240        OBJID
1241        :000000 100644 OBJID OBJID A    foo/bar/baz
1242        :000000 100644 OBJID OBJID A    foo/bar/qux
1243        EOF
1244        empty_tree=$(git mktree </dev/null) &&
1245        cat >input <<-INPUT_END &&
1246        commit refs/heads/N-delete
1247        committer $GIT_COMMITTER_NAME <$GIT_COMMITTER_EMAIL> $GIT_COMMITTER_DATE
1248        data <<COMMIT
1249        collect data to be deleted
1250        COMMIT
1251        deleteall
1253        M 100644 inline foo/bar/baz
1254        data <<DATA_END
1255        hello
1256        DATA_END
1257        C "foo/bar/baz" "foo/bar/qux"
1258        C "foo/bar/baz" "foo/bar/quux/1"
1259        C "foo/bar/baz" "foo/bar/quuux"
1260        M 040000 $empty_tree foo/bar/quux
1261        M 040000 $empty_tree foo/bar/quuux
1262        commit refs/heads/N-delete
1264        committer $GIT_COMMITTER_NAME <$GIT_COMMITTER_EMAIL> $GIT_COMMITTER_DATE
1265        data <<COMMIT
1266        delete subdirectory
1267        COMMIT
1268        M 040000 $empty_tree foo/bar/qux
1270        INPUT_END
1271        git fast-import <input &&
1272        git rev-list N-delete |
1273                git diff-tree -r --stdin --root --always |
1274                sed -e "s/$OID_REGEX/OBJID/g" >actual &&
1275        test_cmp expect actual
1276'
1277test_expect_success 'N: modify copied tree' '
1279        cat >expect <<-\EOF &&
1280        :100644 100644 fcf778cda181eaa1cbc9e9ce3a2e15ee9f9fe791 fcf778cda181eaa1cbc9e9ce3a2e15ee9f9fe791 C100   newdir/interesting      file3/file5
1281        :100755 100755 f1fb5da718392694d0076d677d6d0e364c79b0bc f1fb5da718392694d0076d677d6d0e364c79b0bc C100   file2/newf      file3/newf
1282        :100644 100644 7123f7f44e39be127c5eb701e5968176ee9d78b1 7123f7f44e39be127c5eb701e5968176ee9d78b1 C100   file2/oldf      file3/oldf
1283        EOF
1284        subdir=$(git rev-parse refs/heads/branch^0:file2) &&
1285        cat >input <<-INPUT_END &&
1286        commit refs/heads/N5
1287        committer $GIT_COMMITTER_NAME <$GIT_COMMITTER_EMAIL> $GIT_COMMITTER_DATE
1288        data <<COMMIT
1289        copy by tree hash
1290        COMMIT
1291        from refs/heads/branch^0
1293        M 040000 $subdir file3
1294        commit refs/heads/N5
1296        committer $GIT_COMMITTER_NAME <$GIT_COMMITTER_EMAIL> $GIT_COMMITTER_DATE
1297        data <<COMMIT
1298        modify directory copy
1299        COMMIT
1300        M 644 inline file3/file5
1302        data <<EOF
1303        $file5_data
1304        EOF
1305        INPUT_END
1306        git fast-import <input &&
1307        git diff-tree -C --find-copies-harder -r N5^^ N5 >actual &&
1308        compare_diff_raw expect actual
1309'
1310test_expect_success 'N: reject foo/ syntax' '
1312        subdir=$(git rev-parse refs/heads/branch^0:file2) &&
1313        test_must_fail git fast-import <<-INPUT_END
1314        commit refs/heads/N5B
1315        committer $GIT_COMMITTER_NAME <$GIT_COMMITTER_EMAIL> $GIT_COMMITTER_DATE
1316        data <<COMMIT
1317        copy with invalid syntax
1318        COMMIT
1319        from refs/heads/branch^0
1321        M 040000 $subdir file3/
1322        INPUT_END
1323'
1324test_expect_success 'N: reject foo/ syntax in copy source' '
1326        test_must_fail git fast-import <<-INPUT_END
1327        commit refs/heads/N5C
1328        committer $GIT_COMMITTER_NAME <$GIT_COMMITTER_EMAIL> $GIT_COMMITTER_DATE
1329        data <<COMMIT
1330        copy with invalid syntax
1331        COMMIT
1332        from refs/heads/branch^0
1334        C file2/ file3
1335        INPUT_END
1336'
1337test_expect_success 'N: reject foo/ syntax in rename source' '
1339        test_must_fail git fast-import <<-INPUT_END
1340        commit refs/heads/N5D
1341        committer $GIT_COMMITTER_NAME <$GIT_COMMITTER_EMAIL> $GIT_COMMITTER_DATE
1342        data <<COMMIT
1343        rename with invalid syntax
1344        COMMIT
1345        from refs/heads/branch^0
1347        R file2/ file3
1348        INPUT_END
1349'
1350test_expect_success 'N: reject foo/ syntax in ls argument' '
1352        test_must_fail git fast-import <<-INPUT_END
1353        commit refs/heads/N5E
1354        committer $GIT_COMMITTER_NAME <$GIT_COMMITTER_EMAIL> $GIT_COMMITTER_DATE
1355        data <<COMMIT
1356        copy with invalid syntax
1357        COMMIT
1358        from refs/heads/branch^0
1360        ls "file2/"
1361        INPUT_END
1362'
1363test_expect_success 'N: copy to root by id and modify' '
1365        echo "hello, world" >expect.foo &&
1366        echo hello >expect.bar &&
1367        git fast-import <<-SETUP_END &&
1368        commit refs/heads/N7
1369        committer $GIT_COMMITTER_NAME <$GIT_COMMITTER_EMAIL> $GIT_COMMITTER_DATE
1370        data <<COMMIT
1371        hello, tree
1372        COMMIT
1373        deleteall
1375        M 644 inline foo/bar
1376        data <<EOF
1377        hello
1378        EOF
1379        SETUP_END
1380        tree=$(git rev-parse --verify N7:) &&
1382        git fast-import <<-INPUT_END &&
1383        commit refs/heads/N8
1384        committer $GIT_COMMITTER_NAME <$GIT_COMMITTER_EMAIL> $GIT_COMMITTER_DATE
1385        data <<COMMIT
1386        copy to root by id and modify
1387        COMMIT
1388        M 040000 $tree ""
1390        M 644 inline foo/foo
1391        data <<EOF
1392        hello, world
1393        EOF
1394        INPUT_END
1395        git show N8:foo/foo >actual.foo &&
1396        git show N8:foo/bar >actual.bar &&
1397        test_cmp expect.foo actual.foo &&
1398        test_cmp expect.bar actual.bar
1399'
1400test_expect_success 'N: extract subtree' '
1402        branch=$(git rev-parse --verify refs/heads/branch^{tree}) &&
1403        cat >input <<-INPUT_END &&
1404        commit refs/heads/N9
1405        committer $GIT_COMMITTER_NAME <$GIT_COMMITTER_EMAIL> $GIT_COMMITTER_DATE
1406        data <<COMMIT
1407        extract subtree branch:newdir
1408        COMMIT
1409        M 040000 $branch ""
1411        C "newdir" ""
1412        INPUT_END
1413        git fast-import <input &&
1414        git diff --exit-code branch:newdir N9
1415'
1416test_expect_success 'N: modify subtree, extract it, and modify again' '
1418        echo hello >expect.baz &&
1419        echo hello, world >expect.qux &&
1420        git fast-import <<-SETUP_END &&
1421        commit refs/heads/N10
1422        committer $GIT_COMMITTER_NAME <$GIT_COMMITTER_EMAIL> $GIT_COMMITTER_DATE
1423        data <<COMMIT
1424        hello, tree
1425        COMMIT
1426        deleteall
1428        M 644 inline foo/bar/baz
1429        data <<EOF
1430        hello
1431        EOF
1432        SETUP_END
1433        tree=$(git rev-parse --verify N10:) &&
1435        git fast-import <<-INPUT_END &&
1436        commit refs/heads/N11
1437        committer $GIT_COMMITTER_NAME <$GIT_COMMITTER_EMAIL> $GIT_COMMITTER_DATE
1438        data <<COMMIT
1439        copy to root by id and modify
1440        COMMIT
1441        M 040000 $tree ""
1443        M 100644 inline foo/bar/qux
1444        data <<EOF
1445        hello, world
1446        EOF
1447        R "foo" ""
1448        C "bar/qux" "bar/quux"
1449        INPUT_END
1450        git show N11:bar/baz >actual.baz &&
1451        git show N11:bar/qux >actual.qux &&
1452        git show N11:bar/quux >actual.quux &&
1453        test_cmp expect.baz actual.baz &&
1454        test_cmp expect.qux actual.qux &&
1455        test_cmp expect.qux actual.quux'
1456###
1458### series O
1459###
1460test_expect_success 'O: comments are all skipped' '
1462        cat >input <<-INPUT_END &&
1463        #we will
1464        commit refs/heads/O1
1465        # -- ignore all of this text
1466        committer $GIT_COMMITTER_NAME <$GIT_COMMITTER_EMAIL> $GIT_COMMITTER_DATE
1467        # $GIT_COMMITTER_NAME has inserted here for his benefit.
1468        data <<COMMIT
1469        dirty directory copy
1470        COMMIT
1471        # do not forget the import blank line!
1473        #
1474        # yes, we started from our usual base of branch^0.
1475        # i like branch^0.
1476        from refs/heads/branch^0
1477        # and we need to reuse file2/file5 from N3 above.
1478        M 644 inline file2/file5
1479        # otherwise the tree will be different
1480        data <<EOF
1481        $file5_data
1482        EOF
1483        # do not forget to copy file2 to file3
1485        C file2 file3
1486        #
1487        # or to delete file5 from file2.
1488        D file2/file5
1489        # are we done yet?
1490        INPUT_END
1492        git fast-import <input &&
1494        test $(git rev-parse N3) = $(git rev-parse O1)
1495'
1496test_expect_success 'O: blank lines not necessary after data commands' '
1498        cat >input <<-INPUT_END &&
1499        commit refs/heads/O2
1500        committer $GIT_COMMITTER_NAME <$GIT_COMMITTER_EMAIL> $GIT_COMMITTER_DATE
1501        data <<COMMIT
1502        dirty directory copy
1503        COMMIT
1504        from refs/heads/branch^0
1505        M 644 inline file2/file5
1506        data <<EOF
1507        $file5_data
1508        EOF
1509        C file2 file3
1510        D file2/file5
1511        INPUT_END
1513        git fast-import <input &&
1515        test $(git rev-parse N3) = $(git rev-parse O2)
1516'
1517test_expect_success 'O: repack before next test' '
1519        git repack -a -d
1520'
1521test_expect_success 'O: blank lines not necessary after other commands' '
1523        cat >input <<-INPUT_END &&
1524        commit refs/heads/O3
1525        committer $GIT_COMMITTER_NAME <$GIT_COMMITTER_EMAIL> $GIT_COMMITTER_DATE
1526        data <<COMMIT
1527        zstring
1528        COMMIT
1529        commit refs/heads/O3
1530        committer $GIT_COMMITTER_NAME <$GIT_COMMITTER_EMAIL> $GIT_COMMITTER_DATE
1531        data <<COMMIT
1532        zof
1533        COMMIT
1534        checkpoint
1535        commit refs/heads/O3
1536        mark :5
1537        committer $GIT_COMMITTER_NAME <$GIT_COMMITTER_EMAIL> $GIT_COMMITTER_DATE
1538        data <<COMMIT
1539        zempty
1540        COMMIT
1541        checkpoint
1542        commit refs/heads/O3
1543        committer $GIT_COMMITTER_NAME <$GIT_COMMITTER_EMAIL> $GIT_COMMITTER_DATE
1544        data <<COMMIT
1545        zcommits
1546        COMMIT
1547        reset refs/tags/O3-2nd
1548        from :5
1549        reset refs/tags/O3-3rd
1550        from :5
1551        INPUT_END
1552        cat >expect <<-INPUT_END &&
1554        string
1555        of
1556        empty
1557        commits
1558        INPUT_END
1559        git fast-import <input &&
1561        test 8 = $(find .git/objects/pack -type f | wc -l) &&
1562        test $(git rev-parse refs/tags/O3-2nd) = $(git rev-parse O3^) &&
1563        git log --reverse --pretty=oneline O3 | sed s/^.*z// >actual &&
1564        test_cmp expect actual
1565'
1566test_expect_success 'O: progress outputs as requested by input' '
1568        cat >input <<-INPUT_END &&
1569        commit refs/heads/O4
1570        committer $GIT_COMMITTER_NAME <$GIT_COMMITTER_EMAIL> $GIT_COMMITTER_DATE
1571        data <<COMMIT
1572        zstring
1573        COMMIT
1574        commit refs/heads/O4
1575        committer $GIT_COMMITTER_NAME <$GIT_COMMITTER_EMAIL> $GIT_COMMITTER_DATE
1576        data <<COMMIT
1577        zof
1578        COMMIT
1579        progress Two commits down, 2 to go!
1580        commit refs/heads/O4
1581        committer $GIT_COMMITTER_NAME <$GIT_COMMITTER_EMAIL> $GIT_COMMITTER_DATE
1582        data <<COMMIT
1583        zempty
1584        COMMIT
1585        progress Three commits down, 1 to go!
1586        commit refs/heads/O4
1587        committer $GIT_COMMITTER_NAME <$GIT_COMMITTER_EMAIL> $GIT_COMMITTER_DATE
1588        data <<COMMIT
1589        zcommits
1590        COMMIT
1591        progress done!
1592        INPUT_END
1593        git fast-import <input >actual &&
1594        grep "progress " <input >expect &&
1595        test_cmp expect actual
1596'
1597###
1599### series P (gitlinks)
1600###
1601test_expect_success 'P: superproject & submodule mix' '
1603        cat >input <<-INPUT_END &&
1604        blob
1605        mark :1
1606        data 10
1607        test file
1608        reset refs/heads/sub
1610        commit refs/heads/sub
1611        mark :2
1612        committer $GIT_COMMITTER_NAME <$GIT_COMMITTER_EMAIL> $GIT_COMMITTER_DATE
1613        data 12
1614        sub_initial
1615        M 100644 :1 file
1616        blob
1618        mark :3
1619        data <<DATAEND
1620        [submodule "sub"]
1621                path = sub
1622                url = "$(pwd)/sub"
1623        DATAEND
1624        commit refs/heads/subuse1
1626        mark :4
1627        committer $GIT_COMMITTER_NAME <$GIT_COMMITTER_EMAIL> $GIT_COMMITTER_DATE
1628        data 8
1629        initial
1630        from refs/heads/master
1631        M 100644 :3 .gitmodules
1632        M 160000 :2 sub
1633        blob
1635        mark :5
1636        data 20
1637        test file
1638        more data
1639        commit refs/heads/sub
1641        mark :6
1642        committer $GIT_COMMITTER_NAME <$GIT_COMMITTER_EMAIL> $GIT_COMMITTER_DATE
1643        data 11
1644        sub_second
1645        from :2
1646        M 100644 :5 file
1647        commit refs/heads/subuse1
1649        mark :7
1650        committer $GIT_COMMITTER_NAME <$GIT_COMMITTER_EMAIL> $GIT_COMMITTER_DATE
1651        data 7
1652        second
1653        from :4
1654        M 160000 :6 sub
1655        INPUT_END
1657        git fast-import <input &&
1659        git checkout subuse1 &&
1660        rm -rf sub &&
1661        mkdir sub &&
1662        (
1663                cd sub &&
1664                git init &&
1665                git fetch --update-head-ok .. refs/heads/sub:refs/heads/master &&
1666                git checkout master
1667        ) &&
1668        git submodule init &&
1669        git submodule update
1670'
1671test_expect_success 'P: verbatim SHA gitlinks' '
1673        SUBLAST=$(git rev-parse --verify sub) &&
1674        SUBPREV=$(git rev-parse --verify sub^) &&
1675        cat >input <<-INPUT_END &&
1677        blob
1678        mark :1
1679        data <<DATAEND
1680        [submodule "sub"]
1681                path = sub
1682                url = "$(pwd)/sub"
1683        DATAEND
1684        commit refs/heads/subuse2
1686        mark :2
1687        committer $GIT_COMMITTER_NAME <$GIT_COMMITTER_EMAIL> $GIT_COMMITTER_DATE
1688        data 8
1689        initial
1690        from refs/heads/master
1691        M 100644 :1 .gitmodules
1692        M 160000 $SUBPREV sub
1693        commit refs/heads/subuse2
1695        mark :3
1696        committer $GIT_COMMITTER_NAME <$GIT_COMMITTER_EMAIL> $GIT_COMMITTER_DATE
1697        data 7
1698        second
1699        from :2
1700        M 160000 $SUBLAST sub
1701        INPUT_END
1703        git branch -D sub &&
1705        git gc &&
1706        git prune &&
1707        git fast-import <input &&
1708        test $(git rev-parse --verify subuse2) = $(git rev-parse --verify subuse1)
1709'
1710test_expect_success 'P: fail on inline gitlink' '
1712        test_tick &&
1713        cat >input <<-INPUT_END &&
1714        commit refs/heads/subuse3
1715        mark :1
1716        committer $GIT_COMMITTER_NAME <$GIT_COMMITTER_EMAIL> $GIT_COMMITTER_DATE
1717        data <<COMMIT
1718        corrupt
1719        COMMIT
1720        from refs/heads/subuse2
1722        M 160000 inline sub
1723        data <<DATA
1724        $SUBPREV
1725        DATA
1726        INPUT_END
1728        test_must_fail git fast-import <input
1730'
1731test_expect_success 'P: fail on blob mark in gitlink' '
1733        test_tick &&
1734        cat >input <<-INPUT_END &&
1735        blob
1736        mark :1
1737        data <<DATA
1738        $SUBPREV
1739        DATA
1740        commit refs/heads/subuse3
1742        mark :2
1743        committer $GIT_COMMITTER_NAME <$GIT_COMMITTER_EMAIL> $GIT_COMMITTER_DATE
1744        data <<COMMIT
1745        corrupt
1746        COMMIT
1747        from refs/heads/subuse2
1749        M 160000 :1 sub
1750        INPUT_END
1752        test_must_fail git fast-import <input
1754'
1755###
1757### series Q (notes)
1758###
1759test_expect_success 'Q: commit notes' '
1761        note1_data="The first note for the first commit" &&
1762        note2_data="The first note for the second commit" &&
1763        note3_data="The first note for the third commit" &&
1764        note1b_data="The second note for the first commit" &&
1765        note1c_data="The third note for the first commit" &&
1766        note2b_data="The second note for the second commit" &&
1767        test_tick &&
1769        cat >input <<-INPUT_END &&
1770        blob
1771        mark :2
1772        data <<EOF
1773        $file2_data
1774        EOF
1775        commit refs/heads/notes-test
1777        mark :3
1778        committer $GIT_COMMITTER_NAME <$GIT_COMMITTER_EMAIL> $GIT_COMMITTER_DATE
1779        data <<COMMIT
1780        first (:3)
1781        COMMIT
1782        M 644 :2 file2
1784        blob
1786        mark :4
1787        data $file4_len
1788        $file4_data
1789        commit refs/heads/notes-test
1790        mark :5
1791        committer $GIT_COMMITTER_NAME <$GIT_COMMITTER_EMAIL> $GIT_COMMITTER_DATE
1792        data <<COMMIT
1793        second (:5)
1794        COMMIT
1795        M 644 :4 file4
1797        commit refs/heads/notes-test
1799        mark :6
1800        committer $GIT_COMMITTER_NAME <$GIT_COMMITTER_EMAIL> $GIT_COMMITTER_DATE
1801        data <<COMMIT
1802        third (:6)
1803        COMMIT
1804        M 644 inline file5
1806        data <<EOF
1807        $file5_data
1808        EOF
1809        M 755 inline file6
1811        data <<EOF
1812        $file6_data
1813        EOF
1814        blob
1816        mark :7
1817        data <<EOF
1818        $note1_data
1819        EOF
1820        blob
1822        mark :8
1823        data <<EOF
1824        $note2_data
1825        EOF
1826        commit refs/notes/foobar
1828        mark :9
1829        committer $GIT_COMMITTER_NAME <$GIT_COMMITTER_EMAIL> $GIT_COMMITTER_DATE
1830        data <<COMMIT
1831        notes (:9)
1832        COMMIT
1833        N :7 :3
1835        N :8 :5
1836        N inline :6
1837        data <<EOF
1838        $note3_data
1839        EOF
1840        commit refs/notes/foobar
1842        mark :10
1843        committer $GIT_COMMITTER_NAME <$GIT_COMMITTER_EMAIL> $GIT_COMMITTER_DATE
1844        data <<COMMIT
1845        notes (:10)
1846        COMMIT
1847        N inline :3
1849        data <<EOF
1850        $note1b_data
1851        EOF
1852        commit refs/notes/foobar2
1854        mark :11
1855        committer $GIT_COMMITTER_NAME <$GIT_COMMITTER_EMAIL> $GIT_COMMITTER_DATE
1856        data <<COMMIT
1857        notes (:11)
1858        COMMIT
1859        N inline :3
1861        data <<EOF
1862        $note1c_data
1863        EOF
1864        commit refs/notes/foobar
1866        mark :12
1867        committer $GIT_COMMITTER_NAME <$GIT_COMMITTER_EMAIL> $GIT_COMMITTER_DATE
1868        data <<COMMIT
1869        notes (:12)
1870        COMMIT
1871        deleteall
1873        N inline :5
1874        data <<EOF
1875        $note2b_data
1876        EOF
1877        INPUT_END
1879        git fast-import <input &&
1881        git whatchanged notes-test
1882'
1883test_expect_success 'Q: verify pack' '
1885        verify_packs
1886'
1887test_expect_success 'Q: verify first commit' '
1889        commit1=$(git rev-parse notes-test~2) &&
1890        commit2=$(git rev-parse notes-test^) &&
1891        commit3=$(git rev-parse notes-test) &&
1892        cat >expect <<-EOF &&
1894        author $GIT_COMMITTER_NAME <$GIT_COMMITTER_EMAIL> $GIT_COMMITTER_DATE
1895        committer $GIT_COMMITTER_NAME <$GIT_COMMITTER_EMAIL> $GIT_COMMITTER_DATE
1896        first (:3)
1898        EOF
1899        git cat-file commit notes-test~2 | sed 1d >actual &&
1900        test_cmp expect actual
1901'
1902test_expect_success 'Q: verify second commit' '
1904        cat >expect <<-EOF &&
1905        parent $commit1
1906        author $GIT_COMMITTER_NAME <$GIT_COMMITTER_EMAIL> $GIT_COMMITTER_DATE
1907        committer $GIT_COMMITTER_NAME <$GIT_COMMITTER_EMAIL> $GIT_COMMITTER_DATE
1908        second (:5)
1910        EOF
1911        git cat-file commit notes-test^ | sed 1d >actual &&
1912        test_cmp expect actual
1913'
1914test_expect_success 'Q: verify third commit' '
1916        cat >expect <<-EOF &&
1917        parent $commit2
1918        author $GIT_COMMITTER_NAME <$GIT_COMMITTER_EMAIL> $GIT_COMMITTER_DATE
1919        committer $GIT_COMMITTER_NAME <$GIT_COMMITTER_EMAIL> $GIT_COMMITTER_DATE
1920        third (:6)
1922        EOF
1923        git cat-file commit notes-test | sed 1d >actual &&
1924        test_cmp expect actual
1925'
1926test_expect_success 'Q: verify first notes commit' '
1928        cat >expect <<-EOF &&
1929        author $GIT_COMMITTER_NAME <$GIT_COMMITTER_EMAIL> $GIT_COMMITTER_DATE
1930        committer $GIT_COMMITTER_NAME <$GIT_COMMITTER_EMAIL> $GIT_COMMITTER_DATE
1931        notes (:9)
1933        EOF
1934        git cat-file commit refs/notes/foobar~2 | sed 1d >actual &&
1935        test_cmp expect actual
1936'
1937test_expect_success 'Q: verify first notes tree' '
1939        cat >expect.unsorted <<-EOF &&
1940        100644 blob $commit1
1941        100644 blob $commit2
1942        100644 blob $commit3
1943        EOF
1944        cat expect.unsorted | sort >expect &&
1945        git cat-file -p refs/notes/foobar~2^{tree} | sed "s/ [0-9a-f]*  / /" >actual &&
1946        test_cmp expect actual
1947'
1948test_expect_success 'Q: verify first note for first commit' '
1950        echo "$note1_data" >expect &&
1951        git cat-file blob refs/notes/foobar~2:$commit1 >actual &&
1952        test_cmp expect actual
1953'
1954test_expect_success 'Q: verify first note for second commit' '
1956        echo "$note2_data" >expect &&
1957        git cat-file blob refs/notes/foobar~2:$commit2 >actual &&
1958        test_cmp expect actual
1959'
1960test_expect_success 'Q: verify first note for third commit' '
1962        echo "$note3_data" >expect &&
1963        git cat-file blob refs/notes/foobar~2:$commit3 >actual &&
1964        test_cmp expect actual
1965'
1966test_expect_success 'Q: verify second notes commit' '
1968        cat >expect <<-EOF &&
1969        parent $(git rev-parse --verify refs/notes/foobar~2)
1970        author $GIT_COMMITTER_NAME <$GIT_COMMITTER_EMAIL> $GIT_COMMITTER_DATE
1971        committer $GIT_COMMITTER_NAME <$GIT_COMMITTER_EMAIL> $GIT_COMMITTER_DATE
1972        notes (:10)
1974        EOF
1975        git cat-file commit refs/notes/foobar^ | sed 1d >actual &&
1976        test_cmp expect actual
1977'
1978test_expect_success 'Q: verify second notes tree' '
1980        cat >expect.unsorted <<-EOF &&
1981        100644 blob $commit1
1982        100644 blob $commit2
1983        100644 blob $commit3
1984        EOF
1985        cat expect.unsorted | sort >expect &&
1986        git cat-file -p refs/notes/foobar^^{tree} | sed "s/ [0-9a-f]*   / /" >actual &&
1987        test_cmp expect actual
1988'
1989test_expect_success 'Q: verify second note for first commit' '
1991        echo "$note1b_data" >expect &&
1992        git cat-file blob refs/notes/foobar^:$commit1 >actual &&
1993        test_cmp expect actual
1994'
1995test_expect_success 'Q: verify first note for second commit' '
1997        echo "$note2_data" >expect &&
1998        git cat-file blob refs/notes/foobar^:$commit2 >actual &&
1999        test_cmp expect actual
2000'
2001test_expect_success 'Q: verify first note for third commit' '
2003        echo "$note3_data" >expect &&
2004        git cat-file blob refs/notes/foobar^:$commit3 >actual &&
2005        test_cmp expect actual
2006'
2007test_expect_success 'Q: verify third notes commit' '
2009        cat >expect <<-EOF &&
2010        author $GIT_COMMITTER_NAME <$GIT_COMMITTER_EMAIL> $GIT_COMMITTER_DATE
2011        committer $GIT_COMMITTER_NAME <$GIT_COMMITTER_EMAIL> $GIT_COMMITTER_DATE
2012        notes (:11)
2014        EOF
2015        git cat-file commit refs/notes/foobar2 | sed 1d >actual &&
2016        test_cmp expect actual
2017'
2018test_expect_success 'Q: verify third notes tree' '
2020        cat >expect.unsorted <<-EOF &&
2021        100644 blob $commit1
2022        EOF
2023        cat expect.unsorted | sort >expect &&
2024        git cat-file -p refs/notes/foobar2^{tree} | sed "s/ [0-9a-f]*   / /" >actual &&
2025        test_cmp expect actual
2026'
2027test_expect_success 'Q: verify third note for first commit' '
2029        echo "$note1c_data" >expect &&
2030        git cat-file blob refs/notes/foobar2:$commit1 >actual &&
2031        test_cmp expect actual
2032'
2033test_expect_success 'Q: verify fourth notes commit' '
2035        cat >expect <<-EOF &&
2036        parent $(git rev-parse --verify refs/notes/foobar^)
2037        author $GIT_COMMITTER_NAME <$GIT_COMMITTER_EMAIL> $GIT_COMMITTER_DATE
2038        committer $GIT_COMMITTER_NAME <$GIT_COMMITTER_EMAIL> $GIT_COMMITTER_DATE
2039        notes (:12)
2041        EOF
2042        git cat-file commit refs/notes/foobar | sed 1d >actual &&
2043        test_cmp expect actual
2044'
2045test_expect_success 'Q: verify fourth notes tree' '
2047        cat >expect.unsorted <<-EOF &&
2048        100644 blob $commit2
2049        EOF
2050        cat expect.unsorted | sort >expect &&
2051        git cat-file -p refs/notes/foobar^{tree} | sed "s/ [0-9a-f]*    / /" >actual &&
2052        test_cmp expect actual
2053'
2054test_expect_success 'Q: verify second note for second commit' '
2056        echo "$note2b_data" >expect &&
2057        git cat-file blob refs/notes/foobar:$commit2 >actual &&
2058        test_cmp expect actual
2059'
2060test_expect_success 'Q: deny note on empty branch' '
2062        cat >input <<-EOF &&
2063        reset refs/heads/Q0
2064        commit refs/heads/note-Q0
2066        committer $GIT_COMMITTER_NAME <$GIT_COMMITTER_EMAIL> $GIT_COMMITTER_DATE
2067        data <<COMMIT
2068        Note for an empty branch.
2069        COMMIT
2070        N inline refs/heads/Q0
2072        data <<NOTE
2073        some note
2074        NOTE
2075        EOF
2076        test_must_fail git fast-import <input
2077'
2078###
2079### series R (feature and option)
2080###
2081test_expect_success 'R: abort on unsupported feature' '
2083        cat >input <<-EOF &&
2084        feature no-such-feature-exists
2085        EOF
2086        test_must_fail git fast-import <input
2088'
2089test_expect_success 'R: supported feature is accepted' '
2091        cat >input <<-EOF &&
2092        feature date-format=now
2093        EOF
2094        git fast-import <input
2096'
2097test_expect_success 'R: abort on receiving feature after data command' '
2099        cat >input <<-EOF &&
2100        blob
2101        data 3
2102        hi
2103        feature date-format=now
2104        EOF
2105        test_must_fail git fast-import <input
2107'
2108test_expect_success 'R: only one import-marks feature allowed per stream' '
2110        cat >input <<-EOF &&
2111        feature import-marks=git.marks
2112        feature import-marks=git2.marks
2113        EOF
2114        test_must_fail git fast-import <input
2116'
2117test_expect_success 'R: export-marks feature results in a marks file being created' '
2119        cat >input <<-EOF &&
2120        feature export-marks=git.marks
2121        blob
2122        mark :1
2123        data 3
2124        hi
2125        EOF
2127        cat input | git fast-import &&
2129        grep :1 git.marks
2130'
2131test_expect_success 'R: export-marks options can be overridden by commandline options' '
2133        cat input | git fast-import --export-marks=other.marks &&
2134        grep :1 other.marks
2135'
2136test_expect_success 'R: catch typo in marks file name' '
2138        test_must_fail git fast-import --import-marks=nonexistent.marks </dev/null &&
2139        echo "feature import-marks=nonexistent.marks" |
2140        test_must_fail git fast-import
2141'
2142test_expect_success 'R: import and output marks can be the same file' '
2144        rm -f io.marks &&
2145        blob=$(echo hi | git hash-object --stdin) &&
2146        cat >expect <<-EOF &&
2147        :1 $blob
2148        :2 $blob
2149        EOF
2150        git fast-import --export-marks=io.marks <<-\EOF &&
2151        blob
2152        mark :1
2153        data 3
2154        hi
2155        EOF
2157        git fast-import --import-marks=io.marks --export-marks=io.marks <<-\EOF &&
2158        blob
2159        mark :2
2160        data 3
2161        hi
2162        EOF
2164        test_cmp expect io.marks
2165'
2166test_expect_success 'R: --import-marks=foo --output-marks=foo to create foo fails' '
2168        rm -f io.marks &&
2169        test_must_fail git fast-import --import-marks=io.marks --export-marks=io.marks <<-\EOF
2170        blob
2171        mark :1
2172        data 3
2173        hi
2174        EOF
2176'
2177test_expect_success 'R: --import-marks-if-exists' '
2179        rm -f io.marks &&
2180        blob=$(echo hi | git hash-object --stdin) &&
2181        echo ":1 $blob" >expect &&
2182        git fast-import --import-marks-if-exists=io.marks --export-marks=io.marks <<-\EOF &&
2183        blob
2184        mark :1
2185        data 3
2186        hi
2187        EOF
2189        test_cmp expect io.marks
2190'
2191test_expect_success 'R: feature import-marks-if-exists' '
2193        rm -f io.marks &&
2194        git fast-import --export-marks=io.marks <<-\EOF &&
2196        feature import-marks-if-exists=not_io.marks
2197        EOF
2198        test_must_be_empty io.marks &&
2199        blob=$(echo hi | git hash-object --stdin) &&
2201        echo ":1 $blob" >io.marks &&
2203        echo ":1 $blob" >expect &&
2204        echo ":2 $blob" >>expect &&
2205        git fast-import --export-marks=io.marks <<-\EOF &&
2207        feature import-marks-if-exists=io.marks
2208        blob
2209        mark :2
2210        data 3
2211        hi
2212        EOF
2214        test_cmp expect io.marks &&
2215        echo ":3 $blob" >>expect &&
2217        git fast-import --import-marks=io.marks \
2219                        --export-marks=io.marks <<-\EOF &&
2220        feature import-marks-if-exists=not_io.marks
2221        blob
2222        mark :3
2223        data 3
2224        hi
2225        EOF
2227        test_cmp expect io.marks &&
2228        git fast-import --import-marks-if-exists=not_io.marks \
2230                        --export-marks=io.marks <<-\EOF &&
2231        feature import-marks-if-exists=io.marks
2232        EOF
2233        test_must_be_empty io.marks
2234'
2235test_expect_success 'R: import to output marks works without any content' '
2237        cat >input <<-EOF &&
2238        feature import-marks=marks.out
2239        feature export-marks=marks.new
2240        EOF
2241        cat input | git fast-import &&
2243        test_cmp marks.out marks.new
2244'
2245test_expect_success 'R: import marks prefers commandline marks file over the stream' '
2247        cat >input <<-EOF &&
2248        feature import-marks=nonexistent.marks
2249        feature export-marks=marks.new
2250        EOF
2251        cat input | git fast-import --import-marks=marks.out &&
2253        test_cmp marks.out marks.new
2254'
2255test_expect_success 'R: multiple --import-marks= should be honoured' '
2258        cat >input <<-EOF &&
2259        feature import-marks=nonexistent.marks
2260        feature export-marks=combined.marks
2261        EOF
2262        head -n2 marks.out > one.marks &&
2264        tail -n +3 marks.out > two.marks &&
2265        git fast-import --import-marks=one.marks --import-marks=two.marks <input &&
2266        test_cmp marks.out combined.marks
2267'
2268test_expect_success 'R: feature relative-marks should be honoured' '
2270        cat >input <<-EOF &&
2271        feature relative-marks
2272        feature import-marks=relative.in
2273        feature export-marks=relative.out
2274        EOF
2275        mkdir -p .git/info/fast-import/ &&
2277        cp marks.new .git/info/fast-import/relative.in &&
2278        git fast-import <input &&
2279        test_cmp marks.new .git/info/fast-import/relative.out
2280'
2281test_expect_success 'R: feature no-relative-marks should be honoured' '
2283        cat >input <<-EOF &&
2284        feature relative-marks
2285        feature import-marks=relative.in
2286        feature no-relative-marks
2287        feature export-marks=non-relative.out
2288        EOF
2289        git fast-import <input &&
2291        test_cmp marks.new non-relative.out
2292'
2293test_expect_success 'R: feature ls supported' '
2295        echo "feature ls" |
2296        git fast-import
2297'
2298test_expect_success 'R: feature cat-blob supported' '
2300        echo "feature cat-blob" |
2301        git fast-import
2302'
2303test_expect_success 'R: cat-blob-fd must be a nonnegative integer' '
2305        test_must_fail git fast-import --cat-blob-fd=-1 </dev/null
2306'
2307test_expect_success !MINGW 'R: print old blob' '
2309        blob=$(echo "yes it can" | git hash-object -w --stdin) &&
2310        cat >expect <<-EOF &&
2311        ${blob} blob 11
2312        yes it can
2313        EOF
2315        echo "cat-blob $blob" |
2316        git fast-import --cat-blob-fd=6 6>actual &&
2317        test_cmp expect actual
2318'
2319test_expect_success !MINGW 'R: in-stream cat-blob-fd not respected' '
2321        echo hello >greeting &&
2322        blob=$(git hash-object -w greeting) &&
2323        cat >expect <<-EOF &&
2324        ${blob} blob 6
2325        hello
2326        EOF
2328        git fast-import --cat-blob-fd=3 3>actual.3 >actual.1 <<-EOF &&
2329        cat-blob $blob
2330        EOF
2331        test_cmp expect actual.3 &&
2332        test_must_be_empty actual.1 &&
2333        git fast-import 3>actual.3 >actual.1 <<-EOF &&
2334        option cat-blob-fd=3
2335        cat-blob $blob
2336        EOF
2337        test_must_be_empty actual.3 &&
2338        test_cmp expect actual.1
2339'
2340test_expect_success !MINGW 'R: print mark for new blob' '
2342        echo "effluentish" | git hash-object --stdin >expect &&
2343        git fast-import --cat-blob-fd=6 6>actual <<-\EOF &&
2344        blob
2345        mark :1
2346        data <<BLOB_END
2347        effluentish
2348        BLOB_END
2349        get-mark :1
2350        EOF
2351        test_cmp expect actual
2352'
2353test_expect_success !MINGW 'R: print new blob' '
2355        blob=$(echo "yep yep yep" | git hash-object --stdin) &&
2356        cat >expect <<-EOF &&
2357        ${blob} blob 12
2358        yep yep yep
2359        EOF
2361        git fast-import --cat-blob-fd=6 6>actual <<-\EOF &&
2362        blob
2363        mark :1
2364        data <<BLOB_END
2365        yep yep yep
2366        BLOB_END
2367        cat-blob :1
2368        EOF
2369        test_cmp expect actual
2370'
2371test_expect_success !MINGW 'R: print new blob by sha1' '
2373        blob=$(echo "a new blob named by sha1" | git hash-object --stdin) &&
2374        cat >expect <<-EOF &&
2375        ${blob} blob 25
2376        a new blob named by sha1
2377        EOF
2379        git fast-import --cat-blob-fd=6 6>actual <<-EOF &&
2380        blob
2381        data <<BLOB_END
2382        a new blob named by sha1
2383        BLOB_END
2384        cat-blob $blob
2385        EOF
2386        test_cmp expect actual
2387'
2388test_expect_success 'setup: big file' '
2390        (
2391                echo "the quick brown fox jumps over the lazy dog" >big &&
2392                for i in 1 2 3
2393                do
2394                        cat big big big big >bigger &&
2395                        cat bigger bigger bigger bigger >big ||
2396                        exit
2397                done
2398        )
2399'
2400test_expect_success 'R: print two blobs to stdout' '
2402        blob1=$(git hash-object big) &&
2403        blob1_len=$(wc -c <big) &&
2404        blob2=$(echo hello | git hash-object --stdin) &&
2405        {
2406                echo ${blob1} blob $blob1_len &&
2407                cat big &&
2408                cat <<-EOF
2409                ${blob2} blob 6
2411                hello
2412                EOF
2414        } >expect &&
2415        {
2416                cat <<-\END_PART1 &&
2417                        blob
2418                        mark :1
2419                        data <<data_end
2420                END_PART1
2421                cat big &&
2422                cat <<-\EOF
2423                        data_end
2424                        blob
2425                        mark :2
2426                        data <<data_end
2427                        hello
2428                        data_end
2429                        cat-blob :1
2430                        cat-blob :2
2431                EOF
2432        } |
2433        git fast-import >actual &&
2434        test_cmp expect actual
2435'
2436test_expect_success PIPE 'R: copy using cat-file' '
2438        expect_id=$(git hash-object big) &&
2439        expect_len=$(wc -c <big) &&
2440        echo $expect_id blob $expect_len >expect.response &&
2441        rm -f blobs &&
2443        cat >frontend <<-\FRONTEND_END &&
2444        #!/bin/sh
2445        FRONTEND_END
2446        mkfifo blobs &&
2448        (
2449                export GIT_COMMITTER_NAME GIT_COMMITTER_EMAIL GIT_COMMITTER_DATE &&
2450                cat <<-\EOF &&
2451                feature cat-blob
2452                blob
2453                mark :1
2454                data <<BLOB
2455                EOF
2456                cat big &&
2457                cat <<-\EOF &&
2458                BLOB
2459                cat-blob :1
2460                EOF
2461                read blob_id type size <&3 &&
2463                echo "$blob_id $type $size" >response &&
2464                test_copy_bytes $size >blob <&3 &&
2465                read newline <&3 &&
2466                cat <<-EOF &&
2468                commit refs/heads/copied
2469                committer $GIT_COMMITTER_NAME <$GIT_COMMITTER_EMAIL> $GIT_COMMITTER_DATE
2470                data <<COMMIT
2471                copy big file as file3
2472                COMMIT
2473                M 644 inline file3
2474                data <<BLOB
2475                EOF
2476                cat blob &&
2477                echo BLOB
2478        ) 3<blobs |
2479        git fast-import --cat-blob-fd=3 3>blobs &&
2480        git show copied:file3 >actual &&
2481        test_cmp expect.response response &&
2482        test_cmp big actual
2483'
2484test_expect_success PIPE 'R: print blob mid-commit' '
2486        rm -f blobs &&
2487        echo "A blob from _before_ the commit." >expect &&
2488        mkfifo blobs &&
2489        (
2490                exec 3<blobs &&
2491                cat <<-EOF &&
2492                feature cat-blob
2493                blob
2494                mark :1
2495                data <<BLOB
2496                A blob from _before_ the commit.
2497                BLOB
2498                commit refs/heads/temporary
2499                committer $GIT_COMMITTER_NAME <$GIT_COMMITTER_EMAIL> $GIT_COMMITTER_DATE
2500                data <<COMMIT
2501                Empty commit
2502                COMMIT
2503                cat-blob :1
2504                EOF
2505                read blob_id type size <&3 &&
2507                test_copy_bytes $size >actual <&3 &&
2508                read newline <&3 &&
2509                echo
2511        ) |
2512        git fast-import --cat-blob-fd=3 3>blobs &&
2513        test_cmp expect actual
2514'
2515test_expect_success PIPE 'R: print staged blob within commit' '
2517        rm -f blobs &&
2518        echo "A blob from _within_ the commit." >expect &&
2519        mkfifo blobs &&
2520        (
2521                exec 3<blobs &&
2522                cat <<-EOF &&
2523                feature cat-blob
2524                commit refs/heads/within
2525                committer $GIT_COMMITTER_NAME <$GIT_COMMITTER_EMAIL> $GIT_COMMITTER_DATE
2526                data <<COMMIT
2527                Empty commit
2528                COMMIT
2529                M 644 inline within
2530                data <<BLOB
2531                A blob from _within_ the commit.
2532                BLOB
2533                EOF
2534                to_get=$(
2536                        echo "A blob from _within_ the commit." |
2537                        git hash-object --stdin
2538                ) &&
2539                echo "cat-blob $to_get" &&
2540                read blob_id type size <&3 &&
2542                test_copy_bytes $size >actual <&3 &&
2543                read newline <&3 &&
2544                echo deleteall
2546        ) |
2547        git fast-import --cat-blob-fd=3 3>blobs &&
2548        test_cmp expect actual
2549'
2550test_expect_success 'R: quiet option results in no stats being output' '
2552        cat >input <<-EOF &&
2553        option git quiet
2554        blob
2555        data 3
2556        hi
2557        EOF
2559        cat input | git fast-import 2> output &&
2561        test_must_be_empty output
2562'
2563test_expect_success 'R: feature done means terminating "done" is mandatory' '
2565        echo feature done | test_must_fail git fast-import &&
2566        test_must_fail git fast-import --done </dev/null
2567'
2568test_expect_success 'R: terminating "done" with trailing gibberish is ok' '
2570        git fast-import <<-\EOF &&
2571        feature done
2572        done
2573        trailing gibberish
2574        EOF
2575        git fast-import <<-\EOF
2576        done
2577        more trailing gibberish
2578        EOF
2579'
2580test_expect_success 'R: terminating "done" within commit' '
2582        cat >expect <<-\EOF &&
2583        OBJID
2584        :000000 100644 OBJID OBJID A    hello.c
2585        :000000 100644 OBJID OBJID A    hello2.c
2586        EOF
2587        git fast-import <<-EOF &&
2588        commit refs/heads/done-ends
2589        committer $GIT_COMMITTER_NAME <$GIT_COMMITTER_EMAIL> $GIT_COMMITTER_DATE
2590        data <<EOT
2591        Commit terminated by "done" command
2592        EOT
2593        M 100644 inline hello.c
2594        data <<EOT
2595        Hello, world.
2596        EOT
2597        C hello.c hello2.c
2598        done
2599        EOF
2600        git rev-list done-ends |
2601        git diff-tree -r --stdin --root --always |
2602        sed -e "s/$OID_REGEX/OBJID/g" >actual &&
2603        test_cmp expect actual
2604'
2605test_expect_success 'R: die on unknown option' '
2607        cat >input <<-EOF &&
2608        option git non-existing-option
2609        EOF
2610        test_must_fail git fast-import <input
2612'
2613test_expect_success 'R: unknown commandline options are rejected' '\
2615        test_must_fail git fast-import --non-existing-option < /dev/null
2616'
2617test_expect_success 'R: die on invalid option argument' '
2619        echo "option git active-branches=-5" |
2620        test_must_fail git fast-import &&
2621        echo "option git depth=" |
2622        test_must_fail git fast-import &&
2623        test_must_fail git fast-import --depth="5 elephants" </dev/null
2624'
2625test_expect_success 'R: ignore non-git options' '
2627        cat >input <<-EOF &&
2628        option non-existing-vcs non-existing-option
2629        EOF
2630        git fast-import <input
2632'
2633test_expect_success 'R: corrupt lines do not mess marks file' '
2635        rm -f io.marks &&
2636        blob=$(echo hi | git hash-object --stdin) &&
2637        cat >expect <<-EOF &&
2638        :3 0000000000000000000000000000000000000000
2639        :1 $blob
2640        :2 $blob
2641        EOF
2642        cp expect io.marks &&
2643        test_must_fail git fast-import --import-marks=io.marks --export-marks=io.marks <<-\EOF &&
2644        EOF
2646        test_cmp expect io.marks
2647'
2648##
2650## R: very large blobs
2651##
2652test_expect_success 'R: blob bigger than threshold' '
2653        blobsize=$((2*1024*1024 + 53)) &&
2654        test-tool genrandom bar $blobsize >expect &&
2655        cat >input <<-INPUT_END &&
2656        commit refs/heads/big-file
2657        committer $GIT_COMMITTER_NAME <$GIT_COMMITTER_EMAIL> $GIT_COMMITTER_DATE
2658        data <<COMMIT
2659        R - big file
2660        COMMIT
2661        M 644 inline big1
2663        data $blobsize
2664        INPUT_END
2665        cat expect >>input &&
2666        cat >>input <<-INPUT_END &&
2667        M 644 inline big2
2668        data $blobsize
2669        INPUT_END
2670        cat expect >>input &&
2671        echo >>input &&
2672        test_create_repo R &&
2674        git --git-dir=R/.git config fastimport.unpackLimit 0 &&
2675        git --git-dir=R/.git fast-import --big-file-threshold=1 <input
2676'
2677test_expect_success 'R: verify created pack' '
2679        (
2680                cd R &&
2681                verify_packs -v > ../verify
2682        )
2683'
2684test_expect_success 'R: verify written objects' '
2686        git --git-dir=R/.git cat-file blob big-file:big1 >actual &&
2687        test_cmp_bin expect actual &&
2688        a=$(git --git-dir=R/.git rev-parse big-file:big1) &&
2689        b=$(git --git-dir=R/.git rev-parse big-file:big2) &&
2690        test $a = $b
2691'
2692test_expect_success 'R: blob appears only once' '
2694        n=$(grep $a verify | wc -l) &&
2695        test 1 = $n
2696'
2697###
2699### series S
2700###
2701#
2702# Make sure missing spaces and EOLs after mark references
2703# cause errors.
2704#
2705# Setup:
2706#
2707#   1--2--4
2708#    \   /
2709#     -3-
2710#
2711#   commit marks:  301, 302, 303, 304
2712#   blob marks:              403, 404, resp.
2713#   note mark:          202
2714#
2715# The error message when a space is missing not at the
2716# end of the line is:
2717#
2718#   Missing space after ..
2719#
2720# or when extra characters come after the mark at the end
2721# of the line:
2722#
2723#   Garbage after ..
2724#
2725# or when the dataref is neither "inline " or a known SHA1,
2726#
2727#   Invalid dataref ..
2728#
2729test_expect_success 'S: initialize for S tests' '
2730        test_tick &&
2731        cat >input <<-INPUT_END &&
2733        commit refs/heads/S
2734        mark :301
2735        committer $GIT_COMMITTER_NAME <$GIT_COMMITTER_EMAIL> $GIT_COMMITTER_DATE
2736        data <<COMMIT
2737        commit 1
2738        COMMIT
2739        M 100644 inline hello.c
2740        data <<BLOB
2741        blob 1
2742        BLOB
2743        commit refs/heads/S
2745        mark :302
2746        committer $GIT_COMMITTER_NAME <$GIT_COMMITTER_EMAIL> $GIT_COMMITTER_DATE
2747        data <<COMMIT
2748        commit 2
2749        COMMIT
2750        from :301
2751        M 100644 inline hello.c
2752        data <<BLOB
2753        blob 2
2754        BLOB
2755        blob
2757        mark :403
2758        data <<BLOB
2759        blob 3
2760        BLOB
2761        blob
2763        mark :202
2764        data <<BLOB
2765        note 2
2766        BLOB
2767        INPUT_END
2768        git fast-import --export-marks=marks <input
2770'
2771#
2773# filemodify, three datarefs
2774#
2775test_expect_success 'S: filemodify with garbage after mark must fail' '
2776        test_must_fail git fast-import --import-marks=marks <<-EOF 2>err &&
2777        commit refs/heads/S
2778        committer $GIT_COMMITTER_NAME <$GIT_COMMITTER_EMAIL> $GIT_COMMITTER_DATE
2779        data <<COMMIT
2780        commit N
2781        COMMIT
2782        M 100644 :403x hello.c
2783        EOF
2784        cat err &&
2785        test_i18ngrep "space after mark" err
2786'
2787# inline is misspelled; fast-import thinks it is some unknown dataref
2789test_expect_success 'S: filemodify with garbage after inline must fail' '
2790        test_must_fail git fast-import --import-marks=marks <<-EOF 2>err &&
2791        commit refs/heads/S
2792        committer $GIT_COMMITTER_NAME <$GIT_COMMITTER_EMAIL> $GIT_COMMITTER_DATE
2793        data <<COMMIT
2794        commit N
2795        COMMIT
2796        M 100644 inlineX hello.c
2797        data <<BLOB
2798        inline
2799        BLOB
2800        EOF
2801        cat err &&
2802        test_i18ngrep "nvalid dataref" err
2803'
2804test_expect_success 'S: filemodify with garbage after sha1 must fail' '
2806        sha1=$(grep :403 marks | cut -d\  -f2) &&
2807        test_must_fail git fast-import --import-marks=marks <<-EOF 2>err &&
2808        commit refs/heads/S
2809        committer $GIT_COMMITTER_NAME <$GIT_COMMITTER_EMAIL> $GIT_COMMITTER_DATE
2810        data <<COMMIT
2811        commit N
2812        COMMIT
2813        M 100644 ${sha1}x hello.c
2814        EOF
2815        cat err &&
2816        test_i18ngrep "space after SHA1" err
2817'
2818#
2820# notemodify, three ways to say dataref
2821#
2822test_expect_success 'S: notemodify with garbage after mark dataref must fail' '
2823        test_must_fail git fast-import --import-marks=marks <<-EOF 2>err &&
2824        commit refs/heads/S
2825        committer $GIT_COMMITTER_NAME <$GIT_COMMITTER_EMAIL> $GIT_COMMITTER_DATE
2826        data <<COMMIT
2827        commit S note dataref markref
2828        COMMIT
2829        N :202x :302
2830        EOF
2831        cat err &&
2832        test_i18ngrep "space after mark" err
2833'
2834test_expect_success 'S: notemodify with garbage after inline dataref must fail' '
2836        test_must_fail git fast-import --import-marks=marks <<-EOF 2>err &&
2837        commit refs/heads/S
2838        committer $GIT_COMMITTER_NAME <$GIT_COMMITTER_EMAIL> $GIT_COMMITTER_DATE
2839        data <<COMMIT
2840        commit S note dataref inline
2841        COMMIT
2842        N inlineX :302
2843        data <<BLOB
2844        note blob
2845        BLOB
2846        EOF
2847        cat err &&
2848        test_i18ngrep "nvalid dataref" err
2849'
2850test_expect_success 'S: notemodify with garbage after sha1 dataref must fail' '
2852        sha1=$(grep :202 marks | cut -d\  -f2) &&
2853        test_must_fail git fast-import --import-marks=marks <<-EOF 2>err &&
2854        commit refs/heads/S
2855        committer $GIT_COMMITTER_NAME <$GIT_COMMITTER_EMAIL> $GIT_COMMITTER_DATE
2856        data <<COMMIT
2857        commit S note dataref sha1
2858        COMMIT
2859        N ${sha1}x :302
2860        EOF
2861        cat err &&
2862        test_i18ngrep "space after SHA1" err
2863'
2864#
2866# notemodify, mark in commit-ish
2867#
2868test_expect_success 'S: notemodify with garbage after mark commit-ish must fail' '
2869        test_must_fail git fast-import --import-marks=marks <<-EOF 2>err &&
2870        commit refs/heads/Snotes
2871        committer $GIT_COMMITTER_NAME <$GIT_COMMITTER_EMAIL> $GIT_COMMITTER_DATE
2872        data <<COMMIT
2873        commit S note commit-ish
2874        COMMIT
2875        N :202 :302x
2876        EOF
2877        cat err &&
2878        test_i18ngrep "after mark" err
2879'
2880#
2882# from
2883#
2884test_expect_success 'S: from with garbage after mark must fail' '
2885        test_must_fail \
2886        git fast-import --import-marks=marks --export-marks=marks <<-EOF 2>err &&
2887        commit refs/heads/S2
2888        mark :303
2889        committer $GIT_COMMITTER_NAME <$GIT_COMMITTER_EMAIL> $GIT_COMMITTER_DATE
2890        data <<COMMIT
2891        commit 3
2892        COMMIT
2893        from :301x
2894        M 100644 :403 hello.c
2895        EOF
2896        # go create the commit, need it for merge test
2899        git fast-import --import-marks=marks --export-marks=marks <<-EOF &&
2900        commit refs/heads/S2
2901        mark :303
2902        committer $GIT_COMMITTER_NAME <$GIT_COMMITTER_EMAIL> $GIT_COMMITTER_DATE
2903        data <<COMMIT
2904        commit 3
2905        COMMIT
2906        from :301
2907        M 100644 :403 hello.c
2908        EOF
2909        # now evaluate the error
2911        cat err &&
2912        test_i18ngrep "after mark" err
2913'
2914#
2917# merge
2918#
2919test_expect_success 'S: merge with garbage after mark must fail' '
2920        test_must_fail git fast-import --import-marks=marks <<-EOF 2>err &&
2921        commit refs/heads/S
2922        mark :304
2923        committer $GIT_COMMITTER_NAME <$GIT_COMMITTER_EMAIL> $GIT_COMMITTER_DATE
2924        data <<COMMIT
2925        merge 4
2926        COMMIT
2927        from :302
2928        merge :303x
2929        M 100644 :403 hello.c
2930        EOF
2931        cat err &&
2932        test_i18ngrep "after mark" err
2933'
2934#
2936# tag, from markref
2937#
2938test_expect_success 'S: tag with garbage after mark must fail' '
2939        test_must_fail git fast-import --import-marks=marks <<-EOF 2>err &&
2940        tag refs/tags/Stag
2941        from :302x
2942        tagger $GIT_COMMITTER_NAME <$GIT_COMMITTER_EMAIL> $GIT_COMMITTER_DATE
2943        data <<TAG
2944        tag S
2945        TAG
2946        EOF
2947        cat err &&
2948        test_i18ngrep "after mark" err
2949'
2950#
2952# cat-blob markref
2953#
2954test_expect_success 'S: cat-blob with garbage after mark must fail' '
2955        test_must_fail git fast-import --import-marks=marks <<-EOF 2>err &&
2956        cat-blob :403x
2957        EOF
2958        cat err &&
2959        test_i18ngrep "after mark" err
2960'
2961#
2963# ls markref
2964#
2965test_expect_success 'S: ls with garbage after mark must fail' '
2966        test_must_fail git fast-import --import-marks=marks <<-EOF 2>err &&
2967        ls :302x hello.c
2968        EOF
2969        cat err &&
2970        test_i18ngrep "space after mark" err
2971'
2972test_expect_success 'S: ls with garbage after sha1 must fail' '
2974        sha1=$(grep :302 marks | cut -d\  -f2) &&
2975        test_must_fail git fast-import --import-marks=marks <<-EOF 2>err &&
2976        ls ${sha1}x hello.c
2977        EOF
2978        cat err &&
2979        test_i18ngrep "space after tree-ish" err
2980'
2981###
2983### series T (ls)
2984###
2985# Setup is carried over from series S.
2986test_expect_success 'T: ls root tree' '
2988        sed -e "s/Z\$//" >expect <<-EOF &&
2989        040000 tree $(git rev-parse S^{tree})   Z
2990        EOF
2991        sha1=$(git rev-parse --verify S) &&
2992        git fast-import --import-marks=marks <<-EOF >actual &&
2993        ls $sha1 ""
2994        EOF
2995        test_cmp expect actual
2996'
2997test_expect_success 'T: delete branch' '
2999        git branch to-delete &&
3000        git fast-import <<-EOF &&
3001        reset refs/heads/to-delete
3002        from 0000000000000000000000000000000000000000
3003        EOF
3004        test_must_fail git rev-parse --verify refs/heads/to-delete
3005'
3006test_expect_success 'T: empty reset doesnt delete branch' '
3008        git branch not-to-delete &&
3009        git fast-import <<-EOF &&
3010        reset refs/heads/not-to-delete
3011        EOF
3012        git show-ref &&
3013        git rev-parse --verify refs/heads/not-to-delete
3014'
3015###
3017### series U (filedelete)
3018###
3019test_expect_success 'U: initialize for U tests' '
3021        cat >input <<-INPUT_END &&
3022        commit refs/heads/U
3023        committer $GIT_COMMITTER_NAME <$GIT_COMMITTER_EMAIL> $GIT_COMMITTER_DATE
3024        data <<COMMIT
3025        test setup
3026        COMMIT
3027        M 100644 inline hello.c
3028        data <<BLOB
3029        blob 1
3030        BLOB
3031        M 100644 inline good/night.txt
3032        data <<BLOB
3033        sleep well
3034        BLOB
3035        M 100644 inline good/bye.txt
3036        data <<BLOB
3037        au revoir
3038        BLOB
3039        INPUT_END
3041        git fast-import <input
3043'
3044test_expect_success 'U: filedelete file succeeds' '
3046        cat >input <<-INPUT_END &&
3047        commit refs/heads/U
3048        committer $GIT_COMMITTER_NAME <$GIT_COMMITTER_EMAIL> $GIT_COMMITTER_DATE
3049        data <<COMMIT
3050        delete good/night.txt
3051        COMMIT
3052        from refs/heads/U^0
3053        D good/night.txt
3054        INPUT_END
3056        git fast-import <input
3058'
3059test_expect_success 'U: validate file delete result' '
3061        cat >expect <<-EOF &&
3062        :100644 000000 2907ebb4bf85d91bf0716bb3bd8a68ef48d6da76 0000000000000000000000000000000000000000 D      good/night.txt
3063        EOF
3064        git diff-tree -M -r U^1 U >actual &&
3066        compare_diff_raw expect actual
3068'
3069test_expect_success 'U: filedelete directory succeeds' '
3071        cat >input <<-INPUT_END &&
3072        commit refs/heads/U
3073        committer $GIT_COMMITTER_NAME <$GIT_COMMITTER_EMAIL> $GIT_COMMITTER_DATE
3074        data <<COMMIT
3075        delete good dir
3076        COMMIT
3077        from refs/heads/U^0
3078        D good
3079        INPUT_END
3081        git fast-import <input
3083'
3084test_expect_success 'U: validate directory delete result' '
3086        cat >expect <<-EOF &&
3087        :100644 000000 69cb75792f55123d8389c156b0b41c2ff00ed507 0000000000000000000000000000000000000000 D      good/bye.txt
3088        EOF
3089        git diff-tree -M -r U^1 U >actual &&
3091        compare_diff_raw expect actual
3093'
3094test_expect_success 'U: filedelete root succeeds' '
3096        cat >input <<-INPUT_END &&
3097        commit refs/heads/U
3098        committer $GIT_COMMITTER_NAME <$GIT_COMMITTER_EMAIL> $GIT_COMMITTER_DATE
3099        data <<COMMIT
3100        must succeed
3101        COMMIT
3102        from refs/heads/U^0
3103        D ""
3104        INPUT_END
3106        git fast-import <input
3108'
3109test_expect_success 'U: validate root delete result' '
3111        cat >expect <<-EOF &&
3112        :100644 000000 c18147dc648481eeb65dc5e66628429a64843327 0000000000000000000000000000000000000000 D      hello.c
3113        EOF
3114        git diff-tree -M -r U^1 U >actual &&
3116        compare_diff_raw expect actual
3118'
3119###
3121### series V (checkpoint)
3122###
3123# The commands in input_file should not produce any output on the file
3125# descriptor set with --cat-blob-fd (or stdout if unspecified).
3126#
3127# To make sure you're observing the side effects of checkpoint *before*
3128# fast-import terminates (and thus writes out its state), check that the
3129# fast-import process is still running using background_import_still_running
3130# *after* evaluating the test conditions.
3131background_import_then_checkpoint () {
3132        options=$1
3133        input_file=$2
3134        mkfifo V.input
3136        exec 8<>V.input
3137        rm V.input
3138        mkfifo V.output
3140        exec 9<>V.output
3141        rm V.output
3142        git fast-import $options <&8 >&9 &
3144        echo $! >V.pid
3145        # We don't mind if fast-import has already died by the time the test
3146        # ends.
3147        test_when_finished "
3148                exec 8>&-; exec 9>&-;
3149                kill $(cat V.pid) && wait $(cat V.pid)
3150                true"
3151        # Start in the background to ensure we adhere strictly to (blocking)
3153        # pipes writing sequence. We want to assume that the write below could
3154        # block, e.g. if fast-import blocks writing its own output to &9
3155        # because there is no reader on &9 yet.
3156        (
3157                cat "$input_file"
3158                echo "checkpoint"
3159                echo "progress checkpoint"
3160        ) >&8 &
3161        error=1 ;# assume the worst
3163        while read output <&9
3164        do
3165                if test "$output" = "progress checkpoint"
3166                then
3167                        error=0
3168                        break
3169                fi
3170                # otherwise ignore cruft
3171                echo >&2 "cruft: $output"
3172        done
3173        if test $error -eq 1
3175        then
3176                false
3177        fi
3178}
3179background_import_still_running () {
3181        if ! kill -0 "$(cat V.pid)"
3182        then
3183                echo >&2 "background fast-import terminated too early"
3184                false
3185        fi
3186}
3187test_expect_success PIPE 'V: checkpoint helper does not get stuck with extra output' '
3189        cat >input <<-INPUT_END &&
3190        progress foo
3191        progress bar
3192        INPUT_END
3194        background_import_then_checkpoint "" input &&
3196        background_import_still_running
3197'
3198test_expect_success PIPE 'V: checkpoint updates refs after reset' '
3200        cat >input <<-\INPUT_END &&
3201        reset refs/heads/V
3202        from refs/heads/U
3203        INPUT_END
3205        background_import_then_checkpoint "" input &&
3207        test "$(git rev-parse --verify V)" = "$(git rev-parse --verify U)" &&
3208        background_import_still_running
3209'
3210test_expect_success PIPE 'V: checkpoint updates refs and marks after commit' '
3212        cat >input <<-INPUT_END &&
3213        commit refs/heads/V
3214        mark :1
3215        committer $GIT_COMMITTER_NAME <$GIT_COMMITTER_EMAIL> $GIT_COMMITTER_DATE
3216        data 0
3217        from refs/heads/U
3218        INPUT_END
3220        background_import_then_checkpoint "--export-marks=marks.actual" input &&
3222        echo ":1 $(git rev-parse --verify V)" >marks.expected &&
3224        test "$(git rev-parse --verify V^)" = "$(git rev-parse --verify U)" &&
3226        test_cmp marks.expected marks.actual &&
3227        background_import_still_running
3228'
3229# Re-create the exact same commit, but on a different branch: no new object is
3231# created in the database, but the refs and marks still need to be updated.
3232test_expect_success PIPE 'V: checkpoint updates refs and marks after commit (no new objects)' '
3233        cat >input <<-INPUT_END &&
3234        commit refs/heads/V2
3235        mark :2
3236        committer $GIT_COMMITTER_NAME <$GIT_COMMITTER_EMAIL> $GIT_COMMITTER_DATE
3237        data 0
3238        from refs/heads/U
3239        INPUT_END
3241        background_import_then_checkpoint "--export-marks=marks.actual" input &&
3243        echo ":2 $(git rev-parse --verify V2)" >marks.expected &&
3245        test "$(git rev-parse --verify V2)" = "$(git rev-parse --verify V)" &&
3247        test_cmp marks.expected marks.actual &&
3248        background_import_still_running
3249'
3250test_expect_success PIPE 'V: checkpoint updates tags after tag' '
3252        cat >input <<-INPUT_END &&
3253        tag Vtag
3254        from refs/heads/V
3255        tagger $GIT_COMMITTER_NAME <$GIT_COMMITTER_EMAIL> $GIT_COMMITTER_DATE
3256        data 0
3257        INPUT_END
3259        background_import_then_checkpoint "" input &&
3261        git show-ref -d Vtag &&
3262        background_import_still_running
3263'
3264test_done