/*
* This program is free software; you can redistribute it and/or
* modify it under the terms of the GNU Lesser General Public
* License as published by the Free Software Foundation; either
* version 2 of the License, or (at your option) version 3.
*
* 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
* Lesser General Public License for more details.
*
* You should have received a copy of the GNU Lesser General Public
* License along with the program; if not, see
*
*
* Authors:
* Iain Holmes
* Michael Zucchi
*
* Copyright (C) 1999-2008 Novell, Inc. (www.novell.com)
*
*/
#include
#include
#include
#include
#include
#include
#include
#include
#include
#include
#include "mail-importer.h"
#include "mail/e-mail-backend.h"
#include "shell/e-shell.h"
#define d(x)
struct _elm_import_msg {
MailMsg base;
EImport *import;
EImportTargetHome *target;
GMutex status_lock;
gchar *status_what;
gint status_pc;
gint status_timeout_id;
GCancellable *status;
};
static GHashTable *
parse_elm_rc (const gchar *elmrc)
{
gchar line[4096];
FILE *handle;
GHashTable *prefs;
prefs = g_hash_table_new_full (
g_str_hash, g_str_equal,
(GDestroyNotify) g_free,
(GDestroyNotify) g_free);
if (!g_file_test (elmrc, G_FILE_TEST_IS_REGULAR))
return prefs;
handle = fopen (elmrc, "r");
if (handle == NULL)
return prefs;
while (fgets (line, 4096, handle) != NULL) {
gchar *linestart, *end;
gchar *key, *value;
if (*line == '#' &&
(line[1] != '#' && line[2] != '#')) {
continue;
} else if (*line == '\n') {
continue;
} else if (*line == '#' && line[1] == '#' && line[2] == '#') {
linestart = line + 4;
} else {
linestart = line;
}
end = strstr (linestart, " = ");
if (end == NULL) {
g_warning ("Broken line");
continue;
}
*end = 0;
key = g_strdup (linestart);
linestart = end + 3;
end = strchr (linestart, '\n');
if (end == NULL) {
g_warning ("Broken line");
g_free (key);
continue;
}
*end = 0;
value = g_strdup (linestart);
g_hash_table_insert (prefs, key, value);
}
fclose (handle);
return prefs;
}
static gchar *
elm_get_rc (EImport *ei,
const gchar *name)
{
GHashTable *prefs;
gchar *elmrc;
prefs = g_object_get_data ((GObject *) ei, "elm-rc");
if (prefs == NULL) {
elmrc = g_build_filename (g_get_home_dir (), ".elm/elmrc", NULL);
prefs = parse_elm_rc (elmrc);
g_free (elmrc);
g_object_set_data ((GObject *) ei, "elm-rc", prefs);
}
if (prefs == NULL)
return NULL;
else
return g_hash_table_lookup (prefs, name);
}
static gboolean
elm_supported (EImport *ei,
EImportTarget *target,
EImportImporter *im)
{
const gchar *maildir;
gchar *elmdir;
gboolean mailexists, exists;
if (target->type != E_IMPORT_TARGET_HOME)
return FALSE;
elmdir = g_build_filename (g_get_home_dir (), ".elm", NULL);
exists = g_file_test (elmdir, G_FILE_TEST_IS_DIR);
g_free (elmdir);
if (!exists)
return FALSE;
maildir = elm_get_rc (ei, "maildir");
if (maildir == NULL)
maildir = "Mail";
if (!g_path_is_absolute (maildir))
elmdir = g_build_filename (g_get_home_dir (), maildir, NULL);
else
elmdir = g_strdup (maildir);
mailexists = g_file_test (elmdir, G_FILE_TEST_IS_DIR);
g_free (elmdir);
return mailexists;
}
static gchar *
elm_import_desc (struct _elm_import_msg *m)
{
return g_strdup (_("Importing Elm data"));
}
static MailImporterSpecial elm_special_folders[] = {
{ "received", "Inbox" },
{ NULL },
};
static void
elm_import_exec (struct _elm_import_msg *m,
GCancellable *cancellable,
GError **error)
{
EShell *shell;
EShellBackend *shell_backend;
EMailSession *session;
const gchar *maildir;
gchar *elmdir;
/* XXX Dig up the EMailSession from the default EShell.
* Since the EImport framework doesn't allow for user
* data, I don't see how else to get to it. */
shell = e_shell_get_default ();
shell_backend = e_shell_get_backend_by_name (shell, "mail");
session = e_mail_backend_get_session (E_MAIL_BACKEND (shell_backend));
maildir = elm_get_rc (m->import, "maildir");
if (maildir == NULL)
maildir = "Mail";
if (!g_path_is_absolute (maildir))
elmdir = g_build_filename (g_get_home_dir (), maildir, NULL);
else
elmdir = g_strdup (maildir);
mail_importer_import_folders_sync (
session, elmdir, elm_special_folders, 0, m->status);
g_free (elmdir);
}
static void
elm_import_done (struct _elm_import_msg *m)
{
e_import_complete (m->import, (EImportTarget *) m->target);
}
static void
elm_import_free (struct _elm_import_msg *m)
{
g_object_unref (m->status);
g_free (m->status_what);
g_mutex_clear (&m->status_lock);
g_source_remove (m->status_timeout_id);
m->status_timeout_id = 0;
g_object_unref (m->import);
}
static void
elm_status (CamelOperation *op,
const gchar *what,
gint pc,
gpointer data)
{
struct _elm_import_msg *importer = data;
g_mutex_lock (&importer->status_lock);
g_free (importer->status_what);
importer->status_what = g_strdup (what);
importer->status_pc = pc;
g_mutex_unlock (&importer->status_lock);
}
static gboolean
elm_status_timeout (gpointer data)
{
struct _elm_import_msg *importer = data;
gint pc;
gchar *what;
if (importer->status_what) {
g_mutex_lock (&importer->status_lock);
what = importer->status_what;
importer->status_what = NULL;
pc = importer->status_pc;
g_mutex_unlock (&importer->status_lock);
e_import_status (
importer->import, (EImportTarget *)
importer->target, what, pc);
}
return TRUE;
}
static MailMsgInfo elm_import_info = {
sizeof (struct _elm_import_msg),
(MailMsgDescFunc) elm_import_desc,
(MailMsgExecFunc) elm_import_exec,
(MailMsgDoneFunc) elm_import_done,
(MailMsgFreeFunc) elm_import_free
};
static gint
mail_importer_elm_import (EImport *ei,
EImportTarget *target)
{
struct _elm_import_msg *m;
gint id;
m = mail_msg_new (&elm_import_info);
g_datalist_set_data (&target->data, "elm-msg", m);
m->import = ei;
g_object_ref (m->import);
m->target = (EImportTargetHome *) target;
m->status_timeout_id =
e_named_timeout_add (100, elm_status_timeout, m);
g_mutex_init (&m->status_lock);
m->status = camel_operation_new ();
g_signal_connect (
m->status, "status",
G_CALLBACK (elm_status), m);
id = m->base.seq;
mail_msg_fast_ordered_push (m);
return id;
}
static void
checkbox_toggle_cb (GtkToggleButton *tb,
EImportTarget *target)
{
g_datalist_set_data (
&target->data, "elm-do-mail",
GINT_TO_POINTER (gtk_toggle_button_get_active (tb)));
}
static GtkWidget *
elm_getwidget (EImport *ei,
EImportTarget *target,
EImportImporter *im)
{
GtkWidget *box, *w;
g_datalist_set_data (
&target->data, "elm-do-mail", GINT_TO_POINTER (TRUE));
box = gtk_box_new (GTK_ORIENTATION_VERTICAL, 2);
w = gtk_check_button_new_with_label (_("Mail"));
gtk_toggle_button_set_active ((GtkToggleButton *) w, TRUE);
g_signal_connect (
w, "toggled",
G_CALLBACK (checkbox_toggle_cb), target);
gtk_box_pack_start ((GtkBox *) box, w, FALSE, FALSE, 0);
gtk_widget_show_all (box);
return box;
}
static void
elm_import (EImport *ei,
EImportTarget *target,
EImportImporter *im)
{
if (GPOINTER_TO_INT (g_datalist_get_data (&target->data, "elm-do-mail")))
mail_importer_elm_import (ei, target);
else
e_import_complete (ei, target);
}
static void
elm_cancel (EImport *ei,
EImportTarget *target,
EImportImporter *im)
{
struct _elm_import_msg *m = g_datalist_get_data (&target->data, "elm-msg");
if (m)
g_cancellable_cancel (m->status);
}
static EImportImporter elm_importer = {
E_IMPORT_TARGET_HOME,
0,
elm_supported,
elm_getwidget,
elm_import,
elm_cancel,
NULL, /* get_preview */
};
EImportImporter *
elm_importer_peek (void)
{
elm_importer.name = _("Evolution Elm importer");
elm_importer.description = _("Import mail from Elm.");
return &elm_importer;
}