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 expeced 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        apply_patch --whitespace=error-all && return 1
 103        test -s target && return 1
 104        : happy
 105'
 107test_expect_success 'whitespace=error-all, no rule' '
 109        git config core.whitespace -trailing,-space-before,-indent &&
 111        apply_patch --whitespace=error-all &&
 112        test_cmp file target
 113'
 115test_expect_success 'whitespace=error-all, no rule (attribute)' '
 117        git config --unset core.whitespace &&
 119        echo "target -whitespace" >.gitattributes &&
 120        apply_patch --whitespace=error-all &&
 121        test_cmp file target
 122'
 124test_expect_success 'spaces inserted by tab-in-indent' '
 126        git config core.whitespace -trailing,-space,-indent,tab &&
 128        rm -f .gitattributes &&
 129        test_fix % &&
 130        sed -e "s/_/ /g" -e "s/>/       /" <<-\EOF >expect &&
 131                An_SP in an ordinary line>and a HT.
 132                ________A HT (%).
 133                ________A SP and a HT (@%).
 134                _________A SP, a HT and a SP (@%).
 135                _______Seven SP.
 136                ________Eight SP (#).
 137                ________Seven SP and a HT (@%).
 138                ________________Eight SP and a HT (@#%).
 139                _________Seven SP, a HT and a SP (@%).
 140                _________________Eight SP, a HT and a SP (@#%).
 141                _______________Fifteen SP (#).
 142                ________________Fifteen SP and a HT (@#%).
 143                ________________Sixteen SP (#=).
 144                ________________________Sixteen SP and a HT (@#%=).
 145                _____a__Five SP, a non WS, two SP.
 146                A line with a (!) trailing SP_
 147                A line with a (!) trailing HT>
 148        EOF
 149        test_cmp expect target
 150'
 152for t in - ''
 154do
 155        case "$t" in '') tt='!' ;; *) tt= ;; esac
 156        for s in - ''
 157        do
 158                case "$s" in '') ts='@' ;; *) ts= ;; esac
 159                for i in - ''
 160                do
 161                        case "$i" in '') ti='#' ti16='=';; *) ti= ti16= ;; esac
 162                        for h in - ''
 163                        do
 164                                [ -z "$h$i" ] && continue
 165                                case "$h" in '') th='%' ;; *) th= ;; esac
 166                                rule=${t}trailing,${s}space,${i}indent,${h}tab
 167                                rm -f .gitattributes
 169                                test_expect_success "rule=$rule" '
 170                                        git config core.whitespace "$rule" &&
 171                                        test_fix "$tt$ts$ti$th"
 172                                '
 173                                test_expect_success "rule=$rule,tabwidth=16" '
 175                                        git config core.whitespace "$rule,tabwidth=16" &&
 176                                        test_fix "$tt$ts$ti16$th"
 177                                '
 178                                test_expect_success "rule=$rule (attributes)" '
 180                                        git config --unset core.whitespace &&
 181                                        echo "target whitespace=$rule" >.gitattributes &&
 182                                        test_fix "$tt$ts$ti$th"
 183                                '
 184                                test_expect_success "rule=$rule,tabwidth=16 (attributes)" '
 186                                        echo "target whitespace=$rule,tabwidth=16" >.gitattributes &&
 187                                        test_fix "$tt$ts$ti16$th"
 188                                '
 189                        done
 191                done
 192        done
 193done
 194create_patch () {
 196        sed -e "s/_/ /" <<-\EOF
 197                diff --git a/target b/target
 198                index e69de29..8bd6648 100644
 199                --- a/target
 200                +++ b/target
 201                @@ -0,0 +1,3 @@
 202                +An empty line follows
 203                +
 204                +A line with trailing whitespace and no newline_
 205                \ No newline at end of file
 206        EOF
 207}
 208test_expect_success 'trailing whitespace & no newline at the end of file' '
 210        >target &&
 211        create_patch >patch-file &&
 212        git apply --whitespace=fix patch-file &&
 213        grep "newline$" target &&
 214        grep "^$" target
 215'
 216test_expect_success 'blank at EOF with --whitespace=fix (1)' '
 218        test_might_fail git config --unset core.whitespace &&
 219        rm -f .gitattributes &&
 220        { echo a; echo b; echo c; } >one &&
 222        git add one &&
 223        { echo a; echo b; echo c; } >expect &&
 224        { cat expect; echo; } >one &&
 225        git diff -- one >patch &&
 226        git checkout one &&
 228        git apply --whitespace=fix patch &&
 229        test_cmp expect one
 230'
 231test_expect_success 'blank at EOF with --whitespace=fix (2)' '
 233        { echo a; echo b; echo c; } >one &&
 234        git add one &&
 235        { echo a; echo c; } >expect &&
 236        { cat expect; echo; echo; } >one &&
 237        git diff -- one >patch &&
 238        git checkout one &&
 240        git apply --whitespace=fix patch &&
 241        test_cmp expect one
 242'
 243test_expect_success 'blank at EOF with --whitespace=fix (3)' '
 245        { echo a; echo b; echo; } >one &&
 246        git add one &&
 247        { echo a; echo c; echo; } >expect &&
 248        { cat expect; echo; echo; } >one &&
 249        git diff -- one >patch &&
 250        git checkout one &&
 252        git apply --whitespace=fix patch &&
 253        test_cmp expect one
 254'
 255test_expect_success 'blank at end of hunk, not at EOF with --whitespace=fix' '
 257        { echo a; echo b; echo; echo; echo; echo; echo; echo d; } >one &&
 258        git add one &&
 259        { echo a; echo c; echo; echo; echo; echo; echo; echo; echo d; } >expect &&
 260        cp expect one &&
 261        git diff -- one >patch &&
 262        git checkout one &&
 264        git apply --whitespace=fix patch &&
 265        test_cmp expect one
 266'
 267test_expect_success 'blank at EOF with --whitespace=warn' '
 269        { echo a; echo b; echo c; } >one &&
 270        git add one &&
 271        echo >>one &&
 272        cat one >expect &&
 273        git diff -- one >patch &&
 274        git checkout one &&
 276        git apply --whitespace=warn patch 2>error &&
 277        test_cmp expect one &&
 278        grep "new blank line at EOF" error
 279'
 280test_expect_success 'blank at EOF with --whitespace=error' '
 282        { echo a; echo b; echo c; } >one &&
 283        git add one &&
 284        cat one >expect &&
 285        echo >>one &&
 286        git diff -- one >patch &&
 287        git checkout one &&
 289        test_must_fail git apply --whitespace=error patch 2>error &&
 290        test_cmp expect one &&
 291        grep "new blank line at EOF" error
 292'
 293test_expect_success 'blank but not empty at EOF' '
 295        { echo a; echo b; echo c; } >one &&
 296        git add one &&
 297        echo "   " >>one &&
 298        cat one >expect &&
 299        git diff -- one >patch &&
 300        git checkout one &&
 302        git apply --whitespace=warn patch 2>error &&
 303        test_cmp expect one &&
 304        grep "new blank line at EOF" error
 305'
 306test_expect_success 'applying beyond EOF requires one non-blank context line' '
 308        { echo; echo; echo; echo; } >one &&
 309        git add one &&
 310        { echo b; } >>one &&
 311        git diff -- one >patch &&
 312        git checkout one &&
 314        { echo a; echo; } >one &&
 315        cp one expect &&
 316        test_must_fail git apply --whitespace=fix patch &&
 317        test_cmp one expect &&
 318        test_must_fail git apply --ignore-space-change --whitespace=fix patch &&
 319        test_cmp one expect
 320'
 321test_expect_success 'tons of blanks at EOF should not apply' '
 323        for i in 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16; do
 324                echo; echo; echo; echo;
 325        done >one &&
 326        git add one &&
 327        echo a >>one &&
 328        git diff -- one >patch &&
 329        >one &&
 331        test_must_fail git apply --whitespace=fix patch &&
 332        test_must_fail git apply --ignore-space-change --whitespace=fix patch
 333'
 334test_expect_success 'missing blank line at end with --whitespace=fix' '
 336        echo a >one &&
 337        echo >>one &&
 338        git add one &&
 339        echo b >>one &&
 340        cp one expect &&
 341        git diff -- one >patch &&
 342        echo a >one &&
 343        cp one saved-one &&
 344        test_must_fail git apply patch &&
 345        git apply --whitespace=fix patch &&
 346        test_cmp one expect &&
 347        mv saved-one one &&
 348        git apply --ignore-space-change --whitespace=fix patch &&
 349        test_cmp one expect
 350'
 351test_expect_success 'two missing blank lines at end with --whitespace=fix' '
 353        { echo a; echo; echo b; echo c; } >one &&
 354        cp one no-blank-lines &&
 355        { echo; echo; } >>one &&
 356        git add one &&
 357        echo d >>one &&
 358        cp one expect &&
 359        echo >>one &&
 360        git diff -- one >patch &&
 361        cp no-blank-lines one &&
 362        test_must_fail git apply patch &&
 363        git apply --whitespace=fix patch &&
 364        test_cmp one expect &&
 365        mv no-blank-lines one &&
 366        test_must_fail git apply patch &&
 367        git apply --ignore-space-change --whitespace=fix patch &&
 368        test_cmp one expect
 369'
 370test_expect_success 'missing blank line at end, insert before end, --whitespace=fix' '
 372        { echo a; echo; } >one &&
 373        git add one &&
 374        { echo b; echo a; echo; } >one &&
 375        cp one expect &&
 376        git diff -- one >patch &&
 377        echo a >one &&
 378        test_must_fail git apply patch &&
 379        git apply --whitespace=fix patch &&
 380        test_cmp one expect
 381'
 382test_expect_success 'shrink file with tons of missing blanks at end of file' '
 384        { echo a; echo b; echo c; } >one &&
 385        cp one no-blank-lines &&
 386        for i in 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16; do
 387                echo; echo; echo; echo;
 388        done >>one &&
 389        git add one &&
 390        echo a >one &&
 391        cp one expect &&
 392        git diff -- one >patch &&
 393        cp no-blank-lines one &&
 394        test_must_fail git apply patch &&
 395        git apply --whitespace=fix patch &&
 396        test_cmp one expect &&
 397        mv no-blank-lines one &&
 398        git apply --ignore-space-change --whitespace=fix patch &&
 399        test_cmp one expect
 400'
 401test_expect_success 'missing blanks at EOF must only match blank lines' '
 403        { echo a; echo b; } >one &&
 404        git add one &&
 405        { echo c; echo d; } >>one &&
 406        git diff -- one >patch &&
 407        echo a >one &&
 409        test_must_fail git apply patch &&
 410        test_must_fail git apply --whitespace=fix patch &&
 411        test_must_fail git apply --ignore-space-change --whitespace=fix patch
 412'
 413sed -e's/Z//' >one <<EOF
 415a
 416b
 417c
 418                      Z
 419EOF
 420test_expect_success 'missing blank line should match context line with spaces' '
 422        git add one &&
 423        echo d >>one &&
 424        git diff -- one >patch &&
 425        { echo a; echo b; echo c; } >one &&
 426        cp one expect &&
 427        { echo; echo d; } >>expect &&
 428        git add one &&
 429        git apply --whitespace=fix patch &&
 431        test_cmp one expect
 432'
 433sed -e's/Z//' >one <<EOF
 435a
 436b
 437c
 438                      Z
 439EOF
 440test_expect_success 'same, but with the --ignore-space-option' '
 442        git add one &&
 443        echo d >>one &&
 444        cp one expect &&
 445        git diff -- one >patch &&
 446        { echo a; echo b; echo c; } >one &&
 447        git add one &&
 448        git checkout-index -f one &&
 450        git apply --ignore-space-change --whitespace=fix patch &&
 451        test_cmp one expect
 452'
 453test_expect_success 'same, but with CR-LF line endings && cr-at-eol set' '
 455        git config core.whitespace cr-at-eol &&
 456        printf "a\r\n" >one &&
 457        printf "b\r\n" >>one &&
 458        printf "c\r\n" >>one &&
 459        cp one save-one &&
 460        printf "                 \r\n" >>one &&
 461        git add one &&
 462        printf "d\r\n" >>one &&
 463        cp one expect &&
 464        git diff -- one >patch &&
 465        mv save-one one &&
 466        git apply --ignore-space-change --whitespace=fix patch &&
 468        test_cmp one expect
 469'
 470test_expect_success 'same, but with CR-LF line endings && cr-at-eol unset' '
 472        git config --unset core.whitespace &&
 473        printf "a\r\n" >one &&
 474        printf "b\r\n" >>one &&
 475        printf "c\r\n" >>one &&
 476        cp one save-one &&
 477        printf "                 \r\n" >>one &&
 478        git add one &&
 479        cp one expect &&
 480        printf "d\r\n" >>one &&
 481        git diff -- one >patch &&
 482        mv save-one one &&
 483        echo d >>expect &&
 484        git apply --ignore-space-change --whitespace=fix patch &&
 486        test_cmp one expect
 487'
 488test_done