3 test_description='git commit porcelain-ish'
 
   8         expect=commit_msg_is.expect
 
   9         actual=commit_msg_is.actual
 
  11         printf "%s" "$(git log --pretty=format:%s%b -1)" >$actual &&
 
  12         printf "%s" "$1" >$expect &&
 
  13         test_i18ncmp $expect $actual
 
  16 # Arguments: [<prefix] [<commit message>] [<commit options>]
 
  17 check_summary_oneline() {
 
  19         git commit ${3+"$3"} -m "$2" | head -1 > act &&
 
  22         SUMMARY_PREFIX="$(git name-rev --name-only HEAD)" &&
 
  24         # append the "special" prefix, like "root-commit", "detached HEAD"
 
  27                 SUMMARY_PREFIX="$SUMMARY_PREFIX ($1)"
 
  31         SUMMARY_POSTFIX="$(git log -1 --pretty='format:%h')"
 
  32         echo "[$SUMMARY_PREFIX $SUMMARY_POSTFIX] $2" >exp &&
 
  37 test_expect_success 'output summary format' '
 
  41         check_summary_oneline "root-commit" "initial" &&
 
  43         echo change >>file1 &&
 
  47 test_expect_success 'output summary format: root-commit' '
 
  48         check_summary_oneline "" "a change"
 
  51 test_expect_success 'output summary format for commit with an empty diff' '
 
  53         check_summary_oneline "" "empty" "--allow-empty"
 
  56 test_expect_success 'output summary format for merges' '
 
  58         git checkout -b recursive-base &&
 
  59         test_commit base file1 &&
 
  61         git checkout -b recursive-a recursive-base &&
 
  62         test_commit commit-a file1 &&
 
  64         git checkout -b recursive-b recursive-base &&
 
  65         test_commit commit-b file1 &&
 
  68         git checkout recursive-a &&
 
  69         test_must_fail git merge recursive-b &&
 
  70         # resolve the conflict
 
  71         echo commit-a > file1 &&
 
  73         check_summary_oneline "" "Merge"
 
  76 output_tests_cleanup() {
 
  77         # this is needed for "do not fire editor in the presence of conflicts"
 
  78         git checkout master &&
 
  80         # this is needed for the "partial removal" test to pass
 
  82         git commit -m "cleanup"
 
  85 test_expect_success 'the basics' '
 
  87         output_tests_cleanup &&
 
  89         echo doing partial >"commit is" &&
 
  91         echo very much encouraged but we should >not/forbid &&
 
  92         git add "commit is" not &&
 
  93         echo update added "commit is" file >"commit is" &&
 
  94         echo also update another >not/forbid &&
 
  96         git commit -a -m "initial with -a" &&
 
  98         git cat-file blob HEAD:"commit is" >current.1 &&
 
  99         git cat-file blob HEAD:not/forbid >current.2 &&
 
 101         cmp current.1 "commit is" &&
 
 102         cmp current.2 not/forbid
 
 106 test_expect_success 'partial' '
 
 108         echo another >"commit is" &&
 
 109         echo another >not/forbid &&
 
 111         git commit -m "partial commit to handle a file" "commit is" &&
 
 113         changed=$(git diff-tree --name-only HEAD^ HEAD) &&
 
 114         test "$changed" = "commit is"
 
 118 test_expect_success 'partial modification in a subdirectory' '
 
 121         git commit -m "partial commit to subdirectory" not &&
 
 123         changed=$(git diff-tree -r --name-only HEAD^ HEAD) &&
 
 124         test "$changed" = "not/forbid"
 
 128 test_expect_success 'partial removal' '
 
 131         git commit -m "partial commit to remove not/forbid" not &&
 
 133         changed=$(git diff-tree -r --name-only HEAD^ HEAD) &&
 
 134         test "$changed" = "not/forbid" &&
 
 135         remain=$(git ls-tree -r --name-only HEAD) &&
 
 136         test "$remain" = "commit is"
 
 140 test_expect_success 'sign off' '
 
 144         git commit -s -m "thank you" &&
 
 145         actual=$(git cat-file commit HEAD | sed -ne "s/Signed-off-by: //p") &&
 
 146         expected=$(git var GIT_COMMITTER_IDENT | sed -e "s/>.*/>/") &&
 
 147         test "z$actual" = "z$expected"
 
 151 test_expect_success 'multiple -m' '
 
 155         git commit -m "one" -m "two" -m "three" &&
 
 156         actual=$(git cat-file commit HEAD | sed -e "1,/^\$/d") &&
 
 157         expected=$(echo one; echo; echo two; echo; echo three) &&
 
 158         test "z$actual" = "z$expected"
 
 162 test_expect_success 'verbose' '
 
 164         echo minus >negative &&
 
 166         git status -v | sed -ne "/^diff --git /p" >actual &&
 
 167         echo "diff --git a/negative b/negative" >expect &&
 
 168         test_cmp expect actual
 
 172 test_expect_success 'verbose respects diff config' '
 
 174         git config color.diff always &&
 
 175         git status -v >actual &&
 
 176         grep "\[1mdiff --git" actual &&
 
 177         git config --unset color.diff
 
 180 test_expect_success 'cleanup commit messages (verbatim option,-t)' '
 
 183         { echo;echo "# text";echo; } >expect &&
 
 184         git commit --cleanup=verbatim -t expect -a &&
 
 185         git cat-file -p HEAD |sed -e "1,/^\$/d" |head -n 3 >actual &&
 
 186         test_cmp expect actual
 
 190 test_expect_success 'cleanup commit messages (verbatim option,-F)' '
 
 193         git commit --cleanup=verbatim -F expect -a &&
 
 194         git cat-file -p HEAD |sed -e "1,/^\$/d">actual &&
 
 195         test_cmp expect actual
 
 199 test_expect_success 'cleanup commit messages (verbatim option,-m)' '
 
 202         git commit --cleanup=verbatim -m "$(cat expect)" -a &&
 
 203         git cat-file -p HEAD |sed -e "1,/^\$/d">actual &&
 
 204         test_cmp expect actual
 
 208 test_expect_success 'cleanup commit messages (whitespace option,-F)' '
 
 211         { echo;echo "# text";echo; } >text &&
 
 212         echo "# text" >expect &&
 
 213         git commit --cleanup=whitespace -F text -a &&
 
 214         git cat-file -p HEAD |sed -e "1,/^\$/d">actual &&
 
 215         test_cmp expect actual
 
 219 test_expect_success 'cleanup commit messages (strip option,-F)' '
 
 222         { echo;echo "# text";echo sample;echo; } >text &&
 
 223         echo sample >expect &&
 
 224         git commit --cleanup=strip -F text -a &&
 
 225         git cat-file -p HEAD |sed -e "1,/^\$/d">actual &&
 
 226         test_cmp expect actual
 
 230 test_expect_success 'cleanup commit messages (strip option,-F,-e)' '
 
 233         { echo;echo sample;echo; } >text &&
 
 234         git commit -e -F text -a &&
 
 235         head -n 4 .git/COMMIT_EDITMSG >actual
 
 240 # Please enter the commit message for your changes. Lines starting
 
 241 # with '#' will be ignored, and an empty message aborts the commit." >expect
 
 243 test_expect_success 'cleanup commit messages (strip option,-F,-e): output' '
 
 244         test_i18ncmp expect actual
 
 247 test_expect_success 'cleanup commit message (fail on invalid cleanup mode option)' '
 
 248         test_must_fail git commit --cleanup=non-existent
 
 251 test_expect_success 'cleanup commit message (fail on invalid cleanup mode configuration)' '
 
 252         test_must_fail git -c commit.cleanup=non-existent commit
 
 255 test_expect_success 'cleanup commit message (no config and no option uses default)' '
 
 256         echo content >>file &&
 
 258         test_set_editor "$TEST_DIRECTORY"/t7500/add-content-and-comment &&
 
 259         git commit --no-status &&
 
 260         commit_msg_is "commit message"
 
 263 test_expect_success 'cleanup commit message (option overrides default)' '
 
 264         echo content >>file &&
 
 266         test_set_editor "$TEST_DIRECTORY"/t7500/add-content-and-comment &&
 
 267         git commit --cleanup=whitespace --no-status &&
 
 268         commit_msg_is "commit message # comment"
 
 271 test_expect_success 'cleanup commit message (config overrides default)' '
 
 272         echo content >>file &&
 
 274         test_set_editor "$TEST_DIRECTORY"/t7500/add-content-and-comment &&
 
 275         git -c commit.cleanup=whitespace commit --no-status &&
 
 276         commit_msg_is "commit message # comment"
 
 279 test_expect_success 'cleanup commit message (option overrides config)' '
 
 280         echo content >>file &&
 
 282         test_set_editor "$TEST_DIRECTORY"/t7500/add-content-and-comment &&
 
 283         git -c commit.cleanup=whitespace commit --cleanup=default &&
 
 284         commit_msg_is "commit message"
 
 287 test_expect_success 'cleanup commit message (default, -m)' '
 
 288         echo content >>file &&
 
 290         git commit -m "message #comment " &&
 
 291         commit_msg_is "message #comment"
 
 294 test_expect_success 'cleanup commit message (whitespace option, -m)' '
 
 295         echo content >>file &&
 
 297         git commit --cleanup=whitespace --no-status -m "message #comment " &&
 
 298         commit_msg_is "message #comment"
 
 301 test_expect_success 'cleanup commit message (whitespace config, -m)' '
 
 302         echo content >>file &&
 
 304         git -c commit.cleanup=whitespace commit --no-status -m "message #comment " &&
 
 305         commit_msg_is "message #comment"
 
 308 test_expect_success 'message shows author when it is not equal to committer' '
 
 310         git commit -e -m "sample" -a &&
 
 312           "^# Author: *A U Thor <author@example.com>\$" \
 
 316 test_expect_success AUTOIDENT 'message shows committer when it is automatic' '
 
 320                 sane_unset GIT_COMMITTER_EMAIL &&
 
 321                 sane_unset GIT_COMMITTER_NAME &&
 
 322                 git commit -e -m "sample" -a
 
 324         # the ident is calculated from the system, so we cannot
 
 325         # check the actual value, only that it is there
 
 326         test_i18ngrep "^# Committer: " .git/COMMIT_EDITMSG
 
 329 write_script .git/FAKE_EDITOR <<EOF
 
 330 echo editor started > "$(pwd)/.git/result"
 
 334 test_expect_success !AUTOIDENT 'do not fire editor when committer is bogus' '
 
 340                 sane_unset GIT_COMMITTER_EMAIL &&
 
 341                 sane_unset GIT_COMMITTER_NAME &&
 
 342                 GIT_EDITOR="\"$(pwd)/.git/FAKE_EDITOR\"" &&
 
 344                 test_must_fail git commit -e -m sample -a
 
 346         test_cmp expect .git/result
 
 349 test_expect_success 'do not fire editor in the presence of conflicts' '
 
 354         git commit -m "add g" &&
 
 359         git commit -m "modify g and add h" &&
 
 360         git checkout second &&
 
 363         git commit -m second &&
 
 364         # Must fail due to conflict
 
 365         test_must_fail git cherry-pick -n master &&
 
 366         echo "editor not started" >.git/result &&
 
 368                 GIT_EDITOR="\"$(pwd)/.git/FAKE_EDITOR\"" &&
 
 370                 test_must_fail git commit
 
 372         test "$(cat .git/result)" = "editor not started"
 
 375 write_script .git/FAKE_EDITOR <<EOF
 
 376 # kill -TERM command added below.
 
 379 test_expect_success EXECKEEPSPID 'a SIGTERM should break locks' '
 
 381         ! "$SHELL_PATH" -c '\''
 
 382           echo kill -TERM $$ >> .git/FAKE_EDITOR
 
 383           GIT_EDITOR=.git/FAKE_EDITOR
 
 385           exec git commit -a'\'' &&
 
 386         test ! -f .git/index.lock
 
 389 rm -f .git/MERGE_MSG .git/COMMIT_EDITMSG
 
 392 test_expect_success 'Hand committing of a redundant merge removes dups' '
 
 394         git rev-parse second master >expect &&
 
 395         test_must_fail git merge second master &&
 
 396         git checkout master g &&
 
 397         EDITOR=: git commit -a &&
 
 398         git cat-file commit HEAD | sed -n -e "s/^parent //p" -e "/^$/q" >actual &&
 
 399         test_cmp expect actual
 
 403 test_expect_success 'A single-liner subject with a token plus colon is not a footer' '
 
 406         git commit -s -m "hello: kitty" --allow-empty &&
 
 407         git cat-file commit HEAD | sed -e "1,/^$/d" >actual &&
 
 408         test_line_count = 3 actual
 
 412 write_script .git/FAKE_EDITOR <<\EOF
 
 420 echo '## Custom template' >template
 
 424                 git config --unset-all "$1"
 
 435         GIT_EDITOR=.git/FAKE_EDITOR git commit -a $* $use_template &&
 
 436         case "$use_template" in
 
 438                 test_i18ngrep ! "^## Custom template" .git/COMMIT_EDITMSG ;;
 
 440                 test_i18ngrep "^## Custom template" .git/COMMIT_EDITMSG ;;
 
 444 try_commit_status_combo () {
 
 446         test_expect_success 'commit' '
 
 447                 clear_config commit.status &&
 
 449                 test_i18ngrep "^# Changes to be committed:" .git/COMMIT_EDITMSG
 
 452         test_expect_success 'commit' '
 
 453                 clear_config commit.status &&
 
 455                 test_i18ngrep "^# Changes to be committed:" .git/COMMIT_EDITMSG
 
 458         test_expect_success 'commit --status' '
 
 459                 clear_config commit.status &&
 
 460                 try_commit --status &&
 
 461                 test_i18ngrep "^# Changes to be committed:" .git/COMMIT_EDITMSG
 
 464         test_expect_success 'commit --no-status' '
 
 465                 clear_config commit.status &&
 
 466                 try_commit --no-status &&
 
 467                 test_i18ngrep ! "^# Changes to be committed:" .git/COMMIT_EDITMSG
 
 470         test_expect_success 'commit with commit.status = yes' '
 
 471                 clear_config commit.status &&
 
 472                 git config commit.status yes &&
 
 474                 test_i18ngrep "^# Changes to be committed:" .git/COMMIT_EDITMSG
 
 477         test_expect_success 'commit with commit.status = no' '
 
 478                 clear_config commit.status &&
 
 479                 git config commit.status no &&
 
 481                 test_i18ngrep ! "^# Changes to be committed:" .git/COMMIT_EDITMSG
 
 484         test_expect_success 'commit --status with commit.status = yes' '
 
 485                 clear_config commit.status &&
 
 486                 git config commit.status yes &&
 
 487                 try_commit --status &&
 
 488                 test_i18ngrep "^# Changes to be committed:" .git/COMMIT_EDITMSG
 
 491         test_expect_success 'commit --no-status with commit.status = yes' '
 
 492                 clear_config commit.status &&
 
 493                 git config commit.status yes &&
 
 494                 try_commit --no-status &&
 
 495                 test_i18ngrep ! "^# Changes to be committed:" .git/COMMIT_EDITMSG
 
 498         test_expect_success 'commit --status with commit.status = no' '
 
 499                 clear_config commit.status &&
 
 500                 git config commit.status no &&
 
 501                 try_commit --status &&
 
 502                 test_i18ngrep "^# Changes to be committed:" .git/COMMIT_EDITMSG
 
 505         test_expect_success 'commit --no-status with commit.status = no' '
 
 506                 clear_config commit.status &&
 
 507                 git config commit.status no &&
 
 508                 try_commit --no-status &&
 
 509                 test_i18ngrep ! "^# Changes to be committed:" .git/COMMIT_EDITMSG
 
 514 try_commit_status_combo
 
 516 use_template="-t template"
 
 518 try_commit_status_combo