1#!/bin/sh
   2#
   3# Copyright (c) 2005 Junio C Hamano
   4#
   5test_description='Two way merge with read-tree -m $H $M
   7This test tries two-way merge (aka fast forward with carry forward).
   9There is the head (called H) and another commit (called M), which is
  11simply ahead of H.  The index and the work tree contains a state that
  12is derived from H, but may also have local changes.  This test checks
  13all the combinations described in the two-tree merge "carry forward"
  14rules, found in <Documentation/git-read-tree.txt>.
  15In the test, these paths are used:
  17        bozbar  - in H, stays in M, modified from bozbar to gnusto
  18        frotz   - not in H added in M
  19        nitfol  - in H, stays in M unmodified
  20        rezrov  - in H, deleted in M
  21        yomin   - not in H nor M
  22'
  23. ./test-lib.sh
  24read_tree_twoway () {
  26    git-read-tree -m "$1" "$2" && git-ls-files --stage
  27}
  28_x40='[0-9a-f][0-9a-f][0-9a-f][0-9a-f][0-9a-f]'
  30_x40="$_x40$_x40$_x40$_x40$_x40$_x40$_x40$_x40"
  31compare_change () {
  32        sed -n >current \
  33            -e '/^--- /d; /^+++ /d; /^@@ /d;' \
  34            -e 's/^\([-+][0-7][0-7][0-7][0-7][0-7][0-7]\) '"$_x40"' /\1 X /p' \
  35            "$1"
  36        git diff expected current
  37}
  38check_cache_at () {
  40        clean_if_empty=`git-diff-files -- "$1"`
  41        case "$clean_if_empty" in
  42        '')  echo "$1: clean" ;;
  43        ?*)  echo "$1: dirty" ;;
  44        esac
  45        case "$2,$clean_if_empty" in
  46        clean,)         :     ;;
  47        clean,?*)       false ;;
  48        dirty,)         false ;;
  49        dirty,?*)       :     ;;
  50        esac
  51}
  52cat >bozbar-old <<\EOF
  54This is a sample file used in two-way fast forward merge
  55tests.  Its second line ends with a magic word bozbar
  56which will be modified by the merged head to gnusto.
  57It has some extra lines so that external tools can
  58successfully merge independent changes made to later
  59lines (such as this one), avoiding line conflicts.
  60EOF
  61sed -e 's/bozbar/gnusto (earlier bozbar)/' bozbar-old >bozbar-new
  63test_expect_success \
  65    setup \
  66    'echo frotz >frotz &&
  67     echo nitfol >nitfol &&
  68     cat bozbar-old >bozbar &&
  69     echo rezrov >rezrov &&
  70     echo yomin >yomin &&
  71     git-update-index --add nitfol bozbar rezrov &&
  72     treeH=`git-write-tree` &&
  73     echo treeH $treeH &&
  74     git-ls-tree $treeH &&
  75     cat bozbar-new >bozbar &&
  77     git-update-index --add frotz bozbar --force-remove rezrov &&
  78     git-ls-files --stage >M.out &&
  79     treeM=`git-write-tree` &&
  80     echo treeM $treeM &&
  81     git-ls-tree $treeM &&
  82     git-diff-tree $treeH $treeM'
  83test_expect_success \
  85    '1, 2, 3 - no carry forward' \
  86    'rm -f .git/index &&
  87     read_tree_twoway $treeH $treeM &&
  88     git-ls-files --stage >1-3.out &&
  89     git diff M.out 1-3.out &&
  90     check_cache_at bozbar dirty &&
  91     check_cache_at frotz dirty &&
  92     check_cache_at nitfol dirty'
  93echo '+100644 X 0       yomin' >expected
  95test_expect_success \
  97    '4 - carry forward local addition.' \
  98    'rm -f .git/index &&
  99     git-read-tree $treeH &&
 100     git-checkout-index -u -f -q -a &&
 101     git-update-index --add yomin &&
 102     read_tree_twoway $treeH $treeM &&
 103     git-ls-files --stage >4.out || return 1
 104     git diff M.out 4.out >4diff.out
 105     compare_change 4diff.out expected &&
 106     check_cache_at yomin clean'
 107test_expect_success \
 109    '5 - carry forward local addition.' \
 110    'rm -f .git/index &&
 111     git-read-tree $treeH &&
 112     git-checkout-index -u -f -q -a &&
 113     echo yomin >yomin &&
 114     git-update-index --add yomin &&
 115     echo yomin yomin >yomin &&
 116     read_tree_twoway $treeH $treeM &&
 117     git-ls-files --stage >5.out || return 1
 118     git diff M.out 5.out >5diff.out
 119     compare_change 5diff.out expected &&
 120     check_cache_at yomin dirty'
 121test_expect_success \
 123    '6 - local addition already has the same.' \
 124    'rm -f .git/index &&
 125     git-read-tree $treeH &&
 126     git-checkout-index -u -f -q -a &&
 127     git-update-index --add frotz &&
 128     read_tree_twoway $treeH $treeM &&
 129     git-ls-files --stage >6.out &&
 130     git diff M.out 6.out &&
 131     check_cache_at frotz clean'
 132test_expect_success \
 134    '7 - local addition already has the same.' \
 135    'rm -f .git/index &&
 136     git-read-tree $treeH &&
 137     git-checkout-index -u -f -q -a &&
 138     echo frotz >frotz &&
 139     git-update-index --add frotz &&
 140     echo frotz frotz >frotz &&
 141     read_tree_twoway $treeH $treeM &&
 142     git-ls-files --stage >7.out &&
 143     git diff M.out 7.out &&
 144     check_cache_at frotz dirty'
 145test_expect_success \
 147    '8 - conflicting addition.' \
 148    'rm -f .git/index &&
 149     git-read-tree $treeH &&
 150     git-checkout-index -u -f -q -a &&
 151     echo frotz frotz >frotz &&
 152     git-update-index --add frotz &&
 153     if read_tree_twoway $treeH $treeM; then false; else :; fi'
 154test_expect_success \
 156    '9 - conflicting addition.' \
 157    'rm -f .git/index &&
 158     git-read-tree $treeH &&
 159     git-checkout-index -u -f -q -a &&
 160     echo frotz frotz >frotz &&
 161     git-update-index --add frotz &&
 162     echo frotz >frotz &&
 163     if read_tree_twoway $treeH $treeM; then false; else :; fi'
 164test_expect_success \
 166    '10 - path removed.' \
 167    'rm -f .git/index &&
 168     git-read-tree $treeH &&
 169     git-checkout-index -u -f -q -a &&
 170     echo rezrov >rezrov &&
 171     git-update-index --add rezrov &&
 172     read_tree_twoway $treeH $treeM &&
 173     git-ls-files --stage >10.out &&
 174     git diff M.out 10.out'
 175test_expect_success \
 177    '11 - dirty path removed.' \
 178    'rm -f .git/index &&
 179     git-read-tree $treeH &&
 180     git-checkout-index -u -f -q -a &&
 181     echo rezrov >rezrov &&
 182     git-update-index --add rezrov &&
 183     echo rezrov rezrov >rezrov &&
 184     if read_tree_twoway $treeH $treeM; then false; else :; fi'
 185test_expect_success \
 187    '12 - unmatching local changes being removed.' \
 188    'rm -f .git/index &&
 189     git-read-tree $treeH &&
 190     git-checkout-index -u -f -q -a &&
 191     echo rezrov rezrov >rezrov &&
 192     git-update-index --add rezrov &&
 193     if read_tree_twoway $treeH $treeM; then false; else :; fi'
 194test_expect_success \
 196    '13 - unmatching local changes being removed.' \
 197    'rm -f .git/index &&
 198     git-read-tree $treeH &&
 199     git-checkout-index -u -f -q -a &&
 200     echo rezrov rezrov >rezrov &&
 201     git-update-index --add rezrov &&
 202     echo rezrov >rezrov &&
 203     if read_tree_twoway $treeH $treeM; then false; else :; fi'
 204cat >expected <<EOF
 206-100644 X 0     nitfol
 207+100644 X 0     nitfol
 208EOF
 209test_expect_success \
 211    '14 - unchanged in two heads.' \
 212    'rm -f .git/index &&
 213     git-read-tree $treeH &&
 214     git-checkout-index -u -f -q -a &&
 215     echo nitfol nitfol >nitfol &&
 216     git-update-index --add nitfol &&
 217     read_tree_twoway $treeH $treeM &&
 218     git-ls-files --stage >14.out || return 1
 219     git diff M.out 14.out >14diff.out
 220     compare_change 14diff.out expected &&
 221     check_cache_at nitfol clean'
 222test_expect_success \
 224    '15 - unchanged in two heads.' \
 225    'rm -f .git/index &&
 226     git-read-tree $treeH &&
 227     git-checkout-index -u -f -q -a &&
 228     echo nitfol nitfol >nitfol &&
 229     git-update-index --add nitfol &&
 230     echo nitfol nitfol nitfol >nitfol &&
 231     read_tree_twoway $treeH $treeM &&
 232     git-ls-files --stage >15.out || return 1
 233     git diff M.out 15.out >15diff.out
 234     compare_change 15diff.out expected &&
 235     check_cache_at nitfol dirty'
 236test_expect_success \
 238    '16 - conflicting local change.' \
 239    'rm -f .git/index &&
 240     git-read-tree $treeH &&
 241     git-checkout-index -u -f -q -a &&
 242     echo bozbar bozbar >bozbar &&
 243     git-update-index --add bozbar &&
 244     if read_tree_twoway $treeH $treeM; then false; else :; fi'
 245test_expect_success \
 247    '17 - conflicting local change.' \
 248    'rm -f .git/index &&
 249     git-read-tree $treeH &&
 250     git-checkout-index -u -f -q -a &&
 251     echo bozbar bozbar >bozbar &&
 252     git-update-index --add bozbar &&
 253     echo bozbar bozbar bozbar >bozbar &&
 254     if read_tree_twoway $treeH $treeM; then false; else :; fi'
 255test_expect_success \
 257    '18 - local change already having a good result.' \
 258    'rm -f .git/index &&
 259     git-read-tree $treeH &&
 260     git-checkout-index -u -f -q -a &&
 261     cat bozbar-new >bozbar &&
 262     git-update-index --add bozbar &&
 263     read_tree_twoway $treeH $treeM &&
 264     git-ls-files --stage >18.out &&
 265     git diff M.out 18.out &&
 266     check_cache_at bozbar clean'
 267test_expect_success \
 269    '19 - local change already having a good result, further modified.' \
 270    'rm -f .git/index &&
 271     git-read-tree $treeH &&
 272     git-checkout-index -u -f -q -a &&
 273     cat bozbar-new >bozbar &&
 274     git-update-index --add bozbar &&
 275     echo gnusto gnusto >bozbar &&
 276     read_tree_twoway $treeH $treeM &&
 277     git-ls-files --stage >19.out &&
 278     git diff M.out 19.out &&
 279     check_cache_at bozbar dirty'
 280test_expect_success \
 282    '20 - no local change, use new tree.' \
 283    'rm -f .git/index &&
 284     git-read-tree $treeH &&
 285     git-checkout-index -u -f -q -a &&
 286     cat bozbar-old >bozbar &&
 287     git-update-index --add bozbar &&
 288     read_tree_twoway $treeH $treeM &&
 289     git-ls-files --stage >20.out &&
 290     git diff M.out 20.out &&
 291     check_cache_at bozbar dirty'
 292test_expect_success \
 294    '21 - no local change, dirty cache.' \
 295    'rm -f .git/index &&
 296     git-read-tree $treeH &&
 297     git-checkout-index -u -f -q -a &&
 298     cat bozbar-old >bozbar &&
 299     git-update-index --add bozbar &&
 300     echo gnusto gnusto >bozbar &&
 301     if read_tree_twoway $treeH $treeM; then false; else :; fi'
 302# This fails with straight two-way fast forward.
 304test_expect_success \
 305    '22 - local change cache updated.' \
 306    'rm -f .git/index &&
 307     git-read-tree $treeH &&
 308     git-checkout-index -u -f -q -a &&
 309     sed -e "s/such as/SUCH AS/" bozbar-old >bozbar &&
 310     git-update-index --add bozbar &&
 311     if read_tree_twoway $treeH $treeM; then false; else :; fi'
 312# Also make sure we did not break DF vs DF/DF case.
 314test_expect_success \
 315    'DF vs DF/DF case setup.' \
 316    'rm -f .git/index &&
 317     echo DF >DF &&
 318     git-update-index --add DF &&
 319     treeDF=`git-write-tree` &&
 320     echo treeDF $treeDF &&
 321     git-ls-tree $treeDF &&
 322     rm -f DF &&
 324     mkdir DF &&
 325     echo DF/DF >DF/DF &&
 326     git-update-index --add --remove DF DF/DF &&
 327     treeDFDF=`git-write-tree` &&
 328     echo treeDFDF $treeDFDF &&
 329     git-ls-tree $treeDFDF &&
 330     git-ls-files --stage >DFDF.out'
 331test_expect_success \
 333    'DF vs DF/DF case test.' \
 334    'rm -f .git/index &&
 335     rm -fr DF &&
 336     echo DF >DF &&
 337     git-update-index --add DF &&
 338     read_tree_twoway $treeDF $treeDFDF &&
 339     git-ls-files --stage >DFDFcheck.out &&
 340     git diff DFDF.out DFDFcheck.out &&
 341     check_cache_at DF/DF dirty &&
 342     :'
 343test_done