/* -*- Mode: C; tab-width: 8; indent-tabs-mode: t; c-basic-offset: 8 -*- */
/* evolution-mbox-importer.c
*
* Authors: Iain Holmes <iain@ximian.com>
*
* Copyright (C) 2001 Ximian, Inc.
*
* This program is free software; you can redistribute it and/or
* modify it under the terms of version 2 of the GNU General Public
* License as published by the Free Software Foundation.
*
* This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
* General Public License for more details.
*
* You should have received a copy of the GNU General Public
* License along with this program; if not, write to the
* Free Software Foundation, Inc., 59 Temple Place - Suite 330,
* Boston, MA 02111-1307, USA.
*/
#ifdef HAVE_CONFIG_H
#include <config.h>
#endif
#include <stdio.h>
#include <ctype.h>
#include <bonobo/bonobo-object.h>
#include <bonobo/bonobo-generic-factory.h>
#include <camel/camel-exception.h>
#include <camel/camel-mime-message.h>
#include <camel/camel-mime-parser.h>
#include <camel/camel-mime-part.h>
#include <importer/evolution-importer.h>
#include <importer/GNOME_Evolution_Importer.h>
#include "mozilla-status-headers.h"
#include "mail/mail-importer.h"
#include "mail-tools.h"
#include "e-util/e-path.h"
/* #define IMPORTER_DEBUG */
#ifdef IMPORTER_DEBUG
#define IN g_print ("=====> %s (%d)\n", __FUNCTION__, __LINE__)
#define OUT g_print ("<==== %s (%d)\n", __FUNCTION__, __LINE__)
#else
#define IN
#define OUT
#endif
#define MBOX_FACTORY_IID "OAFIID:GNOME_Evolution_Mail_Mbox_ImporterFactory"
typedef struct {
MailImporter importer; /* Parent */
char *filename;
int num;
CamelMimeParser *mp;
gboolean is_folder;
} MboxImporter;
void mail_importer_module_init (void);
/* EvolutionImporter methods */
static int
string_to_int (const char *str)
{
int result = 0;
char *s;
for (s = (char *) str; *s; s++) {
char c = toupper (*s);
result *= 16;
if (c >= '0' && c <= '9') {
result += (c - '0');
} else if (c >= 'A' && c <= 'F') {
result += (c - 'A');
}
}
g_print ("%s became %d\n", str, result);
return result;
}
static CamelMessageInfo *
get_info_from_mozilla (const char *mozilla_status,
gboolean *deleted)
{
int status;
CamelMessageInfo *info;
info = g_new0 (CamelMessageInfo, 1);
*deleted = FALSE;
status = string_to_int (mozilla_status);
if (status == 0) {
return info;
}
if (status & MSG_FLAG_EXPUNGED) {
*deleted = TRUE;
g_free (info);
return NULL;
}
if (status & MSG_FLAG_READ) {
info->flags |= CAMEL_MESSAGE_SEEN;
}
if (status & MSG_FLAG_MARKED) {
info->flags |= CAMEL_MESSAGE_FLAGGED;
}
if (status & MSG_FLAG_REPLIED) {
info->flags |= CAMEL_MESSAGE_ANSWERED;
}
return info;
}
static void
process_item_fn (EvolutionImporter *eimporter,
CORBA_Object listener,
void *closure,
CORBA_Environment *ev)
{
MboxImporter *mbi = (MboxImporter *) closure;
MailImporter *importer = (MailImporter *) mbi;
gboolean done = FALSE;
CamelException *ex;
const char *mozilla_status;
if (importer->folder == NULL) {
GNOME_Evolution_ImporterListener_notifyResult (listener,
GNOME_Evolution_ImporterListener_NOT_READY,
TRUE, ev);
return;
}
if (mbi->is_folder == TRUE) {
GNOME_Evolution_ImporterListener_notifyResult (listener,
GNOME_Evolution_ImporterListener_OK,
FALSE, ev);
return;
}
ex = camel_exception_new ();
if (camel_mime_parser_step (mbi->mp, 0, 0) == HSCAN_FROM) {
/* Import the next message */
CamelMimeMessage *msg;
CamelMessageInfo *info;
gboolean deleted;
IN;
msg = camel_mime_message_new ();
if (camel_mime_part_construct_from_parser (CAMEL_MIME_PART (msg),
mbi->mp) == -1) {
g_warning ("Failed message %d", mbi->num);
camel_object_unref (CAMEL_OBJECT (msg));
done = TRUE;
}
mozilla_status = camel_medium_get_header (CAMEL_MEDIUM (msg), "X-Mozilla-Status");
if (mozilla_status != NULL) {
g_print ("Got Mozilla status header: %s\n", mozilla_status);
info = get_info_from_mozilla (mozilla_status, &deleted);
} else {
info = g_new0 (CamelMessageInfo, 1);
}
if (deleted == FALSE) {
/* write the mesg */
camel_folder_append_message (importer->folder, msg, info, NULL, ex);
g_free (info);
}
camel_object_unref (CAMEL_OBJECT (msg));
if (camel_exception_is_set (ex)) {
g_warning ("Failed message %d", mbi->num);
done = TRUE;
}
OUT;
} else {
IN;
/* all messages have now been imported */
camel_folder_sync (importer->folder, FALSE, ex);
camel_folder_thaw (importer->folder);
importer->frozen = FALSE;
done = TRUE;
OUT;
}
if (!done) {
camel_mime_parser_step (mbi->mp, 0, 0);
}
camel_exception_free (ex);
GNOME_Evolution_ImporterListener_notifyResult (listener,
GNOME_Evolution_ImporterListener_OK,
!done, ev);
return;
}
static gboolean
support_format_fn (EvolutionImporter *importer,
const char *filename,
void *closure)
{
char signature[6];
gboolean ret = FALSE;
int fd, n;
fd = open (filename, O_RDONLY);
if (fd == -1)
return FALSE;
n = read (fd, signature, 5);
if (n > 0) {
signature[n] = '\0';
if (!g_strncasecmp (signature, "From ", 5))
ret = TRUE;
}
close (fd);
return ret;
}
static void
importer_destroy_cb (GtkObject *object,
MboxImporter *mbi)
{
MailImporter *importer;
importer = (MailImporter *) mbi;
if (importer->frozen) {
camel_folder_sync (importer->folder, FALSE, NULL);
camel_folder_thaw (importer->folder);
}
if (importer->folder)
camel_object_unref (CAMEL_OBJECT (importer->folder));
g_free (mbi->filename);
if (mbi->mp)
camel_object_unref (CAMEL_OBJECT (mbi->mp));
g_free (mbi);
}
static void
folder_created_cb (BonoboListener *listener,
const char *event_name,
const BonoboArg *event_data,
CORBA_Environment *ev,
MailImporter *importer)
{
char *fullpath;
GNOME_Evolution_Storage_FolderResult *result;
CamelException *ex;
if (strcmp (event_name, "evolution-shell:folder_created") != 0) {
return; /* Unknown event */
}
result = event_data->_value;
fullpath = g_strconcat ("file://", result->path, NULL);
ex = camel_exception_new ();
importer->folder = mail_tool_uri_to_folder (fullpath, CAMEL_STORE_FOLDER_CREATE, ex);
if (camel_exception_is_set (ex)) {
g_warning ("Error opening %s", fullpath);
camel_exception_free (ex);
g_free (fullpath);
return;
}
camel_folder_freeze (importer->folder);
importer->frozen = TRUE;
g_free (fullpath);
bonobo_object_unref (BONOBO_OBJECT (listener));
}
static gboolean
load_file_fn (EvolutionImporter *eimporter,
const char *filename,
const char *folderpath,
void *closure)
{
MboxImporter *mbi;
MailImporter *importer;
gboolean delayed = FALSE;
struct stat buf;
int fd;
mbi = (MboxImporter *) closure;
importer = (MailImporter *) mbi;
mbi->filename = g_strdup (filename);
fd = open (filename, O_RDONLY);
if (fd == -1) {
g_warning ("Cannot open file");
return FALSE;
}
fstat (fd, &buf);
if (S_ISREG (buf.st_mode)) {
mbi->mp = camel_mime_parser_new ();
camel_mime_parser_scan_from (mbi->mp, TRUE);
if (camel_mime_parser_init_with_fd (mbi->mp, fd) == -1) {
g_warning ("Unable to process spool folder");
goto fail;
}
mbi->is_folder = FALSE;
} else {
mbi->is_folder = TRUE;
}
importer->mstream = NULL;
if (folderpath == NULL || *folderpath == '\0') {
importer->folder = mail_tool_get_local_inbox (NULL);
} else {
char *parent, *tmp, *fullpath, *homedir;
const char *name;
BonoboListener *listener;
CamelException *ex;
tmp = gnome_util_prepend_user_home ("evolution/local");
homedir = g_strconcat ("file://", tmp, NULL);
g_free (tmp);
fullpath = e_path_to_physical (homedir, folderpath);
ex = camel_exception_new ();
importer->folder = mail_tool_uri_to_folder (fullpath, 0, ex);
g_free (homedir);
if (camel_exception_is_set (ex) || importer->folder == NULL) {
/* Make a new directory */
name = strrchr (folderpath, '/');
if (name == NULL) {
parent = g_strdup ("/");
name = folderpath;
} else {
name += 1;
parent = g_dirname (folderpath);
}
listener = bonobo_listener_new (NULL, NULL);
gtk_signal_connect (GTK_OBJECT (listener), "event-notify",
GTK_SIGNAL_FUNC (folder_created_cb),
importer);
mail_importer_create_folder (parent, name, NULL, listener);
camel_exception_free (ex);
ex = camel_exception_new ();
importer->folder = NULL;
g_print ("No folder yet\n");
delayed = TRUE;
g_free (parent);
}
camel_exception_free (ex);
g_free (fullpath);
}
if (importer->folder == NULL && delayed == FALSE){
g_warning ("Bad folder\n");
goto fail;
}
if (importer->folder != NULL) {
camel_folder_freeze (importer->folder);
importer->frozen = TRUE;
}
return TRUE;
fail:
camel_object_unref (CAMEL_OBJECT (mbi->mp));
mbi->mp = NULL;
return FALSE;
}
static BonoboObject *
mbox_factory_fn (BonoboGenericFactory *_factory,
void *closure)
{
EvolutionImporter *importer;
MboxImporter *mbox;
mbox = g_new0 (MboxImporter, 1);
importer = evolution_importer_new (support_format_fn, load_file_fn,
process_item_fn, NULL, mbox);
gtk_signal_connect (GTK_OBJECT (importer), "destroy",
GTK_SIGNAL_FUNC (importer_destroy_cb), mbox);
return BONOBO_OBJECT (importer);
}
/* Entry point */
void
mail_importer_module_init (void)
{
static gboolean initialised = FALSE;
BonoboGenericFactory *factory;
if (initialised == TRUE)
return;
factory = bonobo_generic_factory_new (MBOX_FACTORY_IID,
mbox_factory_fn, NULL);
if (factory == NULL)
g_warning ("Could not initialise Outlook importer factory.");
initialised = TRUE;
}