aboutsummaryrefslogtreecommitdiffstats
path: root/mail/mail-local.c
diff options
context:
space:
mode:
Diffstat (limited to 'mail/mail-local.c')
-rw-r--r--mail/mail-local.c1334
1 files changed, 710 insertions, 624 deletions
diff --git a/mail/mail-local.c b/mail/mail-local.c
index 3f9c9bdd74..c5b10fde0f 100644
--- a/mail/mail-local.c
+++ b/mail/mail-local.c
@@ -26,40 +26,27 @@
* USA
*/
-/*
- TODO:
-
- If we are going to have all this LocalStore stuff, then the LocalStore
- should have a reconfigure_folder method on it, as, in reality, it is
- the maintainer of this information.
-
-*/
-
#ifdef HAVE_CONFIG_H
#include <config.h>
#endif
+#include <unistd.h>
+#include <errno.h>
+
#include <gnome-xml/xmlmemory.h>
#include <libgnomeui/gnome-dialog.h>
#include <libgnomeui/gnome-dialog-util.h>
#include <glade/glade.h>
+#include "gal/widgets/e-gui-utils.h"
+#include "e-util/e-path.h"
+
#include "Evolution.h"
#include "evolution-storage.h"
#include "evolution-shell-component.h"
#include "evolution-storage-listener.h"
-#include "gal/widgets/e-gui-utils.h"
-#include "e-util/e-path.h"
-
#include "camel/camel.h"
-#include "camel/camel-vee-store.h"
-#include "camel/camel-vee-folder.h"
-#include "camel/camel-vtrash-folder.h"
-
-#include "filter/vfolder-context.h"
-#include "filter/vfolder-rule.h"
-#include "filter/vfolder-editor.h"
#include "mail.h"
#include "mail-local.h"
@@ -69,83 +56,131 @@
#include "mail-folder-cache.h"
#include "mail-vfolder.h"
-#define d(x)
+#define d(x) x
+/* ** MailLocalStore ** (protos) ************************************************** */
-/* Local folder metainfo */
+#define MAIL_LOCAL_STORE_TYPE (mail_local_store_get_type ())
+#define MAIL_LOCAL_STORE(obj) (CAMEL_CHECK_CAST((obj), MAIL_LOCAL_STORE_TYPE, MailLocalStore))
+#define MAIL_LOCAL_STORE_CLASS(k) (CAMEL_CHECK_CLASS_CAST ((k), MAIL_LOCAL_STORE_TYPE, MailLocalStoreClass))
+#define MAIL_IS_LOCAL_STORE(o) (CAMEL_CHECK_TYPE((o), MAIL_LOCAL_STORE_TYPE))
+
+typedef struct {
+ CamelStore parent_object;
+} MailLocalStore;
+
+typedef struct {
+ CamelStoreClass parent_class;
+} MailLocalStoreClass;
+
+static CamelType mail_local_store_get_type (void);
+
+/* ** MailLocalFolder ** (protos) ************************************************* */
+
+#define MAIL_LOCAL_FOLDER_TYPE (mail_local_folder_get_type ())
+#define MAIL_LOCAL_FOLDER(obj) (CAMEL_CHECK_CAST((obj), MAIL_LOCAL_FOLDER_TYPE, MailLocalFolder))
+#define MAIL_LOCAL_FOLDER_CLASS(k) (CAMEL_CHECK_CLASS_CAST ((k), MAIL_LOCAL_FOLDER_TYPE, MailLocalFolderClass))
+#define MAIL_IS_LOCAL_FOLDER(o) (CAMEL_CHECK_TYPE((o), MAIL_LOCAL_FOLDER_TYPE))
struct _local_meta {
- char *path; /* path of metainfo file */
+ char *path; /* path of metainfo */
char *format; /* format of mailbox */
- char *name; /* name of mbox itself */
- int indexed; /* do we index the body? */
+ char *name; /* name of actual mbox */
+ int indexed; /* is body indexed? */
};
+typedef struct {
+ CamelFolder parent_object;
+
+ CamelFolder *real_folder;
+ CamelStore *real_store;
+
+ struct _local_meta *meta;
+
+ GMutex *real_folder_lock; /* no way to use the CamelFolder's lock, so... */
+} MailLocalFolder;
+
+typedef struct {
+ CamelFolderClass parent_class;
+} MailLocalFolderClass;
+
+static CamelType mail_local_folder_get_type (void);
+
+#ifdef ENABLE_THREADS
+#define LOCAL_FOLDER_LOCK(folder) (g_mutex_lock (((MailLocalFolder *)folder)->real_folder_lock))
+#define LOCAL_FOLDER_UNLOCK(folder) (g_mutex_unlock (((MailLocalFolder *)folder)->real_folder_lock))
+#else
+#define LOCAL_FOLDER_LOCK(folder)
+#define LOCAL_FOLDER_UNLOCK(folder)
+#endif
+
+/* ** MailLocalFolder ************************************************************* */
+
static struct _local_meta *
-load_metainfo (const char *path)
+load_metainfo(const char *path)
{
xmlDocPtr doc;
xmlNodePtr node;
struct _local_meta *meta;
-
- meta = g_malloc0 (sizeof (*meta));
- meta->path = g_strdup (path);
-
- d(printf("Loading folder metainfo from : %s\n", meta->path));
-
- doc = xmlParseFile (meta->path);
- if (doc == NULL) {
+
+ d(printf("Loading folder metainfo from : %s\n", path));
+
+ meta = g_malloc0(sizeof(*meta));
+ meta->path = g_strdup(path);
+
+ doc = xmlParseFile(path);
+ if (doc == NULL)
goto dodefault;
- }
+
node = doc->root;
- if (strcmp (node->name, "folderinfo")) {
+ if (strcmp(node->name, "folderinfo"))
goto dodefault;
- }
+
node = node->childs;
while (node) {
- if (!strcmp (node->name, "folder")) {
+ if (!strcmp(node->name, "folder")) {
char *index, *txt;
- txt = xmlGetProp (node, "type");
- meta->format = g_strdup (txt ? txt : "mbox");
- xmlFree (txt);
+ txt = xmlGetProp(node, "type");
+ meta->format = g_strdup(txt?txt:"mbox");
+ xmlFree(txt);
- txt = xmlGetProp (node, "name");
- meta->name = g_strdup (txt ? txt : "mbox");
- xmlFree (txt);
+ txt = xmlGetProp(node, "name");
+ meta->name = g_strdup(txt?txt:"mbox");
+ xmlFree(txt);
- index = xmlGetProp (node, "index");
+ index = xmlGetProp(node, "index");
if (index) {
- meta->indexed = atoi (index);
- xmlFree (index);
+ meta->indexed = atoi(index);
+ xmlFree(index);
} else
meta->indexed = TRUE;
}
node = node->next;
}
- xmlFreeDoc (doc);
+ xmlFreeDoc(doc);
return meta;
dodefault:
- meta->format = g_strdup ("mbox"); /* defaults */
- meta->name = g_strdup ("mbox");
+ meta->format = g_strdup("mbox"); /* defaults */
+ meta->name = g_strdup("mbox");
meta->indexed = TRUE;
- xmlFreeDoc (doc);
+ xmlFreeDoc(doc);
return meta;
}
static void
-free_metainfo (struct _local_meta *meta)
+free_metainfo(struct _local_meta *meta)
{
- g_free (meta->path);
- g_free (meta->format);
- g_free (meta->name);
- g_free (meta);
+ g_free(meta->path);
+ g_free(meta->format);
+ g_free(meta->name);
+ g_free(meta);
}
-static int
+static gboolean
save_metainfo(struct _local_meta *meta)
{
xmlDocPtr doc;
@@ -165,455 +200,561 @@ save_metainfo(struct _local_meta *meta)
ret = xmlSaveFile(meta->path, doc);
xmlFreeDoc(doc);
+
return ret;
}
+/* forward a bunch of functions to the real folder. This pretty
+ * much sucks but I haven't found a better way of doing it.
+ */
-/* MailLocalStore implementation */
-#define MAIL_LOCAL_STORE_TYPE (mail_local_store_get_type ())
-#define MAIL_LOCAL_STORE(obj) (CAMEL_CHECK_CAST((obj), MAIL_LOCAL_STORE_TYPE, MailLocalStore))
-#define MAIL_LOCAL_STORE_CLASS(k) (CAMEL_CHECK_CLASS_CAST ((k), MAIL_LOCAL_STORE_TYPE, MailLocalStoreClass))
-#define MAIL_IS_LOCAL_STORE(o) (CAMEL_CHECK_TYPE((o), MAIL_LOCAL_STORE_TYPE))
-
-typedef struct {
- CamelStore parent_object;
-
- EvolutionStorage *storage;
- GNOME_Evolution_Storage corba_storage;
- EvolutionStorageListener *local_storage_listener;
-
- char *local_path;
- int local_pathlen;
- GHashTable *folders; /* points to MailLocalFolder */
-} MailLocalStore;
+/* We need to do it without having locked our folder, otherwise
+ we can get sync hangs with vfolders/trash */
+static void
+mlf_refresh_info(CamelFolder *folder, CamelException *ex)
+{
+ MailLocalFolder *mlf = MAIL_LOCAL_FOLDER(folder);
+ CamelFolder *f;
-typedef struct {
- CamelStoreClass parent_class;
-} MailLocalStoreClass;
+ LOCAL_FOLDER_LOCK(mlf);
+ f = mlf->real_folder;
+ camel_object_ref((CamelObject *)f);
+ LOCAL_FOLDER_UNLOCK(mlf);
-typedef struct {
- CamelFolder *folder;
- MailLocalStore *local_store;
- char *path, *name, *uri;
-} MailLocalFolder;
+ camel_folder_refresh_info(f, ex);
+ camel_object_unref((CamelObject *)f);
+}
-static MailLocalStore *local_store;
+static void
+mlf_sync(CamelFolder *folder, gboolean expunge, CamelException *ex)
+{
+ MailLocalFolder *mlf = MAIL_LOCAL_FOLDER(folder);
+ CamelFolder *f;
-CamelType mail_local_store_get_type (void);
+ LOCAL_FOLDER_LOCK(mlf);
+ f = mlf->real_folder;
+ camel_object_ref((CamelObject *)f);
+ LOCAL_FOLDER_UNLOCK(mlf);
-static char *get_name (CamelService *service, gboolean brief);
+ camel_folder_sync(f, expunge, ex);
+ camel_object_unref((CamelObject *)f);
+}
-static CamelFolder *get_folder (CamelStore *store, const char *folder_name,
- guint32 flags, CamelException *ex);
-static CamelFolderInfo *get_folder_info (CamelStore *store, const char *top,
- guint32 flags, CamelException *ex);
-static void delete_folder (CamelStore *store, const char *folder_name,
- CamelException *ex);
-static void rename_folder (CamelStore *store, const char *old_name,
- const char *new_name, CamelException *ex);
+static void
+mlf_expunge(CamelFolder *folder, CamelException *ex)
+{
+ MailLocalFolder *mlf = MAIL_LOCAL_FOLDER(folder);
+ CamelFolder *f;
-static void init_trash (CamelStore *store);
+ LOCAL_FOLDER_LOCK(mlf);
+ f = mlf->real_folder;
+ camel_object_ref((CamelObject *)f);
+ LOCAL_FOLDER_UNLOCK(mlf);
-static CamelStoreClass *local_parent_class;
+ camel_folder_expunge(f, ex);
+ camel_object_unref((CamelObject *)f);
+}
static void
-mail_local_store_class_init (MailLocalStoreClass *mail_local_store_class)
+mlf_append_message(CamelFolder *folder, CamelMimeMessage *message, const CamelMessageInfo *info, CamelException *ex)
{
- CamelStoreClass *camel_store_class =
- CAMEL_STORE_CLASS (mail_local_store_class);
- CamelServiceClass *camel_service_class =
- CAMEL_SERVICE_CLASS (mail_local_store_class);
-
- /* virtual method overload */
- camel_service_class->get_name = get_name;
-
- /* Don't cache folders */
- camel_store_class->hash_folder_name = NULL;
- camel_store_class->compare_folder_name = NULL;
-
- camel_store_class->init_trash = init_trash;
- camel_store_class->get_folder = get_folder;
- camel_store_class->get_folder_info = get_folder_info;
- camel_store_class->free_folder_info = camel_store_free_folder_info_full;
- camel_store_class->delete_folder = delete_folder;
- camel_store_class->rename_folder = rename_folder;
-
- local_parent_class = (CamelStoreClass *)camel_type_get_global_classfuncs (camel_store_get_type ());
+ MailLocalFolder *mlf = MAIL_LOCAL_FOLDER(folder);
+ CamelFolder *f;
+
+ LOCAL_FOLDER_LOCK(mlf);
+ f = mlf->real_folder;
+ camel_object_ref((CamelObject *)f);
+ LOCAL_FOLDER_UNLOCK(mlf);
+
+ camel_folder_append_message(f, message, info, ex);
+ camel_object_unref((CamelObject *)f);
}
-static void
-mail_local_store_init (gpointer object, gpointer klass)
+static CamelMimeMessage *
+mlf_get_message(CamelFolder *folder, const char *uid, CamelException *ex)
{
- MailLocalStore *local_store = MAIL_LOCAL_STORE (object);
-
- local_store->corba_storage = CORBA_OBJECT_NIL;
+ MailLocalFolder *mlf = MAIL_LOCAL_FOLDER(folder);
+ CamelMimeMessage *ret;
+ CamelFolder *f;
+
+ LOCAL_FOLDER_LOCK(mlf);
+ f = mlf->real_folder;
+ camel_object_ref((CamelObject *)f);
+ LOCAL_FOLDER_UNLOCK(mlf);
+
+ ret = camel_folder_get_message(f, uid, ex);
+ camel_object_unref((CamelObject *)f);
+
+ return ret;
}
-static void
-free_local_folder (MailLocalFolder *lf)
+static GPtrArray *
+mlf_search_by_expression(CamelFolder *folder, const char *expression, CamelException *ex)
{
- if (lf->folder)
- camel_object_unref ((CamelObject *)lf->folder);
-
- g_free (lf->path);
- g_free (lf->name);
- g_free (lf->uri);
- camel_object_unref ((CamelObject *)lf->local_store);
+ MailLocalFolder *mlf = MAIL_LOCAL_FOLDER (folder);
+ GPtrArray *ret;
+ CamelFolder *f;
+
+ LOCAL_FOLDER_LOCK(mlf);
+ f = mlf->real_folder;
+ camel_object_ref((CamelObject *)f);
+ LOCAL_FOLDER_UNLOCK(mlf);
+
+ ret = camel_folder_search_by_expression(f, expression, ex);
+ camel_object_unref((CamelObject *)f);
+
+ return ret;
}
static void
-free_folder (gpointer key, gpointer data, gpointer user_data)
+mlf_search_free(CamelFolder *folder, GPtrArray *result)
{
- MailLocalFolder *lf = data;
-
- g_free (key);
- free_local_folder (lf);
+ MailLocalFolder *mlf = MAIL_LOCAL_FOLDER(folder);
+ CamelFolder *f;
+
+ LOCAL_FOLDER_LOCK(mlf);
+ f = mlf->real_folder;
+ camel_object_ref((CamelObject *)f);
+ LOCAL_FOLDER_UNLOCK(mlf);
+
+ camel_folder_search_free(f, result);
+ camel_object_unref((CamelObject *)f);
}
static void
-mail_local_store_finalize (gpointer object)
+mlf_set_message_user_flag(CamelFolder *folder, const char *uid, const char *name, gboolean value)
{
- MailLocalStore *local_store = MAIL_LOCAL_STORE (object);
- CORBA_Environment ev;
-
- CORBA_exception_init (&ev);
- if (!CORBA_Object_is_nil (local_store->corba_storage, &ev))
- bonobo_object_release_unref (local_store->corba_storage, &ev);
- CORBA_exception_free (&ev);
-
- if (local_store->local_storage_listener)
- gtk_object_unref (GTK_OBJECT (local_store->local_storage_listener));
+ MailLocalFolder *mlf = MAIL_LOCAL_FOLDER(folder);
+ CamelFolder *f;
- g_hash_table_foreach (local_store->folders, free_folder, NULL);
- g_hash_table_destroy (local_store->folders);
+ LOCAL_FOLDER_LOCK(mlf);
+ f = mlf->real_folder;
+ camel_object_ref((CamelObject *)f);
+ LOCAL_FOLDER_UNLOCK(mlf);
- g_free (local_store->local_path);
+ camel_folder_set_message_user_flag(mlf->real_folder, uid, name, value);
+ camel_object_unref((CamelObject *)f);
}
-CamelType
-mail_local_store_get_type (void)
+static void
+mlf_set_message_user_tag(CamelFolder *folder, const char *uid, const char *name, const char *value)
{
- static CamelType mail_local_store_type = CAMEL_INVALID_TYPE;
+ MailLocalFolder *mlf = MAIL_LOCAL_FOLDER(folder);
+ CamelFolder *f;
- if (mail_local_store_type == CAMEL_INVALID_TYPE) {
- mail_local_store_type = camel_type_register (
- CAMEL_STORE_TYPE, "MailLocalStore",
- sizeof (MailLocalStore),
- sizeof (MailLocalStoreClass),
- (CamelObjectClassInitFunc) mail_local_store_class_init,
- NULL,
- (CamelObjectInitFunc) mail_local_store_init,
- (CamelObjectFinalizeFunc) mail_local_store_finalize);
- }
+ LOCAL_FOLDER_LOCK(mlf);
+ f = mlf->real_folder;
+ camel_object_ref((CamelObject *)f);
+ LOCAL_FOLDER_UNLOCK(mlf);
- return mail_local_store_type;
+ camel_folder_set_message_user_tag(mlf->real_folder, uid, name, value);
+ camel_object_unref((CamelObject *)f);
}
-static CamelFolder *
-get_folder (CamelStore *store,
- const char *folder_name,
- guint32 flags,
- CamelException *ex)
+/* and, conversely, forward the real folder's signals. */
+
+static void
+mlf_proxy_message_changed(CamelObject *real_folder, gpointer event_data, gpointer user_data)
{
- MailLocalStore *local_store = (MailLocalStore *)store;
- MailLocalFolder *local_folder;
- CamelFolder *folder;
-
- local_folder = g_hash_table_lookup (local_store->folders, folder_name);
-
- if (local_folder) {
- folder = local_folder->folder;
- camel_object_ref (CAMEL_OBJECT (folder));
- } else {
- folder = NULL;
- camel_exception_setv (ex, CAMEL_EXCEPTION_STORE_NO_FOLDER,
- _("No such folder /%s"), folder_name);
- }
-
- return folder;
+ camel_object_trigger_event((CamelObject *)user_data, "message_changed", event_data);
}
static void
-trash_add_folder (gpointer key, gpointer value, gpointer data)
+mlf_proxy_folder_changed(CamelObject *real_folder, gpointer event_data, gpointer user_data)
{
- MailLocalFolder *local_folder = (MailLocalFolder *) value;
- CamelFolder *folder = local_folder->folder;
- CamelStore *store = CAMEL_STORE (data);
-
- camel_vee_folder_add_folder (CAMEL_VEE_FOLDER (store->vtrash), folder);
+ camel_object_trigger_event((CamelObject *)user_data, "folder_changed", event_data);
}
static void
-trash_finalize (CamelObject *trash, gpointer event_data, gpointer user_data)
+mlf_unset_folder (MailLocalFolder *mlf)
{
- CamelStore *store = CAMEL_STORE (user_data);
-
- store->vtrash = NULL;
+ CamelFolder *folder = (CamelFolder *)mlf;
+
+ g_assert(mlf->real_folder);
+
+ camel_object_unhook_event(CAMEL_OBJECT(mlf->real_folder),
+ "message_changed",
+ mlf_proxy_message_changed,
+ mlf);
+ camel_object_unhook_event(CAMEL_OBJECT(mlf->real_folder),
+ "folder_changed",
+ mlf_proxy_folder_changed,
+ mlf);
+
+ camel_object_unref((CamelObject *)folder->summary);
+ folder->summary = NULL;
+ camel_object_unref((CamelObject *)mlf->real_folder);
+ mlf->real_folder = NULL;
+ camel_object_unref((CamelObject *)mlf->real_store);
+ mlf->real_store = NULL;
+
+ folder->permanent_flags = 0;
+ folder->has_summary_capability = 0;
+ folder->has_search_capability = 0;
}
-
-static void
-init_trash (CamelStore *store)
+
+static gboolean
+mlf_set_folder(MailLocalFolder *mlf, guint32 flags, CamelException *ex)
{
- MailLocalStore *local_store = MAIL_LOCAL_STORE (store);
-
- store->vtrash = camel_vtrash_folder_new (store, CAMEL_VTRASH_NAME);
-
- if (store->vtrash) {
- /* attach to the finalize event of the vtrash */
- camel_object_hook_event (CAMEL_OBJECT (store->vtrash), "finalize",
- trash_finalize, store);
-
- /* add all the pre-opened folders to the vtrash */
- if (local_store->folders) {
- /* lock? */
- g_hash_table_foreach (local_store->folders, trash_add_folder, store);
- /* unlock? */
- }
+ CamelFolder *folder = (CamelFolder *)mlf;
+ char *uri;
+
+ g_assert(mlf->real_folder == NULL);
- /* would prefer not to special-case this, but... */
- mail_folder_cache_note_folder ("vtrash:file:/", store->vtrash);
- mail_folder_cache_set_update_lstorage ("vtrash:file:/",
- local_store->corba_storage,
- "/Trash");
+ uri = g_strdup_printf("%s:%s%s", mlf->meta->format, ((CamelService *)folder->parent_store)->url->path, folder->full_name);
+ d(printf("opening real store: %s\n", uri));
+ mlf->real_store = camel_session_get_store(session, uri, ex);
+ g_free(uri);
+ if (mlf->real_store == NULL)
+ return FALSE;
+
+ if (mlf->meta->indexed)
+ flags |= CAMEL_STORE_FOLDER_BODY_INDEX;
+
+ mlf->real_folder = camel_store_get_folder(mlf->real_store, mlf->meta->name, flags, ex);
+ if (mlf->real_folder == NULL)
+ return FALSE;
+
+ if (mlf->real_folder->has_summary_capability) {
+ folder->summary = mlf->real_folder->summary;
+ camel_object_ref((CamelObject *)mlf->real_folder->summary);
}
-}
-static void
-populate_folders (gpointer key, gpointer data, gpointer user_data)
-{
- GPtrArray *folders = user_data;
- MailLocalFolder *folder;
- CamelFolderInfo *fi;
-
- folder = data;
-
- fi = g_new0 (CamelFolderInfo, 1);
- fi->full_name = g_strdup (folder->path);
- fi->name = g_strdup (folder->name);
- fi->url = g_strdup (folder->uri);
- fi->unread_message_count = -1;
-
- g_ptr_array_add (folders, fi);
+ folder->permanent_flags = mlf->real_folder->permanent_flags;
+ folder->has_summary_capability = mlf->real_folder->has_summary_capability;
+ folder->has_search_capability = mlf->real_folder->has_search_capability;
+
+ camel_object_hook_event((CamelObject *)mlf->real_folder, "message_changed", mlf_proxy_message_changed, mlf);
+ camel_object_hook_event((CamelObject *)mlf->real_folder, "folder_changed", mlf_proxy_folder_changed, mlf);
+
+ return TRUE;
}
-static CamelFolderInfo *
-get_folder_info (CamelStore *store, const char *top,
- guint32 flags, CamelException *ex)
+static void
+mlf_class_init (CamelObjectClass *camel_object_class)
{
- MailLocalStore *local_store = MAIL_LOCAL_STORE (store);
- CamelFolderInfo *fi = NULL;
- GPtrArray *folders;
-
- folders = g_ptr_array_new ();
- g_hash_table_foreach (local_store->folders, populate_folders, folders);
-
- fi = camel_folder_info_build (folders, top, '/', TRUE);
- g_ptr_array_free (folders, TRUE);
-
- return fi;
+ CamelFolderClass *camel_folder_class = CAMEL_FOLDER_CLASS (camel_object_class);
+
+ /* override all the functions subclassed in providers/local/ */
+
+ camel_folder_class->refresh_info = mlf_refresh_info;
+ camel_folder_class->sync = mlf_sync;
+ camel_folder_class->expunge = mlf_expunge;
+ camel_folder_class->append_message = mlf_append_message;
+ camel_folder_class->get_message = mlf_get_message;
+ camel_folder_class->search_free = mlf_search_free;
+
+ camel_folder_class->search_by_expression = mlf_search_by_expression;
+ camel_folder_class->set_message_user_flag = mlf_set_message_user_flag;
+ camel_folder_class->set_message_user_tag = mlf_set_message_user_tag;
}
static void
-delete_folder (CamelStore *store, const char *folder_name, CamelException *ex)
+mlf_init (CamelObject *obj)
{
- /* No-op. The shell local storage deals with this. */
+ MailLocalFolder *mlf = MAIL_LOCAL_FOLDER (obj);
+
+#ifdef ENABLE_THREADS
+ mlf->real_folder_lock = g_mutex_new();
+#endif
}
static void
-rename_folder (CamelStore *store, const char *old, const char *new,
- CamelException *ex)
+mlf_finalize (CamelObject *obj)
{
- /* Probable no-op... */
-}
+ MailLocalFolder *mlf = MAIL_LOCAL_FOLDER (obj);
-static char *
-get_name (CamelService *service, gboolean brief)
-{
- return g_strdup ("Local mail folders");
+ if (mlf->real_folder)
+ mlf_unset_folder(mlf);
+
+ free_metainfo(mlf->meta);
+
+#ifdef ENABLE_THREADS
+ g_mutex_free (mlf->real_folder_lock);
+#endif
}
+static CamelType
+mail_local_folder_get_type (void)
+{
+ static CamelType mail_local_folder_type = CAMEL_INVALID_TYPE;
+
+ if (mail_local_folder_type == CAMEL_INVALID_TYPE) {
+ mail_local_folder_type = camel_type_register(CAMEL_FOLDER_TYPE,
+ "MailLocalFolder",
+ sizeof (MailLocalFolder),
+ sizeof (MailLocalFolderClass),
+ mlf_class_init,
+ NULL,
+ mlf_init,
+ mlf_finalize);
+ }
-/* Callbacks for the EvolutionStorageListner signals. */
+ return mail_local_folder_type;
+}
-static void
-local_storage_destroyed_cb (EvolutionStorageListener *storage_listener,
- void *data)
+static MailLocalFolder *
+mail_local_folder_construct(MailLocalFolder *mlf, MailLocalStore *parent_store, const char *full_name, CamelException *ex)
{
- /* FIXME: Dunno how to handle this yet. */
- g_warning ("%s -- The LocalStorage has gone?!", __FILE__);
-}
+ const char *name;
+ char *metapath;
-/* ********************************************************************** */
-/* Register folder */
+ name = strrchr(full_name, '/');
+ if (name == NULL)
+ name = full_name;
+ name = name + 1;
-struct _register_msg {
- struct _mail_msg msg;
+ d(printf("constructing local folder: full = %s, name = %s\n", full_name, name));
- MailLocalFolder *local_folder;
-};
+ camel_folder_construct(CAMEL_FOLDER (mlf), CAMEL_STORE(parent_store), full_name, name);
-static char *
-register_folder_desc (struct _mail_msg *mm, int done)
-{
- struct _register_msg *m = (struct _register_msg *)mm;
-
- return g_strdup_printf (_("Registering '%s'"), m->local_folder->uri);
+ metapath = g_strdup_printf("%s/%s/local-metadata.xml", ((CamelService *)parent_store)->url->path, full_name);
+ mlf->meta = load_metainfo(metapath);
+ g_free(metapath);
+
+ return mlf;
}
-static void
-register_folder_register (struct _mail_msg *mm)
+static gboolean
+mail_local_folder_reconfigure (MailLocalFolder *mlf, const char *new_format, CamelException *ex)
{
- struct _register_msg *m = (struct _register_msg *)mm;
- MailLocalFolder *local_folder = m->local_folder;
- char *name, *path = local_folder->uri + 7;
- struct _local_meta *meta;
- CamelStore *store;
- guint32 flags;
+ CamelStore *fromstore = NULL;
+ CamelFolder *fromfolder = NULL;
+ char *oldformat = NULL;
+ char *tmpname;
+ char *store_uri;
+ GPtrArray *uids;
+ int real_folder_frozen = FALSE;
+
+ camel_operation_start(NULL, _("Reconfiguring folder"));
+
+ /* first things first */
+ g_assert (ex);
+ LOCAL_FOLDER_LOCK (mlf);
+
+ /* first, 'close' the old folder */
+ if (mlf->real_folder) {
+ camel_folder_sync(mlf->real_folder, FALSE, ex);
+ if (camel_exception_is_set (ex))
+ goto cleanup;
+ mlf_unset_folder(mlf);
+ }
+
+ store_uri = g_strdup_printf("%s:%s%s", mlf->meta->format,
+ ((CamelService *)((CamelFolder *)mlf)->parent_store)->url->path, ((CamelFolder *)mlf)->full_name);
+ fromstore = camel_session_get_store(session, store_uri, ex);
+ g_free(store_uri);
+ if (fromstore == NULL)
+ goto cleanup;
+
+ oldformat = mlf->meta->format;
+ mlf->meta->format = g_strdup(new_format);
+
+ /* rename the old mbox and open it again, without indexing */
+ tmpname = g_strdup_printf ("%s_reconfig", mlf->meta->name);
+ d(printf("renaming %s to %s, and opening it\n", mlf->meta->name, tmpname));
- name = g_strdup_printf ("%s/local-metadata.xml", path);
- meta = load_metainfo (name);
- g_free (name);
+ camel_store_rename_folder(fromstore, mlf->meta->name, tmpname, ex);
+ if (camel_exception_is_set(ex))
+ goto cleanup;
- name = g_strdup_printf ("%s:%s", meta->format, path);
- store = camel_session_get_store (session, name, &mm->ex);
- g_free (name);
+ /* we dont need to set the create flag ... or need an index if it has one */
+ fromfolder = camel_store_get_folder(fromstore, tmpname, 0, ex);
+ if (fromfolder == NULL || camel_exception_is_set(ex)) {
+ /* try and recover ... */
+ camel_exception_clear(ex);
+ camel_store_rename_folder(fromstore, tmpname, mlf->meta->name, ex);
+ goto cleanup;
+ }
- if (!store) {
- free_metainfo (meta);
- return;
+ /* create a new mbox */
+ d(printf("Creating the destination mbox\n"));
+
+ if (!mlf_set_folder(mlf, CAMEL_STORE_FOLDER_CREATE, ex)) {
+ d(printf("cannot open destination folder\n"));
+ /* try and recover ... */
+ camel_exception_clear(ex);
+ camel_store_rename_folder(fromstore, tmpname, mlf->meta->name, ex);
+ goto cleanup;
}
+
+ real_folder_frozen = TRUE;
+ camel_folder_freeze(mlf->real_folder);
+
+ uids = camel_folder_get_uids(fromfolder);
+ camel_folder_move_messages_to(fromfolder, uids, mlf->real_folder, ex);
+ camel_folder_free_uids(fromfolder, uids);
+ if (camel_exception_is_set(ex))
+ goto cleanup;
- flags = CAMEL_STORE_FOLDER_CREATE;
- if (meta->indexed)
- flags |= CAMEL_STORE_FOLDER_BODY_INDEX;
- local_folder->folder = camel_store_get_folder (store, meta->name, flags, &mm->ex);
+ camel_folder_expunge(fromfolder, ex);
+
+ d(printf("delete old mbox ...\n"));
+ camel_object_unref(CAMEL_OBJECT(fromfolder));
+ fromfolder = NULL;
+ camel_store_delete_folder(fromstore, tmpname, ex);
- camel_object_unref (CAMEL_OBJECT (store));
- free_metainfo (meta);
+ /* switch format */
+ g_free(oldformat);
+ oldformat = NULL;
+ if (save_metainfo(mlf->meta) == FALSE) {
+ camel_exception_setv (ex, CAMEL_EXCEPTION_SYSTEM,
+ _("Cannot save folder metainfo; "
+ "you'll probably find you can't\n"
+ "open this folder anymore: %s: %s"),
+ mlf->meta->path, strerror(errno));
+ }
+
+ cleanup:
+ if (oldformat) {
+ g_free(mlf->meta->format);
+ mlf->meta->format = oldformat;
+ }
+ if (mlf->real_folder == NULL)
+ mlf_set_folder (mlf, CAMEL_STORE_FOLDER_CREATE, ex);
+ if (fromfolder)
+ camel_object_unref((CamelObject *)fromfolder);
+ if (fromstore)
+ camel_object_unref((CamelObject *)fromstore);
+
+ LOCAL_FOLDER_UNLOCK (mlf);
+
+ if (real_folder_frozen)
+ camel_folder_thaw(mlf->real_folder);
+
+ camel_operation_end(NULL);
+
+ return !camel_exception_is_set(ex);
}
+
+/* ******************************************************************************** */
-static void
-register_folder_registered (struct _mail_msg *mm)
+static CamelObjectClass *local_store_parent_class = NULL;
+
+static CamelFolder *
+mls_get_folder(CamelStore *store, const char *folder_name, guint32 flags, CamelException *ex)
{
- struct _register_msg *m = (struct _register_msg *)mm;
- MailLocalFolder *local_folder = m->local_folder;
-
- if (local_folder->folder) {
- gchar *name;
-
- g_hash_table_insert (local_folder->local_store->folders,
- local_folder->uri + 8,
- local_folder);
-
- /* Remove the circular ref once the local store knows about the folder */
- camel_object_unref ((CamelObject *)local_folder->local_store);
-
- /* add the folder to the vfolder lists FIXME: merge stuff above with this */
- vfolder_register_source (local_folder->folder);
-
- mail_folder_cache_set_update_lstorage (local_folder->uri,
- local_folder->local_store->corba_storage,
- local_folder->path);
-
- name = strrchr (local_folder->path, '/');
- if (name) /* should always be true... */ {
- name += 1; /* skip the slash */
- mail_folder_cache_note_name (local_folder->uri, name);
+ MailLocalStore *local_store = MAIL_LOCAL_STORE (store);
+ MailLocalFolder *folder;
+ char *physical_uri;
+
+ d(printf("get_folder: %s", folder_name));
+
+ folder = (MailLocalFolder *)camel_object_new(MAIL_LOCAL_FOLDER_TYPE);
+ folder = mail_local_folder_construct(folder, local_store, folder_name, ex);
+ if (folder == NULL)
+ return NULL;
+
+ if (!mlf_set_folder(folder, flags, ex)) {
+ camel_object_unref(CAMEL_OBJECT(folder));
+ return NULL;
+ }
+
+ if (flags & CAMEL_STORE_FOLDER_CREATE) {
+ if (save_metainfo(folder->meta) == -1) {
+ camel_exception_setv (ex, CAMEL_EXCEPTION_SYSTEM,
+ _("Cannot save folder metainfo to %s: %s"),
+ folder->meta->path, strerror(errno));
+ camel_object_unref(CAMEL_OBJECT (folder));
+ return NULL;
}
-
- /* Do this after specifying the name so it isn't 'mbox' */
- mail_folder_cache_note_folder (local_folder->uri, local_folder->folder);
-
- m->local_folder = NULL;
}
+
+ /* FIXME: why is this here? */
+ physical_uri = g_strdup_printf("file://%s/%s", ((CamelService *)store)->url->path, folder_name);
+ mail_folder_cache_note_folder(physical_uri, CAMEL_FOLDER (folder));
+ g_free(physical_uri);
+
+ return (CamelFolder *)folder;
}
-static void
-register_folder_free (struct _mail_msg *mm)
+static void
+mls_delete_folder(CamelStore *store, const char *folder_name, CamelException *ex)
{
- struct _register_msg *m = (struct _register_msg *)mm;
-
- if (m->local_folder)
- free_local_folder (m->local_folder);
+ CamelStore *real_store;
+ char *metapath, *uri;
+ CamelException local_ex;
+ struct _local_meta *meta;
+
+ d(printf("Deleting folder: %s %s\n", ((CamelService *)store)->url->path, folder_name));
+
+ camel_exception_init(&local_ex);
+
+ /* find the real store for this folder, and proxy the call */
+ metapath = g_strdup_printf("%s%s/local-metadata.xml", ((CamelService *)store)->url->path, folder_name);
+ meta = load_metainfo(metapath);
+ uri = g_strdup_printf("%s:%s%s", meta->format, ((CamelService *)store)->url->path, folder_name);
+ real_store = (CamelStore *)camel_session_get_service(session, uri, CAMEL_PROVIDER_STORE, ex);
+ g_free(uri);
+ if (real_store == NULL) {
+ g_free(metapath);
+ free_metainfo(meta);
+ return;
+ }
+
+ camel_store_delete_folder(real_store, meta->name, &local_ex);
+ if (camel_exception_is_set(&local_ex)) {
+ camel_exception_xfer(ex, &local_ex);
+ g_free(metapath);
+ free_metainfo(meta);
+ return;
+ }
+
+ free_metainfo(meta);
+
+ if (unlink(metapath) == -1) {
+ camel_exception_setv(ex, CAMEL_EXCEPTION_SYSTEM,
+ _("Cannot delete folder metadata %s: %s"),
+ metapath, strerror(errno));
+ }
+
+ g_free(metapath);
}
-static struct _mail_msg_op register_folder_op = {
- register_folder_desc,
- register_folder_register,
- register_folder_registered,
- register_folder_free,
-};
+static char *
+mls_get_name (CamelService *service, gboolean brief)
+{
+ if (brief)
+ return g_strdup("local");
+
+ return g_strdup("Local mail folders");
+}
static void
-local_storage_new_folder_cb (EvolutionStorageListener *storage_listener,
- const char *path,
- const GNOME_Evolution_Folder *folder,
- void *data)
+mls_class_init (CamelObjectClass *camel_object_class)
{
- MailLocalStore *local_store = data;
- MailLocalFolder *local_folder;
- struct _register_msg *m;
- int id;
-
- if (strcmp (folder->type, "mail") != 0 ||
- strncmp (folder->physicalUri, "file://", 7) != 0 ||
- strncmp (folder->physicalUri + 7, local_store->local_path,
- local_store->local_pathlen) != 0)
- return;
-
- local_folder = g_new0 (MailLocalFolder, 1);
- local_folder->name = g_strdup (strrchr (path, '/') + 1);
- local_folder->path = g_strdup (path);
- local_folder->uri = g_strdup (folder->physicalUri);
- local_folder->local_store = local_store;
- camel_object_ref ((CamelObject *)local_store);
-
- m = mail_msg_new (&register_folder_op, NULL, sizeof (*m));
-
- m->local_folder = local_folder;
+ CamelStoreClass *camel_store_class = CAMEL_STORE_CLASS(camel_object_class);
+ CamelServiceClass *camel_service_class = CAMEL_SERVICE_CLASS(camel_object_class);
- /* run synchronous, the shell expects it (I think) */
- id = m->msg.seq;
- e_thread_put (mail_thread_queued, (EMsg *)m);
- mail_msg_wait (id);
+ /* virtual method overload -- the bare minimum */
+ camel_service_class->get_name = mls_get_name;
+ camel_store_class->get_folder = mls_get_folder;
+ camel_store_class->delete_folder = mls_delete_folder;
+
+ local_store_parent_class = camel_type_get_global_classfuncs (CAMEL_STORE_TYPE);
}
-static void
-local_storage_removed_folder_cb (EvolutionStorageListener *storage_listener,
- const char *path,
- void *data)
+static CamelType
+mail_local_store_get_type (void)
{
- MailLocalStore *local_store = data;
- MailLocalFolder *local_folder;
- char *physical_path;
- char *tmpname;
-
- physical_path = e_path_to_physical (local_store->local_path, path);
-
- if (strncmp (physical_path, local_store->local_path,
- local_store->local_pathlen) != 0)
- return;
-
- tmpname = strchr (physical_path, '/');
- if (tmpname) {
- while (*tmpname == '/')
- tmpname++;
- local_folder = g_hash_table_lookup (local_store->folders, tmpname);
- camel_object_ref ((CamelObject *)local_store); /* When we go to free_local_folder() the
- local_store will be unref'd */
- }
- else
- local_folder = NULL;
-
- if (local_folder) {
- g_hash_table_remove (local_store->folders, tmpname);
-
- free_local_folder (local_folder);
+ static CamelType mail_local_store_type = CAMEL_INVALID_TYPE;
+
+ if (mail_local_store_type == CAMEL_INVALID_TYPE) {
+ mail_local_store_type = camel_type_register (
+ CAMEL_STORE_TYPE, "MailLocalStore",
+ sizeof (MailLocalStore),
+ sizeof (MailLocalStoreClass),
+ (CamelObjectClassInitFunc) mls_class_init,
+ NULL,
+ NULL,
+ NULL);
}
-
- g_free (physical_path);
+
+ return mail_local_store_type;
}
+/* ** Local Provider ************************************************************** */
+
static CamelProvider local_provider = {
"file", "Local mail", NULL, "mail",
CAMEL_PROVIDER_IS_STORAGE, CAMEL_URL_NEED_PATH,
@@ -633,33 +774,69 @@ non_equal (gconstpointer a, gconstpointer b)
return TRUE;
}
-void
-mail_local_storage_startup (EvolutionShellClient *shellclient,
- const char *evolution_path)
+static void
+mail_local_provider_init (void)
{
- EvolutionStorageListener *local_storage_listener;
- GNOME_Evolution_StorageListener corba_local_storage_listener;
- CORBA_Environment ev;
-
/* Register with Camel to handle file: URLs */
- local_provider.object_types[CAMEL_PROVIDER_STORE] =
- mail_local_store_get_type();
+ local_provider.object_types[CAMEL_PROVIDER_STORE] = MAIL_LOCAL_STORE_TYPE;
local_provider.service_cache = g_hash_table_new (non_hash, non_equal);
camel_session_register_provider (session, &local_provider);
-
- /* Now build the storage. */
- local_store = (MailLocalStore *)camel_session_get_service (
- session, "file:/", CAMEL_PROVIDER_STORE, NULL);
- if (!local_store) {
- g_warning ("No local store!");
+}
+
+/* ** Local Storage Listener ****************************************************** */
+
+static void
+local_storage_destroyed_cb (EvolutionStorageListener *storage_listener,
+ void *data)
+{
+ CORBA_Environment ev;
+
+ CORBA_exception_init (&ev);
+ bonobo_object_release_unref (data, &ev);
+ CORBA_exception_free (&ev);
+}
+
+
+static void
+local_storage_new_folder_cb (EvolutionStorageListener *storage_listener,
+ const char *path,
+ const GNOME_Evolution_Folder *folder,
+ void *data)
+{
+ if (strcmp (folder->type, "mail") != 0)
return;
- }
-
- local_store->corba_storage = evolution_shell_client_get_local_storage (shellclient);
- if (local_store->corba_storage == CORBA_OBJECT_NIL) {
- g_warning ("No local storage!");
- camel_object_unref (CAMEL_OBJECT (local_store));
+
+ mail_folder_cache_set_update_lstorage (folder->physicalUri,
+ data,
+ path);
+ mail_folder_cache_note_name (folder->physicalUri, folder->displayName);
+}
+
+#if 0
+static void
+local_storage_removed_folder_cb (EvolutionStorageListener *storage_listener,
+ const char *path,
+ void *data)
+{
+ if (strcmp (folder->type, "mail") != 0)
+ return;
+
+ /* anything to do? */
+}
+#endif
+
+static void
+storage_listener_startup (EvolutionShellClient *shellclient)
+{
+ EvolutionStorageListener *local_storage_listener;
+ GNOME_Evolution_StorageListener corba_local_storage_listener;
+ GNOME_Evolution_Storage corba_storage;
+ CORBA_Environment ev;
+
+ corba_storage = evolution_shell_client_get_local_storage (shellclient);
+ if (corba_storage == CORBA_OBJECT_NIL) {
+ g_warning ("No local storage available from shell client!");
return;
}
@@ -670,36 +847,47 @@ mail_local_storage_startup (EvolutionShellClient *shellclient,
gtk_signal_connect (GTK_OBJECT (local_storage_listener),
"destroyed",
GTK_SIGNAL_FUNC (local_storage_destroyed_cb),
- local_store);
+ corba_storage);
gtk_signal_connect (GTK_OBJECT (local_storage_listener),
"new_folder",
GTK_SIGNAL_FUNC (local_storage_new_folder_cb),
- local_store);
- gtk_signal_connect (GTK_OBJECT (local_storage_listener),
- "removed_folder",
- GTK_SIGNAL_FUNC (local_storage_removed_folder_cb),
- local_store);
-
- local_store->local_storage_listener = local_storage_listener;
-
- local_store->local_path = g_strdup_printf ("%s/local",
- evolution_path);
- local_store->local_pathlen = strlen (local_store->local_path);
-
- local_store->folders = g_hash_table_new (g_str_hash, g_str_equal);
+ corba_storage);
+ /*gtk_signal_connect (GTK_OBJECT (local_storage_listener),
+ * "removed_folder",
+ * GTK_SIGNAL_FUNC (local_storage_removed_folder_cb),
+ * corba_storage);
+ */
CORBA_exception_init (&ev);
- GNOME_Evolution_Storage_addListener (local_store->corba_storage,
+ GNOME_Evolution_Storage_addListener (corba_storage,
corba_local_storage_listener, &ev);
if (ev._major != CORBA_NO_EXCEPTION) {
g_warning ("Cannot add a listener to the Local Storage.");
- camel_object_unref (CAMEL_OBJECT (local_store));
CORBA_exception_free (&ev);
return;
}
CORBA_exception_free (&ev);
}
+/* ** The rest ******************************************************************** */
+
+static MailLocalStore *global_local_store;
+
+void
+mail_local_storage_startup (EvolutionShellClient *shellclient, const char *evolution_path)
+{
+ mail_local_provider_init ();
+
+ global_local_store = MAIL_LOCAL_STORE(camel_session_get_service (session, "file:/", CAMEL_PROVIDER_STORE, NULL));
+
+ if (!global_local_store) {
+ g_warning ("No local store!");
+ return;
+ }
+
+ storage_listener_startup (shellclient);
+}
+
/*----------------------------------------------------------------------
* Local folder reconfiguration stuff
@@ -724,8 +912,6 @@ mail_local_storage_startup (EvolutionShellClient *shellclient,
*/
-/* ******************** */
-
/* we should have our own progress bar for this */
struct _reconfigure_msg {
@@ -737,6 +923,7 @@ struct _reconfigure_msg {
GtkWidget *apply;
GtkWidget *cancel;
GtkOptionMenu *optionlist;
+ CamelFolder *folder_out;
};
static char *
@@ -753,174 +940,47 @@ static void
reconfigure_folder_reconfigure (struct _mail_msg *mm)
{
struct _reconfigure_msg *m = (struct _reconfigure_msg *)mm;
- MailLocalFolder *local_folder = NULL;
- CamelStore *fromstore = NULL, *tostore = NULL;
- char *fromurl = NULL, *tourl = NULL;
- CamelFolder *fromfolder = NULL, *tofolder = NULL;
- GPtrArray *uids;
- char *metapath;
- char *tmpname;
- CamelURL *url = NULL;
- struct _local_meta *meta = NULL;
- guint32 flags;
-
+ CamelFolder *local_folder = NULL;
+
d(printf("reconfiguring folder: %s to type %s\n", m->fb->uri, m->newtype));
- /* NOTE: This var is cleared by the folder_browser via the set_uri method */
- m->fb->reconfigure = TRUE;
-
- /* get the actual location of the mailbox */
- url = camel_url_new (m->fb->uri, &mm->ex);
- if (camel_exception_is_set (&mm->ex)) {
- g_warning ("%s is not a workable url!", m->fb->uri);
- goto cleanup;
- }
-
- tmpname = strchr (m->fb->uri, '/');
- if (tmpname) {
- while (*tmpname == '/')
- tmpname++;
- local_folder = g_hash_table_lookup (local_store->folders, tmpname);
- } else
- local_folder = NULL;
- if (!local_folder) {
- g_warning ("%s is not a registered local folder!", m->fb->uri);
- goto cleanup;
+ if (strncmp (m->fb->uri, "file:", 5)) {
+ camel_exception_setv (&mm->ex, CAMEL_EXCEPTION_SERVICE_UNAVAILABLE,
+ _("%s may not be reconfigured because it is not a local folder"),
+ m->fb->uri);
+ return;
}
-
- metapath = g_strdup_printf ("%s/local-metadata.xml", url->path);
- meta = load_metainfo (metapath);
- g_free (metapath);
-
- /* first, 'close' the old folder */
- camel_folder_sync (local_folder->folder, FALSE, &mm->ex);
-
- /* Once for the FolderBrowser, once for the local store */
- camel_object_unref (CAMEL_OBJECT (local_folder->folder));
- camel_object_unref (CAMEL_OBJECT (local_folder->folder));
- local_folder->folder = m->fb->folder = NULL;
-
- camel_url_set_protocol (url, meta->format);
- fromurl = camel_url_to_string (url, CAMEL_URL_HIDE_ALL);
- camel_url_set_protocol (url, m->newtype);
- tourl = camel_url_to_string (url, CAMEL_URL_HIDE_ALL);
-
- d(printf("opening stores %s and %s\n", fromurl, tourl));
-
- fromstore = camel_session_get_store (session, fromurl, &mm->ex);
-
- if (camel_exception_is_set (&mm->ex))
- goto cleanup;
-
- tostore = camel_session_get_store (session, tourl, &mm->ex);
- if (camel_exception_is_set (&mm->ex))
- goto cleanup;
-
- /* rename the old mbox and open it again, without indexing */
- tmpname = g_strdup_printf ("%s_reconfig", meta->name);
- d(printf("renaming %s to %s, and opening it\n", meta->name, tmpname));
-
- camel_store_rename_folder (fromstore, meta->name, tmpname, &mm->ex);
+
+ local_folder = mail_tool_uri_to_folder (m->fb->uri, &mm->ex);
if (camel_exception_is_set (&mm->ex)) {
- goto cleanup;
- }
-
- /* we dont need to set the create flag ... or need an index if it has one */
- fromfolder = camel_store_get_folder (fromstore, tmpname, 0, &mm->ex);
- if (fromfolder == NULL || camel_exception_is_set (&mm->ex)) {
- /* try and recover ... */
- camel_exception_clear (&mm->ex);
- camel_store_rename_folder (fromstore, tmpname, meta->name, &mm->ex);
- goto cleanup;
- }
-
- /* create a new mbox */
- d(printf("Creating the destination mbox\n"));
-
- flags = CAMEL_STORE_FOLDER_CREATE;
- if (meta->indexed)
- flags |= CAMEL_STORE_FOLDER_BODY_INDEX;
- tofolder = camel_store_get_folder (tostore, meta->name, flags, &mm->ex);
- if (tofolder == NULL || camel_exception_is_set (&mm->ex)) {
- d(printf("cannot open destination folder\n"));
- /* try and recover ... */
- camel_exception_clear (&mm->ex);
- camel_store_rename_folder (fromstore, tmpname, meta->name, &mm->ex);
- goto cleanup;
- }
-
- uids = camel_folder_get_uids (fromfolder);
- camel_folder_move_messages_to (fromfolder, uids, tofolder, &mm->ex);
- camel_folder_free_uids (fromfolder, uids);
- if (camel_exception_is_set (&mm->ex))
- goto cleanup;
-
- camel_folder_expunge (fromfolder, &mm->ex);
-
- d(printf("delete old mbox ...\n"));
- camel_store_delete_folder (fromstore, tmpname, &mm->ex);
-
- /* switch format */
- g_free (meta->format);
- meta->format = g_strdup (m->newtype);
- if (save_metainfo (meta) == -1) {
- camel_exception_setv (&mm->ex, CAMEL_EXCEPTION_SYSTEM,
- _("Cannot save folder metainfo; "
- "you'll probably find you can't\n"
- "open this folder anymore: %s"),
- tourl);
- }
-
- cleanup:
- if (local_folder && !local_folder->folder) {
- struct _register_msg *rm = mail_msg_new (&register_folder_op, NULL, sizeof (*m));
-
- /* fake the internal part of this operation, nasty hackish thing */
- rm->local_folder = local_folder;
- register_folder_register ((struct _mail_msg *)rm);
- rm->local_folder = NULL;
- mail_msg_free ((struct _mail_msg *)rm);
+ g_warning ("Can't resolve URI \"%s\" for reconfiguration!", m->fb->uri);
+ return;
}
-
- if (tofolder)
- camel_object_unref (CAMEL_OBJECT (tofolder));
- if (fromfolder)
- camel_object_unref (CAMEL_OBJECT (fromfolder));
- if (fromstore)
- camel_object_unref (CAMEL_OBJECT (fromstore));
- if (tostore)
- camel_object_unref (CAMEL_OBJECT (tostore));
- if (meta)
- free_metainfo (meta);
- g_free (fromurl);
- g_free (tourl);
- if (url)
- camel_url_free (url);
+
+ mail_local_folder_reconfigure (MAIL_LOCAL_FOLDER (local_folder), m->newtype, &mm->ex);
+ m->folder_out = local_folder;
}
static void
reconfigure_folder_reconfigured (struct _mail_msg *mm)
{
struct _reconfigure_msg *m = (struct _reconfigure_msg *)mm;
- char *uri;
+ /*char *uri;*/
if (camel_exception_is_set (&mm->ex)) {
gnome_error_dialog (_("If you can no longer open this mailbox, then\n"
"you may need to repair it manually."));
}
-
- /* force a reload of the newly formatted folder */
- d(printf("opening new source\n"));
- uri = g_strdup (m->fb->uri);
- folder_browser_set_uri (m->fb, uri);
- g_free (uri);
+
+ message_list_set_folder (m->fb->message_list, m->folder_out, FALSE);
}
static void
reconfigure_folder_free (struct _mail_msg *mm)
{
struct _reconfigure_msg *m = (struct _reconfigure_msg *)mm;
-
+
+ camel_object_unref (CAMEL_OBJECT (m->folder_out));
gtk_object_unref (GTK_OBJECT (m->fb));
g_free (m->newtype);
}
@@ -939,48 +999,57 @@ static void
reconfigure_clicked (GnomeDialog *dialog, int button, struct _reconfigure_msg *m)
{
if (button == 0) {
- GtkWidget *menu;
- int type;
- char *types[] = { "mbox", "maildir", "mh" };
+ GtkWidget *menu, *item;
/* hack to clear the message list during update */
- message_list_set_folder (m->fb->message_list, NULL, FALSE);
-
- menu = gtk_option_menu_get_menu (m->optionlist);
- type = g_list_index (GTK_MENU_SHELL (menu)->children,
- gtk_menu_get_active (GTK_MENU (menu)));
- if (type < 0 || type > 2)
- type = 0;
+ /* we need to do this because the message list caches
+ * CamelMessageInfos from the old folder. */
+ message_list_set_folder(m->fb->message_list, NULL, FALSE);
+ menu = gtk_option_menu_get_menu(m->optionlist);
+ item = gtk_menu_get_active(GTK_MENU(menu));
+ m->newtype = g_strdup(gtk_object_get_data((GtkObject *)item, "type"));
+
gtk_widget_set_sensitive (m->frame, FALSE);
gtk_widget_set_sensitive (m->apply, FALSE);
gtk_widget_set_sensitive (m->cancel, FALSE);
- m->newtype = g_strdup (types[type]);
e_thread_put (mail_thread_queued, (EMsg *)m);
} else
mail_msg_free ((struct _mail_msg *)m);
- if (button != -1)
+ if (button != -1) {
+ /* remove this folder from our hash since we are done with it */
+ g_hash_table_remove (reconfigure_folder_hash, m->fb->folder);
+ if (g_hash_table_size (reconfigure_folder_hash) == 0) {
+ /* additional cleanup */
+ g_hash_table_destroy (reconfigure_folder_hash);
+ reconfigure_folder_hash = NULL;
+ }
+
gnome_dialog_close (dialog);
+ }
}
void
mail_local_reconfigure_folder (FolderBrowser *fb)
{
- CamelStore *store;
GladeXML *gui;
GnomeDialog *gd;
struct _reconfigure_msg *m;
char *name, *title;
-
+ GList *p;
+ GtkWidget *menu;
+ char *currentformat;
+ int index=0, history=0;
+
if (fb->folder == NULL) {
g_warning ("Trying to reconfigure nonexistant folder");
return;
}
if (!reconfigure_folder_hash)
- reconfigure_folder_hash = g_hash_table_new (g_direct_hash, g_direct_equal);
+ reconfigure_folder_hash = g_hash_table_new (NULL, NULL);
if ((gd = g_hash_table_lookup (reconfigure_folder_hash, fb->folder))) {
gdk_window_raise (GTK_WIDGET (gd)->window);
@@ -988,22 +1057,13 @@ mail_local_reconfigure_folder (FolderBrowser *fb)
}
/* check if we can work on this folder */
- name = strchr (fb->uri, '/');
- if (name) {
- while (*name == '/')
- name++;
- /* we just want to see if it's NULL or not */
- name = (char *) g_hash_table_lookup (local_store->folders, name);
- }
-
- if (name == NULL) {
+ if (!MAIL_IS_LOCAL_FOLDER (fb->folder)) {
e_notice (NULL, GNOME_MESSAGE_BOX_WARNING,
_("You cannot change the format of a non-local folder."));
return;
}
m = mail_msg_new (&reconfigure_folder_op, NULL, sizeof (*m));
- store = camel_folder_get_parent_store (fb->folder);
gui = glade_xml_new (EVOLUTION_GLADEDIR "/local-config.glade", "dialog_format");
gd = (GnomeDialog *)glade_xml_get_widget (gui, "dialog_format");
@@ -1020,23 +1080,49 @@ mail_local_reconfigure_folder (FolderBrowser *fb)
m->optionlist = (GtkOptionMenu *)glade_xml_get_widget (gui, "option_format");
m->newtype = NULL;
m->fb = fb;
+ m->folder_out = NULL;
gtk_object_ref (GTK_OBJECT (fb));
-
+
+ /* dynamically create the folder type list from camel */
+ /* we assume the list is static and never freed */
+ currentformat = MAIL_LOCAL_FOLDER (fb->folder)->meta->format;
+ p = camel_session_list_providers(session, TRUE);
+ menu = gtk_menu_new();
+ while (p) {
+ CamelProvider *cp = p->data;
+
+ /* we only want local storages */
+ /* hack it so we also dont see file or spool stores, which aren't really meant for this */
+ if ((cp->flags & (CAMEL_PROVIDER_IS_REMOTE|CAMEL_PROVIDER_IS_STORAGE)) == CAMEL_PROVIDER_IS_STORAGE
+ && strcmp(cp->protocol, "file")
+ && strcmp(cp->protocol, "spool")) {
+ GtkWidget *item;
+ char *label;
+
+ if (strcmp(cp->protocol, currentformat) == 0)
+ history = index;
+
+ label = g_strdup_printf("%s (%s)", cp->protocol, _(cp->name));
+ item = gtk_menu_item_new_with_label(label);
+ g_free(label);
+ gtk_object_set_data((GtkObject *)item, "type", cp->protocol);
+ gtk_widget_show(item);
+ gtk_menu_append(GTK_MENU(menu), item);
+ index++;
+ }
+ p = p->next;
+ }
+ gtk_option_menu_remove_menu (GTK_OPTION_MENU(m->optionlist));
+ gtk_option_menu_set_menu (GTK_OPTION_MENU(m->optionlist), menu);
+ gtk_option_menu_set_history(GTK_OPTION_MENU(m->optionlist), history);
+
gtk_label_set_text ((GtkLabel *)glade_xml_get_widget (gui, "label_format"),
- ((CamelService *)store)->url->protocol);
+ MAIL_LOCAL_FOLDER (fb->folder)->meta->format);
gtk_signal_connect (GTK_OBJECT (gd), "clicked", reconfigure_clicked, m);
gtk_object_unref (GTK_OBJECT (gui));
g_hash_table_insert (reconfigure_folder_hash, (gpointer) fb->folder, (gpointer) gd);
- gnome_dialog_run_and_close (GNOME_DIALOG (gd));
-
- /* remove this folder from our hash since we are done with it */
- g_hash_table_remove (reconfigure_folder_hash, fb->folder);
- if (g_hash_table_size (reconfigure_folder_hash) == 0) {
- /* additional cleanup */
- g_hash_table_destroy (reconfigure_folder_hash);
- reconfigure_folder_hash = NULL;
- }
+ gnome_dialog_run (GNOME_DIALOG (gd));
}