/* -*- Mode: C; indent-tabs-mode: t; c-basic-offset: 8; tab-width: 8 -*- */ /* e-local-storage.c * * Copyright (C) 2000, 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. * * Author: Ettore Perazzoli */ /* FIXMEs: * * - If we have `.' or `..' as path elements, we lose. * * - If the LocalStorage is destroyed and an async operation on a shell component is * pending, we get a callback on a bogus object. We need support for cancelling * operations on the shell component. */ #ifdef HAVE_CONFIG_H #include <config.h> #endif #include <errno.h> #include <string.h> #include <sys/param.h> #include <sys/types.h> #include <sys/stat.h> #include <unistd.h> #include <dirent.h> #include <gtk/gtksignal.h> #include <libgnome/gnome-i18n.h> #include <libgnome/gnome-util.h> #include <gal/util/e-util.h> #include "e-util/e-path.h" #include "e-local-folder.h" #include "e-shell-constants.h" #include "evolution-storage.h" #include "e-local-storage.h" #include <bonobo/bonobo-exception.h> #define PARENT_TYPE E_TYPE_STORAGE static EStorageClass *parent_class = NULL; struct _ELocalStoragePrivate { EFolderTypeRegistry *folder_type_registry; char *base_path; EvolutionStorage *bonobo_interface; }; /* EStorageResult <-> errno mapping. */ static EStorageResult errno_to_storage_result (void) { EStorageResult storage_result; switch (errno) { case EACCES: case EROFS: storage_result = E_STORAGE_PERMISSIONDENIED; break; case EEXIST: storage_result = E_STORAGE_EXISTS; break; case ENOSPC: storage_result = E_STORAGE_NOSPACE; break; default: storage_result = E_STORAGE_GENERICERROR; } return storage_result; } static EStorageResult shell_component_result_to_storage_result (EvolutionShellComponentResult result) { /* FIXME: Maybe we need better mapping here. */ switch (result) { case EVOLUTION_SHELL_COMPONENT_OK: return E_STORAGE_OK; case EVOLUTION_SHELL_COMPONENT_NOTFOUND: return E_STORAGE_NOTFOUND; case EVOLUTION_SHELL_COMPONENT_UNSUPPORTEDTYPE: return E_STORAGE_UNSUPPORTEDTYPE; case EVOLUTION_SHELL_COMPONENT_UNSUPPORTEDOPERATION: return E_STORAGE_UNSUPPORTEDOPERATION; case EVOLUTION_SHELL_COMPONENT_EXISTS: return E_STORAGE_EXISTS; case EVOLUTION_SHELL_COMPONENT_PERMISSIONDENIED: return E_STORAGE_PERMISSIONDENIED; case EVOLUTION_SHELL_COMPONENT_ALREADYOWNED: case EVOLUTION_SHELL_COMPONENT_BUSY: case EVOLUTION_SHELL_COMPONENT_CORBAERROR: case EVOLUTION_SHELL_COMPONENT_HASSUBFOLDERS: case EVOLUTION_SHELL_COMPONENT_INTERNALERROR: case EVOLUTION_SHELL_COMPONENT_INTERRUPTED: case EVOLUTION_SHELL_COMPONENT_INVALIDARG: case EVOLUTION_SHELL_COMPONENT_INVALIDURI: case EVOLUTION_SHELL_COMPONENT_NOSPACE: case EVOLUTION_SHELL_COMPONENT_NOTOWNED: case EVOLUTION_SHELL_COMPONENT_UNKNOWNERROR: default: return E_STORAGE_GENERICERROR; } } /* Utility functions. */ static void new_folder (ELocalStorage *local_storage, const char *path, EFolder *folder) { ELocalStoragePrivate *priv; priv = local_storage->priv; e_storage_new_folder (E_STORAGE (local_storage), path, folder); evolution_storage_new_folder (EVOLUTION_STORAGE (priv->bonobo_interface), path, e_folder_get_name (folder), e_folder_get_type_string (folder), e_folder_get_physical_uri (folder), e_folder_get_description (folder), e_folder_get_custom_icon_name (folder), e_folder_get_unread_count (folder), FALSE, 0); } static gboolean setup_folder_as_stock (ELocalStorage *local_storage, const char *path, const char *name, const char *custom_icon_name) { EFolder *folder; folder = e_storage_get_folder (E_STORAGE (local_storage), path); if (folder == NULL) return FALSE; e_folder_set_name (folder, name); e_folder_set_is_stock (folder, TRUE); e_folder_set_custom_icon (folder, custom_icon_name); return TRUE; } static void setup_stock_folders (ELocalStorage *local_storage) { setup_folder_as_stock (local_storage, "/Calendar", _("Calendar"), NULL); setup_folder_as_stock (local_storage, "/Contacts", _("Contacts"), NULL); setup_folder_as_stock (local_storage, "/Drafts", _("Drafts"), NULL); setup_folder_as_stock (local_storage, "/Inbox", _("Inbox"), "inbox"); setup_folder_as_stock (local_storage, "/Outbox", _("Outbox"), "outbox"); setup_folder_as_stock (local_storage, "/Sent", _("Sent"), NULL); setup_folder_as_stock (local_storage, "/Tasks", _("Tasks"), NULL); setup_folder_as_stock (local_storage, "/Trash", _("Trash"), NULL); } static gboolean load_folder (const char *physical_path, const char *path, void *data) { ELocalStorage *local_storage; EFolder *folder; local_storage = E_LOCAL_STORAGE (data); folder = e_local_folder_new_from_path (physical_path); if (folder == NULL) { /* g_warning ("No folder metadata in %s... ignoring", physical_path); FIXME */ return TRUE; } e_storage_new_folder ((EStorage *)local_storage, path, folder); return TRUE; } static void setup_corba_storage (ELocalStorage *local_storage, const char *path) { GList *subfolder_paths; EFolder *folder; GList *p; folder = e_storage_get_folder (E_STORAGE (local_storage), path); if (folder != NULL) evolution_storage_new_folder (EVOLUTION_STORAGE (local_storage->priv->bonobo_interface), path, e_folder_get_name (folder), e_folder_get_type_string (folder), e_folder_get_physical_uri (folder), e_folder_get_description (folder), e_folder_get_custom_icon_name (folder), e_folder_get_unread_count (folder), FALSE, 0); subfolder_paths = e_storage_get_subfolder_paths (E_STORAGE (local_storage), path); for (p = subfolder_paths; p != NULL; p = p->next) setup_corba_storage (local_storage, (const char *) p->data); e_free_string_list (subfolder_paths); } static gboolean load_all_folders (ELocalStorage *local_storage) { const char *base_path; base_path = e_local_storage_get_base_path (local_storage); /* Ignore errors, so we set up the local storage even if there is stale data that we don't understand in ~/evolution. */ e_path_find_folders (base_path, load_folder, local_storage); setup_stock_folders (local_storage); setup_corba_storage (local_storage, "/"); return TRUE; } static EStorageResult storage_result_from_component_result (EvolutionShellComponentResult result) { switch (result) { case EVOLUTION_SHELL_COMPONENT_PERMISSIONDENIED: return E_STORAGE_PERMISSIONDENIED; case EVOLUTION_SHELL_COMPONENT_NOSPACE: return E_STORAGE_NOSPACE; default: return E_STORAGE_GENERICERROR; } } /* Callbacks for the async methods invoked on the `Evolution::ShellComponent's. */ static void notify_listener (const Bonobo_Listener listener, EStorageResult result, const char *physical_path) { CORBA_any any; GNOME_Evolution_Storage_FolderResult folder_result; CORBA_Environment ev; folder_result.result = result; folder_result.path = CORBA_string_dup (physical_path ? physical_path : ""); any._type = TC_GNOME_Evolution_Storage_FolderResult; any._value = &folder_result; CORBA_exception_init (&ev); Bonobo_Listener_event (listener, "evolution-shell:folder_created", &any, &ev); if (BONOBO_EX (&ev)) { g_warning ("Exception notifing listener: %s\n", CORBA_exception_id (&ev)); } CORBA_exception_free (&ev); } struct _AsyncCreateFolderCallbackData { EStorage *storage; Bonobo_Listener listener; char *path; char *display_name; char *type; char *description; char *physical_uri; char *physical_path; EStorageResultCallback callback; void *callback_data; }; typedef struct _AsyncCreateFolderCallbackData AsyncCreateFolderCallbackData; static void component_async_create_folder_callback (EvolutionShellComponentClient *shell_component_client, EvolutionShellComponentResult result, void *data) { AsyncCreateFolderCallbackData *callback_data; EStorageResult storage_result; callback_data = (AsyncCreateFolderCallbackData *) data; storage_result = shell_component_result_to_storage_result (result); if (result != EVOLUTION_SHELL_COMPONENT_OK) { /* XXX: This assumes the component won't leave any files in the directory. */ rmdir (callback_data->physical_path); } else { EFolder *folder; folder = e_local_folder_new (callback_data->display_name, callback_data->type, callback_data->description); e_folder_set_physical_uri (folder, callback_data->physical_uri); if (e_local_folder_save (E_LOCAL_FOLDER (folder))) { new_folder (E_LOCAL_STORAGE (callback_data->storage), callback_data->path, folder); } else { rmdir (callback_data->physical_path); g_object_unref (folder); storage_result = E_STORAGE_IOERROR; } } g_object_unref (shell_component_client); if (callback_data->listener != CORBA_OBJECT_NIL) notify_listener (callback_data->listener, storage_result, callback_data->physical_path); if (callback_data->callback != NULL) (* callback_data->callback) (callback_data->storage, storage_result, callback_data->callback_data); g_free (callback_data->path); g_free (callback_data->display_name); g_free (callback_data->type); g_free (callback_data->description); g_free (callback_data->physical_uri); g_free (callback_data->physical_path); g_free (callback_data); } /* Implementation for the folder operations. */ static EStorageResult create_folder_directory (ELocalStorage *local_storage, const char *path, const char *type, const char *description, char **physical_path_return) { EStorage *storage; ELocalStoragePrivate *priv; char *parent_path; char *physical_path; storage = E_STORAGE (local_storage); priv = local_storage->priv; *physical_path_return = NULL; g_assert (g_path_is_absolute (path)); parent_path = g_path_get_dirname(path); if (strlen(parent_path) > 1) { char *subfolders_directory_physical_path; char *parent; /* Create the `subfolders' subdirectory under the parent. */ parent = g_strdup_printf ("%s/", parent_path); subfolders_directory_physical_path = e_path_to_physical (priv->base_path, parent); if (mkdir (subfolders_directory_physical_path, 0700) == -1 && errno != EEXIST) { g_free (subfolders_directory_physical_path); g_free (parent); return errno_to_storage_result (); } g_free (subfolders_directory_physical_path); g_free (parent); } g_free (parent_path); physical_path = e_path_to_physical (priv->base_path, path); /* Create the directory that holds the folder. */ *physical_path_return = physical_path; if (mkdir (physical_path, 0700) == -1) { return errno_to_storage_result (); } return E_STORAGE_OK; } static void create_folder (ELocalStorage *local_storage, const Bonobo_Listener listener, const char *path, const char *type, const char *description, EStorageResultCallback callback, void *data) { EStorage *storage; ELocalStoragePrivate *priv; EvolutionShellComponentClient *component_client; AsyncCreateFolderCallbackData *callback_data; EStorageResult result; char *folder_name; char *physical_path; char *physical_uri; storage = E_STORAGE (local_storage); priv = local_storage->priv; component_client = e_folder_type_registry_get_handler_for_type (priv->folder_type_registry, type); if (component_client == NULL) { if (listener != CORBA_OBJECT_NIL) notify_listener (listener, E_STORAGE_INVALIDTYPE, NULL); if (callback != NULL) (* callback) (storage, E_STORAGE_INVALIDTYPE, data); return; } g_assert (g_path_is_absolute (path)); result = create_folder_directory (local_storage, path, type, description, &physical_path); if (result != E_STORAGE_OK) { if (callback != NULL) (* callback) (storage, result, data); if (listener != CORBA_OBJECT_NIL) notify_listener (listener, result, NULL); g_free (physical_path); return; } folder_name = g_path_get_basename (path); /* Finally tell the component to do the job of creating the physical files in it. */ /* FIXME: We should put the operations on a queue so that we can cancel them when the ELocalStorage is destroyed. */ physical_uri = g_strconcat ("file://", physical_path, NULL); callback_data = g_new (AsyncCreateFolderCallbackData, 1); callback_data->storage = E_STORAGE (local_storage); callback_data->path = g_strdup (path); callback_data->display_name = g_strdup (folder_name); callback_data->type = g_strdup (type); callback_data->description = g_strdup (description); callback_data->physical_uri = physical_uri; callback_data->physical_path = physical_path; callback_data->listener = listener; callback_data->callback = callback; callback_data->callback_data = data; g_object_ref (component_client); evolution_shell_component_client_async_create_folder (component_client, physical_uri, type, component_async_create_folder_callback, callback_data); g_free (folder_name); } struct _AsyncRemoveFolderCallbackData { EStorage *storage; GList *next_paths_to_delete; }; typedef struct _AsyncRemoveFolderCallbackData AsyncRemoveFolderCallbackData; static EStorageResult remove_folder_directory (ELocalStorage *local_storage, const char *path) { EStorage *storage; ELocalStoragePrivate *priv; char *folder_name; char *file_name; char *physical_path; priv = local_storage->priv; storage = E_STORAGE (local_storage); folder_name = g_path_get_basename (path); /* Delete the metadata file associated with this folder. */ physical_path = e_path_to_physical (priv->base_path, path); file_name = g_build_filename (physical_path, E_LOCAL_FOLDER_METADATA_FILE_NAME, NULL); unlink (file_name); g_free (file_name); /* Delete the physical directory. */ if (rmdir (physical_path) == -1) { g_free (physical_path); g_free (folder_name); return E_STORAGE_GENERICERROR; } g_free (physical_path); /* Delete the 'subfolders' directory that this folder lies in */ if (folder_name != path + 1) { char *subfolders_directory_physical_path; char *parent_path; parent_path = g_strndup (path, strlen (path) - strlen (folder_name)); subfolders_directory_physical_path = e_path_to_physical (priv->base_path, parent_path); g_free (parent_path); rmdir (subfolders_directory_physical_path); g_free (subfolders_directory_physical_path); } g_free (folder_name); return E_STORAGE_OK; } static gboolean remove_folder_step (AsyncRemoveFolderCallbackData *callback_data); static void component_async_remove_folder_callback (EvolutionShellComponentClient *shell_component_client, EvolutionShellComponentResult result, void *data) { ELocalStoragePrivate *priv; AsyncRemoveFolderCallbackData *callback_data; EStorageResult storage_result; gboolean success; const char *path; callback_data = (AsyncRemoveFolderCallbackData *) data; priv = E_LOCAL_STORAGE (callback_data->storage)->priv; path = (const char *) callback_data->next_paths_to_delete->data; storage_result = shell_component_result_to_storage_result (result); if (result == EVOLUTION_SHELL_COMPONENT_OK) { result = remove_folder_directory (E_LOCAL_STORAGE (callback_data->storage), path); e_storage_removed_folder (E_STORAGE (callback_data->storage), path); evolution_storage_removed_folder (EVOLUTION_STORAGE (priv->bonobo_interface), path); } else { /* FIXME: Handle errors. */ g_print ("...Error removing %s!\n", path); } g_object_unref (shell_component_client); /* Now go on and delete the next subfolder in the list that still exists, deallocating the elements in the list in the process. */ do { char *path; path = callback_data->next_paths_to_delete->data; g_free (path); callback_data->next_paths_to_delete = g_list_remove_link (callback_data->next_paths_to_delete, callback_data->next_paths_to_delete); /* Check if we are done. */ if (callback_data->next_paths_to_delete == NULL) { g_free (callback_data); return; } /* Remove the folder; if the folder has disappeared from the tree for some reason (this is an async callback!), just go on with the next one. */ success = remove_folder_step (callback_data); } while (! success); } static gboolean remove_folder_step (AsyncRemoveFolderCallbackData *callback_data) { EvolutionShellComponentClient *client; ELocalStoragePrivate *priv; EFolder *folder; const char *path; const char *type; char *physical_path; char *physical_uri; g_assert (callback_data->next_paths_to_delete != NULL); path = (const char *) callback_data->next_paths_to_delete->data; folder = e_storage_get_folder (callback_data->storage, path); if (folder == NULL) return FALSE; priv = E_LOCAL_STORAGE (callback_data->storage)->priv; physical_path = e_path_to_physical (priv->base_path, path); physical_uri = g_strconcat ("file://", physical_path, NULL); type = e_folder_get_type_string (folder); client = e_folder_type_registry_get_handler_for_type (priv->folder_type_registry, type); g_object_ref (client); evolution_shell_component_client_async_remove_folder (client, physical_uri, type, component_async_remove_folder_callback, callback_data); g_free (physical_path); g_free (physical_uri); return TRUE; } static GList * create_subfolder_list (ELocalStorage *local_storage, const char *path) { GList *subfolders; GList *list; GList *p; subfolders = e_storage_get_subfolder_paths (E_STORAGE (local_storage), path); list = NULL; for (p = subfolders; p != NULL; p = p->next) { char *path; path = (char *) p->data; list = g_list_concat (list, create_subfolder_list (local_storage, path)); list = g_list_append (list, path); } g_list_free (subfolders); return list; } static EStorageResult remove_folder (ELocalStorage *local_storage, const char *path) { ELocalStoragePrivate *priv; EStorage *storage; AsyncRemoveFolderCallbackData *callback_data; EvolutionShellComponentClient *component_client; EFolder *folder; GList *next_paths_to_delete; priv = local_storage->priv; storage = E_STORAGE (local_storage); folder = e_storage_get_folder (storage, path); if (e_folder_get_is_stock (folder)) return E_STORAGE_CANTCHANGESTOCKFOLDER; component_client = e_folder_type_registry_get_handler_for_type (priv->folder_type_registry, e_folder_get_type_string (folder)); if (component_client == NULL) return E_STORAGE_INVALIDTYPE; next_paths_to_delete = create_subfolder_list (E_LOCAL_STORAGE (storage), path); next_paths_to_delete = g_list_append (next_paths_to_delete, g_strdup (path)); callback_data = g_new (AsyncRemoveFolderCallbackData, 1); callback_data->storage = E_STORAGE (local_storage); callback_data->next_paths_to_delete = next_paths_to_delete; if (! remove_folder_step (callback_data)) { /* Eek, something wacky happened. */ return EVOLUTION_SHELL_COMPONENT_UNKNOWNERROR; } return EVOLUTION_SHELL_COMPONENT_OK; } /* GtkObject methods. */ static void impl_dispose (GObject *object) { ELocalStorage *local_storage; ELocalStoragePrivate *priv; CORBA_Environment ev; local_storage = E_LOCAL_STORAGE (object); priv = local_storage->priv; CORBA_exception_init (&ev); if (priv->folder_type_registry != NULL) { g_object_unref (priv->folder_type_registry); priv->folder_type_registry = NULL; } if (priv->bonobo_interface != NULL) { bonobo_object_unref (BONOBO_OBJECT (priv->bonobo_interface)); priv->bonobo_interface = NULL; } CORBA_exception_free (&ev); (* G_OBJECT_CLASS (parent_class)->dispose) (object); } static void impl_finalize (GObject *object) { ELocalStorage *local_storage; ELocalStoragePrivate *priv; local_storage = E_LOCAL_STORAGE (object); priv = local_storage->priv; g_free (priv->base_path); g_free (priv); (* G_OBJECT_CLASS (parent_class)->finalize) (object); } /* Creating folders. */ static void impl_async_create_folder (EStorage *storage, const char *path, const char *type, const char *description, EStorageResultCallback callback, void *data) { ELocalStorage *local_storage; local_storage = E_LOCAL_STORAGE (storage); create_folder (local_storage, CORBA_OBJECT_NIL, path, type, description, callback, data); } /* Removing folders. */ static void impl_async_remove_folder (EStorage *storage, const char *path, EStorageResultCallback callback, void *data) { ELocalStorage *local_storage; EStorageResult result; local_storage = E_LOCAL_STORAGE (storage); result = remove_folder (local_storage, path); if (callback != NULL) (* callback) (E_STORAGE (local_storage), result, data); } /* Transferring folders. */ struct _XferItem { char *source_path; char *destination_path; }; typedef struct _XferItem XferItem; static XferItem * xfer_item_new (char *source_path, char *destination_path) { XferItem *new; new = g_new (XferItem, 1); new->source_path = source_path; new->destination_path = destination_path; return new; } static void xfer_item_free (XferItem *item) { g_free (item->source_path); g_free (item->destination_path); g_free (item); } static void append_xfer_item_list (EStorage *storage, char *source_path, char *destination_path, GList **list) { GList *subfolders; GList *p; *list = g_list_prepend (*list, xfer_item_new (source_path, destination_path)); subfolders = e_storage_get_subfolder_paths (storage, source_path); for (p = subfolders; p != NULL; p = p->next) { char *base_name; char *source_subpath; char *destination_subpath; source_subpath = g_strdup ((const char *) p->data); base_name = g_path_get_basename (source_subpath); destination_subpath = g_build_filename (destination_path, base_name, NULL); append_xfer_item_list (storage, source_subpath, destination_subpath, list); g_free (base_name); } e_free_string_list (subfolders); } struct _XferData { /* The storage on which we are performing the xfer operation. */ ELocalStorage *local_storage; /* List of source/destination path couples to copy, in the right order. */ GList *folder_items; /* Pointer into `folder_items'. The folder item pointed by this is the one handled by the previous CORBA invocation. */ GList *current_folder_item; /* Whether we want to remove the source too. */ gboolean remove_source; /* The callback, with its data. */ EStorageResultCallback callback; void *callback_data; }; typedef struct _XferData XferData; static void async_xfer_folder_step (ELocalStorage *local_storage, const char *source_path, const char *destination_path, gboolean remove_source, EvolutionShellComponentClientCallback component_client_callback, void *component_client_callback_data) { ELocalStoragePrivate *priv; EFolder *source_folder; EvolutionShellComponentClient *component_client; char *physical_path; char *physical_uri; priv = local_storage->priv; source_folder = e_storage_get_folder (E_STORAGE (local_storage), source_path); g_assert (source_folder != NULL); create_folder_directory (local_storage, destination_path, e_folder_get_type_string (source_folder), e_folder_get_description (source_folder), &physical_path); physical_uri = g_strconcat ("file://", physical_path, NULL); g_free (physical_path); component_client = e_folder_type_registry_get_handler_for_type (priv->folder_type_registry, e_folder_get_type_string (source_folder)); g_assert (component_client != NULL); evolution_shell_component_client_async_xfer_folder (component_client, e_folder_get_physical_uri (source_folder), physical_uri, e_folder_get_type_string (source_folder), remove_source, component_client_callback, component_client_callback_data); g_free (physical_uri); } static void async_xfer_folder_complete (XferData *xfer_data, gboolean success) { ELocalStorage *local_storage; GList *p; local_storage = xfer_data->local_storage; if (success && xfer_data->remove_source) { EStorageResult result; /* Remove all the source physical directories, and also the corresponding folders from the folder tree. */ for (p = g_list_last (xfer_data->folder_items); p != NULL; p = p->prev) { XferItem *item; item = (XferItem *) p->data; result = remove_folder_directory (local_storage, item->source_path); /* FIXME handle failure differently? This should be n unlikely situation. */ if (result == E_STORAGE_OK) { e_storage_removed_folder (E_STORAGE (local_storage), item->source_path); evolution_storage_removed_folder (EVOLUTION_STORAGE (local_storage->priv->bonobo_interface), item->source_path); } } } /* Free the data. */ for (p = xfer_data->folder_items; p != NULL; p = p->next) { XferItem *item; item = (XferItem *) p->data; xfer_item_free (item); } g_list_free (xfer_data->folder_items); g_free (xfer_data); } static void async_xfer_folder_callback (EvolutionShellComponentClient *shell_component_client, EvolutionShellComponentResult result, void *callback_data) { XferData *xfer_data; XferItem *item; EFolder *source_folder; EFolder *destination_folder; char *dest_physical_path; char *new_physical_uri; xfer_data = (XferData *) callback_data; item = (XferItem *) xfer_data->current_folder_item->data; if (result != EVOLUTION_SHELL_COMPONENT_OK) { (* xfer_data->callback) (E_STORAGE (xfer_data->local_storage), storage_result_from_component_result (result), xfer_data->callback_data); async_xfer_folder_complete (xfer_data, FALSE); return; } source_folder = e_storage_get_folder (E_STORAGE (xfer_data->local_storage), item->source_path); destination_folder = e_local_folder_new (e_folder_get_name (source_folder), e_folder_get_type_string (source_folder), e_folder_get_description (source_folder)); dest_physical_path = e_path_to_physical (xfer_data->local_storage->priv->base_path, item->destination_path); new_physical_uri = g_strconcat ("file://", dest_physical_path, NULL); g_free (dest_physical_path); e_folder_set_physical_uri (destination_folder, new_physical_uri); g_free (new_physical_uri); e_local_folder_save (E_LOCAL_FOLDER (destination_folder)); /* FIXME check for errors */ new_folder (xfer_data->local_storage, item->destination_path, destination_folder); xfer_data->current_folder_item = xfer_data->current_folder_item->next; if (xfer_data->current_folder_item == NULL) { (* xfer_data->callback) (E_STORAGE (xfer_data->local_storage), E_STORAGE_OK, xfer_data->callback_data); async_xfer_folder_complete (xfer_data, TRUE); return; } item = (XferItem *) xfer_data->current_folder_item->data; async_xfer_folder_step (xfer_data->local_storage, item->source_path, item->destination_path, xfer_data->remove_source, async_xfer_folder_callback, xfer_data); } static void impl_async_xfer_folder (EStorage *storage, const char *source_path, const char *destination_path, gboolean remove_source, EStorageResultCallback callback, void *callback_data) { ELocalStorage *local_storage; ELocalStoragePrivate *priv; XferData *xfer_data; GList *folder_items; /* <XferItem> */ XferItem *first_item; local_storage = E_LOCAL_STORAGE (storage); priv = local_storage->priv; if (remove_source && e_folder_get_is_stock (e_storage_get_folder (storage, source_path))) { (* callback) (storage, E_STORAGE_CANTCHANGESTOCKFOLDER, callback_data); return; } folder_items = NULL; append_xfer_item_list (storage, g_strdup (source_path), g_strdup (destination_path), &folder_items); folder_items = g_list_reverse (folder_items); /* lame */ xfer_data = g_new (XferData, 1); xfer_data->local_storage = local_storage; xfer_data->folder_items = folder_items; xfer_data->current_folder_item = folder_items; xfer_data->remove_source = remove_source; xfer_data->callback = callback; xfer_data->callback_data = callback_data; first_item = (XferItem *) xfer_data->folder_items->data; async_xfer_folder_step (E_LOCAL_STORAGE (storage), first_item->source_path, first_item->destination_path, remove_source, async_xfer_folder_callback, xfer_data); } /* Callbacks for the `Evolution::Storage' interface we are exposing to the outside world. */ static void bonobo_interface_create_folder_cb (EvolutionStorage *storage, const Bonobo_Listener listener, const char *path, const char *type, const char *description, const char *parent_physical_uri, void *data) { ELocalStorage *local_storage; local_storage = E_LOCAL_STORAGE (data); create_folder (local_storage, listener, path, type, description, NULL, NULL); } static int bonobo_interface_remove_folder_cb (EvolutionStorage *storage, const Bonobo_Listener listener, const char *path, const char *physical_uri, void *data) { ELocalStorage *local_storage; local_storage = E_LOCAL_STORAGE (data); return remove_folder (local_storage, path); } static void bonobo_interface_update_folder_cb (EvolutionStorage *storage, const char *path, int unread_count, void *data) { ELocalStorage *local_storage; EFolder *folder; local_storage = E_LOCAL_STORAGE (data); folder = e_storage_get_folder (E_STORAGE (local_storage), path); if (folder == NULL) return; e_folder_set_unread_count (folder, unread_count); return; } /* Initialization. */ static void class_init (ELocalStorageClass *class) { EStorageClass *storage_class; GObjectClass *object_class; parent_class = g_type_class_ref(e_storage_get_type ()); object_class = G_OBJECT_CLASS (class); storage_class = E_STORAGE_CLASS (class); object_class->finalize = impl_finalize; object_class->dispose = impl_dispose; storage_class->async_create_folder = impl_async_create_folder; storage_class->async_remove_folder = impl_async_remove_folder; storage_class->async_xfer_folder = impl_async_xfer_folder; } static void init (ELocalStorage *local_storage) { ELocalStoragePrivate *priv; priv = g_new (ELocalStoragePrivate, 1); priv->base_path = NULL; priv->folder_type_registry = NULL; priv->bonobo_interface = NULL; local_storage->priv = priv; } static gboolean construct (ELocalStorage *local_storage, EFolderTypeRegistry *folder_type_registry, const char *base_path) { ELocalStoragePrivate *priv; EFolder *root_folder; int base_path_len; char *uri; root_folder = e_folder_new (_("Local Folders"), "noselect", ""); uri = g_strdup_printf("file://%s;noselect", base_path); e_folder_set_physical_uri(root_folder, uri); g_free(uri); e_storage_construct (E_STORAGE (local_storage), E_LOCAL_STORAGE_NAME, root_folder); priv = local_storage->priv; base_path_len = strlen (base_path); while (base_path_len > 0 && base_path[base_path_len - 1] == E_PATH_SEPARATOR) base_path_len--; g_return_val_if_fail (base_path_len != 0, FALSE); g_assert (priv->folder_type_registry == NULL); g_object_ref (folder_type_registry); priv->folder_type_registry = folder_type_registry; g_assert (priv->base_path == NULL); priv->base_path = g_strndup (base_path, base_path_len); g_assert (priv->bonobo_interface == NULL); priv->bonobo_interface = evolution_storage_new (E_LOCAL_STORAGE_NAME, FALSE); g_signal_connect (priv->bonobo_interface, "create_folder", G_CALLBACK (bonobo_interface_create_folder_cb), local_storage); g_signal_connect (priv->bonobo_interface, "remove_folder", G_CALLBACK (bonobo_interface_remove_folder_cb), local_storage); g_signal_connect (priv->bonobo_interface, "update_folder", G_CALLBACK (bonobo_interface_update_folder_cb), local_storage); return load_all_folders (local_storage); } EStorage * e_local_storage_open (EFolderTypeRegistry *folder_type_registry, const char *base_path) { EStorage *new; g_return_val_if_fail (folder_type_registry != NULL, NULL); g_return_val_if_fail (E_IS_FOLDER_TYPE_REGISTRY (folder_type_registry), NULL); g_return_val_if_fail (base_path != NULL, NULL); new = g_object_new (e_local_storage_get_type (), NULL); if (! construct (E_LOCAL_STORAGE (new), folder_type_registry, base_path)) { g_object_unref (new); return NULL; } return new; } const char * e_local_storage_get_base_path (ELocalStorage *local_storage) { g_return_val_if_fail (local_storage != NULL, NULL); g_return_val_if_fail (E_IS_LOCAL_STORAGE (local_storage), NULL); return local_storage->priv->base_path; } const GNOME_Evolution_Storage e_local_storage_get_corba_interface (ELocalStorage *local_storage) { ELocalStoragePrivate *priv; GNOME_Evolution_Storage corba_interface; g_return_val_if_fail (local_storage != NULL, NULL); g_return_val_if_fail (E_IS_LOCAL_STORAGE (local_storage), NULL); priv = local_storage->priv; corba_interface = bonobo_object_corba_objref (BONOBO_OBJECT (priv->bonobo_interface)); return corba_interface; } E_MAKE_TYPE (e_local_storage, "ELocalStorage", ELocalStorage, class_init, init, PARENT_TYPE)