1#!/bin/sh
   2test_description='core.whitespace rules and git apply'
   4. ./test-lib.sh
   6prepare_test_file () {
   8        # A line that has character X is touched iff RULE is in effect:
  10        #       X  RULE
  11        #       !  trailing-space
  12        #       @  space-before-tab
  13        #       #  indent-with-non-tab (default tab width 8)
  14        #       =  indent-with-non-tab,tabwidth=16
  15        #       %  tab-in-indent
  16        sed -e "s/_/ /g" -e "s/>/       /" <<-\EOF
  17                An_SP in an ordinary line>and a HT.
  18                >A HT (%).
  19                _>A SP and a HT (@%).
  20                _>_A SP, a HT and a SP (@%).
  21                _______Seven SP.
  22                ________Eight SP (#).
  23                _______>Seven SP and a HT (@%).
  24                ________>Eight SP and a HT (@#%).
  25                _______>_Seven SP, a HT and a SP (@%).
  26                ________>_Eight SP, a HT and a SP (@#%).
  27                _______________Fifteen SP (#).
  28                _______________>Fifteen SP and a HT (@#%).
  29                ________________Sixteen SP (#=).
  30                ________________>Sixteen SP and a HT (@#%=).
  31                _____a__Five SP, a non WS, two SP.
  32                A line with a (!) trailing SP_
  33                A line with a (!) trailing HT>
  34        EOF
  35}
  36apply_patch () {
  38        >target &&
  39        sed -e "s|\([ab]\)/file|\1/target|" <patch |
  40        git apply "$@"
  41}
  42test_fix () {
  44        # fix should not barf
  45        apply_patch --whitespace=fix || return 1
  46        # find touched lines
  48        $DIFF file target | sed -n -e "s/^> //p" >fixed
  49        # the changed lines are all expected to change
  51        fixed_cnt=$(wc -l <fixed)
  52        case "$1" in
  53        '') expect_cnt=$fixed_cnt ;;
  54        ?*) expect_cnt=$(grep "[$1]" <fixed | wc -l) ;;
  55        esac
  56        test $fixed_cnt -eq $expect_cnt || return 1
  57        # and we are not missing anything
  59        case "$1" in
  60        '') expect_cnt=0 ;;
  61        ?*) expect_cnt=$(grep "[$1]" <file | wc -l) ;;
  62        esac
  63        test $fixed_cnt -eq $expect_cnt || return 1
  64        # Get the patch actually applied
  66        git diff-files -p target >fixed-patch
  67        test -s fixed-patch && return 0
  68        # Make sure it is complaint-free
  70        >target
  71        git apply --whitespace=error-all <fixed-patch
  72}
  74test_expect_success setup '
  76        >file &&
  78        git add file &&
  79        prepare_test_file >file &&
  80        git diff-files -p >patch &&
  81        >target &&
  82        git add target
  83'
  85test_expect_success 'whitespace=nowarn, default rule' '
  87        apply_patch --whitespace=nowarn &&
  89        test_cmp file target
  90'
  92test_expect_success 'whitespace=warn, default rule' '
  94        apply_patch --whitespace=warn &&
  96        test_cmp file target
  97'
  99test_expect_success 'whitespace=error-all, default rule' '
 101        test_must_fail apply_patch --whitespace=error-all &&
 103        ! test -s target
 104'
 106test_expect_success 'whitespace=error-all, no rule' '
 108        git config core.whitespace -trailing,-space-before,-indent &&
 110        apply_patch --whitespace=error-all &&
 111        test_cmp file target
 112'
 114test_expect_success 'whitespace=error-all, no rule (attribute)' '
 116        git config --unset core.whitespace &&
 118        echo "target -whitespace" >.gitattributes &&
 119        apply_patch --whitespace=error-all &&
 120        test_cmp file target
 121'
 123test_expect_success 'spaces inserted by tab-in-indent' '
 125        git config core.whitespace -trailing,-space,-indent,tab &&
 127        rm -f .gitattributes &&
 128        test_fix % &&
 129        sed -e "s/_/ /g" -e "s/>/       /" <<-\EOF >expect &&
 130                An_SP in an ordinary line>and a HT.
 131                ________A HT (%).
 132                ________A SP and a HT (@%).
 133                _________A SP, a HT and a SP (@%).
 134                _______Seven SP.
 135                ________Eight SP (#).
 136                ________Seven SP and a HT (@%).
 137                ________________Eight SP and a HT (@#%).
 138                _________Seven SP, a HT and a SP (@%).
 139                _________________Eight SP, a HT and a SP (@#%).
 140                _______________Fifteen SP (#).
 141                ________________Fifteen SP and a HT (@#%).
 142                ________________Sixteen SP (#=).
 143                ________________________Sixteen SP and a HT (@#%=).
 144                _____a__Five SP, a non WS, two SP.
 145                A line with a (!) trailing SP_
 146                A line with a (!) trailing HT>
 147        EOF
 148        test_cmp expect target
 149'
 151for t in - ''
 153do
 154        case "$t" in '') tt='!' ;; *) tt= ;; esac
 155        for s in - ''
 156        do
 157                case "$s" in '') ts='@' ;; *) ts= ;; esac
 158                for i in - ''
 159                do
 160                        case "$i" in '') ti='#' ti16='=';; *) ti= ti16= ;; esac
 161                        for h in - ''
 162                        do
 163                                [ -z "$h$i" ] && continue
 164                                case "$h" in '') th='%' ;; *) th= ;; esac
 165                                rule=${t}trailing,${s}space,${i}indent,${h}tab
 166                                rm -f .gitattributes
 168                                test_expect_success "rule=$rule" '
 169                                        git config core.whitespace "$rule" &&
 170                                        test_fix "$tt$ts$ti$th"
 171                                '
 172                                test_expect_success "rule=$rule,tabwidth=16" '
 174                                        git config core.whitespace "$rule,tabwidth=16" &&
 175                                        test_fix "$tt$ts$ti16$th"
 176                                '
 177                                test_expect_success "rule=$rule (attributes)" '
 179                                        git config --unset core.whitespace &&
 180                                        echo "target whitespace=$rule" >.gitattributes &&
 181                                        test_fix "$tt$ts$ti$th"
 182                                '
 183                                test_expect_success "rule=$rule,tabwidth=16 (attributes)" '
 185                                        echo "target whitespace=$rule,tabwidth=16" >.gitattributes &&
 186                                        test_fix "$tt$ts$ti16$th"
 187                                '
 188                        done
 190                done
 191        done
 192done
 193create_patch () {
 195        sed -e "s/_/ /" <<-\EOF
 196                diff --git a/target b/target
 197                index e69de29..8bd6648 100644
 198                --- a/target
 199                +++ b/target
 200                @@ -0,0 +1,3 @@
 201                +An empty line follows
 202                +
 203                +A line with trailing whitespace and no newline_
 204                \ No newline at end of file
 205        EOF
 206}
 207test_expect_success 'trailing whitespace & no newline at the end of file' '
 209        >target &&
 210        create_patch >patch-file &&
 211        git apply --whitespace=fix patch-file &&
 212        grep "newline$" target &&
 213        grep "^$" target
 214'
 215test_expect_success 'blank at EOF with --whitespace=fix (1)' '
 217        test_might_fail git config --unset core.whitespace &&
 218        rm -f .gitattributes &&
 219        { echo a; echo b; echo c; } >one &&
 221        git add one &&
 222        { echo a; echo b; echo c; } >expect &&
 223        { cat expect; echo; } >one &&
 224        git diff -- one >patch &&
 225        git checkout one &&
 227        git apply --whitespace=fix patch &&
 228        test_cmp expect one
 229'
 230test_expect_success 'blank at EOF with --whitespace=fix (2)' '
 232        { echo a; echo b; echo c; } >one &&
 233        git add one &&
 234        { echo a; echo c; } >expect &&
 235        { cat expect; echo; echo; } >one &&
 236        git diff -- one >patch &&
 237        git checkout one &&
 239        git apply --whitespace=fix patch &&
 240        test_cmp expect one
 241'
 242test_expect_success 'blank at EOF with --whitespace=fix (3)' '
 244        { echo a; echo b; echo; } >one &&
 245        git add one &&
 246        { echo a; echo c; echo; } >expect &&
 247        { cat expect; echo; echo; } >one &&
 248        git diff -- one >patch &&
 249        git checkout one &&
 251        git apply --whitespace=fix patch &&
 252        test_cmp expect one
 253'
 254test_expect_success 'blank at end of hunk, not at EOF with --whitespace=fix' '
 256        { echo a; echo b; echo; echo; echo; echo; echo; echo d; } >one &&
 257        git add one &&
 258        { echo a; echo c; echo; echo; echo; echo; echo; echo; echo d; } >expect &&
 259        cp expect one &&
 260        git diff -- one >patch &&
 261        git checkout one &&
 263        git apply --whitespace=fix patch &&
 264        test_cmp expect one
 265'
 266test_expect_success 'blank at EOF with --whitespace=warn' '
 268        { echo a; echo b; echo c; } >one &&
 269        git add one &&
 270        echo >>one &&
 271        cat one >expect &&
 272        git diff -- one >patch &&
 273        git checkout one &&
 275        git apply --whitespace=warn patch 2>error &&
 276        test_cmp expect one &&
 277        grep "new blank line at EOF" error
 278'
 279test_expect_success 'blank at EOF with --whitespace=error' '
 281        { echo a; echo b; echo c; } >one &&
 282        git add one &&
 283        cat one >expect &&
 284        echo >>one &&
 285        git diff -- one >patch &&
 286        git checkout one &&
 288        test_must_fail git apply --whitespace=error patch 2>error &&
 289        test_cmp expect one &&
 290        grep "new blank line at EOF" error
 291'
 292test_expect_success 'blank but not empty at EOF' '
 294        { echo a; echo b; echo c; } >one &&
 295        git add one &&
 296        echo "   " >>one &&
 297        cat one >expect &&
 298        git diff -- one >patch &&
 299        git checkout one &&
 301        git apply --whitespace=warn patch 2>error &&
 302        test_cmp expect one &&
 303        grep "new blank line at EOF" error
 304'
 305test_expect_success 'applying beyond EOF requires one non-blank context line' '
 307        { echo; echo; echo; echo; } >one &&
 308        git add one &&
 309        { echo b; } >>one &&
 310        git diff -- one >patch &&
 311        git checkout one &&
 313        { echo a; echo; } >one &&
 314        cp one expect &&
 315        test_must_fail git apply --whitespace=fix patch &&
 316        test_cmp one expect &&
 317        test_must_fail git apply --ignore-space-change --whitespace=fix patch &&
 318        test_cmp one expect
 319'
 320test_expect_success 'tons of blanks at EOF should not apply' '
 322        for i in 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16; do
 323                echo; echo; echo; echo;
 324        done >one &&
 325        git add one &&
 326        echo a >>one &&
 327        git diff -- one >patch &&
 328        >one &&
 330        test_must_fail git apply --whitespace=fix patch &&
 331        test_must_fail git apply --ignore-space-change --whitespace=fix patch
 332'
 333test_expect_success 'missing blank line at end with --whitespace=fix' '
 335        echo a >one &&
 336        echo >>one &&
 337        git add one &&
 338        echo b >>one &&
 339        cp one expect &&
 340        git diff -- one >patch &&
 341        echo a >one &&
 342        cp one saved-one &&
 343        test_must_fail git apply patch &&
 344        git apply --whitespace=fix patch &&
 345        test_cmp one expect &&
 346        mv saved-one one &&
 347        git apply --ignore-space-change --whitespace=fix patch &&
 348        test_cmp one expect
 349'
 350test_expect_success 'two missing blank lines at end with --whitespace=fix' '
 352        { echo a; echo; echo b; echo c; } >one &&
 353        cp one no-blank-lines &&
 354        { echo; echo; } >>one &&
 355        git add one &&
 356        echo d >>one &&
 357        cp one expect &&
 358        echo >>one &&
 359        git diff -- one >patch &&
 360        cp no-blank-lines one &&
 361        test_must_fail git apply patch &&
 362        git apply --whitespace=fix patch &&
 363        test_cmp one expect &&
 364        mv no-blank-lines one &&
 365        test_must_fail git apply patch &&
 366        git apply --ignore-space-change --whitespace=fix patch &&
 367        test_cmp one expect
 368'
 369test_expect_success 'missing blank line at end, insert before end, --whitespace=fix' '
 371        { echo a; echo; } >one &&
 372        git add one &&
 373        { echo b; echo a; echo; } >one &&
 374        cp one expect &&
 375        git diff -- one >patch &&
 376        echo a >one &&
 377        test_must_fail git apply patch &&
 378        git apply --whitespace=fix patch &&
 379        test_cmp one expect
 380'
 381test_expect_success 'shrink file with tons of missing blanks at end of file' '
 383        { echo a; echo b; echo c; } >one &&
 384        cp one no-blank-lines &&
 385        for i in 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16; do
 386                echo; echo; echo; echo;
 387        done >>one &&
 388        git add one &&
 389        echo a >one &&
 390        cp one expect &&
 391        git diff -- one >patch &&
 392        cp no-blank-lines one &&
 393        test_must_fail git apply patch &&
 394        git apply --whitespace=fix patch &&
 395        test_cmp one expect &&
 396        mv no-blank-lines one &&
 397        git apply --ignore-space-change --whitespace=fix patch &&
 398        test_cmp one expect
 399'
 400test_expect_success 'missing blanks at EOF must only match blank lines' '
 402        { echo a; echo b; } >one &&
 403        git add one &&
 404        { echo c; echo d; } >>one &&
 405        git diff -- one >patch &&
 406        echo a >one &&
 408        test_must_fail git apply patch &&
 409        test_must_fail git apply --whitespace=fix patch &&
 410        test_must_fail git apply --ignore-space-change --whitespace=fix patch
 411'
 412sed -e's/Z//' >one <<EOF
 414a
 415b
 416c
 417                      Z
 418EOF
 419test_expect_success 'missing blank line should match context line with spaces' '
 421        git add one &&
 422        echo d >>one &&
 423        git diff -- one >patch &&
 424        { echo a; echo b; echo c; } >one &&
 425        cp one expect &&
 426        { echo; echo d; } >>expect &&
 427        git add one &&
 428        git apply --whitespace=fix patch &&
 430        test_cmp one expect
 431'
 432sed -e's/Z//' >one <<EOF
 434a
 435b
 436c
 437                      Z
 438EOF
 439test_expect_success 'same, but with the --ignore-space-option' '
 441        git add one &&
 442        echo d >>one &&
 443        cp one expect &&
 444        git diff -- one >patch &&
 445        { echo a; echo b; echo c; } >one &&
 446        git add one &&
 447        git checkout-index -f one &&
 449        git apply --ignore-space-change --whitespace=fix patch &&
 450        test_cmp one expect
 451'
 452test_expect_success 'same, but with CR-LF line endings && cr-at-eol set' '
 454        git config core.whitespace cr-at-eol &&
 455        printf "a\r\n" >one &&
 456        printf "b\r\n" >>one &&
 457        printf "c\r\n" >>one &&
 458        cp one save-one &&
 459        printf "                 \r\n" >>one &&
 460        git add one &&
 461        printf "d\r\n" >>one &&
 462        cp one expect &&
 463        git diff -- one >patch &&
 464        mv save-one one &&
 465        git apply --ignore-space-change --whitespace=fix patch &&
 467        test_cmp one expect
 468'
 469test_expect_success 'CR-LF line endings && add line && text=auto' '
 471        git config --unset core.whitespace &&
 472        printf "a\r\n" >one &&
 473        cp one save-one &&
 474        git add one &&
 475        printf "b\r\n" >>one &&
 476        cp one expect &&
 477        git diff -- one >patch &&
 478        mv save-one one &&
 479        echo "one text=auto" >.gitattributes &&
 480        git apply patch &&
 481        test_cmp one expect
 482'
 483test_expect_success 'CR-LF line endings && change line && text=auto' '
 485        printf "a\r\n" >one &&
 486        cp one save-one &&
 487        git add one &&
 488        printf "b\r\n" >one &&
 489        cp one expect &&
 490        git diff -- one >patch &&
 491        mv save-one one &&
 492        echo "one text=auto" >.gitattributes &&
 493        git apply patch &&
 494        test_cmp one expect
 495'
 496test_expect_success 'LF in repo, CRLF in worktree && change line && text=auto' '
 498        printf "a\n" >one &&
 499        git add one &&
 500        printf "b\r\n" >one &&
 501        git diff -- one >patch &&
 502        printf "a\r\n" >one &&
 503        echo "one text=auto" >.gitattributes &&
 504        git -c core.eol=CRLF apply patch &&
 505        printf "b\r\n" >expect &&
 506        test_cmp one expect
 507'
 508test_expect_success 'whitespace=fix to expand' '
 510        qz_to_tab_space >preimage <<-\EOF &&
 511        QQa
 512        QQb
 513        QQc
 514        ZZZZZZZZZZZZZZZZd
 515        QQe
 516        QQf
 517        QQg
 518        EOF
 519        qz_to_tab_space >patch <<-\EOF &&
 520        diff --git a/preimage b/preimage
 521        --- a/preimage
 522        +++ b/preimage
 523        @@ -1,7 +1,6 @@
 524         QQa
 525         QQb
 526         QQc
 527        -QQd
 528         QQe
 529         QQf
 530         QQg
 531        EOF
 532        git -c core.whitespace=tab-in-indent apply --whitespace=fix patch
 533'
 534test_expect_success 'whitespace check skipped for excluded paths' '
 536        git config core.whitespace blank-at-eol &&
 537        >used &&
 538        >unused &&
 539        git add used unused &&
 540        echo "used" >used &&
 541        echo "unused " >unused &&
 542        git diff-files -p used unused >patch &&
 543        git apply --include=used --stat --whitespace=error <patch
 544'
 545test_done