1#!/bin/sh
   2test_description='git blame corner cases'
   4. ./test-lib.sh
   5pick_fc='s/^[0-9a-f^]* *\([^ ]*\) *(\([^ ]*\) .*/\1-\2/'
   7test_expect_success setup '
   9        echo A A A A A >one &&
  11        echo B B B B B >two &&
  12        echo C C C C C >tres &&
  13        echo ABC >mouse &&
  14        for i in 1 2 3 4 5 6 7 8 9
  15        do
  16                echo $i
  17        done >nine_lines &&
  18        for i in 1 2 3 4 5 6 7 8 9 a
  19        do
  20                echo $i
  21        done >ten_lines &&
  22        git add one two tres mouse nine_lines ten_lines &&
  23        test_tick &&
  24        GIT_AUTHOR_NAME=Initial git commit -m Initial &&
  25        cat one >uno &&
  27        mv two dos &&
  28        cat one >>tres &&
  29        echo DEF >>mouse &&
  30        git add uno dos tres mouse &&
  31        test_tick &&
  32        GIT_AUTHOR_NAME=Second git commit -a -m Second &&
  33        echo GHIJK >>mouse &&
  35        git add mouse &&
  36        test_tick &&
  37        GIT_AUTHOR_NAME=Third git commit -m Third &&
  38        cat mouse >cow &&
  40        git add cow &&
  41        test_tick &&
  42        GIT_AUTHOR_NAME=Fourth git commit -m Fourth &&
  43        cat >cow <<-\EOF &&
  45        ABC
  46        DEF
  47        XXXX
  48        GHIJK
  49        EOF
  50        git add cow &&
  51        test_tick &&
  52        GIT_AUTHOR_NAME=Fifth git commit -m Fifth
  53'
  54test_expect_success 'straight copy without -C' '
  56        git blame uno | grep Second
  58'
  60test_expect_success 'straight move without -C' '
  62        git blame dos | grep Initial
  64'
  66test_expect_success 'straight copy with -C' '
  68        git blame -C1 uno | grep Second
  70'
  72test_expect_success 'straight move with -C' '
  74        git blame -C1 dos | grep Initial
  76'
  78test_expect_success 'straight copy with -C -C' '
  80        git blame -C -C1 uno | grep Initial
  82'
  84test_expect_success 'straight move with -C -C' '
  86        git blame -C -C1 dos | grep Initial
  88'
  90test_expect_success 'append without -C' '
  92        git blame -L2 tres | grep Second
  94'
  96test_expect_success 'append with -C' '
  98        git blame -L2 -C1 tres | grep Second
 100'
 102test_expect_success 'append with -C -C' '
 104        git blame -L2 -C -C1 tres | grep Second
 106'
 108test_expect_success 'append with -C -C -C' '
 110        git blame -L2 -C -C -C1 tres | grep Initial
 112'
 114test_expect_success 'blame wholesale copy' '
 116        git blame -f -C -C1 HEAD^ -- cow | sed -e "$pick_fc" >current &&
 118        cat >expected <<-\EOF &&
 119        mouse-Initial
 120        mouse-Second
 121        mouse-Third
 122        EOF
 123        test_cmp expected current
 124'
 126test_expect_success 'blame wholesale copy and more' '
 128        git blame -f -C -C1 HEAD -- cow | sed -e "$pick_fc" >current &&
 130        cat >expected <<-\EOF &&
 131        mouse-Initial
 132        mouse-Second
 133        cow-Fifth
 134        mouse-Third
 135        EOF
 136        test_cmp expected current
 137'
 139test_expect_success 'blame wholesale copy and more in the index' '
 141        cat >horse <<-\EOF &&
 143        ABC
 144        DEF
 145        XXXX
 146        YYYY
 147        GHIJK
 148        EOF
 149        git add horse &&
 150        test_when_finished "git rm -f horse" &&
 151        git blame -f -C -C1 -- horse | sed -e "$pick_fc" >current &&
 152        cat >expected <<-\EOF &&
 153        mouse-Initial
 154        mouse-Second
 155        cow-Fifth
 156        horse-Not
 157        mouse-Third
 158        EOF
 159        test_cmp expected current
 160'
 162test_expect_success 'blame during cherry-pick with file rename conflict' '
 164        test_when_finished "git reset --hard && git checkout master" &&
 166        git checkout HEAD~3 &&
 167        echo MOUSE >> mouse &&
 168        git mv mouse rodent &&
 169        git add rodent &&
 170        GIT_AUTHOR_NAME=Rodent git commit -m "rodent" &&
 171        git checkout --detach master &&
 172        (git cherry-pick HEAD@{1} || test $? -eq 1) &&
 173        git show HEAD@{1}:rodent > rodent &&
 174        git add rodent &&
 175        git blame -f -C -C1 rodent | sed -e "$pick_fc" >current &&
 176        cat current &&
 177        cat >expected <<-\EOF &&
 178        mouse-Initial
 179        mouse-Second
 180        rodent-Not
 181        EOF
 182        test_cmp expected current
 183'
 184test_expect_success 'blame path that used to be a directory' '
 186        mkdir path &&
 187        echo A A A A A >path/file &&
 188        echo B B B B B >path/elif &&
 189        git add path &&
 190        test_tick &&
 191        git commit -m "path was a directory" &&
 192        rm -fr path &&
 193        echo A A A A A >path &&
 194        git add path &&
 195        test_tick &&
 196        git commit -m "path is a regular file" &&
 197        git blame HEAD^.. -- path
 198'
 199test_expect_success 'blame to a commit with no author name' '
 201  TREE=$(git rev-parse HEAD:) &&
 202  cat >badcommit <<EOF &&
 203tree $TREE
 204author <noname> 1234567890 +0000
 205committer David Reiss <dreiss@facebook.com> 1234567890 +0000
 206some message
 208EOF
 209  COMMIT=$(git hash-object -t commit -w badcommit) &&
 210  git --no-pager blame $COMMIT -- uno >/dev/null
 211'
 212test_expect_success 'blame -L with invalid start' '
 214        test_must_fail git blame -L5 tres 2>errors &&
 215        test_i18ngrep "has only 2 lines" errors
 216'
 217test_expect_success 'blame -L with invalid end' '
 219        git blame -L1,5 tres >out &&
 220        test_line_count = 2 out
 221'
 222test_expect_success 'blame parses <end> part of -L' '
 224        git blame -L1,1 tres >out &&
 225        test_line_count = 1 out
 226'
 227test_expect_success 'blame -Ln,-(n+1)' '
 229        git blame -L3,-4 nine_lines >out &&
 230        test_line_count = 3 out
 231'
 232test_expect_success 'indent of line numbers, nine lines' '
 234        git blame nine_lines >actual &&
 235        test $(grep -c "  " actual) = 0
 236'
 237test_expect_success 'indent of line numbers, ten lines' '
 239        git blame ten_lines >actual &&
 240        test $(grep -c "  " actual) = 9
 241'
 242test_expect_success 'setup file with CRLF newlines' '
 244        git config core.autocrlf false &&
 245        printf "testcase\n" >crlffile &&
 246        git add crlffile &&
 247        git commit -m testcase &&
 248        printf "testcase\r\n" >crlffile
 249'
 250test_expect_success 'blame file with CRLF core.autocrlf true' '
 252        git config core.autocrlf true &&
 253        git blame crlffile >actual &&
 254        grep "A U Thor" actual
 255'
 256test_expect_success 'blame file with CRLF attributes text' '
 258        git config core.autocrlf false &&
 259        echo "crlffile text" >.gitattributes &&
 260        git blame crlffile >actual &&
 261        grep "A U Thor" actual
 262'
 263test_expect_success 'blame file with CRLF core.autocrlf=true' '
 265        git config core.autocrlf false &&
 266        printf "testcase\r\n" >crlfinrepo &&
 267        >.gitattributes &&
 268        git add crlfinrepo &&
 269        git commit -m "add crlfinrepo" &&
 270        git config core.autocrlf true &&
 271        mv crlfinrepo tmp &&
 272        git checkout crlfinrepo &&
 273        rm tmp &&
 274        git blame crlfinrepo >actual &&
 275        grep "A U Thor" actual
 276'
 277test_done