aboutsummaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
-rw-r--r--src/libmime/message.c621
-rw-r--r--src/libmime/message.h21
2 files changed, 4 insertions, 638 deletions
diff --git a/src/libmime/message.c b/src/libmime/message.c
index 952db753f..a1bc2d25f 100644
--- a/src/libmime/message.c
+++ b/src/libmime/message.c
@@ -1265,7 +1265,7 @@ process_message (struct rspamd_task *task)
/* Parse received headers */
first =
- message_get_raw_header (task, "Received", FALSE);
+ message_get_header (task, "Received", FALSE);
cur = first;
while (cur) {
recv =
@@ -1355,7 +1355,7 @@ process_message (struct rspamd_task *task)
}
/* Parse urls inside Subject header */
- cur = message_get_raw_header (task, "Subject", FALSE);
+ cur = message_get_header (task, "Subject", FALSE);
if (cur) {
p = cur->data;
len = strlen (p);
@@ -1400,625 +1400,10 @@ process_message (struct rspamd_task *task)
return 0;
}
-struct gmime_raw_header {
- struct raw_header *next;
- gchar *name;
- gchar *value;
-};
-
-typedef struct _GMimeHeader {
- GHashTable *hash;
- GHashTable *writers;
- struct raw_header *headers;
-} local_GMimeHeader;
-
-
-/* known header field types */
-enum {
- HEADER_FROM = 0,
- HEADER_REPLY_TO,
- HEADER_TO,
- HEADER_CC,
- HEADER_BCC,
- HEADER_SUBJECT,
- HEADER_DATE,
- HEADER_MESSAGE_ID,
- HEADER_UNKNOWN
-};
-/*
- * Iterate throught all headers and make a list
- */
-#ifndef GMIME24
-static void
-header_iterate (rspamd_mempool_t * pool,
- struct gmime_raw_header *h,
- GList ** ret,
- const gchar *field,
- gboolean strong)
-{
- while (h) {
- if (G_LIKELY (!strong)) {
- if (h->value &&
- !g_ascii_strncasecmp (field, h->name, strlen (field))) {
- if (pool != NULL) {
- *ret =
- g_list_prepend (*ret,
- rspamd_mempool_strdup (pool, h->value));
- }
- else {
- *ret = g_list_prepend (*ret, g_strdup (h->value));
- }
- }
- }
- else {
- if (h->value && !strncmp (field, h->name, strlen (field))) {
- if (pool != NULL) {
- *ret =
- g_list_prepend (*ret,
- rspamd_mempool_strdup (pool, h->value));
- }
- else {
- *ret = g_list_prepend (*ret, g_strdup (h->value));
- }
- }
- }
- h = (struct gmime_raw_header *)h->next;
- }
-}
-#else
-static void
-header_iterate (rspamd_mempool_t * pool,
- GMimeHeaderList * ls,
- GList ** ret,
- const gchar *field,
- gboolean strong)
-{
- /* Use iterator in case of gmime 2.4 */
- GMimeHeaderIter *iter;
- const gchar *name;
-
- if (ls == NULL) {
- *ret = NULL;
- return;
- }
-
- iter = g_mime_header_iter_new ();
- if (g_mime_header_list_get_iter (ls,
- iter) && g_mime_header_iter_first (iter)) {
- /* Iterate throught headers */
- while (g_mime_header_iter_is_valid (iter)) {
- name = g_mime_header_iter_get_name (iter);
- if (G_LIKELY (!strong)) {
- if (!g_ascii_strncasecmp (field, name, strlen (name))) {
- if (pool != NULL) {
- *ret =
- g_list_prepend (*ret,
- rspamd_mempool_strdup (pool,
- g_mime_header_iter_get_value (iter)));
- }
- else {
- *ret =
- g_list_prepend (*ret,
- g_strdup (g_mime_header_iter_get_value (iter)));
- }
- }
- }
- else {
- if (!strncmp (field, name, strlen (name))) {
- if (pool != NULL) {
- *ret =
- g_list_prepend (*ret,
- rspamd_mempool_strdup (pool,
- g_mime_header_iter_get_value (iter)));
- }
- else {
- *ret =
- g_list_prepend (*ret,
- g_strdup (g_mime_header_iter_get_value (iter)));
- }
- }
- }
- if (!g_mime_header_iter_next (iter)) {
- break;
- }
- }
- }
- g_mime_header_iter_free (iter);
-}
-#endif
-
-
-struct multipart_cb_data {
- GList *ret;
- rspamd_mempool_t *pool;
- const gchar *field;
- gboolean try_search;
- gboolean strong;
- gint rec;
-};
-
-#define MAX_REC 10
-
-static void
-#ifdef GMIME24
-multipart_iterate (GMimeObject * parent, GMimeObject * part, gpointer user_data)
-#else
-multipart_iterate (GMimeObject * part, gpointer user_data)
-#endif
-{
- struct multipart_cb_data *data = user_data;
-#ifndef GMIME24
- struct gmime_raw_header *h;
-#endif
- GList *l = NULL;
-
- if (data->try_search && part != NULL && GMIME_IS_PART (part)) {
-#ifdef GMIME24
- GMimeHeaderList *ls;
-
- ls = g_mime_object_get_header_list (GMIME_OBJECT (part));
- header_iterate (data->pool, ls, &l, data->field, data->strong);
-#else
- h = (struct gmime_raw_header *)part->headers->headers;
- header_iterate (data->pool, h, &l, data->field, data->strong);
-#endif
- if (l == NULL) {
- /* Header not found, abandon search results */
- data->try_search = FALSE;
- g_list_free (data->ret);
- data->ret = NULL;
- }
- else {
- data->ret = g_list_concat (l, data->ret);
- }
- }
- else if (data->try_search && GMIME_IS_MULTIPART (part)) {
- /* Maybe endless recursion here ? */
- if (data->rec++ < MAX_REC) {
- g_mime_multipart_foreach (GMIME_MULTIPART (
- part), multipart_iterate, data);
- }
- else {
- msg_info ("maximum recurse limit is over, stop recursing, %d",
- data->rec);
- data->try_search = FALSE;
- }
- }
-}
-
-static GList *
-local_message_get_header (rspamd_mempool_t * pool,
- GMimeMessage * message,
- const gchar *field,
- gboolean strong)
-{
- GList *gret = NULL;
- GMimeObject *part;
- struct multipart_cb_data cb = {
- .try_search = TRUE,
- .rec = 0,
- .ret = NULL,
- };
- cb.pool = pool;
- cb.field = field;
- cb.strong = strong;
-
-#ifndef GMIME24
- struct gmime_raw_header *h;
-
- if (field == NULL) {
- return NULL;
- }
-
- msg_debug ("iterate over headers to find header %s", field);
- h = (struct gmime_raw_header *) (GMIME_OBJECT (message)->headers->headers);
- header_iterate (pool, h, &gret, field, strong);
-
- if (gret == NULL) {
- /* Try to iterate with mime part headers */
- msg_debug ("iterate over headers of mime part to find header %s",
- field);
- part = g_mime_message_get_mime_part (message);
- if (part) {
- h = (struct gmime_raw_header *)part->headers->headers;
- header_iterate (pool, h, &gret, field, strong);
- if (gret == NULL && GMIME_IS_MULTIPART (part)) {
- msg_debug (
- "iterate over headers of each multipart's subparts %s",
- field);
- g_mime_multipart_foreach (GMIME_MULTIPART (
- part), multipart_iterate, &cb);
- if (cb.ret != NULL) {
- gret = cb.ret;
- }
- }
-#ifndef GMIME24
- g_object_unref (part);
-#endif
- }
- }
-
- return gret;
-#else
- GMimeHeaderList *ls;
-
- ls = g_mime_object_get_header_list (GMIME_OBJECT (message));
- header_iterate (pool, ls, &gret, field, strong);
- if (gret == NULL) {
- /* Try to iterate with mime part headers */
- part = g_mime_message_get_mime_part (message);
- if (part) {
- ls = g_mime_object_get_header_list (GMIME_OBJECT (part));
- header_iterate (pool, ls, &gret, field, strong);
- if (gret == NULL && GMIME_IS_MULTIPART (part)) {
- g_mime_multipart_foreach (GMIME_MULTIPART (
- part), multipart_iterate, &cb);
- if (cb.ret != NULL) {
- gret = cb.ret;
- }
- }
-#ifndef GMIME24
- g_object_unref (part);
-#endif
- }
- }
-
-
- return gret;
-#endif
-}
-
-/**
- * g_mime_message_set_date_from_string: Set the message sent-date
- * @message: MIME Message
- * @string: A string of date
- *
- * Set the sent-date on a MIME Message.
- **/
-void
-local_mime_message_set_date_from_string (GMimeMessage * message,
- const gchar * string)
-{
- time_t date;
- gint offset = 0;
-
- date = g_mime_utils_header_decode_date (string, &offset);
- g_mime_message_set_date (message, date, offset);
-}
-
-/*
- * Replacements for standart gmime functions but converting adresses to IA
- */
-static const gchar *
-local_message_get_sender (GMimeMessage * message)
-{
- gchar *res;
- const gchar *from = g_mime_message_get_sender (message);
- InternetAddressList *ia;
-
-#ifndef GMIME24
- ia = internet_address_parse_string (from);
-#else
- ia = internet_address_list_parse_string (from);
-#endif
- if (!ia) {
- return NULL;
- }
- res = internet_address_list_to_string (ia, FALSE);
-#ifndef GMIME24
- internet_address_list_destroy (ia);
-#else
- g_object_unref (ia);
-#endif
-
- return res;
-}
-
-#ifdef GMIME24
-static const gchar*
-local_message_get_content_type (GMimeMessage * message)
-{
- GMimeContentType *ct;
-
- ct = g_mime_object_get_content_type (GMIME_OBJECT (message));
- if (ct) {
- return g_mime_content_type_to_string (ct);
- }
- return NULL;
-}
-static void
-local_message_set_content_type (GMimeMessage * message, const gchar *in)
-{
- GMimeContentType *ct;
-
- ct = g_mime_content_type_new_from_string (in);
- if (ct) {
- g_mime_object_set_content_type (GMIME_OBJECT (message), ct);
- }
-}
-#endif
-
-static const gchar *
-local_message_get_reply_to (GMimeMessage * message)
-{
- gchar *res;
- const gchar *from = g_mime_message_get_reply_to (message);
- InternetAddressList *ia;
-
-#ifndef GMIME24
- ia = internet_address_parse_string (from);
-#else
- ia = internet_address_list_parse_string (from);
-#endif
- if (!ia) {
- return NULL;
- }
- res = internet_address_list_to_string (ia, FALSE);
-#ifndef GMIME24
- internet_address_list_destroy (ia);
-#else
- g_object_unref (ia);
-#endif
-
- return res;
-}
-
-#ifdef GMIME24
-
-# define ADD_RECIPIENT_TEMPLATE(type,def) \
- static void \
- local_message_add_recipients_from_string_ ## type (GMimeMessage * message, \
- const gchar * string, \
- const gchar * value) \
- { \
- InternetAddressList *il, *new; \
- \
- il = g_mime_message_get_recipients (message, (def)); \
- new = internet_address_list_parse_string (string); \
- internet_address_list_append (il, new); \
- } \
-
-ADD_RECIPIENT_TEMPLATE (to, GMIME_RECIPIENT_TYPE_TO)
-ADD_RECIPIENT_TEMPLATE (cc, GMIME_RECIPIENT_TYPE_CC)
-ADD_RECIPIENT_TEMPLATE (bcc, GMIME_RECIPIENT_TYPE_BCC)
-# define GET_RECIPIENT_TEMPLATE(type,def) \
- static InternetAddressList * \
- local_message_get_recipients_ ## type (GMimeMessage * message, \
- const gchar * unused) \
- { \
- return g_mime_message_get_recipients (message, (def)); \
- }
-GET_RECIPIENT_TEMPLATE (to, GMIME_RECIPIENT_TYPE_TO)
-GET_RECIPIENT_TEMPLATE (cc, GMIME_RECIPIENT_TYPE_CC)
-GET_RECIPIENT_TEMPLATE (bcc, GMIME_RECIPIENT_TYPE_BCC)
-#endif
-/* different declarations for different types of set and get functions */
-typedef const gchar *(*GetFunc) (GMimeMessage * message);
-typedef InternetAddressList *(*GetRcptFunc) (GMimeMessage * message,
- const gchar *type);
-typedef GList *(*GetListFunc) (rspamd_mempool_t * pool, GMimeMessage * message,
- const gchar *type, gboolean strong);
-typedef void (*SetFunc) (GMimeMessage * message, const gchar *value);
-typedef void (*SetListFunc) (GMimeMessage * message, const gchar *field,
- const gchar *value);
-
-/** different types of functions
- *
- * FUNC_CHARPTR
- * - function with no arguments
- * - get returns gchar*
- *
- * FUNC_IA (from Internet Address)
- * - function with additional "field" argument from the fieldfunc table,
- * - get returns Glist*
- *
- * FUNC_LIST
- * - function with additional "field" argument (given arbitrary header field name)
- * - get returns Glist*
- **/
-enum {
- FUNC_CHARPTR = 0,
- FUNC_CHARFREEPTR,
- FUNC_IA,
- FUNC_LIST
-};
-
-/**
- * fieldfunc struct: structure of MIME fields and corresponding get and set
- * functions.
- **/
-static struct {
- gchar *name;
- GetFunc func;
- GetRcptFunc rcptfunc;
- GetListFunc getlistfunc;
- SetFunc setfunc;
- SetListFunc setlfunc;
- gint functype;
-} fieldfunc[] =
-{
- {
- "From", local_message_get_sender, NULL, NULL, g_mime_message_set_sender,
- NULL, FUNC_CHARFREEPTR
- }, {
- "Reply-To", local_message_get_reply_to, NULL, NULL,
- g_mime_message_set_reply_to, NULL, FUNC_CHARFREEPTR
- },
-#ifndef GMIME24
- {
- "To", NULL, (GetRcptFunc) g_mime_message_get_recipients, NULL, NULL,
- (SetListFunc) g_mime_message_add_recipients_from_string, FUNC_IA
- }, {
- "Cc", NULL, (GetRcptFunc) g_mime_message_get_recipients, NULL, NULL,
- (SetListFunc) g_mime_message_add_recipients_from_string, FUNC_IA
- }, {
- "Bcc", NULL, (GetRcptFunc) g_mime_message_get_recipients, NULL, NULL,
- (SetListFunc) g_mime_message_add_recipients_from_string, FUNC_IA
- }, {
- "Date", (GetFunc) g_mime_message_get_date_string, NULL, NULL,
- local_mime_message_set_date_from_string, NULL, FUNC_CHARFREEPTR
- },
-#else
- {
- "To", NULL, local_message_get_recipients_to, NULL, NULL,
- local_message_add_recipients_from_string_to, FUNC_IA
- }, {
- "Cc", NULL, local_message_get_recipients_cc, NULL, NULL,
- local_message_add_recipients_from_string_cc, FUNC_IA
- }, {
- "Bcc", NULL, local_message_get_recipients_bcc, NULL, NULL,
- local_message_add_recipients_from_string_bcc, FUNC_IA
- }, {
- "Date", (GetFunc)g_mime_message_get_date_as_string, NULL, NULL,
- local_mime_message_set_date_from_string, NULL, FUNC_CHARFREEPTR
- }, {
- "Content-Type", local_message_get_content_type, NULL, NULL,
- local_message_set_content_type, NULL, FUNC_CHARFREEPTR
- },
-#endif
- {
- "Subject", g_mime_message_get_subject, NULL, NULL,
- g_mime_message_set_subject, NULL, FUNC_CHARPTR
- }, {
- "Message-Id", g_mime_message_get_message_id, NULL, NULL,
- g_mime_message_set_message_id, NULL, FUNC_CHARPTR
- },
-#ifndef GMIME24
- {
- NULL, NULL, NULL, local_message_get_header, NULL,
- g_mime_message_add_header, FUNC_LIST
- }
-#else
- {
- NULL, NULL, NULL, local_message_get_header, NULL,
- (SetListFunc)g_mime_object_append_header, FUNC_LIST
- }
-#endif
-};
-
-/**
- * message_set_header: set header of any type excluding special (Content- and MIME-Version:)
- **/
-void
-message_set_header (GMimeMessage * message,
- const gchar *field,
- const gchar *value)
-{
- gint i;
-
- if (!g_ascii_strcasecmp (field,
- "MIME-Version:") || !g_ascii_strncasecmp (field, "Content-", 8)) {
- return;
- }
- for (i = 0; i <= HEADER_UNKNOWN; ++i) {
- if (!fieldfunc[i].name ||
- !g_ascii_strncasecmp (field, fieldfunc[i].name,
- strlen (fieldfunc[i].name))) {
- switch (fieldfunc[i].functype) {
- case FUNC_CHARPTR:
- case FUNC_CHARFREEPTR:
- if (fieldfunc[i].setfunc) {
- (*(fieldfunc[i].setfunc))(message, value);
- }
- break;
- case FUNC_IA:
- (*(fieldfunc[i].setlfunc))(message, fieldfunc[i].name, value);
- break;
- case FUNC_LIST:
- (*(fieldfunc[i].setlfunc))(message, field, value);
- break;
- }
- break;
- }
- }
-}
-
-
-/**
- * message_get_header: returns the list of 'any header' values
- * (except of unsupported yet Content- and MIME-Version special headers)
- *
- * You should free the GList list by yourself.
- **/
-GList *
-message_get_header (rspamd_mempool_t * pool,
- GMimeMessage * message,
- const gchar *field,
- gboolean strong)
-{
- gint i;
- gchar *ret = NULL, *ia_string;
- GList *gret = NULL;
- InternetAddressList *ia_list = NULL, *ia;
-
- for (i = 0; i <= HEADER_UNKNOWN; ++i) {
- if (!fieldfunc[i].name ||
- !g_ascii_strncasecmp (field, fieldfunc[i].name,
- strlen (fieldfunc[i].name))) {
- switch (fieldfunc[i].functype) {
- case FUNC_CHARFREEPTR:
- ret = (gchar *)(*(fieldfunc[i].func))(message);
- break;
- case FUNC_CHARPTR:
- ret = (gchar *)(*(fieldfunc[i].func))(message);
- break;
- case FUNC_IA:
- ia_list = (*(fieldfunc[i].rcptfunc))(message, field);
- ia = ia_list;
-#ifndef GMIME24
- while (ia && ia->address) {
-
- ia_string = internet_address_to_string (
- (InternetAddress *) ia->address,
- FALSE);
- if (pool != NULL) {
- rspamd_mempool_add_destructor (pool,
- (rspamd_mempool_destruct_t) g_free, ia_string);
- }
- gret = g_list_prepend (gret, ia_string);
- ia = ia->next;
- }
-#else
- i = internet_address_list_length (ia);
- while (--i >= 0) {
- ia_string = internet_address_to_string (internet_address_list_get_address (
- ia,
- i),
- FALSE);
- if (pool != NULL) {
- rspamd_mempool_add_destructor (pool,
- (rspamd_mempool_destruct_t) g_free, ia_string);
- }
- gret = g_list_prepend (gret, ia_string);
- }
-#endif
- break;
- case FUNC_LIST:
- gret =
- (*(fieldfunc[i].getlistfunc))(pool, message, field, strong);
- break;
- }
- break;
- }
- }
- if (gret == NULL && ret != NULL) {
- if (pool != NULL) {
- gret = g_list_prepend (gret, rspamd_mempool_strdup (pool, ret));
- }
- else {
- gret = g_list_prepend (gret, g_strdup (ret));
- }
- }
- if (fieldfunc[i].functype == FUNC_CHARFREEPTR && ret) {
- g_free (ret);
- }
-
- return gret;
-}
GList *
-message_get_raw_header (struct rspamd_task *task,
+message_get_header (struct rspamd_task *task,
const gchar *field,
gboolean strong)
{
diff --git a/src/libmime/message.h b/src/libmime/message.h
index 1c4ee981a..35eb3ff1e 100644
--- a/src/libmime/message.h
+++ b/src/libmime/message.h
@@ -65,25 +65,6 @@ struct raw_header {
*/
gint process_message (struct rspamd_task *task);
-/*
- * Set header with specified name and value
- */
-void message_set_header (GMimeMessage *message,
- const gchar *field,
- const gchar *value);
-
-/*
- * Get a list of header's values with specified header's name
- * @param pool if not NULL this pool would be used for storing header's values
- * @param message g_mime_message object
- * @param field header's name
- * @param strong if this flag is TRUE header's name is case sensitive, otherwise it is not
- * @return A list of header's values or NULL. If list is not NULL it MUST be freed. If pool is NULL elements must be freed as well.
- */
-GList * message_get_header (rspamd_mempool_t *pool,
- GMimeMessage *message,
- const gchar *field,
- gboolean strong);
/*
* Get a list of header's values with specified header's name using raw headers
@@ -92,7 +73,7 @@ GList * message_get_header (rspamd_mempool_t *pool,
* @param strong if this flag is TRUE header's name is case sensitive, otherwise it is not
* @return A list of header's values or NULL. Unlike previous function it is NOT required to free list or values. I should rework one of these functions some time.
*/
-GList * message_get_raw_header (struct rspamd_task *task,
+GList * message_get_header (struct rspamd_task *task,
const gchar *field,
gboolean strong);