2011-05-27 14:08:04 -07:00
|
|
|
|
#!/usr/bin/env bash
|
2011-05-16 15:25:25 -07:00
|
|
|
|
test_description="output of multipart message"
|
|
|
|
|
. ./test-lib.sh
|
|
|
|
|
|
2011-09-05 12:01:46 -07:00
|
|
|
|
cat <<EOF > embedded_message
|
|
|
|
|
From: Carl Worth <cworth@cworth.org>
|
|
|
|
|
To: cworth@cworth.org
|
|
|
|
|
Subject: html message
|
|
|
|
|
Date: Fri, 05 Jan 2001 15:42:57 +0000
|
|
|
|
|
User-Agent: Notmuch/0.5 (http://notmuchmail.org) Emacs/23.3.1 (i486-pc-linux-gnu)
|
|
|
|
|
Message-ID: <87liy5ap01.fsf@yoom.home.cworth.org>
|
|
|
|
|
MIME-Version: 1.0
|
|
|
|
|
Content-Type: multipart/alternative; boundary="==-=-=="
|
|
|
|
|
|
|
|
|
|
--==-=-==
|
|
|
|
|
Content-Type: text/html
|
|
|
|
|
|
|
|
|
|
<p>This is an embedded message, with a multipart/alternative part.</p>
|
|
|
|
|
|
|
|
|
|
--==-=-==
|
|
|
|
|
Content-Type: text/plain
|
|
|
|
|
|
|
|
|
|
This is an embedded message, with a multipart/alternative part.
|
|
|
|
|
|
|
|
|
|
--==-=-==--
|
|
|
|
|
EOF
|
|
|
|
|
|
2011-05-16 15:25:25 -07:00
|
|
|
|
cat <<EOF > ${MAIL_DIR}/multipart
|
|
|
|
|
From: Carl Worth <cworth@cworth.org>
|
|
|
|
|
To: cworth@cworth.org
|
|
|
|
|
Subject: Multipart message
|
2011-06-05 17:29:26 -07:00
|
|
|
|
Date: Fri, 05 Jan 2001 15:43:57 +0000
|
2011-05-16 15:25:25 -07:00
|
|
|
|
User-Agent: Notmuch/0.5 (http://notmuchmail.org) Emacs/23.3.1 (i486-pc-linux-gnu)
|
|
|
|
|
Message-ID: <87liy5ap00.fsf@yoom.home.cworth.org>
|
|
|
|
|
MIME-Version: 1.0
|
|
|
|
|
Content-Type: multipart/signed; boundary="==-=-=";
|
|
|
|
|
micalg=pgp-sha1; protocol="application/pgp-signature"
|
|
|
|
|
|
|
|
|
|
--==-=-=
|
|
|
|
|
Content-Type: multipart/mixed; boundary="=-=-="
|
|
|
|
|
|
|
|
|
|
--=-=-=
|
2011-05-23 00:46:44 -07:00
|
|
|
|
Content-Type: message/rfc822
|
|
|
|
|
Content-Disposition: inline
|
2011-05-16 15:25:25 -07:00
|
|
|
|
|
2011-09-05 12:01:46 -07:00
|
|
|
|
EOF
|
|
|
|
|
cat embedded_message >> ${MAIL_DIR}/multipart
|
|
|
|
|
cat <<EOF >> ${MAIL_DIR}/multipart
|
2012-03-06 18:48:38 +00:00
|
|
|
|
|
2011-05-16 15:25:25 -07:00
|
|
|
|
--=-=-=
|
|
|
|
|
Content-Disposition: attachment; filename=attachment
|
|
|
|
|
|
|
|
|
|
This is a text attachment.
|
|
|
|
|
|
|
|
|
|
--=-=-=
|
|
|
|
|
|
|
|
|
|
And this message is signed.
|
|
|
|
|
|
|
|
|
|
-Carl
|
|
|
|
|
|
|
|
|
|
--=-=-=--
|
|
|
|
|
|
|
|
|
|
--==-=-=
|
|
|
|
|
Content-Type: application/pgp-signature
|
|
|
|
|
|
|
|
|
|
-----BEGIN PGP SIGNATURE-----
|
|
|
|
|
Version: GnuPG v1.4.11 (GNU/Linux)
|
|
|
|
|
|
|
|
|
|
iEYEARECAAYFAk3SA/gACgkQ6JDdNq8qSWj0sACghqVJEQJUs3yV8zbTzhgnSIcD
|
|
|
|
|
W6cAmQE4dcYrx/LPLtYLZm1jsGauE5hE
|
|
|
|
|
=zkga
|
|
|
|
|
-----END PGP SIGNATURE-----
|
|
|
|
|
--==-=-=--
|
|
|
|
|
EOF
|
2011-05-31 15:39:26 -07:00
|
|
|
|
|
|
|
|
|
cat <<EOF > ${MAIL_DIR}/base64-part-with-crlf
|
|
|
|
|
From: Carl Worth <cworth@cworth.org>
|
|
|
|
|
To: cworth@cworth.org
|
|
|
|
|
Subject: Test message with a BASE64 encoded binary containing CRLF pair
|
2011-06-05 17:29:26 -07:00
|
|
|
|
Date: Fri, 05 Jan 2001 15:43:57 +0000
|
2011-05-31 15:39:26 -07:00
|
|
|
|
User-Agent: Notmuch/0.5 (http://notmuchmail.org) Emacs/23.3.1 (i486-pc-linux-gnu)
|
|
|
|
|
Message-ID: <base64-part-with-crlf>
|
|
|
|
|
MIME-Version: 1.0
|
|
|
|
|
Content-Type: multipart/mixed; boundary="==-=-=";
|
|
|
|
|
|
|
|
|
|
--==-=-=
|
|
|
|
|
|
|
|
|
|
The attached BASE64-encoded part expands to a binary containing a CRLF
|
|
|
|
|
pair (that is one bye of 0x0D followed by one byte of 0x0A). This is
|
|
|
|
|
designed to ensure that notmuch is not corrupting the output of this
|
|
|
|
|
part by converting the CRLF pair to an LF only (as would be appropriate
|
|
|
|
|
for display of a text part on a Linux system, for example).
|
|
|
|
|
|
|
|
|
|
The part should be a 3-byte file with the following sequence of 3
|
|
|
|
|
hexadecimal bytes:
|
|
|
|
|
|
|
|
|
|
EF 0D 0A
|
|
|
|
|
|
|
|
|
|
--==-=-=
|
|
|
|
|
Content-Type: application/octet-stream
|
|
|
|
|
Content-Disposition: attachment; filename=crlf.bin
|
|
|
|
|
Content-Transfer-Encoding: base64
|
|
|
|
|
|
|
|
|
|
7w0K
|
|
|
|
|
--==-=-=--
|
|
|
|
|
EOF
|
2011-05-16 15:25:25 -07:00
|
|
|
|
notmuch new > /dev/null
|
|
|
|
|
|
2011-05-23 00:46:44 -07:00
|
|
|
|
test_begin_subtest "--format=text --part=0, full message"
|
2011-05-28 14:16:30 -07:00
|
|
|
|
notmuch show --format=text --part=0 'id:87liy5ap00.fsf@yoom.home.cworth.org' >OUTPUT
|
|
|
|
|
cat <<EOF >EXPECTED
|
2012-03-01 22:30:41 +00:00
|
|
|
|
message{ id:87liy5ap00.fsf@yoom.home.cworth.org depth:0 match:1 excluded:0 filename:${MAIL_DIR}/multipart
|
2011-05-16 15:25:25 -07:00
|
|
|
|
header{
|
2011-05-25 18:01:20 -07:00
|
|
|
|
Carl Worth <cworth@cworth.org> (2001-01-05) (attachment inbox signed unread)
|
2011-05-16 15:25:25 -07:00
|
|
|
|
Subject: Multipart message
|
|
|
|
|
From: Carl Worth <cworth@cworth.org>
|
|
|
|
|
To: cworth@cworth.org
|
2011-06-05 17:29:26 -07:00
|
|
|
|
Date: Fri, 05 Jan 2001 15:43:57 +0000
|
2011-05-16 15:25:25 -07:00
|
|
|
|
header}
|
|
|
|
|
body{
|
2011-05-16 22:28:36 -07:00
|
|
|
|
part{ ID: 1, Content-type: multipart/signed
|
|
|
|
|
part{ ID: 2, Content-type: multipart/mixed
|
2011-05-23 00:46:44 -07:00
|
|
|
|
part{ ID: 3, Content-type: message/rfc822
|
2011-09-05 12:01:46 -07:00
|
|
|
|
header{
|
show: Use consistent header ordering in the text format
Previously, top-level message headers were printed as Subject, From,
To, Date, while embedded message headers were printed From, To,
Subject, Date. This makes both cases use the former order and updates
the tests accordingly.
Strangely, the raw format also uses this function, so this also fixes
the two raw format tests affected by this change.
2012-01-22 21:31:12 -05:00
|
|
|
|
Subject: html message
|
2011-09-05 12:01:46 -07:00
|
|
|
|
From: Carl Worth <cworth@cworth.org>
|
|
|
|
|
To: cworth@cworth.org
|
|
|
|
|
Date: Fri, 05 Jan 2001 15:42:57 +0000
|
|
|
|
|
header}
|
|
|
|
|
body{
|
|
|
|
|
part{ ID: 4, Content-type: multipart/alternative
|
|
|
|
|
part{ ID: 5, Content-type: text/html
|
2011-05-23 00:46:44 -07:00
|
|
|
|
Non-text part: text/html
|
|
|
|
|
part}
|
2011-09-05 12:01:46 -07:00
|
|
|
|
part{ ID: 6, Content-type: text/plain
|
|
|
|
|
This is an embedded message, with a multipart/alternative part.
|
|
|
|
|
part}
|
|
|
|
|
part}
|
|
|
|
|
body}
|
2011-05-16 15:25:25 -07:00
|
|
|
|
part}
|
2011-09-05 12:01:46 -07:00
|
|
|
|
attachment{ ID: 7, Filename: attachment, Content-type: text/plain
|
2011-05-16 15:25:25 -07:00
|
|
|
|
This is a text attachment.
|
|
|
|
|
attachment}
|
2011-09-05 12:01:46 -07:00
|
|
|
|
part{ ID: 8, Content-type: text/plain
|
2011-05-16 15:25:25 -07:00
|
|
|
|
And this message is signed.
|
|
|
|
|
|
|
|
|
|
-Carl
|
|
|
|
|
part}
|
notmuch show: Properly nest MIME parts within mulipart parts
Previously, notmuch show flattened all output, losing information
about the nesting of the MIME hierarchy. Now, the output is properly
nested, (both in the --format=text and --format=json output), so that
clients can analyze the original MIME structure.
Internally, this required splitting the final closing delimiter out of
the various show_part functions and putting it into a new
show_part_end function instead. Also, the show_part function now
accepts a new "first" argument that is set not only for the first MIME
part of a message, but also for each first MIME part within a series
of multipart parts. This "first" argument controls the omission of a
preceding comma when printing a part (for json).
Many thanks to David Edmondson <dme@dme.org> for originally
identifying the lack of nesting in the json output and submitting an
early implementation of this feature. Thanks as well to Jameson Graef
Rollins <jrollins@finestructure.net> for carefully shepherding David's
patches through a remarkably long review process, patiently explaining
them, and providing a cleaned up series that led to this final
implementation. Jameson also provided the new emacs code here.
2011-05-17 15:34:57 -07:00
|
|
|
|
part}
|
2011-09-05 12:01:46 -07:00
|
|
|
|
part{ ID: 9, Content-type: application/pgp-signature
|
2011-05-16 15:25:25 -07:00
|
|
|
|
Non-text part: application/pgp-signature
|
|
|
|
|
part}
|
notmuch show: Properly nest MIME parts within mulipart parts
Previously, notmuch show flattened all output, losing information
about the nesting of the MIME hierarchy. Now, the output is properly
nested, (both in the --format=text and --format=json output), so that
clients can analyze the original MIME structure.
Internally, this required splitting the final closing delimiter out of
the various show_part functions and putting it into a new
show_part_end function instead. Also, the show_part function now
accepts a new "first" argument that is set not only for the first MIME
part of a message, but also for each first MIME part within a series
of multipart parts. This "first" argument controls the omission of a
preceding comma when printing a part (for json).
Many thanks to David Edmondson <dme@dme.org> for originally
identifying the lack of nesting in the json output and submitting an
early implementation of this feature. Thanks as well to Jameson Graef
Rollins <jrollins@finestructure.net> for carefully shepherding David's
patches through a remarkably long review process, patiently explaining
them, and providing a cleaned up series that led to this final
implementation. Jameson also provided the new emacs code here.
2011-05-17 15:34:57 -07:00
|
|
|
|
part}
|
2011-05-16 15:25:25 -07:00
|
|
|
|
body}
|
2011-05-28 14:16:30 -07:00
|
|
|
|
message}
|
|
|
|
|
EOF
|
|
|
|
|
test_expect_equal_file OUTPUT EXPECTED
|
2011-05-16 15:25:25 -07:00
|
|
|
|
|
2011-06-05 17:29:25 -07:00
|
|
|
|
test_begin_subtest "--format=text --part=1, message body"
|
|
|
|
|
notmuch show --format=text --part=1 'id:87liy5ap00.fsf@yoom.home.cworth.org' >OUTPUT
|
2011-05-28 14:16:30 -07:00
|
|
|
|
cat <<EOF >EXPECTED
|
|
|
|
|
part{ ID: 1, Content-type: multipart/signed
|
2011-05-23 00:46:44 -07:00
|
|
|
|
part{ ID: 2, Content-type: multipart/mixed
|
|
|
|
|
part{ ID: 3, Content-type: message/rfc822
|
2011-09-05 12:01:46 -07:00
|
|
|
|
header{
|
show: Use consistent header ordering in the text format
Previously, top-level message headers were printed as Subject, From,
To, Date, while embedded message headers were printed From, To,
Subject, Date. This makes both cases use the former order and updates
the tests accordingly.
Strangely, the raw format also uses this function, so this also fixes
the two raw format tests affected by this change.
2012-01-22 21:31:12 -05:00
|
|
|
|
Subject: html message
|
2011-09-05 12:01:46 -07:00
|
|
|
|
From: Carl Worth <cworth@cworth.org>
|
|
|
|
|
To: cworth@cworth.org
|
|
|
|
|
Date: Fri, 05 Jan 2001 15:42:57 +0000
|
|
|
|
|
header}
|
|
|
|
|
body{
|
|
|
|
|
part{ ID: 4, Content-type: multipart/alternative
|
|
|
|
|
part{ ID: 5, Content-type: text/html
|
2011-05-23 00:46:44 -07:00
|
|
|
|
Non-text part: text/html
|
|
|
|
|
part}
|
2011-09-05 12:01:46 -07:00
|
|
|
|
part{ ID: 6, Content-type: text/plain
|
|
|
|
|
This is an embedded message, with a multipart/alternative part.
|
|
|
|
|
part}
|
2011-05-23 00:46:44 -07:00
|
|
|
|
part}
|
2011-09-05 12:01:46 -07:00
|
|
|
|
body}
|
|
|
|
|
part}
|
|
|
|
|
attachment{ ID: 7, Filename: attachment, Content-type: text/plain
|
2011-05-23 00:46:44 -07:00
|
|
|
|
This is a text attachment.
|
|
|
|
|
attachment}
|
2011-09-05 12:01:46 -07:00
|
|
|
|
part{ ID: 8, Content-type: text/plain
|
2011-05-23 00:46:44 -07:00
|
|
|
|
And this message is signed.
|
2011-05-16 15:25:25 -07:00
|
|
|
|
|
2011-05-23 00:46:44 -07:00
|
|
|
|
-Carl
|
|
|
|
|
part}
|
|
|
|
|
part}
|
2011-09-05 12:01:46 -07:00
|
|
|
|
part{ ID: 9, Content-type: application/pgp-signature
|
2011-05-23 00:46:44 -07:00
|
|
|
|
Non-text part: application/pgp-signature
|
|
|
|
|
part}
|
2011-05-28 14:16:30 -07:00
|
|
|
|
part}
|
|
|
|
|
EOF
|
|
|
|
|
test_expect_equal_file OUTPUT EXPECTED
|
2011-05-23 00:46:44 -07:00
|
|
|
|
|
|
|
|
|
test_begin_subtest "--format=text --part=2, multipart/mixed"
|
2011-05-28 14:16:30 -07:00
|
|
|
|
notmuch show --format=text --part=2 'id:87liy5ap00.fsf@yoom.home.cworth.org' >OUTPUT
|
|
|
|
|
cat <<EOF >EXPECTED
|
|
|
|
|
part{ ID: 2, Content-type: multipart/mixed
|
2011-05-23 00:46:44 -07:00
|
|
|
|
part{ ID: 3, Content-type: message/rfc822
|
2011-09-05 12:01:46 -07:00
|
|
|
|
header{
|
show: Use consistent header ordering in the text format
Previously, top-level message headers were printed as Subject, From,
To, Date, while embedded message headers were printed From, To,
Subject, Date. This makes both cases use the former order and updates
the tests accordingly.
Strangely, the raw format also uses this function, so this also fixes
the two raw format tests affected by this change.
2012-01-22 21:31:12 -05:00
|
|
|
|
Subject: html message
|
2011-09-05 12:01:46 -07:00
|
|
|
|
From: Carl Worth <cworth@cworth.org>
|
|
|
|
|
To: cworth@cworth.org
|
|
|
|
|
Date: Fri, 05 Jan 2001 15:42:57 +0000
|
|
|
|
|
header}
|
|
|
|
|
body{
|
|
|
|
|
part{ ID: 4, Content-type: multipart/alternative
|
|
|
|
|
part{ ID: 5, Content-type: text/html
|
2011-05-23 00:46:44 -07:00
|
|
|
|
Non-text part: text/html
|
|
|
|
|
part}
|
2011-09-05 12:01:46 -07:00
|
|
|
|
part{ ID: 6, Content-type: text/plain
|
|
|
|
|
This is an embedded message, with a multipart/alternative part.
|
|
|
|
|
part}
|
2011-05-23 00:46:44 -07:00
|
|
|
|
part}
|
2011-09-05 12:01:46 -07:00
|
|
|
|
body}
|
|
|
|
|
part}
|
|
|
|
|
attachment{ ID: 7, Filename: attachment, Content-type: text/plain
|
2011-05-23 00:46:44 -07:00
|
|
|
|
This is a text attachment.
|
|
|
|
|
attachment}
|
2011-09-05 12:01:46 -07:00
|
|
|
|
part{ ID: 8, Content-type: text/plain
|
2011-05-23 00:46:44 -07:00
|
|
|
|
And this message is signed.
|
|
|
|
|
|
|
|
|
|
-Carl
|
|
|
|
|
part}
|
2011-05-28 14:16:30 -07:00
|
|
|
|
part}
|
|
|
|
|
EOF
|
|
|
|
|
test_expect_equal_file OUTPUT EXPECTED
|
2011-05-23 00:46:44 -07:00
|
|
|
|
|
2011-09-05 12:01:46 -07:00
|
|
|
|
test_begin_subtest "--format=text --part=3, rfc822 part"
|
2011-05-28 14:16:30 -07:00
|
|
|
|
notmuch show --format=text --part=3 'id:87liy5ap00.fsf@yoom.home.cworth.org' >OUTPUT
|
|
|
|
|
cat <<EOF >EXPECTED
|
|
|
|
|
part{ ID: 3, Content-type: message/rfc822
|
2011-09-05 12:01:46 -07:00
|
|
|
|
header{
|
show: Use consistent header ordering in the text format
Previously, top-level message headers were printed as Subject, From,
To, Date, while embedded message headers were printed From, To,
Subject, Date. This makes both cases use the former order and updates
the tests accordingly.
Strangely, the raw format also uses this function, so this also fixes
the two raw format tests affected by this change.
2012-01-22 21:31:12 -05:00
|
|
|
|
Subject: html message
|
2011-09-05 12:01:46 -07:00
|
|
|
|
From: Carl Worth <cworth@cworth.org>
|
|
|
|
|
To: cworth@cworth.org
|
|
|
|
|
Date: Fri, 05 Jan 2001 15:42:57 +0000
|
|
|
|
|
header}
|
|
|
|
|
body{
|
|
|
|
|
part{ ID: 4, Content-type: multipart/alternative
|
|
|
|
|
part{ ID: 5, Content-type: text/html
|
2011-05-23 00:46:44 -07:00
|
|
|
|
Non-text part: text/html
|
|
|
|
|
part}
|
2011-09-05 12:01:46 -07:00
|
|
|
|
part{ ID: 6, Content-type: text/plain
|
|
|
|
|
This is an embedded message, with a multipart/alternative part.
|
|
|
|
|
part}
|
|
|
|
|
part}
|
|
|
|
|
body}
|
2011-05-28 14:16:30 -07:00
|
|
|
|
part}
|
|
|
|
|
EOF
|
|
|
|
|
test_expect_equal_file OUTPUT EXPECTED
|
2011-05-23 00:46:44 -07:00
|
|
|
|
|
2011-09-05 12:01:46 -07:00
|
|
|
|
test_begin_subtest "--format=text --part=4, rfc822's multipart"
|
2011-05-28 14:16:30 -07:00
|
|
|
|
notmuch show --format=text --part=4 'id:87liy5ap00.fsf@yoom.home.cworth.org' >OUTPUT
|
|
|
|
|
cat <<EOF >EXPECTED
|
2011-09-05 12:01:46 -07:00
|
|
|
|
part{ ID: 4, Content-type: multipart/alternative
|
|
|
|
|
part{ ID: 5, Content-type: text/html
|
2011-05-23 00:46:44 -07:00
|
|
|
|
Non-text part: text/html
|
2011-05-28 14:16:30 -07:00
|
|
|
|
part}
|
2011-09-05 12:01:46 -07:00
|
|
|
|
part{ ID: 6, Content-type: text/plain
|
|
|
|
|
This is an embedded message, with a multipart/alternative part.
|
|
|
|
|
part}
|
|
|
|
|
part}
|
2011-05-28 14:16:30 -07:00
|
|
|
|
EOF
|
|
|
|
|
test_expect_equal_file OUTPUT EXPECTED
|
2011-05-23 00:46:44 -07:00
|
|
|
|
|
2011-09-05 12:01:46 -07:00
|
|
|
|
test_begin_subtest "--format=text --part=5, rfc822's html part"
|
2011-05-28 14:16:30 -07:00
|
|
|
|
notmuch show --format=text --part=5 'id:87liy5ap00.fsf@yoom.home.cworth.org' >OUTPUT
|
|
|
|
|
cat <<EOF >EXPECTED
|
2011-09-05 12:01:46 -07:00
|
|
|
|
part{ ID: 5, Content-type: text/html
|
|
|
|
|
Non-text part: text/html
|
|
|
|
|
part}
|
2011-05-28 14:16:30 -07:00
|
|
|
|
EOF
|
|
|
|
|
test_expect_equal_file OUTPUT EXPECTED
|
2011-05-23 00:46:44 -07:00
|
|
|
|
|
2011-09-05 12:01:46 -07:00
|
|
|
|
test_begin_subtest "--format=text --part=6, rfc822's text part"
|
2011-05-28 14:16:30 -07:00
|
|
|
|
notmuch show --format=text --part=6 'id:87liy5ap00.fsf@yoom.home.cworth.org' >OUTPUT
|
|
|
|
|
cat <<EOF >EXPECTED
|
|
|
|
|
part{ ID: 6, Content-type: text/plain
|
2011-09-05 12:01:46 -07:00
|
|
|
|
This is an embedded message, with a multipart/alternative part.
|
|
|
|
|
part}
|
|
|
|
|
EOF
|
|
|
|
|
test_expect_equal_file OUTPUT EXPECTED
|
|
|
|
|
|
|
|
|
|
test_begin_subtest "--format=text --part=7, inline attachement"
|
|
|
|
|
notmuch show --format=text --part=7 'id:87liy5ap00.fsf@yoom.home.cworth.org' >OUTPUT
|
|
|
|
|
cat <<EOF >EXPECTED
|
|
|
|
|
attachment{ ID: 7, Filename: attachment, Content-type: text/plain
|
|
|
|
|
This is a text attachment.
|
|
|
|
|
attachment}
|
|
|
|
|
EOF
|
|
|
|
|
test_expect_equal_file OUTPUT EXPECTED
|
|
|
|
|
|
|
|
|
|
test_begin_subtest "--format=text --part=8, plain text part"
|
|
|
|
|
notmuch show --format=text --part=8 'id:87liy5ap00.fsf@yoom.home.cworth.org' >OUTPUT
|
|
|
|
|
cat <<EOF >EXPECTED
|
|
|
|
|
part{ ID: 8, Content-type: text/plain
|
2011-05-23 00:46:44 -07:00
|
|
|
|
And this message is signed.
|
|
|
|
|
|
|
|
|
|
-Carl
|
2011-05-28 14:16:30 -07:00
|
|
|
|
part}
|
|
|
|
|
EOF
|
|
|
|
|
test_expect_equal_file OUTPUT EXPECTED
|
2011-05-23 00:46:44 -07:00
|
|
|
|
|
2011-09-05 12:01:46 -07:00
|
|
|
|
test_begin_subtest "--format=text --part=9, pgp signature (unverified)"
|
|
|
|
|
notmuch show --format=text --part=9 'id:87liy5ap00.fsf@yoom.home.cworth.org' >OUTPUT
|
2011-05-28 14:16:30 -07:00
|
|
|
|
cat <<EOF >EXPECTED
|
2011-09-05 12:01:46 -07:00
|
|
|
|
part{ ID: 9, Content-type: application/pgp-signature
|
2011-05-23 00:46:44 -07:00
|
|
|
|
Non-text part: application/pgp-signature
|
2011-05-28 14:16:30 -07:00
|
|
|
|
part}
|
|
|
|
|
EOF
|
|
|
|
|
test_expect_equal_file OUTPUT EXPECTED
|
2011-05-23 00:46:44 -07:00
|
|
|
|
|
|
|
|
|
test_expect_success \
|
|
|
|
|
"--format=text --part=8, no part, expect error" \
|
|
|
|
|
"notmuch show --format=text --part=8 'id:87liy5ap00.fsf@yoom.home.cworth.org'"
|
2011-05-16 15:25:25 -07:00
|
|
|
|
|
2011-05-23 00:46:44 -07:00
|
|
|
|
test_begin_subtest "--format=json --part=0, full message"
|
2012-08-02 21:14:48 -04:00
|
|
|
|
notmuch show --format=json --part=0 'id:87liy5ap00.fsf@yoom.home.cworth.org' >OUTPUT
|
2011-09-05 12:01:46 -07:00
|
|
|
|
cat <<EOF >EXPECTED
|
2012-03-01 22:30:41 +00:00
|
|
|
|
{"id": "87liy5ap00.fsf@yoom.home.cworth.org", "match": true, "excluded": false, "filename": "${MAIL_DIR}/multipart", "timestamp": 978709437, "date_relative": "2001-01-05", "tags": ["attachment","inbox","signed","unread"], "headers": {"Subject": "Multipart message", "From": "Carl Worth <cworth@cworth.org>", "To": "cworth@cworth.org", "Date": "Fri, 05 Jan 2001 15:43:57 +0000"}, "body": [
|
2011-09-05 12:01:46 -07:00
|
|
|
|
{"id": 1, "content-type": "multipart/signed", "content": [
|
|
|
|
|
{"id": 2, "content-type": "multipart/mixed", "content": [
|
2012-02-19 19:26:25 -05:00
|
|
|
|
{"id": 3, "content-type": "message/rfc822", "content": [{"headers": {"Subject": "html message", "From": "Carl Worth <cworth@cworth.org>", "To": "cworth@cworth.org", "Date": "Fri, 05 Jan 2001 15:42:57 +0000"}, "body": [
|
2011-09-05 12:01:46 -07:00
|
|
|
|
{"id": 4, "content-type": "multipart/alternative", "content": [
|
2012-12-15 14:06:45 +11:00
|
|
|
|
{"id": 5, "content-type": "text/html", "content-length": 71},
|
2011-09-05 12:01:46 -07:00
|
|
|
|
{"id": 6, "content-type": "text/plain", "content": "This is an embedded message, with a multipart/alternative part.\n"}]}]}]},
|
|
|
|
|
{"id": 7, "content-type": "text/plain", "filename": "attachment", "content": "This is a text attachment.\n"},
|
|
|
|
|
{"id": 8, "content-type": "text/plain", "content": "And this message is signed.\n\n-Carl\n"}]},
|
2012-12-15 14:06:45 +11:00
|
|
|
|
{"id": 9, "content-type": "application/pgp-signature", "content-length": 197}]}]}
|
2011-09-05 12:01:46 -07:00
|
|
|
|
EOF
|
2012-08-02 21:14:47 -04:00
|
|
|
|
test_expect_equal_json "$(cat OUTPUT)" "$(cat EXPECTED)"
|
2011-05-23 00:46:44 -07:00
|
|
|
|
|
|
|
|
|
test_begin_subtest "--format=json --part=1, message body"
|
2012-08-02 21:14:48 -04:00
|
|
|
|
notmuch show --format=json --part=1 'id:87liy5ap00.fsf@yoom.home.cworth.org' >OUTPUT
|
2011-09-05 12:01:46 -07:00
|
|
|
|
cat <<EOF >EXPECTED
|
|
|
|
|
{"id": 1, "content-type": "multipart/signed", "content": [
|
|
|
|
|
{"id": 2, "content-type": "multipart/mixed", "content": [
|
2012-02-19 19:26:25 -05:00
|
|
|
|
{"id": 3, "content-type": "message/rfc822", "content": [{"headers": {"Subject": "html message", "From": "Carl Worth <cworth@cworth.org>", "To": "cworth@cworth.org", "Date": "Fri, 05 Jan 2001 15:42:57 +0000"}, "body": [
|
2011-09-05 12:01:46 -07:00
|
|
|
|
{"id": 4, "content-type": "multipart/alternative", "content": [
|
2012-12-15 14:06:45 +11:00
|
|
|
|
{"id": 5, "content-type": "text/html", "content-length": 71},
|
2011-09-05 12:01:46 -07:00
|
|
|
|
{"id": 6, "content-type": "text/plain", "content": "This is an embedded message, with a multipart/alternative part.\n"}]}]}]},
|
|
|
|
|
{"id": 7, "content-type": "text/plain", "filename": "attachment", "content": "This is a text attachment.\n"},
|
|
|
|
|
{"id": 8, "content-type": "text/plain", "content": "And this message is signed.\n\n-Carl\n"}]},
|
2012-12-15 14:06:45 +11:00
|
|
|
|
{"id": 9, "content-type": "application/pgp-signature", "content-length": 197}]}
|
2011-09-05 12:01:46 -07:00
|
|
|
|
EOF
|
2012-08-02 21:14:47 -04:00
|
|
|
|
test_expect_equal_json "$(cat OUTPUT)" "$(cat EXPECTED)"
|
2011-05-23 00:46:44 -07:00
|
|
|
|
|
|
|
|
|
test_begin_subtest "--format=json --part=2, multipart/mixed"
|
2012-08-02 21:14:48 -04:00
|
|
|
|
notmuch show --format=json --part=2 'id:87liy5ap00.fsf@yoom.home.cworth.org' >OUTPUT
|
2011-09-05 12:01:46 -07:00
|
|
|
|
cat <<EOF >EXPECTED
|
|
|
|
|
{"id": 2, "content-type": "multipart/mixed", "content": [
|
2012-02-19 19:26:25 -05:00
|
|
|
|
{"id": 3, "content-type": "message/rfc822", "content": [{"headers": {"Subject": "html message", "From": "Carl Worth <cworth@cworth.org>", "To": "cworth@cworth.org", "Date": "Fri, 05 Jan 2001 15:42:57 +0000"}, "body": [
|
2011-09-05 12:01:46 -07:00
|
|
|
|
{"id": 4, "content-type": "multipart/alternative", "content": [
|
2012-12-15 14:06:45 +11:00
|
|
|
|
{"id": 5, "content-type": "text/html", "content-length": 71},
|
2011-09-05 12:01:46 -07:00
|
|
|
|
{"id": 6, "content-type": "text/plain", "content": "This is an embedded message, with a multipart/alternative part.\n"}]}]}]},
|
|
|
|
|
{"id": 7, "content-type": "text/plain", "filename": "attachment", "content": "This is a text attachment.\n"},
|
|
|
|
|
{"id": 8, "content-type": "text/plain", "content": "And this message is signed.\n\n-Carl\n"}]}
|
|
|
|
|
EOF
|
2012-08-02 21:14:47 -04:00
|
|
|
|
test_expect_equal_json "$(cat OUTPUT)" "$(cat EXPECTED)"
|
2011-09-05 12:01:46 -07:00
|
|
|
|
|
|
|
|
|
test_begin_subtest "--format=json --part=3, rfc822 part"
|
2012-08-02 21:14:48 -04:00
|
|
|
|
notmuch show --format=json --part=3 'id:87liy5ap00.fsf@yoom.home.cworth.org' >OUTPUT
|
2011-09-05 12:01:46 -07:00
|
|
|
|
cat <<EOF >EXPECTED
|
2012-02-19 19:26:25 -05:00
|
|
|
|
{"id": 3, "content-type": "message/rfc822", "content": [{"headers": {"Subject": "html message", "From": "Carl Worth <cworth@cworth.org>", "To": "cworth@cworth.org", "Date": "Fri, 05 Jan 2001 15:42:57 +0000"}, "body": [
|
2011-09-05 12:01:46 -07:00
|
|
|
|
{"id": 4, "content-type": "multipart/alternative", "content": [
|
2012-12-15 14:06:45 +11:00
|
|
|
|
{"id": 5, "content-type": "text/html", "content-length": 71},
|
2011-09-05 12:01:46 -07:00
|
|
|
|
{"id": 6, "content-type": "text/plain", "content": "This is an embedded message, with a multipart/alternative part.\n"}]}]}]}
|
|
|
|
|
EOF
|
2012-08-02 21:14:47 -04:00
|
|
|
|
test_expect_equal_json "$(cat OUTPUT)" "$(cat EXPECTED)"
|
2011-09-05 12:01:46 -07:00
|
|
|
|
|
|
|
|
|
test_begin_subtest "--format=json --part=4, rfc822's multipart/alternative"
|
2012-08-02 21:14:48 -04:00
|
|
|
|
notmuch show --format=json --part=4 'id:87liy5ap00.fsf@yoom.home.cworth.org' >OUTPUT
|
2011-09-05 12:01:46 -07:00
|
|
|
|
cat <<EOF >EXPECTED
|
|
|
|
|
{"id": 4, "content-type": "multipart/alternative", "content": [
|
2012-12-15 14:06:45 +11:00
|
|
|
|
{"id": 5, "content-type": "text/html", "content-length": 71},
|
2011-09-05 12:01:46 -07:00
|
|
|
|
{"id": 6, "content-type": "text/plain", "content": "This is an embedded message, with a multipart/alternative part.\n"}]}
|
|
|
|
|
EOF
|
2012-08-02 21:14:47 -04:00
|
|
|
|
test_expect_equal_json "$(cat OUTPUT)" "$(cat EXPECTED)"
|
2011-09-05 12:01:46 -07:00
|
|
|
|
|
|
|
|
|
test_begin_subtest "--format=json --part=5, rfc822's html part"
|
2012-08-02 21:14:48 -04:00
|
|
|
|
notmuch show --format=json --part=5 'id:87liy5ap00.fsf@yoom.home.cworth.org' >OUTPUT
|
2011-09-05 12:01:46 -07:00
|
|
|
|
cat <<EOF >EXPECTED
|
2012-12-15 14:06:45 +11:00
|
|
|
|
{"id": 5, "content-type": "text/html", "content-length": 71}
|
2011-09-05 12:01:46 -07:00
|
|
|
|
EOF
|
2012-08-02 21:14:47 -04:00
|
|
|
|
test_expect_equal_json "$(cat OUTPUT)" "$(cat EXPECTED)"
|
2011-09-05 12:01:46 -07:00
|
|
|
|
|
|
|
|
|
test_begin_subtest "--format=json --part=6, rfc822's text part"
|
2012-08-02 21:14:48 -04:00
|
|
|
|
notmuch show --format=json --part=6 'id:87liy5ap00.fsf@yoom.home.cworth.org' >OUTPUT
|
2011-09-05 12:01:46 -07:00
|
|
|
|
cat <<EOF >EXPECTED
|
|
|
|
|
{"id": 6, "content-type": "text/plain", "content": "This is an embedded message, with a multipart/alternative part.\n"}
|
|
|
|
|
EOF
|
2012-08-02 21:14:47 -04:00
|
|
|
|
test_expect_equal_json "$(cat OUTPUT)" "$(cat EXPECTED)"
|
2011-09-05 12:01:46 -07:00
|
|
|
|
|
|
|
|
|
test_begin_subtest "--format=json --part=7, inline attachment"
|
2012-08-02 21:14:48 -04:00
|
|
|
|
notmuch show --format=json --part=7 'id:87liy5ap00.fsf@yoom.home.cworth.org' >OUTPUT
|
2011-09-05 12:01:46 -07:00
|
|
|
|
cat <<EOF >EXPECTED
|
|
|
|
|
{"id": 7, "content-type": "text/plain", "filename": "attachment", "content": "This is a text attachment.\n"}
|
|
|
|
|
EOF
|
2012-08-02 21:14:47 -04:00
|
|
|
|
test_expect_equal_json "$(cat OUTPUT)" "$(cat EXPECTED)"
|
2011-09-05 12:01:46 -07:00
|
|
|
|
|
|
|
|
|
test_begin_subtest "--format=json --part=8, plain text part"
|
2012-08-02 21:14:48 -04:00
|
|
|
|
notmuch show --format=json --part=8 'id:87liy5ap00.fsf@yoom.home.cworth.org' >OUTPUT
|
2011-09-05 12:01:46 -07:00
|
|
|
|
cat <<EOF >EXPECTED
|
|
|
|
|
{"id": 8, "content-type": "text/plain", "content": "And this message is signed.\n\n-Carl\n"}
|
|
|
|
|
EOF
|
2012-08-02 21:14:47 -04:00
|
|
|
|
test_expect_equal_json "$(cat OUTPUT)" "$(cat EXPECTED)"
|
2011-09-05 12:01:46 -07:00
|
|
|
|
|
|
|
|
|
test_begin_subtest "--format=json --part=9, pgp signature (unverified)"
|
2012-08-02 21:14:48 -04:00
|
|
|
|
notmuch show --format=json --part=9 'id:87liy5ap00.fsf@yoom.home.cworth.org' >OUTPUT
|
2011-09-05 12:01:46 -07:00
|
|
|
|
cat <<EOF >EXPECTED
|
2012-12-15 14:06:45 +11:00
|
|
|
|
{"id": 9, "content-type": "application/pgp-signature", "content-length": 197}
|
2011-09-05 12:01:46 -07:00
|
|
|
|
EOF
|
2012-08-02 21:14:47 -04:00
|
|
|
|
test_expect_equal_json "$(cat OUTPUT)" "$(cat EXPECTED)"
|
2011-05-23 00:46:44 -07:00
|
|
|
|
|
|
|
|
|
test_expect_success \
|
2011-09-05 12:01:46 -07:00
|
|
|
|
"--format=json --part=10, no part, expect error" \
|
|
|
|
|
"notmuch show --format=json --part=10 'id:87liy5ap00.fsf@yoom.home.cworth.org'"
|
2011-05-23 00:46:44 -07:00
|
|
|
|
|
|
|
|
|
test_begin_subtest "--format=raw"
|
2011-05-28 14:16:30 -07:00
|
|
|
|
notmuch show --format=raw 'id:87liy5ap00.fsf@yoom.home.cworth.org' >OUTPUT
|
|
|
|
|
test_expect_equal_file OUTPUT "${MAIL_DIR}"/multipart
|
2011-05-23 00:46:44 -07:00
|
|
|
|
|
|
|
|
|
test_begin_subtest "--format=raw --part=0, full message"
|
2011-05-28 14:16:30 -07:00
|
|
|
|
notmuch show --format=raw --part=0 'id:87liy5ap00.fsf@yoom.home.cworth.org' >OUTPUT
|
|
|
|
|
test_expect_equal_file OUTPUT "${MAIL_DIR}"/multipart
|
2011-05-23 00:46:44 -07:00
|
|
|
|
|
|
|
|
|
test_begin_subtest "--format=raw --part=1, message body"
|
2011-05-28 14:16:30 -07:00
|
|
|
|
notmuch show --format=raw --part=1 'id:87liy5ap00.fsf@yoom.home.cworth.org' >OUTPUT
|
show: Convert raw format to the new self-recursive style, properly support interior parts
This is fully compatible for root and leaf parts, but now has proper
support for interior parts. This requires some design decisions that
were guided by what I would want if I were to save a part.
Specifically:
- Leaf parts are printed without headers and with transfer decoding.
This is what makes sense for saving attachments. (Furthermore, the
transfer decoding is necessary since, without the headers, the
caller would not be able to interpret non-transfer-decoded output.)
- Message parts are printed with their message headers, but without
enclosing part headers. This is what makes sense for saving a
message as a whole (which is a message part) and for saving attached
messages. This is symmetric for whole messages and for attached
messages, though we special-case the whole message for performance
reasons (and corner-case correctness reasons: given malformed input,
GMime may not be able to reproduce it from the parsed
representation).
- Multipart parts are printed with their headers and all child parts.
It's not clear what the best thing to do for multipart is, but this
was the most natural to implement and can be justified because such
parts can't be interpreted without their headers.
As an added benefit, we can move the special-case code for part 0 into
the raw formatter.
2012-03-06 18:48:43 +00:00
|
|
|
|
test_expect_equal_file OUTPUT "${MAIL_DIR}"/multipart
|
2011-05-23 00:46:44 -07:00
|
|
|
|
|
|
|
|
|
test_begin_subtest "--format=raw --part=2, multipart/mixed"
|
2011-05-28 14:16:30 -07:00
|
|
|
|
notmuch show --format=raw --part=2 'id:87liy5ap00.fsf@yoom.home.cworth.org' >OUTPUT
|
|
|
|
|
cat <<EOF >EXPECTED
|
show: Convert raw format to the new self-recursive style, properly support interior parts
This is fully compatible for root and leaf parts, but now has proper
support for interior parts. This requires some design decisions that
were guided by what I would want if I were to save a part.
Specifically:
- Leaf parts are printed without headers and with transfer decoding.
This is what makes sense for saving attachments. (Furthermore, the
transfer decoding is necessary since, without the headers, the
caller would not be able to interpret non-transfer-decoded output.)
- Message parts are printed with their message headers, but without
enclosing part headers. This is what makes sense for saving a
message as a whole (which is a message part) and for saving attached
messages. This is symmetric for whole messages and for attached
messages, though we special-case the whole message for performance
reasons (and corner-case correctness reasons: given malformed input,
GMime may not be able to reproduce it from the parsed
representation).
- Multipart parts are printed with their headers and all child parts.
It's not clear what the best thing to do for multipart is, but this
was the most natural to implement and can be justified because such
parts can't be interpreted without their headers.
As an added benefit, we can move the special-case code for part 0 into
the raw formatter.
2012-03-06 18:48:43 +00:00
|
|
|
|
Content-Type: multipart/mixed; boundary="=-=-="
|
|
|
|
|
|
|
|
|
|
--=-=-=
|
|
|
|
|
Content-Type: message/rfc822
|
|
|
|
|
Content-Disposition: inline
|
|
|
|
|
|
2011-09-05 12:01:46 -07:00
|
|
|
|
From: Carl Worth <cworth@cworth.org>
|
|
|
|
|
To: cworth@cworth.org
|
show: Convert raw format to the new self-recursive style, properly support interior parts
This is fully compatible for root and leaf parts, but now has proper
support for interior parts. This requires some design decisions that
were guided by what I would want if I were to save a part.
Specifically:
- Leaf parts are printed without headers and with transfer decoding.
This is what makes sense for saving attachments. (Furthermore, the
transfer decoding is necessary since, without the headers, the
caller would not be able to interpret non-transfer-decoded output.)
- Message parts are printed with their message headers, but without
enclosing part headers. This is what makes sense for saving a
message as a whole (which is a message part) and for saving attached
messages. This is symmetric for whole messages and for attached
messages, though we special-case the whole message for performance
reasons (and corner-case correctness reasons: given malformed input,
GMime may not be able to reproduce it from the parsed
representation).
- Multipart parts are printed with their headers and all child parts.
It's not clear what the best thing to do for multipart is, but this
was the most natural to implement and can be justified because such
parts can't be interpreted without their headers.
As an added benefit, we can move the special-case code for part 0 into
the raw formatter.
2012-03-06 18:48:43 +00:00
|
|
|
|
Subject: html message
|
2011-09-05 12:01:46 -07:00
|
|
|
|
Date: Fri, 05 Jan 2001 15:42:57 +0000
|
show: Convert raw format to the new self-recursive style, properly support interior parts
This is fully compatible for root and leaf parts, but now has proper
support for interior parts. This requires some design decisions that
were guided by what I would want if I were to save a part.
Specifically:
- Leaf parts are printed without headers and with transfer decoding.
This is what makes sense for saving attachments. (Furthermore, the
transfer decoding is necessary since, without the headers, the
caller would not be able to interpret non-transfer-decoded output.)
- Message parts are printed with their message headers, but without
enclosing part headers. This is what makes sense for saving a
message as a whole (which is a message part) and for saving attached
messages. This is symmetric for whole messages and for attached
messages, though we special-case the whole message for performance
reasons (and corner-case correctness reasons: given malformed input,
GMime may not be able to reproduce it from the parsed
representation).
- Multipart parts are printed with their headers and all child parts.
It's not clear what the best thing to do for multipart is, but this
was the most natural to implement and can be justified because such
parts can't be interpreted without their headers.
As an added benefit, we can move the special-case code for part 0 into
the raw formatter.
2012-03-06 18:48:43 +00:00
|
|
|
|
User-Agent: Notmuch/0.5 (http://notmuchmail.org) Emacs/23.3.1 (i486-pc-linux-gnu)
|
|
|
|
|
Message-ID: <87liy5ap01.fsf@yoom.home.cworth.org>
|
|
|
|
|
MIME-Version: 1.0
|
|
|
|
|
Content-Type: multipart/alternative; boundary="==-=-=="
|
|
|
|
|
|
|
|
|
|
--==-=-==
|
|
|
|
|
Content-Type: text/html
|
2011-09-05 12:01:46 -07:00
|
|
|
|
|
|
|
|
|
<p>This is an embedded message, with a multipart/alternative part.</p>
|
show: Convert raw format to the new self-recursive style, properly support interior parts
This is fully compatible for root and leaf parts, but now has proper
support for interior parts. This requires some design decisions that
were guided by what I would want if I were to save a part.
Specifically:
- Leaf parts are printed without headers and with transfer decoding.
This is what makes sense for saving attachments. (Furthermore, the
transfer decoding is necessary since, without the headers, the
caller would not be able to interpret non-transfer-decoded output.)
- Message parts are printed with their message headers, but without
enclosing part headers. This is what makes sense for saving a
message as a whole (which is a message part) and for saving attached
messages. This is symmetric for whole messages and for attached
messages, though we special-case the whole message for performance
reasons (and corner-case correctness reasons: given malformed input,
GMime may not be able to reproduce it from the parsed
representation).
- Multipart parts are printed with their headers and all child parts.
It's not clear what the best thing to do for multipart is, but this
was the most natural to implement and can be justified because such
parts can't be interpreted without their headers.
As an added benefit, we can move the special-case code for part 0 into
the raw formatter.
2012-03-06 18:48:43 +00:00
|
|
|
|
|
|
|
|
|
--==-=-==
|
|
|
|
|
Content-Type: text/plain
|
|
|
|
|
|
2011-09-05 12:01:46 -07:00
|
|
|
|
This is an embedded message, with a multipart/alternative part.
|
show: Convert raw format to the new self-recursive style, properly support interior parts
This is fully compatible for root and leaf parts, but now has proper
support for interior parts. This requires some design decisions that
were guided by what I would want if I were to save a part.
Specifically:
- Leaf parts are printed without headers and with transfer decoding.
This is what makes sense for saving attachments. (Furthermore, the
transfer decoding is necessary since, without the headers, the
caller would not be able to interpret non-transfer-decoded output.)
- Message parts are printed with their message headers, but without
enclosing part headers. This is what makes sense for saving a
message as a whole (which is a message part) and for saving attached
messages. This is symmetric for whole messages and for attached
messages, though we special-case the whole message for performance
reasons (and corner-case correctness reasons: given malformed input,
GMime may not be able to reproduce it from the parsed
representation).
- Multipart parts are printed with their headers and all child parts.
It's not clear what the best thing to do for multipart is, but this
was the most natural to implement and can be justified because such
parts can't be interpreted without their headers.
As an added benefit, we can move the special-case code for part 0 into
the raw formatter.
2012-03-06 18:48:43 +00:00
|
|
|
|
|
|
|
|
|
--==-=-==--
|
|
|
|
|
|
|
|
|
|
--=-=-=
|
|
|
|
|
Content-Disposition: attachment; filename=attachment
|
|
|
|
|
|
2011-05-23 00:46:44 -07:00
|
|
|
|
This is a text attachment.
|
show: Convert raw format to the new self-recursive style, properly support interior parts
This is fully compatible for root and leaf parts, but now has proper
support for interior parts. This requires some design decisions that
were guided by what I would want if I were to save a part.
Specifically:
- Leaf parts are printed without headers and with transfer decoding.
This is what makes sense for saving attachments. (Furthermore, the
transfer decoding is necessary since, without the headers, the
caller would not be able to interpret non-transfer-decoded output.)
- Message parts are printed with their message headers, but without
enclosing part headers. This is what makes sense for saving a
message as a whole (which is a message part) and for saving attached
messages. This is symmetric for whole messages and for attached
messages, though we special-case the whole message for performance
reasons (and corner-case correctness reasons: given malformed input,
GMime may not be able to reproduce it from the parsed
representation).
- Multipart parts are printed with their headers and all child parts.
It's not clear what the best thing to do for multipart is, but this
was the most natural to implement and can be justified because such
parts can't be interpreted without their headers.
As an added benefit, we can move the special-case code for part 0 into
the raw formatter.
2012-03-06 18:48:43 +00:00
|
|
|
|
|
|
|
|
|
--=-=-=
|
|
|
|
|
|
2011-05-23 00:46:44 -07:00
|
|
|
|
And this message is signed.
|
|
|
|
|
|
2011-05-28 14:16:30 -07:00
|
|
|
|
-Carl
|
show: Convert raw format to the new self-recursive style, properly support interior parts
This is fully compatible for root and leaf parts, but now has proper
support for interior parts. This requires some design decisions that
were guided by what I would want if I were to save a part.
Specifically:
- Leaf parts are printed without headers and with transfer decoding.
This is what makes sense for saving attachments. (Furthermore, the
transfer decoding is necessary since, without the headers, the
caller would not be able to interpret non-transfer-decoded output.)
- Message parts are printed with their message headers, but without
enclosing part headers. This is what makes sense for saving a
message as a whole (which is a message part) and for saving attached
messages. This is symmetric for whole messages and for attached
messages, though we special-case the whole message for performance
reasons (and corner-case correctness reasons: given malformed input,
GMime may not be able to reproduce it from the parsed
representation).
- Multipart parts are printed with their headers and all child parts.
It's not clear what the best thing to do for multipart is, but this
was the most natural to implement and can be justified because such
parts can't be interpreted without their headers.
As an added benefit, we can move the special-case code for part 0 into
the raw formatter.
2012-03-06 18:48:43 +00:00
|
|
|
|
|
|
|
|
|
--=-=-=--
|
2011-05-28 14:16:30 -07:00
|
|
|
|
EOF
|
|
|
|
|
test_expect_equal_file OUTPUT EXPECTED
|
2011-05-23 00:46:44 -07:00
|
|
|
|
|
2011-09-05 12:01:46 -07:00
|
|
|
|
test_begin_subtest "--format=raw --part=3, rfc822 part"
|
2011-05-28 14:16:30 -07:00
|
|
|
|
notmuch show --format=raw --part=3 'id:87liy5ap00.fsf@yoom.home.cworth.org' >OUTPUT
|
2011-09-05 12:01:46 -07:00
|
|
|
|
test_expect_equal_file OUTPUT embedded_message
|
|
|
|
|
|
2012-03-06 18:48:37 +00:00
|
|
|
|
test_begin_subtest "--format=raw --part=4, rfc822's multipart"
|
2011-09-05 12:01:46 -07:00
|
|
|
|
notmuch show --format=raw --part=4 'id:87liy5ap00.fsf@yoom.home.cworth.org' >OUTPUT
|
2011-05-28 14:16:30 -07:00
|
|
|
|
cat <<EOF >EXPECTED
|
show: Convert raw format to the new self-recursive style, properly support interior parts
This is fully compatible for root and leaf parts, but now has proper
support for interior parts. This requires some design decisions that
were guided by what I would want if I were to save a part.
Specifically:
- Leaf parts are printed without headers and with transfer decoding.
This is what makes sense for saving attachments. (Furthermore, the
transfer decoding is necessary since, without the headers, the
caller would not be able to interpret non-transfer-decoded output.)
- Message parts are printed with their message headers, but without
enclosing part headers. This is what makes sense for saving a
message as a whole (which is a message part) and for saving attached
messages. This is symmetric for whole messages and for attached
messages, though we special-case the whole message for performance
reasons (and corner-case correctness reasons: given malformed input,
GMime may not be able to reproduce it from the parsed
representation).
- Multipart parts are printed with their headers and all child parts.
It's not clear what the best thing to do for multipart is, but this
was the most natural to implement and can be justified because such
parts can't be interpreted without their headers.
As an added benefit, we can move the special-case code for part 0 into
the raw formatter.
2012-03-06 18:48:43 +00:00
|
|
|
|
From: Carl Worth <cworth@cworth.org>
|
|
|
|
|
To: cworth@cworth.org
|
|
|
|
|
Subject: html message
|
|
|
|
|
Date: Fri, 05 Jan 2001 15:42:57 +0000
|
|
|
|
|
User-Agent: Notmuch/0.5 (http://notmuchmail.org) Emacs/23.3.1 (i486-pc-linux-gnu)
|
|
|
|
|
Message-ID: <87liy5ap01.fsf@yoom.home.cworth.org>
|
|
|
|
|
MIME-Version: 1.0
|
|
|
|
|
Content-Type: multipart/alternative; boundary="==-=-=="
|
|
|
|
|
|
|
|
|
|
--==-=-==
|
|
|
|
|
Content-Type: text/html
|
|
|
|
|
|
2011-09-05 12:01:46 -07:00
|
|
|
|
<p>This is an embedded message, with a multipart/alternative part.</p>
|
show: Convert raw format to the new self-recursive style, properly support interior parts
This is fully compatible for root and leaf parts, but now has proper
support for interior parts. This requires some design decisions that
were guided by what I would want if I were to save a part.
Specifically:
- Leaf parts are printed without headers and with transfer decoding.
This is what makes sense for saving attachments. (Furthermore, the
transfer decoding is necessary since, without the headers, the
caller would not be able to interpret non-transfer-decoded output.)
- Message parts are printed with their message headers, but without
enclosing part headers. This is what makes sense for saving a
message as a whole (which is a message part) and for saving attached
messages. This is symmetric for whole messages and for attached
messages, though we special-case the whole message for performance
reasons (and corner-case correctness reasons: given malformed input,
GMime may not be able to reproduce it from the parsed
representation).
- Multipart parts are printed with their headers and all child parts.
It's not clear what the best thing to do for multipart is, but this
was the most natural to implement and can be justified because such
parts can't be interpreted without their headers.
As an added benefit, we can move the special-case code for part 0 into
the raw formatter.
2012-03-06 18:48:43 +00:00
|
|
|
|
|
|
|
|
|
--==-=-==
|
|
|
|
|
Content-Type: text/plain
|
|
|
|
|
|
2011-09-05 12:01:46 -07:00
|
|
|
|
This is an embedded message, with a multipart/alternative part.
|
show: Convert raw format to the new self-recursive style, properly support interior parts
This is fully compatible for root and leaf parts, but now has proper
support for interior parts. This requires some design decisions that
were guided by what I would want if I were to save a part.
Specifically:
- Leaf parts are printed without headers and with transfer decoding.
This is what makes sense for saving attachments. (Furthermore, the
transfer decoding is necessary since, without the headers, the
caller would not be able to interpret non-transfer-decoded output.)
- Message parts are printed with their message headers, but without
enclosing part headers. This is what makes sense for saving a
message as a whole (which is a message part) and for saving attached
messages. This is symmetric for whole messages and for attached
messages, though we special-case the whole message for performance
reasons (and corner-case correctness reasons: given malformed input,
GMime may not be able to reproduce it from the parsed
representation).
- Multipart parts are printed with their headers and all child parts.
It's not clear what the best thing to do for multipart is, but this
was the most natural to implement and can be justified because such
parts can't be interpreted without their headers.
As an added benefit, we can move the special-case code for part 0 into
the raw formatter.
2012-03-06 18:48:43 +00:00
|
|
|
|
|
|
|
|
|
--==-=-==--
|
2011-05-28 14:16:30 -07:00
|
|
|
|
EOF
|
|
|
|
|
test_expect_equal_file OUTPUT EXPECTED
|
2011-05-23 00:46:44 -07:00
|
|
|
|
|
2011-09-05 12:01:46 -07:00
|
|
|
|
test_begin_subtest "--format=raw --part=5, rfc822's html part"
|
|
|
|
|
notmuch show --format=raw --part=5 'id:87liy5ap00.fsf@yoom.home.cworth.org' >OUTPUT
|
2011-05-28 14:16:30 -07:00
|
|
|
|
cat <<EOF >EXPECTED
|
2011-09-05 12:01:46 -07:00
|
|
|
|
<p>This is an embedded message, with a multipart/alternative part.</p>
|
2011-05-28 14:16:30 -07:00
|
|
|
|
EOF
|
|
|
|
|
test_expect_equal_file OUTPUT EXPECTED
|
2011-05-23 00:46:44 -07:00
|
|
|
|
|
2011-09-05 12:01:46 -07:00
|
|
|
|
test_begin_subtest "--format=raw --part=6, rfc822's text part"
|
|
|
|
|
notmuch show --format=raw --part=6 'id:87liy5ap00.fsf@yoom.home.cworth.org' >OUTPUT
|
|
|
|
|
cat <<EOF >EXPECTED
|
|
|
|
|
This is an embedded message, with a multipart/alternative part.
|
|
|
|
|
EOF
|
|
|
|
|
test_expect_equal_file OUTPUT EXPECTED
|
|
|
|
|
|
|
|
|
|
test_begin_subtest "--format=raw --part=7, inline attachment"
|
|
|
|
|
notmuch show --format=raw --part=7 'id:87liy5ap00.fsf@yoom.home.cworth.org' >OUTPUT
|
2011-05-28 14:16:30 -07:00
|
|
|
|
cat <<EOF >EXPECTED
|
|
|
|
|
This is a text attachment.
|
|
|
|
|
EOF
|
|
|
|
|
test_expect_equal_file OUTPUT EXPECTED
|
2011-05-23 00:46:44 -07:00
|
|
|
|
|
2011-09-05 12:01:46 -07:00
|
|
|
|
test_begin_subtest "--format=raw --part=8, plain text part"
|
|
|
|
|
notmuch show --format=raw --part=8 'id:87liy5ap00.fsf@yoom.home.cworth.org' >OUTPUT
|
2011-05-28 14:16:30 -07:00
|
|
|
|
cat <<EOF >EXPECTED
|
|
|
|
|
And this message is signed.
|
2011-05-23 00:46:44 -07:00
|
|
|
|
|
2011-05-28 14:16:30 -07:00
|
|
|
|
-Carl
|
|
|
|
|
EOF
|
|
|
|
|
test_expect_equal_file OUTPUT EXPECTED
|
2011-05-23 00:46:44 -07:00
|
|
|
|
|
2011-09-05 12:01:46 -07:00
|
|
|
|
test_begin_subtest "--format=raw --part=9, pgp signature (unverified)"
|
|
|
|
|
notmuch show --format=raw --part=9 'id:87liy5ap00.fsf@yoom.home.cworth.org' >OUTPUT
|
2011-05-28 14:16:30 -07:00
|
|
|
|
# output should *not* include newline
|
|
|
|
|
echo >>OUTPUT
|
|
|
|
|
cat <<EOF >EXPECTED
|
|
|
|
|
-----BEGIN PGP SIGNATURE-----
|
2011-05-23 00:46:44 -07:00
|
|
|
|
Version: GnuPG v1.4.11 (GNU/Linux)
|
|
|
|
|
|
|
|
|
|
iEYEARECAAYFAk3SA/gACgkQ6JDdNq8qSWj0sACghqVJEQJUs3yV8zbTzhgnSIcD
|
|
|
|
|
W6cAmQE4dcYrx/LPLtYLZm1jsGauE5hE
|
|
|
|
|
=zkga
|
2011-05-28 14:16:30 -07:00
|
|
|
|
-----END PGP SIGNATURE-----
|
|
|
|
|
EOF
|
|
|
|
|
test_expect_equal_file OUTPUT EXPECTED
|
2011-05-23 00:46:44 -07:00
|
|
|
|
|
|
|
|
|
test_expect_success \
|
2011-09-05 12:01:46 -07:00
|
|
|
|
"--format=raw --part=10, no part, expect error" \
|
2011-05-23 00:46:44 -07:00
|
|
|
|
"notmuch show --format=raw --part=8 'id:87liy5ap00.fsf@yoom.home.cworth.org'"
|
|
|
|
|
|
|
|
|
|
test_begin_subtest "--format=mbox"
|
2011-05-28 14:16:30 -07:00
|
|
|
|
notmuch show --format=mbox 'id:87liy5ap00.fsf@yoom.home.cworth.org' >OUTPUT
|
|
|
|
|
printf "From cworth@cworth.org Fri Jan 5 15:43:57 2001\n" >EXPECTED
|
|
|
|
|
cat "${MAIL_DIR}"/multipart >>EXPECTED
|
|
|
|
|
# mbox output is expected to include a blank line
|
|
|
|
|
echo >>EXPECTED
|
|
|
|
|
test_expect_equal_file OUTPUT EXPECTED
|
2011-05-23 00:46:44 -07:00
|
|
|
|
|
|
|
|
|
test_expect_success \
|
|
|
|
|
"--format=mbox --part=1, incompatible, expect error" \
|
|
|
|
|
"! notmuch show --format=mbox --part=1 'id:87liy5ap00.fsf@yoom.home.cworth.org'"
|
|
|
|
|
|
2011-05-24 10:56:38 -07:00
|
|
|
|
test_begin_subtest "'notmuch reply' to a multipart message"
|
2011-05-28 14:16:30 -07:00
|
|
|
|
notmuch reply 'id:87liy5ap00.fsf@yoom.home.cworth.org' >OUTPUT
|
|
|
|
|
cat <<EOF >EXPECTED
|
|
|
|
|
From: Notmuch Test Suite <test_suite@notmuchmail.org>
|
2011-05-24 10:56:38 -07:00
|
|
|
|
Subject: Re: Multipart message
|
|
|
|
|
To: Carl Worth <cworth@cworth.org>, cworth@cworth.org
|
|
|
|
|
In-Reply-To: <87liy5ap00.fsf@yoom.home.cworth.org>
|
|
|
|
|
References: <87liy5ap00.fsf@yoom.home.cworth.org>
|
|
|
|
|
|
2011-06-05 17:29:26 -07:00
|
|
|
|
On Fri, 05 Jan 2001 15:43:57 +0000, Carl Worth <cworth@cworth.org> wrote:
|
2011-09-05 12:01:46 -07:00
|
|
|
|
> From: Carl Worth <cworth@cworth.org>
|
|
|
|
|
> To: cworth@cworth.org
|
|
|
|
|
> Subject: html message
|
|
|
|
|
> Date: Fri, 05 Jan 2001 15:42:57 +0000
|
|
|
|
|
>
|
2011-05-24 10:56:38 -07:00
|
|
|
|
Non-text part: text/html
|
2011-09-05 12:01:46 -07:00
|
|
|
|
> This is an embedded message, with a multipart/alternative part.
|
2011-05-24 10:56:38 -07:00
|
|
|
|
> This is a text attachment.
|
|
|
|
|
> And this message is signed.
|
|
|
|
|
>
|
|
|
|
|
> -Carl
|
2011-05-28 14:16:30 -07:00
|
|
|
|
EOF
|
|
|
|
|
test_expect_equal_file OUTPUT EXPECTED
|
2011-05-24 10:56:38 -07:00
|
|
|
|
|
2012-03-18 10:32:33 -06:00
|
|
|
|
test_begin_subtest "'notmuch reply' to a multipart message with json format"
|
|
|
|
|
notmuch reply --format=json 'id:87liy5ap00.fsf@yoom.home.cworth.org' | notmuch_json_show_sanitize >OUTPUT
|
2013-12-08 16:39:05 +08:00
|
|
|
|
notmuch_json_show_sanitize <<EOF >EXPECTED
|
2012-03-18 10:32:33 -06:00
|
|
|
|
{"reply-headers": {"Subject": "Re: Multipart message",
|
|
|
|
|
"From": "Notmuch Test Suite <test_suite@notmuchmail.org>",
|
2012-08-02 21:14:47 -04:00
|
|
|
|
"To": "Carl Worth <cworth@cworth.org>, cworth@cworth.org",
|
2012-03-18 10:32:33 -06:00
|
|
|
|
"In-reply-to": "<87liy5ap00.fsf@yoom.home.cworth.org>",
|
2013-08-16 11:35:42 -04:00
|
|
|
|
"References": "<87liy5ap00.fsf@yoom.home.cworth.org>"},
|
2012-03-18 10:32:33 -06:00
|
|
|
|
"original": {"id": "XXXXX",
|
|
|
|
|
"match": false,
|
|
|
|
|
"excluded": false,
|
|
|
|
|
"filename": "YYYYY",
|
|
|
|
|
"timestamp": 978709437,
|
|
|
|
|
"date_relative": "2001-01-05",
|
|
|
|
|
"tags": ["attachment","inbox","signed","unread"],
|
|
|
|
|
"headers": {"Subject": "Multipart message",
|
|
|
|
|
"From": "Carl Worth <cworth@cworth.org>",
|
|
|
|
|
"To": "cworth@cworth.org",
|
2012-08-02 21:14:47 -04:00
|
|
|
|
"Date": "Fri, 05 Jan 2001 15:43:57 +0000"},
|
2012-03-18 10:32:33 -06:00
|
|
|
|
"body": [{"id": 1,
|
|
|
|
|
"content-type": "multipart/signed",
|
|
|
|
|
"content": [{"id": 2,
|
|
|
|
|
"content-type": "multipart/mixed",
|
|
|
|
|
"content": [{"id": 3,
|
|
|
|
|
"content-type": "message/rfc822",
|
|
|
|
|
"content": [{"headers": {"Subject": "html message",
|
|
|
|
|
"From": "Carl Worth <cworth@cworth.org>",
|
|
|
|
|
"To": "cworth@cworth.org",
|
2012-08-02 21:14:47 -04:00
|
|
|
|
"Date": "Fri, 05 Jan 2001 15:42:57 +0000"},
|
2012-03-18 10:32:33 -06:00
|
|
|
|
"body": [{"id": 4,
|
|
|
|
|
"content-type": "multipart/alternative",
|
|
|
|
|
"content": [{"id": 5,
|
2012-12-15 14:06:45 +11:00
|
|
|
|
"content-type": "text/html",
|
|
|
|
|
"content-length": 71},
|
2012-03-18 10:32:33 -06:00
|
|
|
|
{"id": 6,
|
|
|
|
|
"content-type": "text/plain",
|
2012-08-02 21:14:47 -04:00
|
|
|
|
"content": "This is an embedded message, with a multipart/alternative part.\n"}]}]}]},
|
2012-03-18 10:32:33 -06:00
|
|
|
|
{"id": 7,
|
|
|
|
|
"content-type": "text/plain",
|
2012-12-15 14:06:42 +11:00
|
|
|
|
"filename": "attachment",
|
2012-03-18 10:32:33 -06:00
|
|
|
|
"content": "This is a text attachment.\n"},
|
|
|
|
|
{"id": 8,
|
|
|
|
|
"content-type": "text/plain",
|
|
|
|
|
"content": "And this message is signed.\n\n-Carl\n"}]},
|
|
|
|
|
{"id": 9,
|
2012-12-15 14:06:45 +11:00
|
|
|
|
"content-type": "application/pgp-signature",
|
|
|
|
|
"content-length": 197}]}]}}
|
2012-03-18 10:32:33 -06:00
|
|
|
|
EOF
|
2012-08-02 21:14:47 -04:00
|
|
|
|
test_expect_equal_json "$(cat OUTPUT)" "$(cat EXPECTED)"
|
2012-03-18 10:32:33 -06:00
|
|
|
|
|
2011-05-31 15:39:26 -07:00
|
|
|
|
test_begin_subtest "'notmuch show --part' does not corrupt a part with CRLF pair"
|
|
|
|
|
notmuch show --format=raw --part=3 id:base64-part-with-crlf > crlf.out
|
|
|
|
|
echo -n -e "\xEF\x0D\x0A" > crlf.expected
|
|
|
|
|
test_expect_equal_file crlf.out crlf.expected
|
2011-05-24 10:56:38 -07:00
|
|
|
|
|
2013-08-24 18:29:30 +03:00
|
|
|
|
|
|
|
|
|
# The ISO-8859-1 encoding of U+00BD is a single byte: octal 275
|
|
|
|
|
# (Portability note: Dollar-Single ($'...', ANSI C-style escape sequences)
|
|
|
|
|
# quoting works on bash, ksh, zsh, *BSD sh but not on dash, ash nor busybox sh)
|
|
|
|
|
readonly u_00bd_latin1=$'\275'
|
|
|
|
|
|
|
|
|
|
# The Unicode fraction symbol 1/2 is U+00BD and is encoded
|
|
|
|
|
# in UTF-8 as two bytes: octal 302 275
|
|
|
|
|
readonly u_00bd_utf8=$'\302\275'
|
|
|
|
|
|
|
|
|
|
cat <<EOF > ${MAIL_DIR}/include-html
|
|
|
|
|
From: A <a@example.com>
|
|
|
|
|
To: B <b@example.com>
|
|
|
|
|
Subject: html message
|
|
|
|
|
Date: Sat, 01 January 2000 00:00:00 +0000
|
|
|
|
|
Message-ID: <htmlmessage>
|
|
|
|
|
MIME-Version: 1.0
|
|
|
|
|
Content-Type: multipart/alternative; boundary="==-=="
|
|
|
|
|
|
|
|
|
|
--==-==
|
|
|
|
|
Content-Type: text/html; charset=UTF-8
|
|
|
|
|
|
|
|
|
|
<p>0.5 equals ${u_00bd_utf8}</p>
|
|
|
|
|
|
|
|
|
|
--==-==
|
|
|
|
|
Content-Type: text/html; charset=ISO-8859-1
|
|
|
|
|
|
|
|
|
|
<p>0.5 equals ${u_00bd_latin1}</p>
|
|
|
|
|
|
|
|
|
|
--==-==
|
|
|
|
|
Content-Type: text/plain; charset=UTF-8
|
|
|
|
|
|
|
|
|
|
0.5 equals ${u_00bd_utf8}
|
|
|
|
|
|
|
|
|
|
--==-==--
|
|
|
|
|
EOF
|
|
|
|
|
|
|
|
|
|
notmuch new > /dev/null
|
|
|
|
|
|
|
|
|
|
cat_expected_head ()
|
|
|
|
|
{
|
|
|
|
|
cat <<EOF
|
|
|
|
|
[[[{"id": "htmlmessage", "match":true, "excluded": false, "date_relative":"2000-01-01",
|
|
|
|
|
"timestamp": 946684800,
|
|
|
|
|
"filename": "${MAIL_DIR}/include-html",
|
|
|
|
|
"tags": ["inbox", "unread"],
|
|
|
|
|
"headers": { "Date": "Sat, 01 Jan 2000 00:00:00 +0000", "From": "A <a@example.com>",
|
|
|
|
|
"Subject": "html message", "To": "B <b@example.com>"},
|
|
|
|
|
"body": [{
|
|
|
|
|
"content-type": "multipart/alternative", "id": 1,
|
|
|
|
|
EOF
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
cat_expected_head > EXPECTED.nohtml
|
|
|
|
|
cat <<EOF >> EXPECTED.nohtml
|
|
|
|
|
"content": [
|
|
|
|
|
{ "id": 2, "content-charset": "UTF-8", "content-length": 21, "content-type": "text/html"},
|
|
|
|
|
{ "id": 3, "content-charset": "ISO-8859-1", "content-length": 20, "content-type": "text/html"},
|
|
|
|
|
{ "id": 4, "content-type": "text/plain", "content": "0.5 equals \\u00bd\\n"}
|
|
|
|
|
]}]},[]]]]
|
|
|
|
|
EOF
|
|
|
|
|
|
|
|
|
|
# Both the UTF-8 and ISO-8859-1 part should have U+00BD
|
|
|
|
|
cat_expected_head > EXPECTED.withhtml
|
|
|
|
|
cat <<EOF >> EXPECTED.withhtml
|
|
|
|
|
"content": [
|
|
|
|
|
{ "id": 2, "content-type": "text/html", "content": "<p>0.5 equals \\u00bd</p>\\n"},
|
|
|
|
|
{ "id": 3, "content-type": "text/html", "content": "<p>0.5 equals \\u00bd</p>\\n"},
|
|
|
|
|
{ "id": 4, "content-type": "text/plain", "content": "0.5 equals \\u00bd\\n"}
|
|
|
|
|
]}]},[]]]]
|
|
|
|
|
EOF
|
|
|
|
|
|
|
|
|
|
test_begin_subtest "html parts excluded by default"
|
|
|
|
|
notmuch show --format=json id:htmlmessage > OUTPUT
|
|
|
|
|
test_expect_equal_json "$(cat OUTPUT)" "$(cat EXPECTED.nohtml)"
|
|
|
|
|
|
|
|
|
|
test_begin_subtest "html parts included"
|
|
|
|
|
notmuch show --format=json --include-html id:htmlmessage > OUTPUT
|
|
|
|
|
test_expect_equal_json "$(cat OUTPUT)" "$(cat EXPECTED.withhtml)"
|
|
|
|
|
|
|
|
|
|
test_done
|