Blame src/em-format/e-mail-parser.c

Packit 15f964
/*
Packit 15f964
 * e-mail-parser.c
Packit 15f964
 *
Packit 15f964
 * This program is free software; you can redistribute it and/or modify it
Packit 15f964
 * under the terms of the GNU Lesser General Public License as published by
Packit 15f964
 * the Free Software Foundation.
Packit 15f964
 *
Packit 15f964
 * This program is distributed in the hope that it will be useful, but
Packit 15f964
 * WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY
Packit 15f964
 * or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU General Public License
Packit 15f964
 * for more details.
Packit 15f964
 *
Packit 15f964
 * You should have received a copy of the GNU Lesser General Public License
Packit 15f964
 * along with this program; if not, see <http://www.gnu.org/licenses/>.
Packit 15f964
 *
Packit 15f964
 */
Packit 15f964
Packit 15f964
#include "evolution-config.h"
Packit 15f964
Packit 15f964
#include "e-mail-parser.h"
Packit 15f964
Packit 15f964
#include <string.h>
Packit 15f964
Packit 15f964
#include <libebackend/libebackend.h>
Packit 15f964
Packit 15f964
#include <shell/e-shell.h>
Packit 15f964
#include <shell/e-shell-window.h>
Packit 15f964
Packit 15f964
#include "e-mail-parser-extension.h"
Packit 15f964
#include "e-mail-part-attachment.h"
Packit 15f964
#include "e-mail-part-utils.h"
Packit 15f964
Packit 15f964
#define E_MAIL_PARSER_GET_PRIVATE(obj) \
Packit 15f964
	(G_TYPE_INSTANCE_GET_PRIVATE \
Packit 15f964
	((obj), E_TYPE_MAIL_PARSER, EMailParserPrivate))
Packit 15f964
Packit 15f964
#define d(x)
Packit 15f964
Packit 15f964
struct _EMailParserPrivate {
Packit 15f964
	GMutex mutex;
Packit 15f964
Packit 15f964
	gint last_error;
Packit 15f964
Packit 15f964
	CamelSession *session;
Packit 15f964
};
Packit 15f964
Packit 15f964
enum {
Packit 15f964
	PROP_0,
Packit 15f964
	PROP_SESSION
Packit 15f964
};
Packit 15f964
Packit 15f964
/* internal parser extensions */
Packit 15f964
GType e_mail_parser_application_mbox_get_type (void);
Packit 15f964
GType e_mail_parser_audio_get_type (void);
Packit 15f964
GType e_mail_parser_headers_get_type (void);
Packit 15f964
GType e_mail_parser_message_get_type (void);
Packit 15f964
GType e_mail_parser_secure_button_get_type (void);
Packit 15f964
GType e_mail_parser_source_get_type (void);
Packit 15f964
GType e_mail_parser_image_get_type (void);
Packit 15f964
GType e_mail_parser_inline_pgp_encrypted_get_type (void);
Packit 15f964
GType e_mail_parser_inline_pgp_signed_get_type (void);
Packit 15f964
GType e_mail_parser_message_delivery_status_get_type (void);
Packit 15f964
GType e_mail_parser_message_external_get_type (void);
Packit 15f964
GType e_mail_parser_message_rfc822_get_type (void);
Packit 15f964
GType e_mail_parser_multipart_alternative_get_type (void);
Packit 15f964
GType e_mail_parser_multipart_apple_double_get_type (void);
Packit 15f964
GType e_mail_parser_multipart_digest_get_type (void);
Packit 15f964
GType e_mail_parser_multipart_encrypted_get_type (void);
Packit 15f964
GType e_mail_parser_multipart_mixed_get_type (void);
Packit 15f964
GType e_mail_parser_multipart_related_get_type (void);
Packit 15f964
GType e_mail_parser_multipart_signed_get_type (void);
Packit 15f964
GType e_mail_parser_text_enriched_get_type (void);
Packit 15f964
GType e_mail_parser_text_html_get_type (void);
Packit 15f964
GType e_mail_parser_text_plain_get_type (void);
Packit 15f964
#ifdef ENABLE_SMIME
Packit 15f964
GType e_mail_parser_application_smime_get_type (void);
Packit 15f964
#endif
Packit 15f964
Packit 15f964
static gpointer parent_class;
Packit 15f964
Packit 15f964
static void
Packit 15f964
mail_parser_run (EMailParser *parser,
Packit 15f964
                 EMailPartList *part_list,
Packit 15f964
                 GCancellable *cancellable)
Packit 15f964
{
Packit 15f964
	EMailExtensionRegistry *reg;
Packit 15f964
	CamelMimeMessage *message;
Packit 15f964
	EMailPart *mail_part;
Packit 15f964
	GQueue *parsers;
Packit 15f964
	GQueue mail_part_queue = G_QUEUE_INIT;
Packit 15f964
	GList *iter;
Packit 15f964
	GString *part_id;
Packit 15f964
Packit 15f964
	message = e_mail_part_list_get_message (part_list);
Packit 15f964
Packit 15f964
	reg = e_mail_parser_get_extension_registry (parser);
Packit 15f964
Packit 15f964
	parsers = e_mail_extension_registry_get_for_mime_type (
Packit 15f964
		reg, "application/vnd.evolution.message");
Packit 15f964
Packit 15f964
	if (parsers == NULL)
Packit 15f964
		parsers = e_mail_extension_registry_get_for_mime_type (
Packit 15f964
			reg, "message/*");
Packit 15f964
Packit 15f964
	/* No parsers means the internal Evolution parser
Packit 15f964
	 * extensions were not loaded. Something is terribly wrong! */
Packit 15f964
	g_return_if_fail (parsers != NULL);
Packit 15f964
Packit 15f964
	part_id = g_string_new (".message");
Packit 15f964
Packit 15f964
	mail_part = e_mail_part_new (CAMEL_MIME_PART (message), ".message");
Packit 15f964
	e_mail_part_list_add_part (part_list, mail_part);
Packit 15f964
	g_object_unref (mail_part);
Packit 15f964
Packit 15f964
	for (iter = parsers->head; iter; iter = iter->next) {
Packit 15f964
		EMailParserExtension *extension;
Packit 15f964
		gboolean message_handled;
Packit 15f964
Packit 15f964
		if (g_cancellable_is_cancelled (cancellable))
Packit 15f964
			break;
Packit 15f964
Packit 15f964
		extension = iter->data;
Packit 15f964
		if (!extension)
Packit 15f964
			continue;
Packit 15f964
Packit 15f964
		message_handled = e_mail_parser_extension_parse (
Packit 15f964
			extension, parser,
Packit 15f964
			CAMEL_MIME_PART (message),
Packit 15f964
			part_id, cancellable, &mail_part_queue);
Packit 15f964
Packit 15f964
		if (message_handled)
Packit 15f964
			break;
Packit 15f964
	}
Packit 15f964
Packit 15f964
	while (!g_queue_is_empty (&mail_part_queue)) {
Packit 15f964
		mail_part = g_queue_pop_head (&mail_part_queue);
Packit 15f964
		e_mail_part_list_add_part (part_list, mail_part);
Packit 15f964
		g_object_unref (mail_part);
Packit 15f964
	}
Packit 15f964
Packit 15f964
	g_string_free (part_id, TRUE);
Packit 15f964
}
Packit 15f964
Packit 15f964
static void
Packit 15f964
shell_gone_cb (gpointer user_data,
Packit 15f964
	       GObject *gone_extension_registry)
Packit 15f964
{
Packit 15f964
	EMailParserClass *class = user_data;
Packit 15f964
Packit 15f964
	g_return_if_fail (class != NULL);
Packit 15f964
Packit 15f964
	g_clear_object (&class->extension_registry);
Packit 15f964
}
Packit 15f964
Packit 15f964
static void
Packit 15f964
mail_parser_set_session (EMailParser *parser,
Packit 15f964
                         CamelSession *session)
Packit 15f964
{
Packit 15f964
	g_return_if_fail (CAMEL_IS_SESSION (session));
Packit 15f964
	g_return_if_fail (parser->priv->session == NULL);
Packit 15f964
Packit 15f964
	parser->priv->session = g_object_ref (session);
Packit 15f964
}
Packit 15f964
Packit 15f964
static void
Packit 15f964
e_mail_parser_set_property (GObject *object,
Packit 15f964
                          guint property_id,
Packit 15f964
                          const GValue *value,
Packit 15f964
                          GParamSpec *pspec)
Packit 15f964
{
Packit 15f964
	switch (property_id) {
Packit 15f964
		case PROP_SESSION:
Packit 15f964
			mail_parser_set_session (
Packit 15f964
				E_MAIL_PARSER (object),
Packit 15f964
				g_value_get_object (value));
Packit 15f964
			return;
Packit 15f964
	}
Packit 15f964
Packit 15f964
	G_OBJECT_WARN_INVALID_PROPERTY_ID (object, property_id, pspec);
Packit 15f964
}
Packit 15f964
Packit 15f964
static void
Packit 15f964
e_mail_parser_get_property (GObject *object,
Packit 15f964
                          guint property_id,
Packit 15f964
                          GValue *value,
Packit 15f964
                          GParamSpec *pspec)
Packit 15f964
{
Packit 15f964
	switch (property_id) {
Packit 15f964
		case PROP_SESSION:
Packit 15f964
			g_value_set_object (
Packit 15f964
				value,
Packit 15f964
				e_mail_parser_get_session (
Packit 15f964
				E_MAIL_PARSER (object)));
Packit 15f964
			return;
Packit 15f964
	}
Packit 15f964
Packit 15f964
	G_OBJECT_WARN_INVALID_PROPERTY_ID (object, property_id, pspec);
Packit 15f964
}
Packit 15f964
Packit 15f964
static void
Packit 15f964
e_mail_parser_finalize (GObject *object)
Packit 15f964
{
Packit 15f964
	EMailParserPrivate *priv;
Packit 15f964
Packit 15f964
	priv = E_MAIL_PARSER_GET_PRIVATE (object);
Packit 15f964
Packit 15f964
	g_clear_object (&priv->session);
Packit 15f964
	g_mutex_clear (&priv->mutex);
Packit 15f964
Packit 15f964
	/* Chain up to parent's finalize() method. */
Packit 15f964
	G_OBJECT_CLASS (parent_class)->finalize (object);
Packit 15f964
}
Packit 15f964
Packit 15f964
static void
Packit 15f964
e_mail_parser_base_init (EMailParserClass *class)
Packit 15f964
{
Packit 15f964
	EShell *shell;
Packit 15f964
Packit 15f964
	/* Register internal extensions. */
Packit 15f964
	g_type_ensure (e_mail_parser_application_mbox_get_type ());
Packit 15f964
	/* This is currently disabled, because the WebKit player requires javascript,
Packit 15f964
	   which is disabled in Evolution. */
Packit 15f964
	/* g_type_ensure (e_mail_parser_audio_get_type ()); */
Packit 15f964
	g_type_ensure (e_mail_parser_headers_get_type ());
Packit 15f964
	g_type_ensure (e_mail_parser_message_get_type ());
Packit 15f964
	g_type_ensure (e_mail_parser_secure_button_get_type ());
Packit 15f964
	g_type_ensure (e_mail_parser_source_get_type ());
Packit 15f964
	g_type_ensure (e_mail_parser_image_get_type ());
Packit 15f964
	g_type_ensure (e_mail_parser_inline_pgp_encrypted_get_type ());
Packit 15f964
	g_type_ensure (e_mail_parser_inline_pgp_signed_get_type ());
Packit 15f964
	g_type_ensure (e_mail_parser_message_delivery_status_get_type ());
Packit 15f964
	g_type_ensure (e_mail_parser_message_external_get_type ());
Packit 15f964
	g_type_ensure (e_mail_parser_message_rfc822_get_type ());
Packit 15f964
	g_type_ensure (e_mail_parser_multipart_alternative_get_type ());
Packit 15f964
	g_type_ensure (e_mail_parser_multipart_apple_double_get_type ());
Packit 15f964
	g_type_ensure (e_mail_parser_multipart_digest_get_type ());
Packit 15f964
	g_type_ensure (e_mail_parser_multipart_encrypted_get_type ());
Packit 15f964
	g_type_ensure (e_mail_parser_multipart_mixed_get_type ());
Packit 15f964
	g_type_ensure (e_mail_parser_multipart_related_get_type ());
Packit 15f964
	g_type_ensure (e_mail_parser_multipart_signed_get_type ());
Packit 15f964
	g_type_ensure (e_mail_parser_text_enriched_get_type ());
Packit 15f964
	g_type_ensure (e_mail_parser_text_html_get_type ());
Packit 15f964
	g_type_ensure (e_mail_parser_text_plain_get_type ());
Packit 15f964
#ifdef ENABLE_SMIME
Packit 15f964
	g_type_ensure (e_mail_parser_application_smime_get_type ());
Packit 15f964
#endif
Packit 15f964
Packit 15f964
	class->extension_registry = g_object_new (
Packit 15f964
		E_TYPE_MAIL_PARSER_EXTENSION_REGISTRY, NULL);
Packit 15f964
Packit 15f964
	e_mail_parser_extension_registry_load (class->extension_registry);
Packit 15f964
Packit 15f964
	e_extensible_load_extensions (E_EXTENSIBLE (class->extension_registry));
Packit 15f964
Packit 15f964
	shell = e_shell_get_default ();
Packit 15f964
	g_object_weak_ref (G_OBJECT (shell), shell_gone_cb, class);
Packit 15f964
}
Packit 15f964
Packit 15f964
static void
Packit 15f964
e_mail_parser_class_init (EMailParserClass *class)
Packit 15f964
{
Packit 15f964
	GObjectClass *object_class;
Packit 15f964
Packit 15f964
	parent_class = g_type_class_peek_parent (class);
Packit 15f964
	g_type_class_add_private (class, sizeof (EMailParserPrivate));
Packit 15f964
Packit 15f964
	object_class = G_OBJECT_CLASS (class);
Packit 15f964
	object_class->finalize = e_mail_parser_finalize;
Packit 15f964
	object_class->set_property = e_mail_parser_set_property;
Packit 15f964
	object_class->get_property = e_mail_parser_get_property;
Packit 15f964
Packit 15f964
	g_object_class_install_property (
Packit 15f964
		object_class,
Packit 15f964
		PROP_SESSION,
Packit 15f964
		g_param_spec_object (
Packit 15f964
			"session",
Packit 15f964
			"Camel Session",
Packit 15f964
			NULL,
Packit 15f964
			CAMEL_TYPE_SESSION,
Packit 15f964
			G_PARAM_READWRITE |
Packit 15f964
			G_PARAM_CONSTRUCT_ONLY));
Packit 15f964
}
Packit 15f964
Packit 15f964
static void
Packit 15f964
e_mail_parser_init (EMailParser *parser)
Packit 15f964
{
Packit 15f964
	parser->priv = E_MAIL_PARSER_GET_PRIVATE (parser);
Packit 15f964
Packit 15f964
	g_mutex_init (&parser->priv->mutex);
Packit 15f964
}
Packit 15f964
Packit 15f964
GType
Packit 15f964
e_mail_parser_get_type (void)
Packit 15f964
{
Packit 15f964
	static GType type = 0;
Packit 15f964
Packit 15f964
	if (G_UNLIKELY (type == 0)) {
Packit 15f964
		static const GTypeInfo type_info = {
Packit 15f964
			sizeof (EMailParserClass),
Packit 15f964
			(GBaseInitFunc) e_mail_parser_base_init,
Packit 15f964
			(GBaseFinalizeFunc) NULL,
Packit 15f964
			(GClassInitFunc) e_mail_parser_class_init,
Packit 15f964
			(GClassFinalizeFunc) NULL,
Packit 15f964
			NULL,  /* class_data */
Packit 15f964
			sizeof (EMailParser),
Packit 15f964
			0,     /* n_preallocs */
Packit 15f964
			(GInstanceInitFunc) e_mail_parser_init,
Packit 15f964
			NULL   /* value_table */
Packit 15f964
		};
Packit 15f964
Packit 15f964
		type = g_type_register_static (
Packit 15f964
			G_TYPE_OBJECT, "EMailParser",
Packit 15f964
			&type_info, 0);
Packit 15f964
	}
Packit 15f964
Packit 15f964
	return type;
Packit 15f964
}
Packit 15f964
Packit 15f964
EMailParser *
Packit 15f964
e_mail_parser_new (CamelSession *session)
Packit 15f964
{
Packit 15f964
	g_return_val_if_fail (CAMEL_IS_SESSION (session), NULL);
Packit 15f964
Packit 15f964
	return g_object_new (
Packit 15f964
		E_TYPE_MAIL_PARSER,
Packit 15f964
		"session", session, NULL);
Packit 15f964
}
Packit 15f964
Packit 15f964
/**
Packit 15f964
 * e_mail_parser_parse_sync:
Packit 15f964
 * @parser: an #EMailParser
Packit 15f964
 * @folder: (allow none) a #CamelFolder containing the @message or %NULL
Packit 15f964
 * @message_uid: (allow none) UID of the @message within the @folder or %NULL
Packit 15f964
 * @message: a #CamelMimeMessage
Packit 15f964
 * @cancellable: (allow-none) a #GCancellable
Packit 15f964
 *
Packit 15f964
 * Parses the @message synchronously. Returns a list of #EMailParts which
Packit 15f964
 * represents structure of the message and additional properties of each part.
Packit 15f964
 *
Packit 15f964
 * Note that this function can block for a while, so it's not a good idea to call
Packit 15f964
 * it from main thread.
Packit 15f964
 *
Packit 15f964
 * Return Value: An #EMailPartsList
Packit 15f964
 */
Packit 15f964
EMailPartList *
Packit 15f964
e_mail_parser_parse_sync (EMailParser *parser,
Packit 15f964
                          CamelFolder *folder,
Packit 15f964
                          const gchar *message_uid,
Packit 15f964
                          CamelMimeMessage *message,
Packit 15f964
                          GCancellable *cancellable)
Packit 15f964
{
Packit 15f964
	EMailPartList *part_list;
Packit 15f964
Packit 15f964
	g_return_val_if_fail (E_IS_MAIL_PARSER (parser), NULL);
Packit 15f964
	g_return_val_if_fail (CAMEL_IS_MIME_MESSAGE (message), NULL);
Packit 15f964
Packit 15f964
	part_list = e_mail_part_list_new (message, message_uid, folder);
Packit 15f964
Packit 15f964
	mail_parser_run (parser, part_list, cancellable);
Packit 15f964
Packit 15f964
	if (camel_debug_start ("emformat:parser")) {
Packit 15f964
		GQueue queue = G_QUEUE_INIT;
Packit 15f964
Packit 15f964
		printf (
Packit 15f964
			"%s finished with EMailPartList:\n",
Packit 15f964
			G_OBJECT_TYPE_NAME (parser));
Packit 15f964
Packit 15f964
		e_mail_part_list_queue_parts (part_list, NULL, &queue);
Packit 15f964
Packit 15f964
		while (!g_queue_is_empty (&queue)) {
Packit 15f964
			EMailPart *part;
Packit 15f964
Packit 15f964
			part = g_queue_pop_head (&queue);
Packit 15f964
Packit 15f964
			printf (
Packit 15f964
				"	id: %s | cid: %s | mime_type: %s | "
Packit 15f964
				"is_hidden: %d | is_attachment: %d\n",
Packit 15f964
				e_mail_part_get_id (part),
Packit 15f964
				e_mail_part_get_cid (part),
Packit 15f964
				e_mail_part_get_mime_type (part),
Packit 15f964
				part->is_hidden ? 1 : 0,
Packit 15f964
				e_mail_part_get_is_attachment (part) ? 1 : 0);
Packit 15f964
Packit 15f964
			g_object_unref (part);
Packit 15f964
		}
Packit 15f964
Packit 15f964
		camel_debug_end ();
Packit 15f964
	}
Packit 15f964
Packit 15f964
	return part_list;
Packit 15f964
}
Packit 15f964
Packit 15f964
static void
Packit 15f964
mail_parser_parse_thread (GSimpleAsyncResult *simple,
Packit 15f964
                          GObject *source_object,
Packit 15f964
                          GCancellable *cancellable)
Packit 15f964
{
Packit 15f964
	EMailPartList *part_list;
Packit 15f964
Packit 15f964
	part_list = g_simple_async_result_get_op_res_gpointer (simple);
Packit 15f964
Packit 15f964
	mail_parser_run (
Packit 15f964
		E_MAIL_PARSER (source_object),
Packit 15f964
		part_list, cancellable);
Packit 15f964
}
Packit 15f964
Packit 15f964
/**
Packit 15f964
 * e_mail_parser_parse:
Packit 15f964
 * @parser: an #EMailParser
Packit 15f964
 * @message: a #CamelMimeMessage
Packit 15f964
 * @callback: a #GAsyncReadyCallback
Packit 15f964
 * @cancellable: (allow-none) a #GCancellable
Packit 15f964
 * @user_data: (allow-none) user data passed to the callback
Packit 15f964
 *
Packit 15f964
 * Asynchronous version of e_mail_parser_parse_sync().
Packit 15f964
 */
Packit 15f964
void
Packit 15f964
e_mail_parser_parse (EMailParser *parser,
Packit 15f964
                     CamelFolder *folder,
Packit 15f964
                     const gchar *message_uid,
Packit 15f964
                     CamelMimeMessage *message,
Packit 15f964
                     GAsyncReadyCallback callback,
Packit 15f964
                     GCancellable *cancellable,
Packit 15f964
                     gpointer user_data)
Packit 15f964
{
Packit 15f964
	GSimpleAsyncResult *simple;
Packit 15f964
	EMailPartList *part_list;
Packit 15f964
Packit 15f964
	g_return_if_fail (E_IS_MAIL_PARSER (parser));
Packit 15f964
	g_return_if_fail (CAMEL_IS_MIME_MESSAGE (message));
Packit 15f964
Packit 15f964
	part_list = e_mail_part_list_new (message, message_uid, folder);
Packit 15f964
Packit 15f964
	simple = g_simple_async_result_new (
Packit 15f964
		G_OBJECT (parser), callback,
Packit 15f964
		user_data, e_mail_parser_parse);
Packit 15f964
Packit 15f964
	g_simple_async_result_set_check_cancellable (simple, cancellable);
Packit 15f964
Packit 15f964
	g_simple_async_result_set_op_res_gpointer (
Packit 15f964
		simple, part_list, (GDestroyNotify) g_object_unref);
Packit 15f964
Packit 15f964
	g_simple_async_result_run_in_thread (
Packit 15f964
		simple, mail_parser_parse_thread,
Packit 15f964
		G_PRIORITY_DEFAULT, cancellable);
Packit 15f964
Packit 15f964
	g_object_unref (simple);
Packit 15f964
}
Packit 15f964
Packit 15f964
EMailPartList *
Packit 15f964
e_mail_parser_parse_finish (EMailParser *parser,
Packit 15f964
                            GAsyncResult *result,
Packit 15f964
                            GError **error)
Packit 15f964
{
Packit 15f964
	GSimpleAsyncResult *simple;
Packit 15f964
	EMailPartList *part_list;
Packit 15f964
Packit 15f964
	g_return_val_if_fail (
Packit 15f964
		g_simple_async_result_is_valid (
Packit 15f964
		result, G_OBJECT (parser), e_mail_parser_parse), NULL);
Packit 15f964
Packit 15f964
	simple = G_SIMPLE_ASYNC_RESULT (result);
Packit 15f964
	part_list = g_simple_async_result_get_op_res_gpointer (simple);
Packit 15f964
Packit 15f964
	if (camel_debug_start ("emformat:parser")) {
Packit 15f964
		GQueue queue = G_QUEUE_INIT;
Packit 15f964
Packit 15f964
		printf (
Packit 15f964
			"%s finished with EMailPartList:\n",
Packit 15f964
			G_OBJECT_TYPE_NAME (parser));
Packit 15f964
Packit 15f964
		e_mail_part_list_queue_parts (part_list, NULL, &queue);
Packit 15f964
Packit 15f964
		while (!g_queue_is_empty (&queue)) {
Packit 15f964
			EMailPart *part;
Packit 15f964
Packit 15f964
			part = g_queue_pop_head (&queue);
Packit 15f964
Packit 15f964
			printf (
Packit 15f964
				"	id: %s | cid: %s | mime_type: %s | "
Packit 15f964
				"is_hidden: %d | is_attachment: %d\n",
Packit 15f964
				e_mail_part_get_id (part),
Packit 15f964
				e_mail_part_get_cid (part),
Packit 15f964
				e_mail_part_get_mime_type (part),
Packit 15f964
				part->is_hidden ? 1 : 0,
Packit 15f964
				e_mail_part_get_is_attachment (part) ? 1 : 0);
Packit 15f964
Packit 15f964
			g_object_unref (part);
Packit 15f964
		}
Packit 15f964
Packit 15f964
		camel_debug_end ();
Packit 15f964
	}
Packit 15f964
Packit 15f964
	return g_object_ref (part_list);
Packit 15f964
}
Packit 15f964
Packit 15f964
GQueue *
Packit 15f964
e_mail_parser_get_parsers_for_part (EMailParser *parser,
Packit 15f964
				    CamelMimePart *part)
Packit 15f964
{
Packit 15f964
	CamelContentType *ct;
Packit 15f964
	gchar *mime_type;
Packit 15f964
	GQueue *parsers;
Packit 15f964
Packit 15f964
	g_return_val_if_fail (E_IS_MAIL_PARSER (parser), NULL);
Packit 15f964
	g_return_val_if_fail (CAMEL_IS_MIME_PART (part), NULL);
Packit 15f964
Packit 15f964
	ct = camel_mime_part_get_content_type (part);
Packit 15f964
	if (!ct) {
Packit 15f964
		mime_type = (gchar *) "application/vnd.evolution.error";
Packit 15f964
	} else {
Packit 15f964
		gchar *tmp;
Packit 15f964
		tmp = camel_content_type_simple (ct);
Packit 15f964
		mime_type = g_ascii_strdown (tmp, -1);
Packit 15f964
		g_free (tmp);
Packit 15f964
	}
Packit 15f964
Packit 15f964
	parsers = e_mail_parser_get_parsers (parser, mime_type);
Packit 15f964
Packit 15f964
	if (ct)
Packit 15f964
		g_free (mime_type);
Packit 15f964
Packit 15f964
	return parsers;
Packit 15f964
}
Packit 15f964
Packit 15f964
GQueue *
Packit 15f964
e_mail_parser_get_parsers (EMailParser *parser,
Packit 15f964
			   const gchar *mime_type)
Packit 15f964
{
Packit 15f964
	EMailExtensionRegistry *reg;
Packit 15f964
	EMailParserClass *parser_class;
Packit 15f964
	gchar *as_mime_type;
Packit 15f964
	GQueue *parsers;
Packit 15f964
Packit 15f964
	g_return_val_if_fail (E_IS_MAIL_PARSER (parser), NULL);
Packit 15f964
Packit 15f964
	parser_class = E_MAIL_PARSER_GET_CLASS (parser);
Packit 15f964
	g_return_val_if_fail (parser_class != NULL, NULL);
Packit 15f964
Packit 15f964
	if (mime_type)
Packit 15f964
		as_mime_type = g_ascii_strdown (mime_type, -1);
Packit 15f964
	else
Packit 15f964
		as_mime_type = NULL;
Packit 15f964
Packit 15f964
	reg = E_MAIL_EXTENSION_REGISTRY (parser_class->extension_registry);
Packit 15f964
Packit 15f964
	parsers = e_mail_extension_registry_get_for_mime_type (reg, as_mime_type);
Packit 15f964
	if (!parsers)
Packit 15f964
		parsers = e_mail_extension_registry_get_fallback (reg, as_mime_type);
Packit 15f964
Packit 15f964
	g_free (as_mime_type);
Packit 15f964
Packit 15f964
	return parsers;
Packit 15f964
}
Packit 15f964
Packit 15f964
gboolean
Packit 15f964
e_mail_parser_parse_part (EMailParser *parser,
Packit 15f964
                          CamelMimePart *part,
Packit 15f964
                          GString *part_id,
Packit 15f964
                          GCancellable *cancellable,
Packit 15f964
                          GQueue *out_mail_parts)
Packit 15f964
{
Packit 15f964
	CamelContentType *ct;
Packit 15f964
	gchar *mime_type;
Packit 15f964
	gint handled;
Packit 15f964
Packit 15f964
	ct = camel_mime_part_get_content_type (part);
Packit 15f964
	if (!ct) {
Packit 15f964
		mime_type = (gchar *) "application/vnd.evolution.error";
Packit 15f964
	} else {
Packit 15f964
		gchar *tmp;
Packit 15f964
		tmp = camel_content_type_simple (ct);
Packit 15f964
		mime_type = g_ascii_strdown (tmp, -1);
Packit 15f964
		g_free (tmp);
Packit 15f964
	}
Packit 15f964
Packit 15f964
	handled = e_mail_parser_parse_part_as (
Packit 15f964
		parser, part, part_id, mime_type,
Packit 15f964
		cancellable, out_mail_parts);
Packit 15f964
Packit 15f964
	if (ct) {
Packit 15f964
		g_free (mime_type);
Packit 15f964
	}
Packit 15f964
Packit 15f964
	return handled;
Packit 15f964
}
Packit 15f964
Packit 15f964
gboolean
Packit 15f964
e_mail_parser_parse_part_as (EMailParser *parser,
Packit 15f964
                             CamelMimePart *part,
Packit 15f964
                             GString *part_id,
Packit 15f964
                             const gchar *mime_type,
Packit 15f964
                             GCancellable *cancellable,
Packit 15f964
                             GQueue *out_mail_parts)
Packit 15f964
{
Packit 15f964
	GQueue *parsers;
Packit 15f964
	GList *iter;
Packit 15f964
	gboolean mime_part_handled = FALSE;
Packit 15f964
Packit 15f964
	parsers = e_mail_parser_get_parsers (parser, mime_type);
Packit 15f964
Packit 15f964
	if (parsers == NULL) {
Packit 15f964
		e_mail_parser_wrap_as_attachment (
Packit 15f964
			parser, part, part_id, out_mail_parts);
Packit 15f964
		return TRUE;
Packit 15f964
	}
Packit 15f964
Packit 15f964
	for (iter = parsers->head; iter; iter = iter->next) {
Packit 15f964
		EMailParserExtension *extension;
Packit 15f964
Packit 15f964
		extension = iter->data;
Packit 15f964
		if (!extension)
Packit 15f964
			continue;
Packit 15f964
Packit 15f964
		mime_part_handled = e_mail_parser_extension_parse (
Packit 15f964
			extension, parser, part, part_id,
Packit 15f964
			cancellable, out_mail_parts);
Packit 15f964
Packit 15f964
		if (mime_part_handled)
Packit 15f964
			break;
Packit 15f964
	}
Packit 15f964
Packit 15f964
	return mime_part_handled;
Packit 15f964
}
Packit 15f964
Packit 15f964
void
Packit 15f964
e_mail_parser_error (EMailParser *parser,
Packit 15f964
                     GQueue *out_mail_parts,
Packit 15f964
                     const gchar *format,
Packit 15f964
                     ...)
Packit 15f964
{
Packit 15f964
	const gchar *mime_type = "application/vnd.evolution.error";
Packit 15f964
	EMailPart *mail_part;
Packit 15f964
	CamelMimePart *part;
Packit 15f964
	gchar *errmsg;
Packit 15f964
	gchar *uri;
Packit 15f964
	va_list ap;
Packit 15f964
Packit 15f964
	g_return_if_fail (E_IS_MAIL_PARSER (parser));
Packit 15f964
	g_return_if_fail (out_mail_parts != NULL);
Packit 15f964
	g_return_if_fail (format != NULL);
Packit 15f964
Packit 15f964
	va_start (ap, format);
Packit 15f964
	errmsg = g_strdup_vprintf (format, ap);
Packit 15f964
Packit 15f964
	part = camel_mime_part_new ();
Packit 15f964
	camel_mime_part_set_content (
Packit 15f964
		part, errmsg, strlen (errmsg), mime_type);
Packit 15f964
	g_free (errmsg);
Packit 15f964
	va_end (ap);
Packit 15f964
Packit 15f964
	g_mutex_lock (&parser->priv->mutex);
Packit 15f964
	parser->priv->last_error++;
Packit 15f964
	uri = g_strdup_printf (".error.%d", parser->priv->last_error);
Packit 15f964
	g_mutex_unlock (&parser->priv->mutex);
Packit 15f964
Packit 15f964
	mail_part = e_mail_part_new (part, uri);
Packit 15f964
	e_mail_part_set_mime_type (mail_part, mime_type);
Packit 15f964
	mail_part->is_error = TRUE;
Packit 15f964
Packit 15f964
	g_free (uri);
Packit 15f964
	g_object_unref (part);
Packit 15f964
Packit 15f964
	g_queue_push_tail (out_mail_parts, mail_part);
Packit 15f964
}
Packit 15f964
Packit 15f964
static void
Packit 15f964
attachment_loaded (EAttachment *attachment,
Packit 15f964
                   GAsyncResult *res,
Packit 15f964
                   gpointer user_data)
Packit 15f964
{
Packit 15f964
	EShell *shell;
Packit 15f964
	GtkWindow *window;
Packit 15f964
Packit 15f964
	shell = e_shell_get_default ();
Packit 15f964
	window = e_shell_get_active_window (shell);
Packit 15f964
Packit 15f964
	e_attachment_load_handle_error (attachment, res, window);
Packit 15f964
Packit 15f964
	g_object_unref (attachment);
Packit 15f964
}
Packit 15f964
Packit 15f964
/* Idle callback */
Packit 15f964
static gboolean
Packit 15f964
load_attachment_idle (EAttachment *attachment)
Packit 15f964
{
Packit 15f964
	e_attachment_load_async (
Packit 15f964
		attachment,
Packit 15f964
		(GAsyncReadyCallback) attachment_loaded, NULL);
Packit 15f964
Packit 15f964
	return FALSE;
Packit 15f964
}
Packit 15f964
Packit 15f964
void
Packit 15f964
e_mail_parser_wrap_as_attachment (EMailParser *parser,
Packit 15f964
                                  CamelMimePart *part,
Packit 15f964
                                  GString *part_id,
Packit 15f964
                                  GQueue *parts_queue)
Packit 15f964
{
Packit 15f964
	EMailPartAttachment *empa;
Packit 15f964
	EAttachment *attachment;
Packit 15f964
	EMailPart *first_part;
Packit 15f964
	const gchar *snoop_mime_type;
Packit 15f964
	GQueue *extensions;
Packit 15f964
	CamelContentType *ct;
Packit 15f964
	gchar *mime_type;
Packit 15f964
	CamelDataWrapper *dw;
Packit 15f964
	GByteArray *ba;
Packit 15f964
	gsize size;
Packit 15f964
	gint part_id_len;
Packit 15f964
Packit 15f964
	ct = camel_mime_part_get_content_type (part);
Packit 15f964
	extensions = NULL;
Packit 15f964
	snoop_mime_type = NULL;
Packit 15f964
	if (ct) {
Packit 15f964
		EMailExtensionRegistry *reg;
Packit 15f964
		mime_type = camel_content_type_simple (ct);
Packit 15f964
Packit 15f964
		reg = e_mail_parser_get_extension_registry (parser);
Packit 15f964
		extensions = e_mail_extension_registry_get_for_mime_type (
Packit 15f964
			reg, mime_type);
Packit 15f964
Packit 15f964
		if (camel_content_type_is (ct, "text", "*") ||
Packit 15f964
		    camel_content_type_is (ct, "message", "*"))
Packit 15f964
			snoop_mime_type = mime_type;
Packit 15f964
		else
Packit 15f964
			g_free (mime_type);
Packit 15f964
	}
Packit 15f964
Packit 15f964
	if (!snoop_mime_type)
Packit 15f964
		snoop_mime_type = e_mail_part_snoop_type (part);
Packit 15f964
Packit 15f964
	if (!extensions) {
Packit 15f964
		EMailExtensionRegistry *reg;
Packit 15f964
Packit 15f964
		reg = e_mail_parser_get_extension_registry (parser);
Packit 15f964
		extensions = e_mail_extension_registry_get_for_mime_type (
Packit 15f964
			reg, snoop_mime_type);
Packit 15f964
Packit 15f964
		if (!extensions) {
Packit 15f964
			extensions = e_mail_extension_registry_get_fallback (
Packit 15f964
				reg, snoop_mime_type);
Packit 15f964
		}
Packit 15f964
	}
Packit 15f964
Packit 15f964
	part_id_len = part_id->len;
Packit 15f964
	g_string_append (part_id, ".attachment");
Packit 15f964
Packit 15f964
	empa = e_mail_part_attachment_new (part, part_id->str);
Packit 15f964
	empa->shown = extensions && (!g_queue_is_empty (extensions) &&
Packit 15f964
		e_mail_part_is_inline (part, extensions));
Packit 15f964
	empa->snoop_mime_type = snoop_mime_type;
Packit 15f964
Packit 15f964
	first_part = g_queue_peek_head (parts_queue);
Packit 15f964
	if (first_part != NULL && !E_IS_MAIL_PART_ATTACHMENT (first_part)) {
Packit 15f964
		const gchar *id = e_mail_part_get_id (first_part);
Packit 15f964
		empa->part_id_with_attachment = g_strdup (id);
Packit 15f964
		first_part->is_hidden = TRUE;
Packit 15f964
	}
Packit 15f964
Packit 15f964
	attachment = e_mail_part_attachment_ref_attachment (empa);
Packit 15f964
Packit 15f964
	e_attachment_set_initially_shown (attachment, empa->shown);
Packit 15f964
	e_attachment_set_can_show (
Packit 15f964
		attachment,
Packit 15f964
		extensions && !g_queue_is_empty (extensions));
Packit 15f964
Packit 15f964
	/* Try to guess size of the attachments */
Packit 15f964
	dw = camel_medium_get_content (CAMEL_MEDIUM (part));
Packit 15f964
	ba = camel_data_wrapper_get_byte_array (dw);
Packit 15f964
	if (ba) {
Packit 15f964
		size = ba->len;
Packit 15f964
Packit 15f964
		if (camel_mime_part_get_encoding (part) == CAMEL_TRANSFER_ENCODING_BASE64)
Packit 15f964
			size = size / 1.37;
Packit 15f964
	} else {
Packit 15f964
		size = 0;
Packit 15f964
	}
Packit 15f964
Packit 15f964
	/* e_attachment_load_async must be called from main thread */
Packit 15f964
	/* Prioritize ahead of GTK+ redraws. */
Packit 15f964
	g_idle_add_full (
Packit 15f964
		G_PRIORITY_HIGH_IDLE,
Packit 15f964
		(GSourceFunc) load_attachment_idle,
Packit 15f964
		g_object_ref (attachment),
Packit 15f964
		NULL);
Packit 15f964
Packit 15f964
	if (size != 0) {
Packit 15f964
		GFileInfo *file_info;
Packit 15f964
Packit 15f964
		file_info = e_attachment_ref_file_info (attachment);
Packit 15f964
Packit 15f964
		if (file_info == NULL) {
Packit 15f964
			file_info = g_file_info_new ();
Packit 15f964
			g_file_info_set_content_type (
Packit 15f964
				file_info, empa->snoop_mime_type);
Packit 15f964
		}
Packit 15f964
Packit 15f964
		g_file_info_set_size (file_info, size);
Packit 15f964
		e_attachment_set_file_info (attachment, file_info);
Packit 15f964
Packit 15f964
		g_object_unref (file_info);
Packit 15f964
	}
Packit 15f964
Packit 15f964
	g_object_unref (attachment);
Packit 15f964
Packit 15f964
	g_string_truncate (part_id, part_id_len);
Packit 15f964
Packit 15f964
	/* Push to head, not tail. */
Packit 15f964
	g_queue_push_head (parts_queue, empa);
Packit 15f964
}
Packit 15f964
Packit 15f964
CamelSession *
Packit 15f964
e_mail_parser_get_session (EMailParser *parser)
Packit 15f964
{
Packit 15f964
	g_return_val_if_fail (E_IS_MAIL_PARSER (parser), NULL);
Packit 15f964
Packit 15f964
	return parser->priv->session;
Packit 15f964
}
Packit 15f964
Packit 15f964
EMailExtensionRegistry *
Packit 15f964
e_mail_parser_get_extension_registry (EMailParser *parser)
Packit 15f964
{
Packit 15f964
	EMailParserClass *parser_class;
Packit 15f964
Packit 15f964
	g_return_val_if_fail (E_IS_MAIL_PARSER (parser), NULL);
Packit 15f964
Packit 15f964
	parser_class = E_MAIL_PARSER_GET_CLASS (parser);
Packit 15f964
	g_return_val_if_fail (parser_class != NULL, NULL);
Packit 15f964
Packit 15f964
	return E_MAIL_EXTENSION_REGISTRY (parser_class->extension_registry);
Packit 15f964
}