aboutsummaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
-rw-r--r--shell/ChangeLog19
-rw-r--r--shell/e-local-storage.c159
2 files changed, 133 insertions, 45 deletions
diff --git a/shell/ChangeLog b/shell/ChangeLog
index 070b72638a..6abf2cc700 100644
--- a/shell/ChangeLog
+++ b/shell/ChangeLog
@@ -1,3 +1,20 @@
+2002-06-25 Ettore Perazzoli <ettore@ximian.com>
+
+ [Fix #10681, Incorrect behavior when deleting a hierarchy of local
+ folders.]
+
+ * e-local-storage.c: Change the ASyncRemoveFolderCallbackData
+ struct to have only storage and next_paths_to_delete members.
+ (remove_folder_step): New.
+ (component_async_remove_folder_callback): Start deletion for the
+ next path in next_paths_to_delete, so we handle recursion
+ properly.
+ (create_subfolder_list): New.
+ (remove_folder): Create the list with create_subfolder_list,
+ initialize the ASyncRemoveFolderCallbackData with the
+ next_paths_to_delete list and do the first deletion. This way we
+ handle recursion properly.
+
2002-06-26 Iain <iain@ximian.com>
* e-shell-importer.c (show_import_wizard): Only let the dialog be
@@ -32,6 +49,8 @@
2002-06-24 Ettore Perazzoli <ettore@ximian.com>
+ * e-local-storage.c (remove_folder): Removed arg @physical_uri.
+
* e-shell-folder-selection-dialog.c (impl_clicked): Pass
default_type to e_shell_show_folder_creation_dialog() properly.
[Patch by Christian Kreibich <christian@whoop.org>.]
diff --git a/shell/e-local-storage.c b/shell/e-local-storage.c
index 46965f43a8..a521f5503a 100644
--- a/shell/e-local-storage.c
+++ b/shell/e-local-storage.c
@@ -449,9 +449,7 @@ create_folder (ELocalStorage *local_storage,
struct _AsyncRemoveFolderCallbackData {
EStorage *storage;
-
- char *path;
- char *physical_path;
+ GList *next_paths_to_delete;
};
typedef struct _AsyncRemoveFolderCallbackData AsyncRemoveFolderCallbackData;
@@ -500,55 +498,139 @@ remove_folder_directory (ELocalStorage *local_storage,
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 == HASSUBFOLDERS then recurse delete the subfolders dir? */
-
- /* FIXME: Handle errors */
if (result == EVOLUTION_SHELL_COMPONENT_OK) {
- ELocalStoragePrivate *priv;
+ result = remove_folder_directory (E_LOCAL_STORAGE (callback_data->storage), path);
+ e_storage_removed_folder (E_STORAGE (callback_data->storage), path);
+
+ g_print ("...Removed %s!\n", path);
+
+ evolution_storage_removed_folder (EVOLUTION_STORAGE (priv->bonobo_interface), path);
+ } else {
+ /* FIXME: Handle errors. */
+ g_print ("...Error removing %s!\n", path);
+ }
+
+ bonobo_object_unref (BONOBO_OBJECT (shell_component_client));
- priv = E_LOCAL_STORAGE (callback_data->storage)->priv;
+ /* 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;
- result = remove_folder_directory (E_LOCAL_STORAGE (callback_data->storage),
- callback_data->path);
+ path = callback_data->next_paths_to_delete->data;
+ g_free (path);
- e_storage_removed_folder (E_STORAGE (callback_data->storage),
- callback_data->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;
+ }
- evolution_storage_removed_folder (EVOLUTION_STORAGE (priv->bonobo_interface),
- callback_data->path);
+ /* 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);
+
+ bonobo_object_ref (BONOBO_OBJECT (client));
+
+ g_print ("Removing %s...\n", path);
+
+ 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);
}
- bonobo_object_unref (BONOBO_OBJECT (shell_component_client));
+ g_list_free (subfolders);
- g_free (callback_data->path);
- g_free (callback_data->physical_path);
- g_free (callback_data);
+ return list;
}
static EStorageResult
remove_folder (ELocalStorage *local_storage,
- const char *path,
- const char *physical_uri)
+ const char *path)
{
ELocalStoragePrivate *priv;
EStorage *storage;
AsyncRemoveFolderCallbackData *callback_data;
EvolutionShellComponentClient *component_client;
EFolder *folder;
- char *physical_path, *physical_uri_mem = NULL;
- GList *subfolder_paths;
+ GList *next_paths_to_delete;
GList *p;
priv = local_storage->priv;
@@ -564,30 +646,17 @@ remove_folder (ELocalStorage *local_storage,
if (component_client == NULL)
return E_STORAGE_INVALIDTYPE;
- physical_path = e_path_to_physical (priv->base_path, path);
-
- if (!physical_uri)
- physical_uri = physical_uri_mem = g_strconcat ("file://", physical_path, NULL);
-
- /* Recursively remove the subfolders */
- subfolder_paths = e_storage_get_subfolder_paths (storage, path);
-
- for (p = subfolder_paths; p; p = p->next)
- remove_folder (local_storage, p->data, NULL);
+ 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->path = g_strdup (path);
- callback_data->physical_path = physical_path;
-
- bonobo_object_ref (BONOBO_OBJECT (component_client));
+ callback_data->storage = E_STORAGE (local_storage);
+ callback_data->next_paths_to_delete = next_paths_to_delete;
- evolution_shell_component_client_async_remove_folder (component_client,
- physical_uri,
- e_folder_get_type_string (folder),
- component_async_remove_folder_callback,
- callback_data);
- g_free (physical_uri_mem);
+ if (! remove_folder_step (callback_data)) {
+ /* Eek, something wacky happened. */
+ return EVOLUTION_SHELL_COMPONENT_UNKNOWNERROR;
+ }
return EVOLUTION_SHELL_COMPONENT_OK;
}
@@ -654,7 +723,7 @@ impl_async_remove_folder (EStorage *storage,
local_storage = E_LOCAL_STORAGE (storage);
- result = remove_folder (local_storage, path, NULL);
+ result = remove_folder (local_storage, path);
if (callback != NULL)
(* callback) (E_STORAGE (local_storage), result, data);
@@ -953,7 +1022,7 @@ bonobo_interface_remove_folder_cb (EvolutionStorage *storage,
local_storage = E_LOCAL_STORAGE (data);
- return remove_folder (local_storage, path, physical_uri);
+ return remove_folder (local_storage, path);
}
static void