cli/reply: return internet address list from get header funcs

Pass in GMimeMessage to simplify To/Cc/Bcc headers. We'll eventually
remove the notmuch message passing altogether, but keep both for now
to not make too big changes at once.

Getting the headers from GMimeMessage using GMime functions fixes the
error on duplicate Cc headers reported by Daniel Kahn Gillmor
<dkg@fifthhorseman.net> in id:87d1ngv95p.fsf@alice.fifthhorseman.net.

Get rid of an intermediate function.

The small annoyance is the ownership differences in the address lists.
This commit is contained in:
Jani Nikula 2016-09-13 20:14:19 +03:00 committed by David Bremner
parent ca82d481a1
commit 536b1f9df9
2 changed files with 30 additions and 44 deletions

View file

@ -227,31 +227,6 @@ scan_address_list (InternetAddressList *list,
return n; return n;
} }
/* Scan addresses in 'recipients'.
*
* See the documentation of scan_address_list() above. This function
* does exactly the same, but converts 'recipients' to an
* InternetAddressList first.
*/
static unsigned int
scan_address_string (const char *recipients,
notmuch_config_t *config,
GMimeMessage *message,
GMimeRecipientType type,
const char **user_from)
{
InternetAddressList *list;
if (recipients == NULL)
return 0;
list = internet_address_list_parse_string (recipients);
if (list == NULL)
return 0;
return scan_address_list (list, config, message, type, user_from);
}
/* Does the address in the Reply-To header of 'message' already appear /* Does the address in the Reply-To header of 'message' already appear
* in either the 'To' or 'Cc' header of the message? * in either the 'To' or 'Cc' header of the message?
*/ */
@ -287,11 +262,12 @@ reply_to_header_is_redundant (notmuch_message_t *message, const char *reply_to)
return 0; return 0;
} }
static const char *get_sender(notmuch_message_t *message) static InternetAddressList *get_sender(notmuch_message_t *message,
GMimeMessage *mime_message)
{ {
const char *reply_to; const char *reply_to;
reply_to = notmuch_message_get_header (message, "reply-to"); reply_to = g_mime_message_get_reply_to (mime_message);
if (reply_to && *reply_to) { if (reply_to && *reply_to) {
/* /*
* Some mailing lists munge the Reply-To header despite it * Some mailing lists munge the Reply-To header despite it
@ -307,25 +283,32 @@ static const char *get_sender(notmuch_message_t *message)
* will always appear in the reply if reply_all is true. * will always appear in the reply if reply_all is true.
*/ */
if (! reply_to_header_is_redundant (message, reply_to)) if (! reply_to_header_is_redundant (message, reply_to))
return reply_to; return internet_address_list_parse_string (reply_to);
} }
return notmuch_message_get_header (message, "from"); return internet_address_list_parse_string (
g_mime_message_get_sender (mime_message));
} }
static const char *get_to(notmuch_message_t *message) static InternetAddressList *get_to(unused(notmuch_message_t *message),
GMimeMessage *mime_message)
{ {
return notmuch_message_get_header (message, "to"); return g_mime_message_get_recipients (mime_message,
GMIME_RECIPIENT_TYPE_TO);
} }
static const char *get_cc(notmuch_message_t *message) static InternetAddressList *get_cc(unused(notmuch_message_t *message),
GMimeMessage *mime_message)
{ {
return notmuch_message_get_header (message, "cc"); return g_mime_message_get_recipients (mime_message,
GMIME_RECIPIENT_TYPE_CC);
} }
static const char *get_bcc(notmuch_message_t *message) static InternetAddressList *get_bcc(unused(notmuch_message_t *message),
GMimeMessage *mime_message)
{ {
return notmuch_message_get_header (message, "bcc"); return g_mime_message_get_recipients (mime_message,
GMIME_RECIPIENT_TYPE_BCC);
} }
/* Augment the recipients of 'reply' from the "Reply-to:", "From:", /* Augment the recipients of 'reply' from the "Reply-to:", "From:",
@ -344,10 +327,12 @@ static const char *
add_recipients_from_message (GMimeMessage *reply, add_recipients_from_message (GMimeMessage *reply,
notmuch_config_t *config, notmuch_config_t *config,
notmuch_message_t *message, notmuch_message_t *message,
GMimeMessage *mime_message,
notmuch_bool_t reply_all) notmuch_bool_t reply_all)
{ {
struct { struct {
const char * (*get_header)(notmuch_message_t *message); InternetAddressList * (*get_header)(notmuch_message_t *message,
GMimeMessage *mime_message);
GMimeRecipientType recipient_type; GMimeRecipientType recipient_type;
} reply_to_map[] = { } reply_to_map[] = {
{ get_sender, GMIME_RECIPIENT_TYPE_TO }, { get_sender, GMIME_RECIPIENT_TYPE_TO },
@ -360,11 +345,11 @@ add_recipients_from_message (GMimeMessage *reply,
unsigned int n = 0; unsigned int n = 0;
for (i = 0; i < ARRAY_SIZE (reply_to_map); i++) { for (i = 0; i < ARRAY_SIZE (reply_to_map); i++) {
const char *recipients; InternetAddressList *recipients;
recipients = reply_to_map[i].get_header (message); recipients = reply_to_map[i].get_header (message, mime_message);
n += scan_address_string (recipients, config, reply, n += scan_address_list (recipients, config, reply,
reply_to_map[i].recipient_type, &from_addr); reply_to_map[i].recipient_type, &from_addr);
if (!reply_all && n) { if (!reply_all && n) {
@ -536,6 +521,7 @@ static GMimeMessage *
create_reply_message(void *ctx, create_reply_message(void *ctx,
notmuch_config_t *config, notmuch_config_t *config,
notmuch_message_t *message, notmuch_message_t *message,
GMimeMessage *mime_message,
notmuch_bool_t reply_all, notmuch_bool_t reply_all,
notmuch_bool_t limited) notmuch_bool_t limited)
{ {
@ -566,8 +552,8 @@ create_reply_message(void *ctx,
g_mime_object_set_header (GMIME_OBJECT (reply), "References", references); g_mime_object_set_header (GMIME_OBJECT (reply), "References", references);
from_addr = add_recipients_from_message (reply, config, from_addr = add_recipients_from_message (reply, config, message,
message, reply_all); mime_message, reply_all);
/* The above is all that is needed for limited headers. */ /* The above is all that is needed for limited headers. */
if (limited) if (limited)
@ -666,7 +652,8 @@ static int do_reply(notmuch_config_t *config,
if (mime_node_open (config, message, &params->crypto, &node)) if (mime_node_open (config, message, &params->crypto, &node))
return 1; return 1;
reply = create_reply_message (config, config, message, reply_all, reply = create_reply_message (config, config, message,
GMIME_MESSAGE (node->part), reply_all,
format == FORMAT_HEADERS_ONLY); format == FORMAT_HEADERS_ONLY);
if (!reply) if (!reply)
return 1; return 1;

View file

@ -254,7 +254,6 @@ test_expect_equal_json "$output" '
}' }'
test_begin_subtest "Reply to a message with multiple Cc headers" test_begin_subtest "Reply to a message with multiple Cc headers"
test_subtest_known_broken
add_email_corpus broken add_email_corpus broken
output=$(notmuch reply id:multiple-cc@example.org) output=$(notmuch reply id:multiple-cc@example.org)
test_expect_equal "$output" "From: Notmuch Test Suite <test_suite@notmuchmail.org> test_expect_equal "$output" "From: Notmuch Test Suite <test_suite@notmuchmail.org>