3 # Copyright (c) 2006 Junio C Hamano
6 test_description='various format-patch tests'
9 . "$TEST_DIRECTORY"/lib-terminal.sh
11 test_expect_success setup '
13 for i in 1 2 3 4 5 6 7 8 9 10; do echo "$i"; done >file &&
17 git commit -m Initial &&
18 git checkout -b side &&
20 for i in 1 2 5 6 A B C 7 8 9 10; do echo "$i"; done >file &&
23 git commit -m "Side changes #1" &&
25 for i in D E F; do echo "$i"; done >>file &&
26 git update-index file &&
28 git commit -m "Side changes #2" &&
31 for i in 5 6 1 2 3 A 4 B C 7 8 9 10 D E F; do echo "$i"; done >file &&
32 git update-index file &&
34 git commit -m "Side changes #3 with \\n backslash-n in it." &&
36 git checkout master &&
37 git diff-tree -p C2 | git apply --index &&
39 git commit -m "Master accepts moral equivalent of #2"
43 test_expect_success "format-patch --ignore-if-in-upstream" '
45 git format-patch --stdout master..side >patch0 &&
46 cnt=$(grep "^From " patch0 | wc -l) &&
51 test_expect_success "format-patch --ignore-if-in-upstream" '
53 git format-patch --stdout \
54 --ignore-if-in-upstream master..side >patch1 &&
55 cnt=$(grep "^From " patch1 | wc -l) &&
60 test_expect_success "format-patch doesn't consider merge commits" '
62 git checkout -b slave master &&
63 echo "Another line" >>file &&
65 git commit -am "Slave change #1" &&
66 echo "Yet another line" >>file &&
68 git commit -am "Slave change #2" &&
69 git checkout -b merger master &&
71 git merge --no-ff slave &&
72 cnt=$(git format-patch -3 --stdout | grep "^From " | wc -l) &&
76 test_expect_success "format-patch result applies" '
78 git checkout -b rebuild-0 master &&
80 cnt=$(git rev-list master.. | wc -l) &&
84 test_expect_success "format-patch --ignore-if-in-upstream result applies" '
86 git checkout -b rebuild-1 master &&
88 cnt=$(git rev-list master.. | wc -l) &&
92 test_expect_success 'commit did not screw up the log message' '
94 git cat-file commit side | grep "^Side .* with .* backslash-n"
98 test_expect_success 'format-patch did not screw up the log message' '
100 grep "^Subject: .*Side changes #3 with .* backslash-n" patch0 &&
101 grep "^Subject: .*Side changes #3 with .* backslash-n" patch1
105 test_expect_success 'replay did not screw up the log message' '
107 git cat-file commit rebuild-1 | grep "^Side .* with .* backslash-n"
111 test_expect_success 'extra headers' '
113 git config format.headers "To: R E Cipient <rcipient@example.com>
115 git config --add format.headers "Cc: S E Cipient <scipient@example.com>
117 git format-patch --stdout master..side > patch2 &&
118 sed -e "/^\$/q" patch2 > hdrs2 &&
119 grep "^To: R E Cipient <rcipient@example.com>\$" hdrs2 &&
120 grep "^Cc: S E Cipient <scipient@example.com>\$" hdrs2
124 test_expect_success 'extra headers without newlines' '
126 git config --replace-all format.headers "To: R E Cipient <rcipient@example.com>" &&
127 git config --add format.headers "Cc: S E Cipient <scipient@example.com>" &&
128 git format-patch --stdout master..side >patch3 &&
129 sed -e "/^\$/q" patch3 > hdrs3 &&
130 grep "^To: R E Cipient <rcipient@example.com>\$" hdrs3 &&
131 grep "^Cc: S E Cipient <scipient@example.com>\$" hdrs3
135 test_expect_success 'extra headers with multiple To:s' '
137 git config --replace-all format.headers "To: R E Cipient <rcipient@example.com>" &&
138 git config --add format.headers "To: S E Cipient <scipient@example.com>" &&
139 git format-patch --stdout master..side > patch4 &&
140 sed -e "/^\$/q" patch4 > hdrs4 &&
141 grep "^To: R E Cipient <rcipient@example.com>,\$" hdrs4 &&
142 grep "^ *S E Cipient <scipient@example.com>\$" hdrs4
145 test_expect_success 'additional command line cc (ascii)' '
147 git config --replace-all format.headers "Cc: R E Cipient <rcipient@example.com>" &&
148 git format-patch --cc="S E Cipient <scipient@example.com>" --stdout master..side | sed -e "/^\$/q" >patch5 &&
149 grep "^Cc: R E Cipient <rcipient@example.com>,\$" patch5 &&
150 grep "^ *S E Cipient <scipient@example.com>\$" patch5
153 test_expect_failure 'additional command line cc (rfc822)' '
155 git config --replace-all format.headers "Cc: R E Cipient <rcipient@example.com>" &&
156 git format-patch --cc="S. E. Cipient <scipient@example.com>" --stdout master..side | sed -e "/^\$/q" >patch5 &&
157 grep "^Cc: R E Cipient <rcipient@example.com>,\$" patch5 &&
158 grep "^ *\"S. E. Cipient\" <scipient@example.com>\$" patch5
161 test_expect_success 'command line headers' '
163 git config --unset-all format.headers &&
164 git format-patch --add-header="Cc: R E Cipient <rcipient@example.com>" --stdout master..side | sed -e "/^\$/q" >patch6 &&
165 grep "^Cc: R E Cipient <rcipient@example.com>\$" patch6
168 test_expect_success 'configuration headers and command line headers' '
170 git config --replace-all format.headers "Cc: R E Cipient <rcipient@example.com>" &&
171 git format-patch --add-header="Cc: S E Cipient <scipient@example.com>" --stdout master..side | sed -e "/^\$/q" >patch7 &&
172 grep "^Cc: R E Cipient <rcipient@example.com>,\$" patch7 &&
173 grep "^ *S E Cipient <scipient@example.com>\$" patch7
176 test_expect_success 'command line To: header (ascii)' '
178 git config --unset-all format.headers &&
179 git format-patch --to="R E Cipient <rcipient@example.com>" --stdout master..side | sed -e "/^\$/q" >patch8 &&
180 grep "^To: R E Cipient <rcipient@example.com>\$" patch8
183 test_expect_failure 'command line To: header (rfc822)' '
185 git format-patch --to="R. E. Cipient <rcipient@example.com>" --stdout master..side | sed -e "/^\$/q" >patch8 &&
186 grep "^To: \"R. E. Cipient\" <rcipient@example.com>\$" patch8
189 test_expect_failure 'command line To: header (rfc2047)' '
191 git format-patch --to="R Ä Cipient <rcipient@example.com>" --stdout master..side | sed -e "/^\$/q" >patch8 &&
192 grep "^To: =?UTF-8?q?R=20=C3=84=20Cipient?= <rcipient@example.com>\$" patch8
195 test_expect_success 'configuration To: header (ascii)' '
197 git config format.to "R E Cipient <rcipient@example.com>" &&
198 git format-patch --stdout master..side | sed -e "/^\$/q" >patch9 &&
199 grep "^To: R E Cipient <rcipient@example.com>\$" patch9
202 test_expect_failure 'configuration To: header (rfc822)' '
204 git config format.to "R. E. Cipient <rcipient@example.com>" &&
205 git format-patch --stdout master..side | sed -e "/^\$/q" >patch9 &&
206 grep "^To: \"R. E. Cipient\" <rcipient@example.com>\$" patch9
209 test_expect_failure 'configuration To: header (rfc2047)' '
211 git config format.to "R Ä Cipient <rcipient@example.com>" &&
212 git format-patch --stdout master..side | sed -e "/^\$/q" >patch9 &&
213 grep "^To: =?UTF-8?q?R=20=C3=84=20Cipient?= <rcipient@example.com>\$" patch9
216 # check_patch <patch>: Verify that <patch> looks like a half-sane
217 # patch email to avoid a false positive with !grep
219 grep -e "^From:" "$1" &&
220 grep -e "^Date:" "$1" &&
221 grep -e "^Subject:" "$1"
224 test_expect_success '--no-to overrides config.to' '
226 git config --replace-all format.to \
227 "R E Cipient <rcipient@example.com>" &&
228 git format-patch --no-to --stdout master..side |
229 sed -e "/^\$/q" >patch10 &&
230 check_patch patch10 &&
231 ! grep "^To: R E Cipient <rcipient@example.com>\$" patch10
234 test_expect_success '--no-to and --to replaces config.to' '
236 git config --replace-all format.to \
237 "Someone <someone@out.there>" &&
238 git format-patch --no-to --to="Someone Else <else@out.there>" \
239 --stdout master..side |
240 sed -e "/^\$/q" >patch11 &&
241 check_patch patch11 &&
242 ! grep "^To: Someone <someone@out.there>\$" patch11 &&
243 grep "^To: Someone Else <else@out.there>\$" patch11
246 test_expect_success '--no-cc overrides config.cc' '
248 git config --replace-all format.cc \
249 "C E Cipient <rcipient@example.com>" &&
250 git format-patch --no-cc --stdout master..side |
251 sed -e "/^\$/q" >patch12 &&
252 check_patch patch12 &&
253 ! grep "^Cc: C E Cipient <rcipient@example.com>\$" patch12
256 test_expect_success '--no-add-header overrides config.headers' '
258 git config --replace-all format.headers \
259 "Header1: B E Cipient <rcipient@example.com>" &&
260 git format-patch --no-add-header --stdout master..side |
261 sed -e "/^\$/q" >patch13 &&
262 check_patch patch13 &&
263 ! grep "^Header1: B E Cipient <rcipient@example.com>\$" patch13
266 test_expect_success 'multiple files' '
270 git format-patch -o patches/ master &&
271 ls patches/0001-Side-changes-1.patch patches/0002-Side-changes-2.patch patches/0003-Side-changes-3-with-n-backslash-n-in-it.patch
274 test_expect_success 'reroll count' '
276 git format-patch -o patches --cover-letter --reroll-count 4 master..side >list &&
277 ! grep -v "^patches/v4-000[0-3]-" list &&
278 sed -n -e "/^Subject: /p" $(cat list) >subjects &&
279 ! grep -v "^Subject: \[PATCH v4 [0-3]/3\] " subjects
282 test_expect_success 'reroll count (-v)' '
284 git format-patch -o patches --cover-letter -v4 master..side >list &&
285 ! grep -v "^patches/v4-000[0-3]-" list &&
286 sed -n -e "/^Subject: /p" $(cat list) >subjects &&
287 ! grep -v "^Subject: \[PATCH v4 [0-3]/3\] " subjects
293 (git format-patch --stdout "$@"; echo $? > status.out) |
294 # Prints everything between the Message-ID and In-Reply-To,
295 # and replaces all Message-ID-lookalikes by a sequence number
297 if (/^(message-id|references|in-reply-to)/i) {
303 $h{$1}=$i++ if (/<([^>]+)>/ and !exists $h{$1});
304 for $k (keys %h) {s/$k/$h{$k}/};
307 print "---\n" if /^From /i;
309 test 0 = "$(cat status.out)" &&
310 test_cmp "$expect" actual
313 cat >> expect.no-threading <<EOF
319 test_expect_success 'no threading' '
321 check_threading expect.no-threading master
324 cat > expect.thread <<EOF
337 test_expect_success 'thread' '
338 check_threading expect.thread --thread master
341 cat > expect.in-reply-to <<EOF
356 test_expect_success 'thread in-reply-to' '
357 check_threading expect.in-reply-to --in-reply-to="<test.message>" \
361 cat > expect.cover-letter <<EOF
378 test_expect_success 'thread cover-letter' '
379 check_threading expect.cover-letter --cover-letter --thread master
382 cat > expect.cl-irt <<EOF
404 test_expect_success 'thread cover-letter in-reply-to' '
405 check_threading expect.cl-irt --cover-letter \
406 --in-reply-to="<test.message>" --thread master
409 test_expect_success 'thread explicit shallow' '
410 check_threading expect.cl-irt --cover-letter \
411 --in-reply-to="<test.message>" --thread=shallow master
414 cat > expect.deep <<EOF
428 test_expect_success 'thread deep' '
429 check_threading expect.deep --thread=deep master
432 cat > expect.deep-irt <<EOF
450 test_expect_success 'thread deep in-reply-to' '
451 check_threading expect.deep-irt --thread=deep \
452 --in-reply-to="<test.message>" master
455 cat > expect.deep-cl <<EOF
475 test_expect_success 'thread deep cover-letter' '
476 check_threading expect.deep-cl --cover-letter --thread=deep master
479 cat > expect.deep-cl-irt <<EOF
504 test_expect_success 'thread deep cover-letter in-reply-to' '
505 check_threading expect.deep-cl-irt --cover-letter \
506 --in-reply-to="<test.message>" --thread=deep master
509 test_expect_success 'thread via config' '
510 test_config format.thread true &&
511 check_threading expect.thread master
514 test_expect_success 'thread deep via config' '
515 test_config format.thread deep &&
516 check_threading expect.deep master
519 test_expect_success 'thread config + override' '
520 test_config format.thread deep &&
521 check_threading expect.thread --thread master
524 test_expect_success 'thread config + --no-thread' '
525 test_config format.thread deep &&
526 check_threading expect.no-threading --no-thread master
529 test_expect_success 'excessive subject' '
533 for i in 5 6 1 2 3 A 4 B C 7 8 9 10 D E F; do echo "$i"; done >>file &&
534 git update-index file &&
535 git commit -m "This is an excessively long subject line for a message due to the habit some projects have of not having a short, one-line subject at the start of the commit message, but rather sticking a whole paragraph right at the start as the only thing in the commit message. It had better not become the filename for the patch." &&
536 git format-patch -o patches/ master..side &&
537 ls patches/0004-This-is-an-excessively-long-subject-line-for-a-messa.patch
540 test_expect_success 'cover-letter inherits diff options' '
544 git format-patch --cover-letter -1 &&
545 check_patch 0000-cover-letter.patch &&
546 ! grep "file => foo .* 0 *\$" 0000-cover-letter.patch &&
547 git format-patch --cover-letter -1 -M &&
548 grep "file => foo .* 0 *\$" 0000-cover-letter.patch
553 This is an excessively long subject line for a message due to the
554 habit some projects have of not having a short, one-line subject at
555 the start of the commit message, but rather sticking a whole
556 paragraph right at the start as the only thing in the commit
557 message. It had better not become the filename for the patch.
562 test_expect_success 'shortlog of cover-letter wraps overly-long onelines' '
564 git format-patch --cover-letter -2 &&
565 sed -e "1,/A U Thor/d" -e "/^\$/q" < 0000-cover-letter.patch > output &&
566 test_cmp expect output
571 index 40f36c6..2dc5c23 100644
582 test_expect_success 'format-patch respects -U' '
584 git format-patch -U4 -2 &&
585 sed -e "1,/^diff/d" -e "/^+5/q" \
586 <0001-This-is-an-excessively-long-subject-line-for-a-messa.patch \
588 test_cmp expect output
594 diff --git a/file b/file
595 index 40f36c6..2dc5c23 100644
605 test_expect_success 'format-patch -p suppresses stat' '
607 git format-patch -p -2 &&
608 sed -e "1,/^\$/d" -e "/^+5/q" < 0001-This-is-an-excessively-long-subject-line-for-a-messa.patch > output &&
609 test_cmp expect output
613 test_expect_success 'format-patch from a subdirectory (1)' '
624 echo "Oops? $filename"
631 test_expect_success 'format-patch from a subdirectory (2)' '
636 git format-patch -1 -o ..
642 echo "Oops? $filename"
646 basename=$(expr "$filename" : ".*/\(.*\)") &&
647 test -f "sub/$basename"
650 test_expect_success 'format-patch from a subdirectory (3)' '
656 git format-patch -1 -o "$TRASH_DIRECTORY"
658 basename=$(expr "$filename" : ".*/\(.*\)") &&
662 test_expect_success 'format-patch --in-reply-to' '
663 git format-patch -1 --stdout --in-reply-to "baz@foo.bar" > patch8 &&
664 grep "^In-Reply-To: <baz@foo.bar>" patch8 &&
665 grep "^References: <baz@foo.bar>" patch8
668 test_expect_success 'format-patch --signoff' '
669 git format-patch -1 --signoff --stdout >out &&
670 grep "^Signed-off-by: $GIT_COMMITTER_NAME <$GIT_COMMITTER_EMAIL>" out
673 test_expect_success 'format-patch --notes --signoff' '
674 git notes --ref test add -m "test message" HEAD &&
675 git format-patch -1 --signoff --stdout --notes=test >out &&
676 # Three dashes must come after S-o-b
677 ! sed "/^Signed-off-by: /q" out | grep "test message" &&
678 sed "1,/^Signed-off-by: /d" out | grep "test message" &&
679 # Notes message must come after three dashes
680 ! sed "/^---$/q" out | grep "test message" &&
681 sed "1,/^---$/d" out | grep "test message"
684 echo "fatal: --name-only does not make sense" > expect.name-only
685 echo "fatal: --name-status does not make sense" > expect.name-status
686 echo "fatal: --check does not make sense" > expect.check
688 test_expect_success 'options no longer allowed for format-patch' '
689 test_must_fail git format-patch --name-only 2> output &&
690 test_i18ncmp expect.name-only output &&
691 test_must_fail git format-patch --name-status 2> output &&
692 test_i18ncmp expect.name-status output &&
693 test_must_fail git format-patch --check 2> output &&
694 test_i18ncmp expect.check output'
696 test_expect_success 'format-patch --numstat should produce a patch' '
697 git format-patch --numstat --stdout master..side > output &&
698 test 6 = $(grep "^diff --git a/" output | wc -l)'
700 test_expect_success 'format-patch -- <path>' '
701 git format-patch master..side -- file 2>error &&
702 ! grep "Use .--" error
705 test_expect_success 'format-patch --ignore-if-in-upstream HEAD' '
706 git format-patch --ignore-if-in-upstream HEAD
709 test_expect_success 'format-patch --signature' '
710 git format-patch --stdout --signature="my sig" -1 >output &&
714 test_expect_success 'format-patch with format.signature config' '
715 git config format.signature "config sig" &&
716 git format-patch --stdout -1 >output &&
717 grep "config sig" output
720 test_expect_success 'format-patch --signature overrides format.signature' '
721 git config format.signature "config sig" &&
722 git format-patch --stdout --signature="overrides" -1 >output &&
723 ! grep "config sig" output &&
724 grep "overrides" output
727 test_expect_success 'format-patch --no-signature ignores format.signature' '
728 git config format.signature "config sig" &&
729 git format-patch --stdout --signature="my sig" --no-signature \
731 check_patch output &&
732 ! grep "config sig" output &&
733 ! grep "my sig" output &&
734 ! grep "^-- \$" output
737 test_expect_success 'format-patch --signature --cover-letter' '
738 git config --unset-all format.signature &&
739 git format-patch --stdout --signature="my sig" --cover-letter \
741 grep "my sig" output &&
742 test 2 = $(grep "my sig" output | wc -l)
745 test_expect_success 'format.signature="" suppresses signatures' '
746 git config format.signature "" &&
747 git format-patch --stdout -1 >output &&
748 check_patch output &&
749 ! grep "^-- \$" output
752 test_expect_success 'format-patch --no-signature suppresses signatures' '
753 git config --unset-all format.signature &&
754 git format-patch --stdout --no-signature -1 >output &&
755 check_patch output &&
756 ! grep "^-- \$" output
759 test_expect_success 'format-patch --signature="" suppresses signatures' '
760 git format-patch --stdout --signature="" -1 >output &&
761 check_patch output &&
762 ! grep "^-- \$" output
765 test_expect_success 'prepare mail-signature input' '
766 cat >mail-signature <<-\EOF
768 Test User <test.email@kernel.org>
769 http://git.kernel.org/cgit/git/git.git
771 git.kernel.org/?p=git/git.git;a=summary
776 test_expect_success '--signature-file=file works' '
777 git format-patch --stdout --signature-file=mail-signature -1 >output &&
778 check_patch output &&
779 sed -e "1,/^-- \$/d" <output >actual &&
781 cat mail-signature && echo
783 test_cmp expect actual
786 test_expect_success 'format.signaturefile works' '
787 test_config format.signaturefile mail-signature &&
788 git format-patch --stdout -1 >output &&
789 check_patch output &&
790 sed -e "1,/^-- \$/d" <output >actual &&
792 cat mail-signature && echo
794 test_cmp expect actual
797 test_expect_success '--no-signature suppresses format.signaturefile ' '
798 test_config format.signaturefile mail-signature &&
799 git format-patch --stdout --no-signature -1 >output &&
800 check_patch output &&
801 ! grep "^-- \$" output
804 test_expect_success '--signature-file overrides format.signaturefile' '
805 cat >other-mail-signature <<-\EOF
806 Use this other signature instead of mail-signature.
808 test_config format.signaturefile mail-signature &&
809 git format-patch --stdout \
810 --signature-file=other-mail-signature -1 >output &&
811 check_patch output &&
812 sed -e "1,/^-- \$/d" <output >actual &&
814 cat other-mail-signature && echo
816 test_cmp expect actual
819 test_expect_success '--signature overrides format.signaturefile' '
820 test_config format.signaturefile mail-signature &&
821 git format-patch --stdout --signature="my sig" -1 >output &&
822 check_patch output &&
826 test_expect_success TTY 'format-patch --stdout paginates' '
828 test_terminal env GIT_PAGER="wc >pager_used" git format-patch --stdout --all &&
829 test_path_is_file pager_used
832 test_expect_success TTY 'format-patch --stdout pagination can be disabled' '
834 test_terminal env GIT_PAGER="wc >pager_used" git --no-pager format-patch --stdout --all &&
835 test_terminal env GIT_PAGER="wc >pager_used" git -c "pager.format-patch=false" format-patch --stdout --all &&
836 test_path_is_missing pager_used &&
837 test_path_is_missing .git/pager_used
840 test_expect_success 'format-patch handles multi-line subjects' '
842 echo content >>file &&
843 for i in one two three; do echo $i; done >msg &&
846 git format-patch -o patches -1 &&
847 grep ^Subject: patches/0001-one.patch >actual &&
848 echo "Subject: [PATCH] one two three" >expect &&
849 test_cmp expect actual
852 test_expect_success 'format-patch handles multi-line encoded subjects' '
854 echo content >>file &&
855 for i in en två tre; do echo $i; done >msg &&
858 git format-patch -o patches -1 &&
859 grep ^Subject: patches/0001-en.patch >actual &&
860 echo "Subject: [PATCH] =?UTF-8?q?en=20tv=C3=A5=20tre?=" >expect &&
861 test_cmp expect actual
865 M64=$M8$M8$M8$M8$M8$M8$M8$M8
866 M512=$M64$M64$M64$M64$M64$M64$M64$M64
868 Subject: [PATCH] foo bar foo bar foo bar foo bar foo bar foo bar foo bar foo
869 bar foo bar foo bar foo bar foo bar foo bar foo bar foo bar foo bar foo bar
870 foo bar foo bar foo bar foo bar foo bar foo bar foo bar foo bar foo bar foo
871 bar foo bar foo bar foo bar foo bar foo bar foo bar foo bar foo bar foo bar
872 foo bar foo bar foo bar foo bar foo bar foo bar foo bar foo bar foo bar foo
873 bar foo bar foo bar foo bar foo bar foo bar foo bar foo bar foo bar foo bar
874 foo bar foo bar foo bar foo bar foo bar foo bar foo bar foo bar foo bar
876 test_expect_success 'format-patch wraps extremely long subject (ascii)' '
877 echo content >>file &&
879 git commit -m "$M512" &&
880 git format-patch --stdout -1 >patch &&
881 sed -n "/^Subject/p; /^ /p; /^$/q" <patch >subject &&
882 test_cmp expect subject
886 M64=$M8$M8$M8$M8$M8$M8$M8$M8
887 M512=$M64$M64$M64$M64$M64$M64$M64$M64
889 Subject: [PATCH] =?UTF-8?q?f=C3=B6=C3=B6=20bar=20f=C3=B6=C3=B6=20bar=20f?=
890 =?UTF-8?q?=C3=B6=C3=B6=20bar=20f=C3=B6=C3=B6=20bar=20f=C3=B6=C3=B6=20bar?=
891 =?UTF-8?q?=20f=C3=B6=C3=B6=20bar=20f=C3=B6=C3=B6=20bar=20f=C3=B6=C3=B6=20?=
892 =?UTF-8?q?bar=20f=C3=B6=C3=B6=20bar=20f=C3=B6=C3=B6=20bar=20f=C3=B6=C3=B6?=
893 =?UTF-8?q?=20bar=20f=C3=B6=C3=B6=20bar=20f=C3=B6=C3=B6=20bar=20f=C3=B6?=
894 =?UTF-8?q?=C3=B6=20bar=20f=C3=B6=C3=B6=20bar=20f=C3=B6=C3=B6=20bar=20f?=
895 =?UTF-8?q?=C3=B6=C3=B6=20bar=20f=C3=B6=C3=B6=20bar=20f=C3=B6=C3=B6=20bar?=
896 =?UTF-8?q?=20f=C3=B6=C3=B6=20bar=20f=C3=B6=C3=B6=20bar=20f=C3=B6=C3=B6=20?=
897 =?UTF-8?q?bar=20f=C3=B6=C3=B6=20bar=20f=C3=B6=C3=B6=20bar=20f=C3=B6=C3=B6?=
898 =?UTF-8?q?=20bar=20f=C3=B6=C3=B6=20bar=20f=C3=B6=C3=B6=20bar=20f=C3=B6?=
899 =?UTF-8?q?=C3=B6=20bar=20f=C3=B6=C3=B6=20bar=20f=C3=B6=C3=B6=20bar=20f?=
900 =?UTF-8?q?=C3=B6=C3=B6=20bar=20f=C3=B6=C3=B6=20bar=20f=C3=B6=C3=B6=20bar?=
901 =?UTF-8?q?=20f=C3=B6=C3=B6=20bar=20f=C3=B6=C3=B6=20bar=20f=C3=B6=C3=B6=20?=
902 =?UTF-8?q?bar=20f=C3=B6=C3=B6=20bar=20f=C3=B6=C3=B6=20bar=20f=C3=B6=C3=B6?=
903 =?UTF-8?q?=20bar=20f=C3=B6=C3=B6=20bar=20f=C3=B6=C3=B6=20bar=20f=C3=B6?=
904 =?UTF-8?q?=C3=B6=20bar=20f=C3=B6=C3=B6=20bar=20f=C3=B6=C3=B6=20bar=20f?=
905 =?UTF-8?q?=C3=B6=C3=B6=20bar=20f=C3=B6=C3=B6=20bar=20f=C3=B6=C3=B6=20bar?=
906 =?UTF-8?q?=20f=C3=B6=C3=B6=20bar=20f=C3=B6=C3=B6=20bar=20f=C3=B6=C3=B6=20?=
907 =?UTF-8?q?bar=20f=C3=B6=C3=B6=20bar=20f=C3=B6=C3=B6=20bar=20f=C3=B6=C3=B6?=
908 =?UTF-8?q?=20bar=20f=C3=B6=C3=B6=20bar=20f=C3=B6=C3=B6=20bar=20f=C3=B6?=
909 =?UTF-8?q?=C3=B6=20bar=20f=C3=B6=C3=B6=20bar=20f=C3=B6=C3=B6=20bar=20f?=
910 =?UTF-8?q?=C3=B6=C3=B6=20bar=20f=C3=B6=C3=B6=20bar=20f=C3=B6=C3=B6=20bar?=
911 =?UTF-8?q?=20f=C3=B6=C3=B6=20bar=20f=C3=B6=C3=B6=20bar=20f=C3=B6=C3=B6=20?=
914 test_expect_success 'format-patch wraps extremely long subject (rfc2047)' '
916 echo content >>file &&
918 git commit -m "$M512" &&
919 git format-patch --stdout -1 >patch &&
920 sed -n "/^Subject/p; /^ /p; /^$/q" <patch >subject &&
921 test_cmp expect subject
925 echo content >>file &&
927 GIT_AUTHOR_NAME=$1 git commit -m author-check &&
928 git format-patch --stdout -1 >patch &&
929 sed -n "/^From: /p; /^ /p; /^$/q" <patch >actual &&
930 test_cmp expect actual
934 From: "Foo B. Bar" <author@example.com>
936 test_expect_success 'format-patch quotes dot in from-headers' '
937 check_author "Foo B. Bar"
941 From: "Foo \"The Baz\" Bar" <author@example.com>
943 test_expect_success 'format-patch quotes double-quote in from-headers' '
944 check_author "Foo \"The Baz\" Bar"
948 From: =?UTF-8?q?F=C3=B6o=20Bar?= <author@example.com>
950 test_expect_success 'format-patch uses rfc2047-encoded from-headers when necessary' '
951 check_author "Föo Bar"
955 From: =?UTF-8?q?F=C3=B6o=20B=2E=20Bar?= <author@example.com>
957 test_expect_success 'rfc2047-encoded from-headers leave no rfc822 specials' '
958 check_author "Föo B. Bar"
962 From: foo_bar_foo_bar_foo_bar_foo_bar_foo_bar_foo_bar_foo_bar_foo_bar_
965 test_expect_success 'format-patch wraps moderately long from-header (ascii)' '
966 check_author "foo_bar_foo_bar_foo_bar_foo_bar_foo_bar_foo_bar_foo_bar_foo_bar_"
970 From: Foo Bar Foo Bar Foo Bar Foo Bar Foo Bar Foo Bar Foo Bar Foo Bar Foo Bar
971 Foo Bar Foo Bar Foo Bar Foo Bar Foo Bar Foo Bar Foo Bar Foo Bar Foo Bar Foo
972 Bar Foo Bar Foo Bar Foo Bar <author@example.com>
974 test_expect_success 'format-patch wraps extremely long from-header (ascii)' '
975 check_author "Foo Bar Foo Bar Foo Bar Foo Bar Foo Bar Foo Bar Foo Bar Foo Bar Foo Bar Foo Bar Foo Bar Foo Bar Foo Bar Foo Bar Foo Bar Foo Bar Foo Bar Foo Bar Foo Bar Foo Bar Foo Bar Foo Bar"
979 From: "Foo.Bar Foo Bar Foo Bar Foo Bar Foo Bar Foo Bar Foo Bar Foo Bar Foo Bar
980 Foo Bar Foo Bar Foo Bar Foo Bar Foo Bar Foo Bar Foo Bar Foo Bar Foo Bar Foo
981 Bar Foo Bar Foo Bar Foo Bar" <author@example.com>
983 test_expect_success 'format-patch wraps extremely long from-header (rfc822)' '
984 check_author "Foo.Bar Foo Bar Foo Bar Foo Bar Foo Bar Foo Bar Foo Bar Foo Bar Foo Bar Foo Bar Foo Bar Foo Bar Foo Bar Foo Bar Foo Bar Foo Bar Foo Bar Foo Bar Foo Bar Foo Bar Foo Bar Foo Bar"
988 From: =?UTF-8?q?Fo=C3=B6=20Bar=20Foo=20Bar=20Foo=20Bar=20Foo=20Bar=20Foo?=
989 =?UTF-8?q?=20Bar=20Foo=20Bar=20Foo=20Bar=20Foo=20Bar=20Foo=20Bar=20Foo=20?=
990 =?UTF-8?q?Bar=20Foo=20Bar=20Foo=20Bar=20Foo=20Bar=20Foo=20Bar=20Foo=20Bar?=
991 =?UTF-8?q?=20Foo=20Bar=20Foo=20Bar=20Foo=20Bar=20Foo=20Bar=20Foo=20Bar=20?=
992 =?UTF-8?q?Foo=20Bar=20Foo=20Bar?= <author@example.com>
994 test_expect_success 'format-patch wraps extremely long from-header (rfc2047)' '
995 check_author "Foö Bar Foo Bar Foo Bar Foo Bar Foo Bar Foo Bar Foo Bar Foo Bar Foo Bar Foo Bar Foo Bar Foo Bar Foo Bar Foo Bar Foo Bar Foo Bar Foo Bar Foo Bar Foo Bar Foo Bar Foo Bar Foo Bar"
999 Subject: header with . in it
1001 test_expect_success 'subject lines do not have 822 atom-quoting' '
1002 echo content >>file &&
1004 git commit -m "header with . in it" &&
1005 git format-patch -k -1 --stdout >patch &&
1006 grep ^Subject: patch >actual &&
1007 test_cmp expect actual
1011 Subject: [PREFIX 1/1] header with . in it
1013 test_expect_success 'subject prefixes have space prepended' '
1014 git format-patch -n -1 --stdout --subject-prefix=PREFIX >patch &&
1015 grep ^Subject: patch >actual &&
1016 test_cmp expect actual
1020 Subject: [1/1] header with . in it
1022 test_expect_success 'empty subject prefix does not have extra space' '
1023 git format-patch -n -1 --stdout --subject-prefix= >patch &&
1024 grep ^Subject: patch >actual &&
1025 test_cmp expect actual
1028 test_expect_success '--from=ident notices bogus ident' '
1029 test_must_fail git format-patch -1 --stdout --from=foo >patch
1032 test_expect_success '--from=ident replaces author' '
1033 git format-patch -1 --stdout --from="Me <me@example.com>" >patch &&
1034 cat >expect <<-\EOF &&
1035 From: Me <me@example.com>
1037 From: A U Thor <author@example.com>
1040 sed -ne "/^From:/p; /^$/p; /^---$/q" <patch >patch.head &&
1041 test_cmp expect patch.head
1044 test_expect_success '--from uses committer ident' '
1045 git format-patch -1 --stdout --from >patch &&
1046 cat >expect <<-\EOF &&
1047 From: C O Mitter <committer@example.com>
1049 From: A U Thor <author@example.com>
1052 sed -ne "/^From:/p; /^$/p; /^---$/q" <patch >patch.head &&
1053 test_cmp expect patch.head
1056 test_expect_success '--from omits redundant in-body header' '
1057 git format-patch -1 --stdout --from="A U Thor <author@example.com>" >patch &&
1058 cat >expect <<-\EOF &&
1059 From: A U Thor <author@example.com>
1062 sed -ne "/^From:/p; /^$/p; /^---$/q" <patch >patch.head &&
1063 test_cmp expect patch.head
1066 test_expect_success 'in-body headers trigger content encoding' '
1067 GIT_AUTHOR_NAME="éxötìc" test_commit exotic &&
1068 test_when_finished "git reset --hard HEAD^" &&
1069 git format-patch -1 --stdout --from >patch &&
1070 cat >expect <<-\EOF &&
1071 From: C O Mitter <committer@example.com>
1072 Content-Type: text/plain; charset=UTF-8
1074 From: éxötìc <author@example.com>
1077 sed -ne "/^From:/p; /^$/p; /^Content-Type/p; /^---$/q" <patch >patch.head &&
1078 test_cmp expect patch.head
1083 C=$(git commit-tree HEAD^^{tree} -p HEAD) &&
1084 git format-patch --stdout --signoff $C^..$C >append_signoff.patch &&
1085 sed -n -e "1,/^---$/p" append_signoff.patch |
1086 egrep -n "^Subject|Sign|^$"
1089 test_expect_success 'signoff: commit with no body' '
1090 append_signoff </dev/null >actual &&
1091 cat <<\EOF | sed "s/EOL$//" >expected &&
1092 4:Subject: [PATCH] EOL
1094 9:Signed-off-by: C O Mitter <committer@example.com>
1096 test_cmp expected actual
1099 test_expect_success 'signoff: commit with only subject' '
1100 echo subject | append_signoff >actual &&
1101 cat >expected <<\EOF &&
1102 4:Subject: [PATCH] subject
1104 9:Signed-off-by: C O Mitter <committer@example.com>
1106 test_cmp expected actual
1109 test_expect_success 'signoff: commit with only subject that does not end with NL' '
1110 printf subject | append_signoff >actual &&
1111 cat >expected <<\EOF &&
1112 4:Subject: [PATCH] subject
1114 9:Signed-off-by: C O Mitter <committer@example.com>
1116 test_cmp expected actual
1119 test_expect_success 'signoff: no existing signoffs' '
1120 append_signoff <<\EOF >actual &&
1125 cat >expected <<\EOF &&
1126 4:Subject: [PATCH] subject
1129 11:Signed-off-by: C O Mitter <committer@example.com>
1131 test_cmp expected actual
1134 test_expect_success 'signoff: no existing signoffs and no trailing NL' '
1135 printf "subject\n\nbody" | append_signoff >actual &&
1136 cat >expected <<\EOF &&
1137 4:Subject: [PATCH] subject
1140 11:Signed-off-by: C O Mitter <committer@example.com>
1142 test_cmp expected actual
1145 test_expect_success 'signoff: some random signoff' '
1146 append_signoff <<\EOF >actual &&
1151 Signed-off-by: my@house
1153 cat >expected <<\EOF &&
1154 4:Subject: [PATCH] subject
1157 11:Signed-off-by: my@house
1158 12:Signed-off-by: C O Mitter <committer@example.com>
1160 test_cmp expected actual
1163 test_expect_success 'signoff: misc conforming footer elements' '
1164 append_signoff <<\EOF >actual &&
1169 Signed-off-by: my@house
1170 (cherry picked from commit da39a3ee5e6b4b0d3255bfef95601890afd80709)
1171 Tested-by: Some One <someone@example.com>
1174 cat >expected <<\EOF &&
1175 4:Subject: [PATCH] subject
1178 11:Signed-off-by: my@house
1179 15:Signed-off-by: C O Mitter <committer@example.com>
1181 test_cmp expected actual
1184 test_expect_success 'signoff: some random signoff-alike' '
1185 append_signoff <<\EOF >actual &&
1189 Fooled-by-me: my@house
1191 cat >expected <<\EOF &&
1192 4:Subject: [PATCH] subject
1195 12:Signed-off-by: C O Mitter <committer@example.com>
1197 test_cmp expected actual
1200 test_expect_success 'signoff: not really a signoff' '
1201 append_signoff <<\EOF >actual &&
1204 I want to mention about Signed-off-by: here.
1206 cat >expected <<\EOF &&
1207 4:Subject: [PATCH] subject
1209 9:I want to mention about Signed-off-by: here.
1211 11:Signed-off-by: C O Mitter <committer@example.com>
1213 test_cmp expected actual
1216 test_expect_success 'signoff: not really a signoff (2)' '
1217 append_signoff <<\EOF >actual &&
1221 Signed-off-by: example happens to be wrapped here.
1223 cat >expected <<\EOF &&
1224 4:Subject: [PATCH] subject
1226 10:Signed-off-by: example happens to be wrapped here.
1228 12:Signed-off-by: C O Mitter <committer@example.com>
1230 test_cmp expected actual
1233 test_expect_success 'signoff: valid S-o-b paragraph in the middle' '
1234 append_signoff <<\EOF >actual &&
1237 Signed-off-by: my@house
1238 Signed-off-by: your@house
1242 cat >expected <<\EOF &&
1243 4:Subject: [PATCH] subject
1245 9:Signed-off-by: my@house
1246 10:Signed-off-by: your@house
1249 14:Signed-off-by: C O Mitter <committer@example.com>
1251 test_cmp expected actual
1254 test_expect_success 'signoff: the same signoff at the end' '
1255 append_signoff <<\EOF >actual &&
1260 Signed-off-by: C O Mitter <committer@example.com>
1262 cat >expected <<\EOF &&
1263 4:Subject: [PATCH] subject
1266 11:Signed-off-by: C O Mitter <committer@example.com>
1268 test_cmp expected actual
1271 test_expect_success 'signoff: the same signoff at the end, no trailing NL' '
1272 printf "subject\n\nSigned-off-by: C O Mitter <committer@example.com>" |
1273 append_signoff >actual &&
1274 cat >expected <<\EOF &&
1275 4:Subject: [PATCH] subject
1277 9:Signed-off-by: C O Mitter <committer@example.com>
1279 test_cmp expected actual
1282 test_expect_success 'signoff: the same signoff NOT at the end' '
1283 append_signoff <<\EOF >actual &&
1288 Signed-off-by: C O Mitter <committer@example.com>
1289 Signed-off-by: my@house
1291 cat >expected <<\EOF &&
1292 4:Subject: [PATCH] subject
1295 11:Signed-off-by: C O Mitter <committer@example.com>
1296 12:Signed-off-by: my@house
1298 test_cmp expected actual
1301 test_expect_success 'signoff: detect garbage in non-conforming footer' '
1302 append_signoff <<\EOF >actual &&
1309 Signed-off-by: C O Mitter <committer@example.com>
1311 cat >expected <<\EOF &&
1312 4:Subject: [PATCH] subject
1315 13:Signed-off-by: C O Mitter <committer@example.com>
1317 15:Signed-off-by: C O Mitter <committer@example.com>
1319 test_cmp expected actual
1322 test_expect_success 'signoff: footer begins with non-signoff without @ sign' '
1323 append_signoff <<\EOF >actual &&
1330 Change-id: Ideadbeef
1331 Signed-off-by: C O Mitter <committer@example.com>
1334 cat >expected <<\EOF &&
1335 4:Subject: [PATCH] subject
1338 14:Signed-off-by: C O Mitter <committer@example.com>
1340 test_cmp expected actual
1343 test_expect_success 'format patch ignores color.ui' '
1344 test_unconfig color.ui &&
1345 git format-patch --stdout -1 >expect &&
1346 test_config color.ui always &&
1347 git format-patch --stdout -1 >actual &&
1348 test_cmp expect actual
1351 test_expect_success 'cover letter using branch description (1)' '
1352 git checkout rebuild-1 &&
1353 test_config branch.rebuild-1.description hello &&
1354 git format-patch --stdout --cover-letter master >actual &&
1355 grep hello actual >/dev/null
1358 test_expect_success 'cover letter using branch description (2)' '
1359 git checkout rebuild-1 &&
1360 test_config branch.rebuild-1.description hello &&
1361 git format-patch --stdout --cover-letter rebuild-1~2..rebuild-1 >actual &&
1362 grep hello actual >/dev/null
1365 test_expect_success 'cover letter using branch description (3)' '
1366 git checkout rebuild-1 &&
1367 test_config branch.rebuild-1.description hello &&
1368 git format-patch --stdout --cover-letter ^master rebuild-1 >actual &&
1369 grep hello actual >/dev/null
1372 test_expect_success 'cover letter using branch description (4)' '
1373 git checkout rebuild-1 &&
1374 test_config branch.rebuild-1.description hello &&
1375 git format-patch --stdout --cover-letter master.. >actual &&
1376 grep hello actual >/dev/null
1379 test_expect_success 'cover letter using branch description (5)' '
1380 git checkout rebuild-1 &&
1381 test_config branch.rebuild-1.description hello &&
1382 git format-patch --stdout --cover-letter -2 HEAD >actual &&
1383 grep hello actual >/dev/null
1386 test_expect_success 'cover letter using branch description (6)' '
1387 git checkout rebuild-1 &&
1388 test_config branch.rebuild-1.description hello &&
1389 git format-patch --stdout --cover-letter -2 >actual &&
1390 grep hello actual >/dev/null
1393 test_expect_success 'cover letter with nothing' '
1394 git format-patch --stdout --cover-letter >actual &&
1395 test_line_count = 0 actual
1398 test_expect_success 'cover letter auto' '
1400 test_when_finished "rm -rf tmp;
1401 git config --unset format.coverletter" &&
1403 git config format.coverletter auto &&
1404 git format-patch -o tmp -1 >list &&
1405 test_line_count = 1 list &&
1406 git format-patch -o tmp -2 >list &&
1407 test_line_count = 3 list
1410 test_expect_success 'cover letter auto user override' '
1412 test_when_finished "rm -rf tmp;
1413 git config --unset format.coverletter" &&
1415 git config format.coverletter auto &&
1416 git format-patch -o tmp --cover-letter -1 >list &&
1417 test_line_count = 2 list &&
1418 git format-patch -o tmp --cover-letter -2 >list &&
1419 test_line_count = 3 list &&
1420 git format-patch -o tmp --no-cover-letter -1 >list &&
1421 test_line_count = 1 list &&
1422 git format-patch -o tmp --no-cover-letter -2 >list &&
1423 test_line_count = 2 list