t / t5505-remote.shon commit oneway_merge(): only lstat() when told to update worktree (686b2de)
   1#!/bin/sh
   2
   3test_description='git remote porcelain-ish'
   4
   5. ./test-lib.sh
   6
   7setup_repository () {
   8        mkdir "$1" && (
   9        cd "$1" &&
  10        git init &&
  11        >file &&
  12        git add file &&
  13        test_tick &&
  14        git commit -m "Initial" &&
  15        git checkout -b side &&
  16        >elif &&
  17        git add elif &&
  18        test_tick &&
  19        git commit -m "Second" &&
  20        git checkout master
  21        )
  22}
  23
  24tokens_match () {
  25        echo "$1" | tr ' ' '\012' | sort | sed -e '/^$/d' >expect &&
  26        echo "$2" | tr ' ' '\012' | sort | sed -e '/^$/d' >actual &&
  27        test_cmp expect actual
  28}
  29
  30check_remote_track () {
  31        actual=$(git remote show "$1" | sed -ne 's|^    \(.*\) tracked$|\1|p')
  32        shift &&
  33        tokens_match "$*" "$actual"
  34}
  35
  36check_tracking_branch () {
  37        f="" &&
  38        r=$(git for-each-ref "--format=%(refname)" |
  39                sed -ne "s|^refs/remotes/$1/||p") &&
  40        shift &&
  41        tokens_match "$*" "$r"
  42}
  43
  44test_expect_success setup '
  45
  46        setup_repository one &&
  47        setup_repository two &&
  48        (
  49                cd two && git branch another
  50        ) &&
  51        git clone one test
  52
  53'
  54
  55test_expect_success 'remote information for the origin' '
  56(
  57        cd test &&
  58        tokens_match origin "$(git remote)" &&
  59        check_remote_track origin master side &&
  60        check_tracking_branch origin HEAD master side
  61)
  62'
  63
  64test_expect_success 'add another remote' '
  65(
  66        cd test &&
  67        git remote add -f second ../two &&
  68        tokens_match "origin second" "$(git remote)" &&
  69        check_remote_track origin master side &&
  70        check_remote_track second master side another &&
  71        check_tracking_branch second master side another &&
  72        git for-each-ref "--format=%(refname)" refs/remotes |
  73        sed -e "/^refs\/remotes\/origin\//d" \
  74            -e "/^refs\/remotes\/second\//d" >actual &&
  75        >expect &&
  76        test_cmp expect actual
  77)
  78'
  79
  80test_expect_success 'remote forces tracking branches' '
  81(
  82        cd test &&
  83        case `git config remote.second.fetch` in
  84        +*) true ;;
  85         *) false ;;
  86        esac
  87)
  88'
  89
  90test_expect_success 'remove remote' '
  91(
  92        cd test &&
  93        git symbolic-ref refs/remotes/second/HEAD refs/remotes/second/master &&
  94        git remote rm second
  95)
  96'
  97
  98test_expect_success 'remove remote' '
  99(
 100        cd test &&
 101        tokens_match origin "$(git remote)" &&
 102        check_remote_track origin master side &&
 103        git for-each-ref "--format=%(refname)" refs/remotes |
 104        sed -e "/^refs\/remotes\/origin\//d" >actual &&
 105        >expect &&
 106        test_cmp expect actual
 107)
 108'
 109
 110test_expect_success 'remove remote protects local branches' '
 111(
 112        cd test &&
 113        { cat >expect1 <<EOF
 114Note: A branch outside the refs/remotes/ hierarchy was not removed;
 115to delete it, use:
 116  git branch -d master
 117EOF
 118        } &&
 119        { cat >expect2 <<EOF
 120Note: Some branches outside the refs/remotes/ hierarchy were not removed;
 121to delete them, use:
 122  git branch -d foobranch
 123  git branch -d master
 124EOF
 125        } &&
 126        git tag footag &&
 127        git config --add remote.oops.fetch "+refs/*:refs/*" &&
 128        git remote remove oops 2>actual1 &&
 129        git branch foobranch &&
 130        git config --add remote.oops.fetch "+refs/*:refs/*" &&
 131        git remote rm oops 2>actual2 &&
 132        git branch -d foobranch &&
 133        git tag -d footag &&
 134        test_cmp expect1 actual1 &&
 135        test_cmp expect2 actual2
 136)
 137'
 138
 139cat > test/expect << EOF
 140* remote origin
 141  Fetch URL: $(pwd)/one
 142  Push  URL: $(pwd)/one
 143  HEAD branch: master
 144  Remote branches:
 145    master new (next fetch will store in remotes/origin)
 146    side   tracked
 147  Local branches configured for 'git pull':
 148    ahead    merges with remote master
 149    master   merges with remote master
 150    octopus  merges with remote topic-a
 151                and with remote topic-b
 152                and with remote topic-c
 153    rebase  rebases onto remote master
 154  Local refs configured for 'git push':
 155    master pushes to master   (local out of date)
 156    master pushes to upstream (create)
 157* remote two
 158  Fetch URL: ../two
 159  Push  URL: ../three
 160  HEAD branch (remote HEAD is ambiguous, may be one of the following):
 161    another
 162    master
 163  Local refs configured for 'git push':
 164    ahead  forces to master  (fast-forwardable)
 165    master pushes to another (up to date)
 166EOF
 167
 168test_expect_success 'show' '
 169        (cd test &&
 170         git config --add remote.origin.fetch refs/heads/master:refs/heads/upstream &&
 171         git fetch &&
 172         git checkout -b ahead origin/master &&
 173         echo 1 >> file &&
 174         test_tick &&
 175         git commit -m update file &&
 176         git checkout master &&
 177         git branch --track octopus origin/master &&
 178         git branch --track rebase origin/master &&
 179         git branch -d -r origin/master &&
 180         git config --add remote.two.url ../two &&
 181         git config --add remote.two.pushurl ../three &&
 182         git config branch.rebase.rebase true &&
 183         git config branch.octopus.merge "topic-a topic-b topic-c" &&
 184         (cd ../one &&
 185          echo 1 > file &&
 186          test_tick &&
 187          git commit -m update file) &&
 188         git config --add remote.origin.push : &&
 189         git config --add remote.origin.push refs/heads/master:refs/heads/upstream &&
 190         git config --add remote.origin.push +refs/tags/lastbackup &&
 191         git config --add remote.two.push +refs/heads/ahead:refs/heads/master &&
 192         git config --add remote.two.push refs/heads/master:refs/heads/another &&
 193         git remote show origin two > output &&
 194         git branch -d rebase octopus &&
 195         test_cmp expect output)
 196'
 197
 198cat > test/expect << EOF
 199* remote origin
 200  Fetch URL: $(pwd)/one
 201  Push  URL: $(pwd)/one
 202  HEAD branch: (not queried)
 203  Remote branches: (status not queried)
 204    master
 205    side
 206  Local branches configured for 'git pull':
 207    ahead  merges with remote master
 208    master merges with remote master
 209  Local refs configured for 'git push' (status not queried):
 210    (matching)           pushes to (matching)
 211    refs/heads/master    pushes to refs/heads/upstream
 212    refs/tags/lastbackup forces to refs/tags/lastbackup
 213EOF
 214
 215test_expect_success 'show -n' '
 216        (mv one one.unreachable &&
 217         cd test &&
 218         git remote show -n origin > output &&
 219         mv ../one.unreachable ../one &&
 220         test_cmp expect output)
 221'
 222
 223test_expect_success 'prune' '
 224        (cd one &&
 225         git branch -m side side2) &&
 226        (cd test &&
 227         git fetch origin &&
 228         git remote prune origin &&
 229         git rev-parse refs/remotes/origin/side2 &&
 230         test_must_fail git rev-parse refs/remotes/origin/side)
 231'
 232
 233test_expect_success 'set-head --delete' '
 234        (cd test &&
 235         git symbolic-ref refs/remotes/origin/HEAD &&
 236         git remote set-head --delete origin &&
 237         test_must_fail git symbolic-ref refs/remotes/origin/HEAD)
 238'
 239
 240test_expect_success 'set-head --auto' '
 241        (cd test &&
 242         git remote set-head --auto origin &&
 243         echo refs/remotes/origin/master >expect &&
 244         git symbolic-ref refs/remotes/origin/HEAD >output &&
 245         test_cmp expect output
 246        )
 247'
 248
 249cat >test/expect <<EOF
 250error: Multiple remote HEAD branches. Please choose one explicitly with:
 251  git remote set-head two another
 252  git remote set-head two master
 253EOF
 254
 255test_expect_success 'set-head --auto fails w/multiple HEADs' '
 256        (cd test &&
 257         test_must_fail git remote set-head --auto two >output 2>&1 &&
 258        test_cmp expect output)
 259'
 260
 261cat >test/expect <<EOF
 262refs/remotes/origin/side2
 263EOF
 264
 265test_expect_success 'set-head explicit' '
 266        (cd test &&
 267         git remote set-head origin side2 &&
 268         git symbolic-ref refs/remotes/origin/HEAD >output &&
 269         git remote set-head origin master &&
 270         test_cmp expect output)
 271'
 272
 273cat > test/expect << EOF
 274Pruning origin
 275URL: $(pwd)/one
 276 * [would prune] origin/side2
 277EOF
 278
 279test_expect_success 'prune --dry-run' '
 280        (cd one &&
 281         git branch -m side2 side) &&
 282        (cd test &&
 283         git remote prune --dry-run origin > output &&
 284         git rev-parse refs/remotes/origin/side2 &&
 285         test_must_fail git rev-parse refs/remotes/origin/side &&
 286        (cd ../one &&
 287         git branch -m side side2) &&
 288         test_cmp expect output)
 289'
 290
 291test_expect_success 'add --mirror && prune' '
 292        (mkdir mirror &&
 293         cd mirror &&
 294         git init --bare &&
 295         git remote add --mirror -f origin ../one) &&
 296        (cd one &&
 297         git branch -m side2 side) &&
 298        (cd mirror &&
 299         git rev-parse --verify refs/heads/side2 &&
 300         test_must_fail git rev-parse --verify refs/heads/side &&
 301         git fetch origin &&
 302         git remote prune origin &&
 303         test_must_fail git rev-parse --verify refs/heads/side2 &&
 304         git rev-parse --verify refs/heads/side)
 305'
 306
 307test_expect_success 'add --mirror=fetch' '
 308        mkdir mirror-fetch &&
 309        git init mirror-fetch/parent &&
 310        (cd mirror-fetch/parent &&
 311         test_commit one) &&
 312        git init --bare mirror-fetch/child &&
 313        (cd mirror-fetch/child &&
 314         git remote add --mirror=fetch -f parent ../parent)
 315'
 316
 317test_expect_success 'fetch mirrors act as mirrors during fetch' '
 318        (cd mirror-fetch/parent &&
 319         git branch new &&
 320         git branch -m master renamed
 321        ) &&
 322        (cd mirror-fetch/child &&
 323         git fetch parent &&
 324         git rev-parse --verify refs/heads/new &&
 325         git rev-parse --verify refs/heads/renamed
 326        )
 327'
 328
 329test_expect_success 'fetch mirrors can prune' '
 330        (cd mirror-fetch/child &&
 331         git remote prune parent &&
 332         test_must_fail git rev-parse --verify refs/heads/master
 333        )
 334'
 335
 336test_expect_success 'fetch mirrors do not act as mirrors during push' '
 337        (cd mirror-fetch/parent &&
 338         git checkout HEAD^0
 339        ) &&
 340        (cd mirror-fetch/child &&
 341         git branch -m renamed renamed2 &&
 342         git push parent
 343        ) &&
 344        (cd mirror-fetch/parent &&
 345         git rev-parse --verify renamed &&
 346         test_must_fail git rev-parse --verify refs/heads/renamed2
 347        )
 348'
 349
 350test_expect_success 'add fetch mirror with specific branches' '
 351        git init --bare mirror-fetch/track &&
 352        (cd mirror-fetch/track &&
 353         git remote add --mirror=fetch -t heads/new parent ../parent
 354        )
 355'
 356
 357test_expect_success 'fetch mirror respects specific branches' '
 358        (cd mirror-fetch/track &&
 359         git fetch parent &&
 360         git rev-parse --verify refs/heads/new &&
 361         test_must_fail git rev-parse --verify refs/heads/renamed
 362        )
 363'
 364
 365test_expect_success 'add --mirror=push' '
 366        mkdir mirror-push &&
 367        git init --bare mirror-push/public &&
 368        git init mirror-push/private &&
 369        (cd mirror-push/private &&
 370         test_commit one &&
 371         git remote add --mirror=push public ../public
 372        )
 373'
 374
 375test_expect_success 'push mirrors act as mirrors during push' '
 376        (cd mirror-push/private &&
 377         git branch new &&
 378         git branch -m master renamed &&
 379         git push public
 380        ) &&
 381        (cd mirror-push/private &&
 382         git rev-parse --verify refs/heads/new &&
 383         git rev-parse --verify refs/heads/renamed &&
 384         test_must_fail git rev-parse --verify refs/heads/master
 385        )
 386'
 387
 388test_expect_success 'push mirrors do not act as mirrors during fetch' '
 389        (cd mirror-push/public &&
 390         git branch -m renamed renamed2 &&
 391         git symbolic-ref HEAD refs/heads/renamed2
 392        ) &&
 393        (cd mirror-push/private &&
 394         git fetch public &&
 395         git rev-parse --verify refs/heads/renamed &&
 396         test_must_fail git rev-parse --verify refs/heads/renamed2
 397        )
 398'
 399
 400test_expect_success 'push mirrors do not allow you to specify refs' '
 401        git init mirror-push/track &&
 402        (cd mirror-push/track &&
 403         test_must_fail git remote add --mirror=push -t new public ../public
 404        )
 405'
 406
 407test_expect_success 'add alt && prune' '
 408        (mkdir alttst &&
 409         cd alttst &&
 410         git init &&
 411         git remote add -f origin ../one &&
 412         git config remote.alt.url ../one &&
 413         git config remote.alt.fetch "+refs/heads/*:refs/remotes/origin/*") &&
 414        (cd one &&
 415         git branch -m side side2) &&
 416        (cd alttst &&
 417         git rev-parse --verify refs/remotes/origin/side &&
 418         test_must_fail git rev-parse --verify refs/remotes/origin/side2 &&
 419         git fetch alt &&
 420         git remote prune alt &&
 421         test_must_fail git rev-parse --verify refs/remotes/origin/side &&
 422         git rev-parse --verify refs/remotes/origin/side2)
 423'
 424
 425cat >test/expect <<\EOF
 426some-tag
 427EOF
 428
 429test_expect_success 'add with reachable tags (default)' '
 430        (cd one &&
 431         >foobar &&
 432         git add foobar &&
 433         git commit -m "Foobar" &&
 434         git tag -a -m "Foobar tag" foobar-tag &&
 435         git reset --hard HEAD~1 &&
 436         git tag -a -m "Some tag" some-tag) &&
 437        (mkdir add-tags &&
 438         cd add-tags &&
 439         git init &&
 440         git remote add -f origin ../one &&
 441         git tag -l some-tag >../test/output &&
 442         git tag -l foobar-tag >>../test/output &&
 443         test_must_fail git config remote.origin.tagopt) &&
 444        test_cmp test/expect test/output
 445'
 446
 447cat >test/expect <<\EOF
 448some-tag
 449foobar-tag
 450--tags
 451EOF
 452
 453test_expect_success 'add --tags' '
 454        (rm -rf add-tags &&
 455         mkdir add-tags &&
 456         cd add-tags &&
 457         git init &&
 458         git remote add -f --tags origin ../one &&
 459         git tag -l some-tag >../test/output &&
 460         git tag -l foobar-tag >>../test/output &&
 461         git config remote.origin.tagopt >>../test/output) &&
 462        test_cmp test/expect test/output
 463'
 464
 465cat >test/expect <<\EOF
 466--no-tags
 467EOF
 468
 469test_expect_success 'add --no-tags' '
 470        (rm -rf add-tags &&
 471         mkdir add-no-tags &&
 472         cd add-no-tags &&
 473         git init &&
 474         git remote add -f --no-tags origin ../one &&
 475         git tag -l some-tag >../test/output &&
 476         git tag -l foobar-tag >../test/output &&
 477         git config remote.origin.tagopt >>../test/output) &&
 478        (cd one &&
 479         git tag -d some-tag foobar-tag) &&
 480        test_cmp test/expect test/output
 481'
 482
 483test_expect_success 'reject --no-no-tags' '
 484        (cd add-no-tags &&
 485         test_must_fail git remote add -f --no-no-tags neworigin ../one)
 486'
 487
 488cat > one/expect << EOF
 489  apis/master
 490  apis/side
 491  drosophila/another
 492  drosophila/master
 493  drosophila/side
 494EOF
 495
 496test_expect_success 'update' '
 497
 498        (cd one &&
 499         git remote add drosophila ../two &&
 500         git remote add apis ../mirror &&
 501         git remote update &&
 502         git branch -r > output &&
 503         test_cmp expect output)
 504
 505'
 506
 507cat > one/expect << EOF
 508  drosophila/another
 509  drosophila/master
 510  drosophila/side
 511  manduca/master
 512  manduca/side
 513  megaloprepus/master
 514  megaloprepus/side
 515EOF
 516
 517test_expect_success 'update with arguments' '
 518
 519        (cd one &&
 520         for b in $(git branch -r)
 521         do
 522                git branch -r -d $b || break
 523         done &&
 524         git remote add manduca ../mirror &&
 525         git remote add megaloprepus ../mirror &&
 526         git config remotes.phobaeticus "drosophila megaloprepus" &&
 527         git config remotes.titanus manduca &&
 528         git remote update phobaeticus titanus &&
 529         git branch -r > output &&
 530         test_cmp expect output)
 531
 532'
 533
 534test_expect_success 'update --prune' '
 535
 536        (cd one &&
 537         git branch -m side2 side3) &&
 538        (cd test &&
 539         git remote update --prune &&
 540         (cd ../one && git branch -m side3 side2) &&
 541         git rev-parse refs/remotes/origin/side3 &&
 542         test_must_fail git rev-parse refs/remotes/origin/side2)
 543'
 544
 545cat > one/expect << EOF
 546  apis/master
 547  apis/side
 548  manduca/master
 549  manduca/side
 550  megaloprepus/master
 551  megaloprepus/side
 552EOF
 553
 554test_expect_success 'update default' '
 555
 556        (cd one &&
 557         for b in $(git branch -r)
 558         do
 559                git branch -r -d $b || break
 560         done &&
 561         git config remote.drosophila.skipDefaultUpdate true &&
 562         git remote update default &&
 563         git branch -r > output &&
 564         test_cmp expect output)
 565
 566'
 567
 568cat > one/expect << EOF
 569  drosophila/another
 570  drosophila/master
 571  drosophila/side
 572EOF
 573
 574test_expect_success 'update default (overridden, with funny whitespace)' '
 575
 576        (cd one &&
 577         for b in $(git branch -r)
 578         do
 579                git branch -r -d $b || break
 580         done &&
 581         git config remotes.default "$(printf "\t drosophila  \n")" &&
 582         git remote update default &&
 583         git branch -r > output &&
 584         test_cmp expect output)
 585
 586'
 587
 588test_expect_success 'update (with remotes.default defined)' '
 589
 590        (cd one &&
 591         for b in $(git branch -r)
 592         do
 593                git branch -r -d $b || break
 594         done &&
 595         git config remotes.default "drosophila" &&
 596         git remote update &&
 597         git branch -r > output &&
 598         test_cmp expect output)
 599
 600'
 601
 602test_expect_success '"remote show" does not show symbolic refs' '
 603
 604        git clone one three &&
 605        (cd three &&
 606         git remote show origin > output &&
 607         ! grep "^ *HEAD$" < output &&
 608         ! grep -i stale < output)
 609
 610'
 611
 612test_expect_success 'reject adding remote with an invalid name' '
 613
 614        test_must_fail git remote add some:url desired-name
 615
 616'
 617
 618# The first three test if the tracking branches are properly renamed,
 619# the last two ones check if the config is updated.
 620
 621test_expect_success 'rename a remote' '
 622
 623        git clone one four &&
 624        (cd four &&
 625         git remote rename origin upstream &&
 626         rmdir .git/refs/remotes/origin &&
 627         test "$(git symbolic-ref refs/remotes/upstream/HEAD)" = "refs/remotes/upstream/master" &&
 628         test "$(git rev-parse upstream/master)" = "$(git rev-parse master)" &&
 629         test "$(git config remote.upstream.fetch)" = "+refs/heads/*:refs/remotes/upstream/*" &&
 630         test "$(git config branch.master.remote)" = "upstream")
 631
 632'
 633
 634test_expect_success 'rename does not update a non-default fetch refspec' '
 635
 636        git clone one four.one &&
 637        (cd four.one &&
 638         git config remote.origin.fetch +refs/heads/*:refs/heads/origin/* &&
 639         git remote rename origin upstream &&
 640         test "$(git config remote.upstream.fetch)" = "+refs/heads/*:refs/heads/origin/*" &&
 641         git rev-parse -q origin/master)
 642
 643'
 644
 645test_expect_success 'rename a remote with name part of fetch spec' '
 646
 647        git clone one four.two &&
 648        (cd four.two &&
 649         git remote rename origin remote &&
 650         git remote rename remote upstream &&
 651         test "$(git config remote.upstream.fetch)" = "+refs/heads/*:refs/remotes/upstream/*")
 652
 653'
 654
 655test_expect_success 'rename a remote with name prefix of other remote' '
 656
 657        git clone one four.three &&
 658        (cd four.three &&
 659         git remote add o git://example.com/repo.git &&
 660         git remote rename o upstream &&
 661         test "$(git rev-parse origin/master)" = "$(git rev-parse master)")
 662
 663'
 664
 665cat > remotes_origin << EOF
 666URL: $(pwd)/one
 667Push: refs/heads/master:refs/heads/upstream
 668Pull: refs/heads/master:refs/heads/origin
 669EOF
 670
 671test_expect_success 'migrate a remote from named file in $GIT_DIR/remotes' '
 672        git clone one five &&
 673        origin_url=$(pwd)/one &&
 674        (cd five &&
 675         git remote remove origin &&
 676         mkdir -p .git/remotes &&
 677         cat ../remotes_origin > .git/remotes/origin &&
 678         git remote rename origin origin &&
 679         ! test -f .git/remotes/origin &&
 680         test "$(git config remote.origin.url)" = "$origin_url" &&
 681         test "$(git config remote.origin.push)" = "refs/heads/master:refs/heads/upstream" &&
 682         test "$(git config remote.origin.fetch)" = "refs/heads/master:refs/heads/origin")
 683'
 684
 685test_expect_success 'migrate a remote from named file in $GIT_DIR/branches' '
 686        git clone one six &&
 687        origin_url=$(pwd)/one &&
 688        (cd six &&
 689         git remote rm origin &&
 690         echo "$origin_url" > .git/branches/origin &&
 691         git remote rename origin origin &&
 692         ! test -f .git/branches/origin &&
 693         test "$(git config remote.origin.url)" = "$origin_url" &&
 694         test "$(git config remote.origin.fetch)" = "refs/heads/master:refs/heads/origin")
 695'
 696
 697test_expect_success 'remote prune to cause a dangling symref' '
 698        git clone one seven &&
 699        (
 700                cd one &&
 701                git checkout side2 &&
 702                git branch -D master
 703        ) &&
 704        (
 705                cd seven &&
 706                git remote prune origin
 707        ) >err 2>&1 &&
 708        grep "has become dangling" err &&
 709
 710        : And the dangling symref will not cause other annoying errors &&
 711        (
 712                cd seven &&
 713                git branch -a
 714        ) 2>err &&
 715        ! grep "points nowhere" err &&
 716        (
 717                cd seven &&
 718                test_must_fail git branch nomore origin
 719        ) 2>err &&
 720        grep "dangling symref" err
 721'
 722
 723test_expect_success 'show empty remote' '
 724
 725        test_create_repo empty &&
 726        git clone empty empty-clone &&
 727        (
 728                cd empty-clone &&
 729                git remote show origin
 730        )
 731'
 732
 733test_expect_success 'remote set-branches requires a remote' '
 734        test_must_fail git remote set-branches &&
 735        test_must_fail git remote set-branches --add
 736'
 737
 738test_expect_success 'remote set-branches' '
 739        echo "+refs/heads/*:refs/remotes/scratch/*" >expect.initial &&
 740        sort <<-\EOF >expect.add &&
 741        +refs/heads/*:refs/remotes/scratch/*
 742        +refs/heads/other:refs/remotes/scratch/other
 743        EOF
 744        sort <<-\EOF >expect.replace &&
 745        +refs/heads/maint:refs/remotes/scratch/maint
 746        +refs/heads/master:refs/remotes/scratch/master
 747        +refs/heads/next:refs/remotes/scratch/next
 748        EOF
 749        sort <<-\EOF >expect.add-two &&
 750        +refs/heads/maint:refs/remotes/scratch/maint
 751        +refs/heads/master:refs/remotes/scratch/master
 752        +refs/heads/next:refs/remotes/scratch/next
 753        +refs/heads/pu:refs/remotes/scratch/pu
 754        +refs/heads/t/topic:refs/remotes/scratch/t/topic
 755        EOF
 756        sort <<-\EOF >expect.setup-ffonly &&
 757        refs/heads/master:refs/remotes/scratch/master
 758        +refs/heads/next:refs/remotes/scratch/next
 759        EOF
 760        sort <<-\EOF >expect.respect-ffonly &&
 761        refs/heads/master:refs/remotes/scratch/master
 762        +refs/heads/next:refs/remotes/scratch/next
 763        +refs/heads/pu:refs/remotes/scratch/pu
 764        EOF
 765
 766        git clone .git/ setbranches &&
 767        (
 768                cd setbranches &&
 769                git remote rename origin scratch &&
 770                git config --get-all remote.scratch.fetch >config-result &&
 771                sort <config-result >../actual.initial &&
 772
 773                git remote set-branches scratch --add other &&
 774                git config --get-all remote.scratch.fetch >config-result &&
 775                sort <config-result >../actual.add &&
 776
 777                git remote set-branches scratch maint master next &&
 778                git config --get-all remote.scratch.fetch >config-result &&
 779                sort <config-result >../actual.replace &&
 780
 781                git remote set-branches --add scratch pu t/topic &&
 782                git config --get-all remote.scratch.fetch >config-result &&
 783                sort <config-result >../actual.add-two &&
 784
 785                git config --unset-all remote.scratch.fetch &&
 786                git config remote.scratch.fetch \
 787                        refs/heads/master:refs/remotes/scratch/master &&
 788                git config --add remote.scratch.fetch \
 789                        +refs/heads/next:refs/remotes/scratch/next &&
 790                git config --get-all remote.scratch.fetch >config-result &&
 791                sort <config-result >../actual.setup-ffonly &&
 792
 793                git remote set-branches --add scratch pu &&
 794                git config --get-all remote.scratch.fetch >config-result &&
 795                sort <config-result >../actual.respect-ffonly
 796        ) &&
 797        test_cmp expect.initial actual.initial &&
 798        test_cmp expect.add actual.add &&
 799        test_cmp expect.replace actual.replace &&
 800        test_cmp expect.add-two actual.add-two &&
 801        test_cmp expect.setup-ffonly actual.setup-ffonly &&
 802        test_cmp expect.respect-ffonly actual.respect-ffonly
 803'
 804
 805test_expect_success 'remote set-branches with --mirror' '
 806        echo "+refs/*:refs/*" >expect.initial &&
 807        echo "+refs/heads/master:refs/heads/master" >expect.replace &&
 808        git clone --mirror .git/ setbranches-mirror &&
 809        (
 810                cd setbranches-mirror &&
 811                git remote rename origin scratch &&
 812                git config --get-all remote.scratch.fetch >../actual.initial &&
 813
 814                git remote set-branches scratch heads/master &&
 815                git config --get-all remote.scratch.fetch >../actual.replace
 816        ) &&
 817        test_cmp expect.initial actual.initial &&
 818        test_cmp expect.replace actual.replace
 819'
 820
 821test_expect_success 'new remote' '
 822        git remote add someremote foo &&
 823        echo foo >expect &&
 824        git config --get-all remote.someremote.url >actual &&
 825        cmp expect actual
 826'
 827
 828test_expect_success 'remote set-url bar' '
 829        git remote set-url someremote bar &&
 830        echo bar >expect &&
 831        git config --get-all remote.someremote.url >actual &&
 832        cmp expect actual
 833'
 834
 835test_expect_success 'remote set-url baz bar' '
 836        git remote set-url someremote baz bar &&
 837        echo baz >expect &&
 838        git config --get-all remote.someremote.url >actual &&
 839        cmp expect actual
 840'
 841
 842test_expect_success 'remote set-url zot bar' '
 843        test_must_fail git remote set-url someremote zot bar &&
 844        echo baz >expect &&
 845        git config --get-all remote.someremote.url >actual &&
 846        cmp expect actual
 847'
 848
 849test_expect_success 'remote set-url --push zot baz' '
 850        test_must_fail git remote set-url --push someremote zot baz &&
 851        echo "YYY" >expect &&
 852        echo baz >>expect &&
 853        test_must_fail git config --get-all remote.someremote.pushurl >actual &&
 854        echo "YYY" >>actual &&
 855        git config --get-all remote.someremote.url >>actual &&
 856        cmp expect actual
 857'
 858
 859test_expect_success 'remote set-url --push zot' '
 860        git remote set-url --push someremote zot &&
 861        echo zot >expect &&
 862        echo "YYY" >>expect &&
 863        echo baz >>expect &&
 864        git config --get-all remote.someremote.pushurl >actual &&
 865        echo "YYY" >>actual &&
 866        git config --get-all remote.someremote.url >>actual &&
 867        cmp expect actual
 868'
 869
 870test_expect_success 'remote set-url --push qux zot' '
 871        git remote set-url --push someremote qux zot &&
 872        echo qux >expect &&
 873        echo "YYY" >>expect &&
 874        echo baz >>expect &&
 875        git config --get-all remote.someremote.pushurl >actual &&
 876        echo "YYY" >>actual &&
 877        git config --get-all remote.someremote.url >>actual &&
 878        cmp expect actual
 879'
 880
 881test_expect_success 'remote set-url --push foo qu+x' '
 882        git remote set-url --push someremote foo qu+x &&
 883        echo foo >expect &&
 884        echo "YYY" >>expect &&
 885        echo baz >>expect &&
 886        git config --get-all remote.someremote.pushurl >actual &&
 887        echo "YYY" >>actual &&
 888        git config --get-all remote.someremote.url >>actual &&
 889        cmp expect actual
 890'
 891
 892test_expect_success 'remote set-url --push --add aaa' '
 893        git remote set-url --push --add someremote aaa &&
 894        echo foo >expect &&
 895        echo aaa >>expect &&
 896        echo "YYY" >>expect &&
 897        echo baz >>expect &&
 898        git config --get-all remote.someremote.pushurl >actual &&
 899        echo "YYY" >>actual &&
 900        git config --get-all remote.someremote.url >>actual &&
 901        cmp expect actual
 902'
 903
 904test_expect_success 'remote set-url --push bar aaa' '
 905        git remote set-url --push someremote bar aaa &&
 906        echo foo >expect &&
 907        echo bar >>expect &&
 908        echo "YYY" >>expect &&
 909        echo baz >>expect &&
 910        git config --get-all remote.someremote.pushurl >actual &&
 911        echo "YYY" >>actual &&
 912        git config --get-all remote.someremote.url >>actual &&
 913        cmp expect actual
 914'
 915
 916test_expect_success 'remote set-url --push --delete bar' '
 917        git remote set-url --push --delete someremote bar &&
 918        echo foo >expect &&
 919        echo "YYY" >>expect &&
 920        echo baz >>expect &&
 921        git config --get-all remote.someremote.pushurl >actual &&
 922        echo "YYY" >>actual &&
 923        git config --get-all remote.someremote.url >>actual &&
 924        cmp expect actual
 925'
 926
 927test_expect_success 'remote set-url --push --delete foo' '
 928        git remote set-url --push --delete someremote foo &&
 929        echo "YYY" >expect &&
 930        echo baz >>expect &&
 931        test_must_fail git config --get-all remote.someremote.pushurl >actual &&
 932        echo "YYY" >>actual &&
 933        git config --get-all remote.someremote.url >>actual &&
 934        cmp expect actual
 935'
 936
 937test_expect_success 'remote set-url --add bbb' '
 938        git remote set-url --add someremote bbb &&
 939        echo "YYY" >expect &&
 940        echo baz >>expect &&
 941        echo bbb >>expect &&
 942        test_must_fail git config --get-all remote.someremote.pushurl >actual &&
 943        echo "YYY" >>actual &&
 944        git config --get-all remote.someremote.url >>actual &&
 945        cmp expect actual
 946'
 947
 948test_expect_success 'remote set-url --delete .*' '
 949        test_must_fail git remote set-url --delete someremote .\* &&
 950        echo "YYY" >expect &&
 951        echo baz >>expect &&
 952        echo bbb >>expect &&
 953        test_must_fail git config --get-all remote.someremote.pushurl >actual &&
 954        echo "YYY" >>actual &&
 955        git config --get-all remote.someremote.url >>actual &&
 956        cmp expect actual
 957'
 958
 959test_expect_success 'remote set-url --delete bbb' '
 960        git remote set-url --delete someremote bbb &&
 961        echo "YYY" >expect &&
 962        echo baz >>expect &&
 963        test_must_fail git config --get-all remote.someremote.pushurl >actual &&
 964        echo "YYY" >>actual &&
 965        git config --get-all remote.someremote.url >>actual &&
 966        cmp expect actual
 967'
 968
 969test_expect_success 'remote set-url --delete baz' '
 970        test_must_fail git remote set-url --delete someremote baz &&
 971        echo "YYY" >expect &&
 972        echo baz >>expect &&
 973        test_must_fail git config --get-all remote.someremote.pushurl >actual &&
 974        echo "YYY" >>actual &&
 975        git config --get-all remote.someremote.url >>actual &&
 976        cmp expect actual
 977'
 978
 979test_expect_success 'remote set-url --add ccc' '
 980        git remote set-url --add someremote ccc &&
 981        echo "YYY" >expect &&
 982        echo baz >>expect &&
 983        echo ccc >>expect &&
 984        test_must_fail git config --get-all remote.someremote.pushurl >actual &&
 985        echo "YYY" >>actual &&
 986        git config --get-all remote.someremote.url >>actual &&
 987        cmp expect actual
 988'
 989
 990test_expect_success 'remote set-url --delete baz' '
 991        git remote set-url --delete someremote baz &&
 992        echo "YYY" >expect &&
 993        echo ccc >>expect &&
 994        test_must_fail git config --get-all remote.someremote.pushurl >actual &&
 995        echo "YYY" >>actual &&
 996        git config --get-all remote.someremote.url >>actual &&
 997        cmp expect actual
 998'
 999
1000test_done