aboutsummaryrefslogblamecommitdiffstats
path: root/mail/importers/evolution-mbox-importer.c
blob: ffa8872fc60ba285531a56e796dbe0284bfcd601 (plain) (tree)
1
2
3
4
5
6
7
8
9
10







                                                                           

                                                                   















                                                                    
                  
                  
 


                                          



                                     



                                              

                                   
                               

                       

                          
                             






                                                               








                                                                           
                           

               
                                      


                               




























































                                                      









                                                      
                                   
 













                                                                                                          




                                                                   
                                 
 
                   







                                                                                 








                                                                                                  
                                    
                                                                                            


                                      




                                                                  
                    
                
                   




                                                                




                                                       























































                                                                                           


















                                                                         
                                                                                             







                                                         
                                               

                                



                                                       


                                           
                                     



                               

                                 












                                               










                                                                         


                                 
                                                        
                                                                    
                

                                                        
                                         







                                                                       
                                                                             
                                 

















                                                                                   

                                                    

                                                    
                                       
                                        
                 

                                          
         
 
                                                          
                                           


                          



                                                       
 




















                                                                           
 





















                                                                             
/* -*- 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;
}