mirror of
https://git.notmuchmail.org/git/notmuch
synced 2025-01-03 15:21:41 +01:00
lib: replace the header parser with gmime
The notmuch library includes a full blown message header parser. Yet the same message headers are parsed by gmime during indexing. Switch to gmime parsing completely. These are the main changes: * Gmime stops header parsing at the first invalid header, and presumes the message body starts from there. The current parser is quite liberal in accepting broken headers. The change means we will be much pickier about accepting invalid messages. * The current parser converts tabs used in header folding to spaces. Gmime preserve the tabs. Due to a broken python library used in mailman, there are plenty of mailing lists that produce headers with tabs in header folding, and we'll see plenty of tabs. (This change has been mitigated in preparatory patches.) * For pure header parsing, the current parser is likely faster than gmime, which parses the whole message rather than just the headers. Since we parse the message and its headers using gmime for indexing anyway, this avoids and extra header parsing round when adding new messages. In case of duplicate messages, we'll end up parsing the full message although just headers would be sufficient. All in all this should still speed up 'notmuch new'. * Calls to notmuch_message_get_header() may be slightly slower than previously for headers that are not indexed in the database, due to parsing of the whole message. Within the notmuch code base, notmuch reply is the only such user.
This commit is contained in:
parent
6812136bf5
commit
473930bb6f
4 changed files with 225 additions and 350 deletions
|
@ -1971,15 +1971,10 @@ notmuch_database_add_message (notmuch_database_t *notmuch,
|
||||||
if (ret)
|
if (ret)
|
||||||
goto DONE;
|
goto DONE;
|
||||||
|
|
||||||
notmuch_message_file_restrict_headers (message_file,
|
/* Parse message up front to get better error status. */
|
||||||
"date",
|
ret = _notmuch_message_file_parse (message_file);
|
||||||
"from",
|
if (ret)
|
||||||
"in-reply-to",
|
goto DONE;
|
||||||
"message-id",
|
|
||||||
"references",
|
|
||||||
"subject",
|
|
||||||
"to",
|
|
||||||
(char *) NULL);
|
|
||||||
|
|
||||||
try {
|
try {
|
||||||
/* Before we do any real work, (especially before doing a
|
/* Before we do any real work, (especially before doing a
|
||||||
|
@ -2066,7 +2061,7 @@ notmuch_database_add_message (notmuch_database_t *notmuch,
|
||||||
date = notmuch_message_file_get_header (message_file, "date");
|
date = notmuch_message_file_get_header (message_file, "date");
|
||||||
_notmuch_message_set_header_values (message, date, from, subject);
|
_notmuch_message_set_header_values (message, date, from, subject);
|
||||||
|
|
||||||
ret = _notmuch_message_index_file (message, filename);
|
ret = _notmuch_message_index_file (message, message_file);
|
||||||
if (ret)
|
if (ret)
|
||||||
goto DONE;
|
goto DONE;
|
||||||
} else {
|
} else {
|
||||||
|
|
54
lib/index.cc
54
lib/index.cc
|
@ -425,45 +425,17 @@ _index_mime_part (notmuch_message_t *message,
|
||||||
|
|
||||||
notmuch_status_t
|
notmuch_status_t
|
||||||
_notmuch_message_index_file (notmuch_message_t *message,
|
_notmuch_message_index_file (notmuch_message_t *message,
|
||||||
const char *filename)
|
notmuch_message_file_t *message_file)
|
||||||
{
|
{
|
||||||
GMimeStream *stream = NULL;
|
GMimeMessage *mime_message;
|
||||||
GMimeParser *parser = NULL;
|
|
||||||
GMimeMessage *mime_message = NULL;
|
|
||||||
InternetAddressList *addresses;
|
InternetAddressList *addresses;
|
||||||
FILE *file = NULL;
|
|
||||||
const char *from, *subject;
|
const char *from, *subject;
|
||||||
notmuch_status_t ret = NOTMUCH_STATUS_SUCCESS;
|
notmuch_status_t status;
|
||||||
static int initialized = 0;
|
|
||||||
char from_buf[5];
|
|
||||||
|
|
||||||
if (! initialized) {
|
status = _notmuch_message_file_get_mime_message (message_file,
|
||||||
g_mime_init (GMIME_ENABLE_RFC2047_WORKAROUNDS);
|
&mime_message);
|
||||||
initialized = 1;
|
if (status)
|
||||||
}
|
return status;
|
||||||
|
|
||||||
file = fopen (filename, "r");
|
|
||||||
if (! file) {
|
|
||||||
fprintf (stderr, "Error opening %s: %s\n", filename, strerror (errno));
|
|
||||||
ret = NOTMUCH_STATUS_FILE_ERROR;
|
|
||||||
goto DONE;
|
|
||||||
}
|
|
||||||
|
|
||||||
/* Is this mbox? */
|
|
||||||
if (fread (from_buf, sizeof (from_buf), 1, file) == 1 &&
|
|
||||||
strncmp (from_buf, "From ", 5) == 0) {
|
|
||||||
ret = NOTMUCH_STATUS_FILE_NOT_EMAIL;
|
|
||||||
goto DONE;
|
|
||||||
}
|
|
||||||
rewind (file);
|
|
||||||
|
|
||||||
/* Evil GMime steals my FILE* here so I won't fclose it. */
|
|
||||||
stream = g_mime_stream_file_new (file);
|
|
||||||
|
|
||||||
parser = g_mime_parser_new_with_stream (stream);
|
|
||||||
g_mime_parser_set_scan_from (parser, FALSE);
|
|
||||||
|
|
||||||
mime_message = g_mime_parser_construct_message (parser);
|
|
||||||
|
|
||||||
from = g_mime_message_get_sender (mime_message);
|
from = g_mime_message_get_sender (mime_message);
|
||||||
|
|
||||||
|
@ -484,15 +456,5 @@ _notmuch_message_index_file (notmuch_message_t *message,
|
||||||
|
|
||||||
_index_mime_part (message, g_mime_message_get_mime_part (mime_message));
|
_index_mime_part (message, g_mime_message_get_mime_part (mime_message));
|
||||||
|
|
||||||
DONE:
|
return NOTMUCH_STATUS_SUCCESS;
|
||||||
if (mime_message)
|
|
||||||
g_object_unref (mime_message);
|
|
||||||
|
|
||||||
if (parser)
|
|
||||||
g_object_unref (parser);
|
|
||||||
|
|
||||||
if (stream)
|
|
||||||
g_object_unref (stream);
|
|
||||||
|
|
||||||
return ret;
|
|
||||||
}
|
}
|
||||||
|
|
|
@ -26,30 +26,15 @@
|
||||||
|
|
||||||
#include <glib.h> /* GHashTable */
|
#include <glib.h> /* GHashTable */
|
||||||
|
|
||||||
typedef struct {
|
|
||||||
char *str;
|
|
||||||
size_t size;
|
|
||||||
size_t len;
|
|
||||||
} header_value_closure_t;
|
|
||||||
|
|
||||||
struct _notmuch_message_file {
|
struct _notmuch_message_file {
|
||||||
/* File object */
|
/* File object */
|
||||||
FILE *file;
|
FILE *file;
|
||||||
|
char *filename;
|
||||||
|
|
||||||
/* Header storage */
|
/* Cache for decoded headers */
|
||||||
int restrict_headers;
|
|
||||||
GHashTable *headers;
|
GHashTable *headers;
|
||||||
int broken_headers;
|
|
||||||
int good_headers;
|
|
||||||
size_t header_size; /* Length of full message header in bytes. */
|
|
||||||
|
|
||||||
/* Parsing state */
|
GMimeMessage *message;
|
||||||
char *line;
|
|
||||||
size_t line_size;
|
|
||||||
header_value_closure_t value;
|
|
||||||
|
|
||||||
int parsing_started;
|
|
||||||
int parsing_finished;
|
|
||||||
};
|
};
|
||||||
|
|
||||||
static int
|
static int
|
||||||
|
@ -76,15 +61,12 @@ strcase_hash (const void *ptr)
|
||||||
static int
|
static int
|
||||||
_notmuch_message_file_destructor (notmuch_message_file_t *message)
|
_notmuch_message_file_destructor (notmuch_message_file_t *message)
|
||||||
{
|
{
|
||||||
if (message->line)
|
|
||||||
free (message->line);
|
|
||||||
|
|
||||||
if (message->value.size)
|
|
||||||
free (message->value.str);
|
|
||||||
|
|
||||||
if (message->headers)
|
if (message->headers)
|
||||||
g_hash_table_destroy (message->headers);
|
g_hash_table_destroy (message->headers);
|
||||||
|
|
||||||
|
if (message->message)
|
||||||
|
g_object_unref (message->message);
|
||||||
|
|
||||||
if (message->file)
|
if (message->file)
|
||||||
fclose (message->file);
|
fclose (message->file);
|
||||||
|
|
||||||
|
@ -102,20 +84,17 @@ _notmuch_message_file_open_ctx (void *ctx, const char *filename)
|
||||||
if (unlikely (message == NULL))
|
if (unlikely (message == NULL))
|
||||||
return NULL;
|
return NULL;
|
||||||
|
|
||||||
|
/* Only needed for error messages during parsing. */
|
||||||
|
message->filename = talloc_strdup (message, filename);
|
||||||
|
if (message->filename == NULL)
|
||||||
|
goto FAIL;
|
||||||
|
|
||||||
talloc_set_destructor (message, _notmuch_message_file_destructor);
|
talloc_set_destructor (message, _notmuch_message_file_destructor);
|
||||||
|
|
||||||
message->file = fopen (filename, "r");
|
message->file = fopen (filename, "r");
|
||||||
if (message->file == NULL)
|
if (message->file == NULL)
|
||||||
goto FAIL;
|
goto FAIL;
|
||||||
|
|
||||||
message->headers = g_hash_table_new_full (strcase_hash,
|
|
||||||
strcase_equal,
|
|
||||||
free,
|
|
||||||
g_free);
|
|
||||||
|
|
||||||
message->parsing_started = 0;
|
|
||||||
message->parsing_finished = 0;
|
|
||||||
|
|
||||||
return message;
|
return message;
|
||||||
|
|
||||||
FAIL:
|
FAIL:
|
||||||
|
@ -137,264 +116,202 @@ notmuch_message_file_close (notmuch_message_file_t *message)
|
||||||
talloc_free (message);
|
talloc_free (message);
|
||||||
}
|
}
|
||||||
|
|
||||||
void
|
static notmuch_bool_t
|
||||||
notmuch_message_file_restrict_headersv (notmuch_message_file_t *message,
|
is_mbox (FILE *file)
|
||||||
va_list va_headers)
|
|
||||||
{
|
{
|
||||||
char *header;
|
char from_buf[5];
|
||||||
|
notmuch_bool_t ret = FALSE;
|
||||||
|
|
||||||
if (message->parsing_started)
|
/* Is this mbox? */
|
||||||
INTERNAL_ERROR ("notmuch_message_file_restrict_headers called after parsing has started");
|
if (fread (from_buf, sizeof (from_buf), 1, file) == 1 &&
|
||||||
|
strncmp (from_buf, "From ", 5) == 0)
|
||||||
|
ret = TRUE;
|
||||||
|
|
||||||
while (1) {
|
rewind (file);
|
||||||
header = va_arg (va_headers, char*);
|
|
||||||
if (header == NULL)
|
|
||||||
break;
|
|
||||||
g_hash_table_insert (message->headers,
|
|
||||||
xstrdup (header), NULL);
|
|
||||||
}
|
|
||||||
|
|
||||||
message->restrict_headers = 1;
|
return ret;
|
||||||
}
|
}
|
||||||
|
|
||||||
void
|
notmuch_status_t
|
||||||
notmuch_message_file_restrict_headers (notmuch_message_file_t *message, ...)
|
_notmuch_message_file_parse (notmuch_message_file_t *message)
|
||||||
{
|
{
|
||||||
va_list va_headers;
|
GMimeStream *stream;
|
||||||
|
GMimeParser *parser;
|
||||||
va_start (va_headers, message);
|
notmuch_status_t status = NOTMUCH_STATUS_SUCCESS;
|
||||||
|
|
||||||
notmuch_message_file_restrict_headersv (message, va_headers);
|
|
||||||
}
|
|
||||||
|
|
||||||
static void
|
|
||||||
copy_header_unfolding (header_value_closure_t *value,
|
|
||||||
const char *chunk)
|
|
||||||
{
|
|
||||||
char *last;
|
|
||||||
|
|
||||||
if (chunk == NULL)
|
|
||||||
return;
|
|
||||||
|
|
||||||
while (*chunk == ' ' || *chunk == '\t')
|
|
||||||
chunk++;
|
|
||||||
|
|
||||||
if (value->len + 1 + strlen (chunk) + 1 > value->size) {
|
|
||||||
unsigned int new_size = value->size;
|
|
||||||
if (value->size == 0)
|
|
||||||
new_size = strlen (chunk) + 1;
|
|
||||||
else
|
|
||||||
while (value->len + 1 + strlen (chunk) + 1 > new_size)
|
|
||||||
new_size *= 2;
|
|
||||||
value->str = xrealloc (value->str, new_size);
|
|
||||||
value->size = new_size;
|
|
||||||
}
|
|
||||||
|
|
||||||
last = value->str + value->len;
|
|
||||||
if (value->len) {
|
|
||||||
*last = ' ';
|
|
||||||
last++;
|
|
||||||
value->len++;
|
|
||||||
}
|
|
||||||
|
|
||||||
strcpy (last, chunk);
|
|
||||||
value->len += strlen (chunk);
|
|
||||||
|
|
||||||
last = value->str + value->len - 1;
|
|
||||||
if (*last == '\n') {
|
|
||||||
*last = '\0';
|
|
||||||
value->len--;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/* As a special-case, a value of NULL for header_desired will force
|
|
||||||
* the entire header to be parsed if it is not parsed already. This is
|
|
||||||
* used by the _notmuch_message_file_get_headers_end function.
|
|
||||||
* Another special case is the Received: header. For this header we
|
|
||||||
* want to concatenate all instances of the header instead of just
|
|
||||||
* hashing the first instance as we use this when analyzing the path
|
|
||||||
* the mail has taken from sender to recipient.
|
|
||||||
*/
|
|
||||||
const char *
|
|
||||||
notmuch_message_file_get_header (notmuch_message_file_t *message,
|
|
||||||
const char *header_desired)
|
|
||||||
{
|
|
||||||
int contains;
|
|
||||||
char *header, *decoded_value, *header_sofar, *combined_header;
|
|
||||||
const char *s, *colon;
|
|
||||||
int match, newhdr, hdrsofar, is_received;
|
|
||||||
static int initialized = 0;
|
static int initialized = 0;
|
||||||
|
|
||||||
is_received = (strcmp(header_desired,"received") == 0);
|
if (message->message)
|
||||||
|
return NOTMUCH_STATUS_SUCCESS;
|
||||||
|
|
||||||
|
/* We no longer support mboxes at all. */
|
||||||
|
if (is_mbox (message->file))
|
||||||
|
return NOTMUCH_STATUS_FILE_NOT_EMAIL;
|
||||||
|
|
||||||
if (! initialized) {
|
if (! initialized) {
|
||||||
g_mime_init (GMIME_ENABLE_RFC2047_WORKAROUNDS);
|
g_mime_init (GMIME_ENABLE_RFC2047_WORKAROUNDS);
|
||||||
initialized = 1;
|
initialized = 1;
|
||||||
}
|
}
|
||||||
|
|
||||||
message->parsing_started = 1;
|
message->headers = g_hash_table_new_full (strcase_hash, strcase_equal,
|
||||||
|
free, g_free);
|
||||||
|
if (! message->headers)
|
||||||
|
return NOTMUCH_STATUS_OUT_OF_MEMORY;
|
||||||
|
|
||||||
if (header_desired == NULL)
|
stream = g_mime_stream_file_new (message->file);
|
||||||
contains = 0;
|
|
||||||
else
|
|
||||||
contains = g_hash_table_lookup_extended (message->headers,
|
|
||||||
header_desired, NULL,
|
|
||||||
(gpointer *) &decoded_value);
|
|
||||||
|
|
||||||
if (contains && decoded_value)
|
/* We'll own and fclose the FILE* ourselves. */
|
||||||
return decoded_value;
|
g_mime_stream_file_set_owner (GMIME_STREAM_FILE (stream), FALSE);
|
||||||
|
|
||||||
if (message->parsing_finished)
|
parser = g_mime_parser_new_with_stream (stream);
|
||||||
return "";
|
g_mime_parser_set_scan_from (parser, FALSE);
|
||||||
|
|
||||||
#define NEXT_HEADER_LINE(closure) \
|
message->message = g_mime_parser_construct_message (parser);
|
||||||
while (1) { \
|
if (! message->message) {
|
||||||
ssize_t bytes_read = getline (&message->line, \
|
status = NOTMUCH_STATUS_FILE_NOT_EMAIL;
|
||||||
&message->line_size, \
|
goto DONE;
|
||||||
message->file); \
|
|
||||||
if (bytes_read == -1) { \
|
|
||||||
message->parsing_finished = 1; \
|
|
||||||
break; \
|
|
||||||
} \
|
|
||||||
if (*message->line == '\n') { \
|
|
||||||
message->parsing_finished = 1; \
|
|
||||||
break; \
|
|
||||||
} \
|
|
||||||
if (closure && \
|
|
||||||
(*message->line == ' ' || *message->line == '\t')) \
|
|
||||||
{ \
|
|
||||||
copy_header_unfolding ((closure), message->line); \
|
|
||||||
} \
|
|
||||||
if (*message->line == ' ' || *message->line == '\t') \
|
|
||||||
message->header_size += strlen (message->line); \
|
|
||||||
else \
|
|
||||||
break; \
|
|
||||||
}
|
}
|
||||||
|
|
||||||
if (message->line == NULL)
|
DONE:
|
||||||
NEXT_HEADER_LINE (NULL);
|
g_object_unref (stream);
|
||||||
|
g_object_unref (parser);
|
||||||
|
|
||||||
while (1) {
|
if (status) {
|
||||||
|
g_hash_table_destroy (message->headers);
|
||||||
|
message->headers = NULL;
|
||||||
|
|
||||||
if (message->parsing_finished)
|
if (message->message) {
|
||||||
break;
|
g_object_unref (message->message);
|
||||||
|
message->message = NULL;
|
||||||
colon = strchr (message->line, ':');
|
|
||||||
|
|
||||||
if (colon == NULL) {
|
|
||||||
message->broken_headers++;
|
|
||||||
/* A simple heuristic for giving up on things that just
|
|
||||||
* don't look like mail messages. */
|
|
||||||
if (message->broken_headers >= 10 &&
|
|
||||||
message->good_headers < 5)
|
|
||||||
{
|
|
||||||
message->parsing_finished = 1;
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
NEXT_HEADER_LINE (NULL);
|
|
||||||
continue;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
message->header_size += strlen (message->line);
|
rewind (message->file);
|
||||||
|
|
||||||
message->good_headers++;
|
|
||||||
|
|
||||||
header = xstrndup (message->line, colon - message->line);
|
|
||||||
|
|
||||||
if (message->restrict_headers &&
|
|
||||||
! g_hash_table_lookup_extended (message->headers,
|
|
||||||
header, NULL, NULL))
|
|
||||||
{
|
|
||||||
free (header);
|
|
||||||
NEXT_HEADER_LINE (NULL);
|
|
||||||
continue;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
s = colon + 1;
|
return status;
|
||||||
while (*s == ' ' || *s == '\t')
|
}
|
||||||
s++;
|
|
||||||
|
notmuch_status_t
|
||||||
message->value.len = 0;
|
_notmuch_message_file_get_mime_message (notmuch_message_file_t *message,
|
||||||
copy_header_unfolding (&message->value, s);
|
GMimeMessage **mime_message)
|
||||||
|
{
|
||||||
NEXT_HEADER_LINE (&message->value);
|
notmuch_status_t status;
|
||||||
|
|
||||||
if (header_desired == NULL)
|
status = _notmuch_message_file_parse (message);
|
||||||
match = 0;
|
if (status)
|
||||||
else
|
return status;
|
||||||
match = (strcasecmp (header, header_desired) == 0);
|
|
||||||
|
*mime_message = message->message;
|
||||||
decoded_value = g_mime_utils_header_decode_text (message->value.str);
|
|
||||||
header_sofar = (char *)g_hash_table_lookup (message->headers, header);
|
return NOTMUCH_STATUS_SUCCESS;
|
||||||
/* we treat the Received: header special - we want to concat ALL of
|
}
|
||||||
* the Received: headers we encounter.
|
|
||||||
* for everything else we return the first instance of a header */
|
/*
|
||||||
if (strcasecmp(header, "received") == 0) {
|
* Get all instances of a header decoded and concatenated.
|
||||||
if (header_sofar == NULL) {
|
*
|
||||||
/* first Received: header we encountered; just add it */
|
* The result must be freed using g_free().
|
||||||
g_hash_table_insert (message->headers, header, decoded_value);
|
*
|
||||||
} else {
|
* Return NULL on errors, empty string for non-existing headers.
|
||||||
/* we need to add the header to those we already collected */
|
*/
|
||||||
newhdr = strlen(decoded_value);
|
static char *
|
||||||
hdrsofar = strlen(header_sofar);
|
_notmuch_message_file_get_combined_header (notmuch_message_file_t *message,
|
||||||
combined_header = g_malloc(hdrsofar + newhdr + 2);
|
const char *header)
|
||||||
strncpy(combined_header,header_sofar,hdrsofar);
|
{
|
||||||
*(combined_header+hdrsofar) = ' ';
|
GMimeHeaderList *headers;
|
||||||
strncpy(combined_header+hdrsofar+1,decoded_value,newhdr+1);
|
GMimeHeaderIter *iter;
|
||||||
g_free (decoded_value);
|
char *combined = NULL;
|
||||||
g_hash_table_insert (message->headers, header, combined_header);
|
|
||||||
}
|
headers = g_mime_object_get_header_list (GMIME_OBJECT (message->message));
|
||||||
} else {
|
if (! headers)
|
||||||
if (header_sofar == NULL) {
|
return NULL;
|
||||||
/* Only insert if we don't have a value for this header, yet. */
|
|
||||||
g_hash_table_insert (message->headers, header, decoded_value);
|
iter = g_mime_header_iter_new ();
|
||||||
} else {
|
if (! iter)
|
||||||
free (header);
|
return NULL;
|
||||||
g_free (decoded_value);
|
|
||||||
decoded_value = header_sofar;
|
if (! g_mime_header_list_get_iter (headers, iter))
|
||||||
}
|
goto DONE;
|
||||||
}
|
|
||||||
/* if we found a match we can bail - unless of course we are
|
do {
|
||||||
* collecting all the Received: headers */
|
const char *value;
|
||||||
if (match && !is_received)
|
char *decoded;
|
||||||
return decoded_value;
|
|
||||||
}
|
if (strcasecmp (g_mime_header_iter_get_name (iter), header) != 0)
|
||||||
|
continue;
|
||||||
if (message->parsing_finished) {
|
|
||||||
fclose (message->file);
|
/* Note that GMime retains ownership of value... */
|
||||||
message->file = NULL;
|
value = g_mime_header_iter_get_value (iter);
|
||||||
}
|
|
||||||
|
/* ... while decoded needs to be freed with g_free(). */
|
||||||
if (message->line)
|
decoded = g_mime_utils_header_decode_text (value);
|
||||||
free (message->line);
|
if (! decoded) {
|
||||||
message->line = NULL;
|
if (combined) {
|
||||||
|
g_free (combined);
|
||||||
if (message->value.size) {
|
combined = NULL;
|
||||||
free (message->value.str);
|
}
|
||||||
message->value.str = NULL;
|
goto DONE;
|
||||||
message->value.size = 0;
|
}
|
||||||
message->value.len = 0;
|
|
||||||
}
|
if (combined) {
|
||||||
|
char *tmp = g_strdup_printf ("%s %s", combined, decoded);
|
||||||
/* For the Received: header we actually might end up here even
|
g_free (decoded);
|
||||||
* though we found the header (as we force continued parsing
|
g_free (combined);
|
||||||
* in that case). So let's check if that's the header we were
|
if (! tmp) {
|
||||||
* looking for and return the value that we found (if any)
|
combined = NULL;
|
||||||
*/
|
goto DONE;
|
||||||
if (is_received)
|
}
|
||||||
return (char *)g_hash_table_lookup (message->headers, "received");
|
|
||||||
|
combined = tmp;
|
||||||
/* We've parsed all headers and never found the one we're looking
|
} else {
|
||||||
* for. It's probably just not there, but let's check that we
|
combined = decoded;
|
||||||
* didn't make a mistake preventing us from seeing it. */
|
}
|
||||||
if (message->restrict_headers && header_desired &&
|
} while (g_mime_header_iter_next (iter));
|
||||||
! g_hash_table_lookup_extended (message->headers,
|
|
||||||
header_desired, NULL, NULL))
|
/* Return empty string for non-existing headers. */
|
||||||
{
|
if (! combined)
|
||||||
INTERNAL_ERROR ("Attempt to get header \"%s\" which was not\n"
|
combined = g_strdup ("");
|
||||||
"included in call to notmuch_message_file_restrict_headers\n",
|
|
||||||
header_desired);
|
DONE:
|
||||||
}
|
g_mime_header_iter_free (iter);
|
||||||
|
|
||||||
return "";
|
return combined;
|
||||||
|
}
|
||||||
|
|
||||||
|
const char *
|
||||||
|
notmuch_message_file_get_header (notmuch_message_file_t *message,
|
||||||
|
const char *header)
|
||||||
|
{
|
||||||
|
const char *value;
|
||||||
|
char *decoded;
|
||||||
|
|
||||||
|
if (_notmuch_message_file_parse (message))
|
||||||
|
return NULL;
|
||||||
|
|
||||||
|
/* If we have a cached decoded value, use it. */
|
||||||
|
value = g_hash_table_lookup (message->headers, header);
|
||||||
|
if (value)
|
||||||
|
return value;
|
||||||
|
|
||||||
|
if (strcasecmp (header, "received") == 0) {
|
||||||
|
/*
|
||||||
|
* The Received: header is special. We concatenate all
|
||||||
|
* instances of the header as we use this when analyzing the
|
||||||
|
* path the mail has taken from sender to recipient.
|
||||||
|
*/
|
||||||
|
decoded = _notmuch_message_file_get_combined_header (message, header);
|
||||||
|
} else {
|
||||||
|
value = g_mime_object_get_header (GMIME_OBJECT (message->message),
|
||||||
|
header);
|
||||||
|
if (value)
|
||||||
|
decoded = g_mime_utils_header_decode_text (value);
|
||||||
|
else
|
||||||
|
decoded = g_strdup ("");
|
||||||
|
}
|
||||||
|
|
||||||
|
if (! decoded)
|
||||||
|
return NULL;
|
||||||
|
|
||||||
|
/* Cache the decoded value. We also own the strings. */
|
||||||
|
g_hash_table_insert (message->headers, xstrdup (header), decoded);
|
||||||
|
|
||||||
|
return decoded;
|
||||||
}
|
}
|
||||||
|
|
|
@ -46,6 +46,8 @@ NOTMUCH_BEGIN_DECLS
|
||||||
|
|
||||||
#include <talloc.h>
|
#include <talloc.h>
|
||||||
|
|
||||||
|
#include <gmime/gmime.h>
|
||||||
|
|
||||||
#include "xutil.h"
|
#include "xutil.h"
|
||||||
#include "error_util.h"
|
#include "error_util.h"
|
||||||
|
|
||||||
|
@ -320,13 +322,6 @@ notmuch_message_set_author (notmuch_message_t *message, const char *author);
|
||||||
const char *
|
const char *
|
||||||
notmuch_message_get_author (notmuch_message_t *message);
|
notmuch_message_get_author (notmuch_message_t *message);
|
||||||
|
|
||||||
|
|
||||||
/* index.cc */
|
|
||||||
|
|
||||||
notmuch_status_t
|
|
||||||
_notmuch_message_index_file (notmuch_message_t *message,
|
|
||||||
const char *filename);
|
|
||||||
|
|
||||||
/* message-file.c */
|
/* message-file.c */
|
||||||
|
|
||||||
/* XXX: I haven't decided yet whether these will actually get exported
|
/* XXX: I haven't decided yet whether these will actually get exported
|
||||||
|
@ -352,31 +347,31 @@ _notmuch_message_file_open_ctx (void *ctx, const char *filename);
|
||||||
void
|
void
|
||||||
notmuch_message_file_close (notmuch_message_file_t *message);
|
notmuch_message_file_close (notmuch_message_file_t *message);
|
||||||
|
|
||||||
/* Restrict 'message' to only save the named headers.
|
/* Parse the message.
|
||||||
*
|
*
|
||||||
* When the caller is only interested in a short list of headers,
|
* This will be done automatically as necessary on other calls
|
||||||
* known in advance, calling this function can avoid wasted time and
|
* depending on it, but an explicit call allows for better error
|
||||||
* memory parsing/saving header values that will never be needed.
|
* status reporting.
|
||||||
*
|
|
||||||
* The variable arguments should be a list of const char * with a
|
|
||||||
* final '(const char *) NULL' to terminate the list.
|
|
||||||
*
|
|
||||||
* If this function is called, it must be called before any calls to
|
|
||||||
* notmuch_message_get_header for this message.
|
|
||||||
*
|
|
||||||
* After calling this function, if notmuch_message_get_header is
|
|
||||||
* called with a header name not in this list, then NULL will be
|
|
||||||
* returned even if that header exists in the actual message.
|
|
||||||
*/
|
*/
|
||||||
void
|
notmuch_status_t
|
||||||
notmuch_message_file_restrict_headers (notmuch_message_file_t *message, ...);
|
_notmuch_message_file_parse (notmuch_message_file_t *message);
|
||||||
|
|
||||||
/* Identical to notmuch_message_restrict_headers but accepting a va_list. */
|
/* Get the gmime message of a message file.
|
||||||
void
|
*
|
||||||
notmuch_message_file_restrict_headersv (notmuch_message_file_t *message,
|
* The message file is parsed as necessary.
|
||||||
va_list va_headers);
|
*
|
||||||
|
* The GMimeMessage* is set to *mime_message on success (which the
|
||||||
|
* caller must not unref).
|
||||||
|
*
|
||||||
|
* XXX: Would be nice to not have to expose GMimeMessage here.
|
||||||
|
*/
|
||||||
|
notmuch_status_t
|
||||||
|
_notmuch_message_file_get_mime_message (notmuch_message_file_t *message,
|
||||||
|
GMimeMessage **mime_message);
|
||||||
|
|
||||||
/* Get the value of the specified header from the message as a UTF-8 string.
|
/* Get the value of the specified header from the message as a UTF-8 string.
|
||||||
|
*
|
||||||
|
* The message file is parsed as necessary.
|
||||||
*
|
*
|
||||||
* The header name is case insensitive.
|
* The header name is case insensitive.
|
||||||
*
|
*
|
||||||
|
@ -387,13 +382,19 @@ notmuch_message_file_restrict_headersv (notmuch_message_file_t *message,
|
||||||
* only until the message is closed. The caller should copy it if
|
* only until the message is closed. The caller should copy it if
|
||||||
* needing to modify the value or to hold onto it for longer.
|
* needing to modify the value or to hold onto it for longer.
|
||||||
*
|
*
|
||||||
* Returns NULL if the message does not contain a header line matching
|
* Returns NULL on errors, empty string if the message does not
|
||||||
* 'header'.
|
* contain a header line matching 'header'.
|
||||||
*/
|
*/
|
||||||
const char *
|
const char *
|
||||||
notmuch_message_file_get_header (notmuch_message_file_t *message,
|
notmuch_message_file_get_header (notmuch_message_file_t *message,
|
||||||
const char *header);
|
const char *header);
|
||||||
|
|
||||||
|
/* index.cc */
|
||||||
|
|
||||||
|
notmuch_status_t
|
||||||
|
_notmuch_message_index_file (notmuch_message_t *message,
|
||||||
|
notmuch_message_file_t *message_file);
|
||||||
|
|
||||||
/* messages.c */
|
/* messages.c */
|
||||||
|
|
||||||
typedef struct _notmuch_message_node {
|
typedef struct _notmuch_message_node {
|
||||||
|
|
Loading…
Reference in a new issue