[evolution-data-server/imap-notify: 15/40] Rework camel_imapx_server_list().



commit 517951ef4b7523c77f1c49ddd7a96dc6e864b336
Author: Matthew Barnes <mbarnes redhat com>
Date:   Wed Aug 7 10:18:06 2013 +0200

    Rework camel_imapx_server_list().
    
    Repurpose camel_imapx_server_list() to create/update CamelIMAPXMailbox
    instances within CamelIMAPXServer.  This function no longer returns an
    array of CamelIMAPXListResponses, but simply a success/failure boolean.
    
    Also, this function no longer takes an extension string parameter.  If
    the server names "LIST-EXTENDED" in its CAPABILITY response, then this
    function automatically issues an extended LIST command:
    
        LIST "" $PATTERN RETURN (CHILDREN SUBSCRIBED)
    
    Otherwise it issues a LIST command followed by an LSUB command:
    
        LIST "" $PATTERN
        LSUB "" $PATTERN
    
    After camel_imapx_server_list() returns, the caller can then pass the
    same $PATTERN string to camel_imapx_server_list_mailboxes() to obtain
    an up-to-date list of CamelIMAPXMailbox instances that match $PATTERN.
    
    Also as part of this, we now explicitly only list mailboxes in personal
    namespaces.  Listing mailboxes in shared or other users namespaces is a
    future enhancement since CamelStore does not yet have a suitable API to
    support this properly.

 camel/camel-imapx-server.c |  141 ++++++++--------
 camel/camel-imapx-server.h |    3 +-
 camel/camel-imapx-store.c  |  418 +++++++++++++++++++-------------------------
 3 files changed, 252 insertions(+), 310 deletions(-)
---
diff --git a/camel/camel-imapx-server.c b/camel/camel-imapx-server.c
index 1fb3b87..b6f9b62 100644
--- a/camel/camel-imapx-server.c
+++ b/camel/camel-imapx-server.c
@@ -134,9 +134,6 @@ struct _CopyMessagesData {
 
 struct _ListData {
        gchar *pattern;
-       CamelStoreGetFolderInfoFlags flags;
-       gchar *ext;
-       GHashTable *folders;
 };
 
 struct _MailboxData {
@@ -323,6 +320,10 @@ struct _CamelIMAPXServerPrivate {
         * STATUS $mailbox_name ($status_data_items) */
        gchar *status_data_items;
 
+       /* Return options for extended LIST commands:
+        * LIST "" $pattern RETURN ($list_return_opts) */
+       gchar *list_return_opts;
+
        /* Untagged SEARCH data gets deposited here.
         * The search command should claim the results
         * when finished and reset the pointer to NULL. */
@@ -660,9 +661,6 @@ static void
 list_data_free (ListData *data)
 {
        g_free (data->pattern);
-       g_free (data->ext);
-
-       g_hash_table_destroy (data->folders);
 
        g_slice_free (ListData, data);
 }
@@ -826,6 +824,14 @@ imapx_server_stash_command_arguments (CamelIMAPXServer *is)
                g_string_append (buffer, " HIGHESTMODSEQ");
        g_free (is->priv->status_data_items);
        is->priv->status_data_items = g_string_free (buffer, FALSE);
+
+       g_free (is->priv->list_return_opts);
+       if (CAMEL_IMAPX_HAVE_CAPABILITY (is->cinfo, LIST_EXTENDED)) {
+               buffer = g_string_new ("CHILDREN SUBSCRIBED");
+               is->priv->list_return_opts = g_string_free (buffer, FALSE);
+       } else {
+               is->priv->list_return_opts = NULL;
+       }
 }
 
 static void
@@ -2414,8 +2420,6 @@ imapx_untagged_list (CamelIMAPXServer *is,
 {
        CamelIMAPXListResponse *response;
        CamelIMAPXMailbox *mailbox = NULL;
-       CamelIMAPXJob *job = NULL;
-       ListData *data = NULL;
        gboolean emit_mailbox_created = FALSE;
        gboolean emit_mailbox_renamed = FALSE;
        gboolean emit_mailbox_updated = FALSE;
@@ -2482,30 +2486,7 @@ imapx_untagged_list (CamelIMAPXServer *is,
                g_signal_emit (is, signals[MAILBOX_UPDATED], 0, mailbox);
 
        g_clear_object (&mailbox);
-
-       job = imapx_match_active_job (is, IMAPX_JOB_LIST, mailbox_name);
-
-       data = camel_imapx_job_get_data (job);
-       g_return_val_if_fail (data != NULL, FALSE);
-
-       // TODO: we want to make sure the names match?
-
-       if (data->flags & CAMEL_STORE_FOLDER_INFO_SUBSCRIBED) {
-               c (is->tagprefix, "lsub: '%s' (%c)\n", mailbox_name, separator);
-       } else {
-               c (is->tagprefix, "list: '%s' (%c)\n", mailbox_name, separator);
-       }
-
-       if (job && !g_hash_table_contains (data->folders, response)) {
-               if (is->priv->context->lsub)
-                       camel_imapx_list_response_add_attribute (
-                               response, CAMEL_IMAPX_LIST_ATTR_SUBSCRIBED);
-               g_hash_table_add (data->folders, g_object_ref (response));
-       } else {
-               g_warning ("got list response but no current listing job happening?\n");
-       }
-
-       g_object_unref (response);
+       g_clear_object (&response);
 
        return TRUE;
 }
@@ -6519,6 +6500,43 @@ imapx_command_list_done (CamelIMAPXServer *is,
        imapx_unregister_job (is, job);
 }
 
+static void
+imapx_command_list_lsub (CamelIMAPXServer *is,
+                         CamelIMAPXCommand *ic)
+{
+       CamelIMAPXJob *job;
+       ListData *data;
+       GError *local_error = NULL;
+
+       job = camel_imapx_command_get_job (ic);
+       g_return_if_fail (CAMEL_IS_IMAPX_JOB (job));
+
+       data = camel_imapx_job_get_data (job);
+       g_return_if_fail (data != NULL);
+
+       if (camel_imapx_command_set_error_if_failed (ic, &local_error)) {
+               g_prefix_error (
+                       &local_error, "%s: ",
+                       _("Error fetching folders"));
+               camel_imapx_job_take_error (job, local_error);
+               imapx_unregister_job (is, job);
+
+       } else {
+               ic = camel_imapx_command_new (
+                       is, "LIST", NULL,
+                       "LSUB \"\" %s",
+                       data->pattern);
+
+               ic->pri = job->pri;
+               camel_imapx_command_set_job (ic, job);
+               ic->complete = imapx_command_list_done;
+
+               imapx_command_queue (is, ic);
+
+               camel_imapx_command_unref (ic);
+       }
+}
+
 static gboolean
 imapx_job_list_start (CamelIMAPXJob *job,
                       CamelIMAPXServer *is,
@@ -6531,20 +6549,23 @@ imapx_job_list_start (CamelIMAPXJob *job,
        data = camel_imapx_job_get_data (job);
        g_return_val_if_fail (data != NULL, FALSE);
 
-       ic = camel_imapx_command_new (
-               is, "LIST", NULL,
-               "%s \"\" %s",
-               (data->flags & CAMEL_STORE_FOLDER_INFO_SUBSCRIBED) ?
-                       "LSUB" : "LIST",
-               data->pattern);
-       if (data->ext) {
-               /* Hm, we need a way to add atoms _without_ quoting or using literals */
-               camel_imapx_command_add (ic, " ");
-               camel_imapx_command_add (ic, data->ext);
+       if (is->priv->list_return_opts != NULL) {
+               ic = camel_imapx_command_new (
+                       is, "LIST", NULL,
+                       "LIST \"\" %s RETURN (%t)",
+                       data->pattern,
+                       is->priv->list_return_opts);
+               ic->complete = imapx_command_list_done;
+       } else {
+               ic = camel_imapx_command_new (
+                       is, "LIST", NULL,
+                       "LIST \"\" %s",
+                       data->pattern);
+               ic->complete = imapx_command_list_lsub;
        }
+
        ic->pri = job->pri;
        camel_imapx_command_set_job (ic, job);
-       ic->complete = imapx_command_list_done;
 
        imapx_command_queue (is, ic);
 
@@ -7642,6 +7663,7 @@ imapx_server_finalize (GObject *object)
        g_mutex_clear (&is->priv->mailboxes_lock);
 
        g_free (is->priv->status_data_items);
+       g_free (is->priv->list_return_opts);
 
        if (is->priv->search_results != NULL)
                g_array_unref (is->priv->search_results);
@@ -8807,30 +8829,22 @@ camel_imapx_server_expunge (CamelIMAPXServer *is,
        return success;
 }
 
-GPtrArray *
+gboolean
 camel_imapx_server_list (CamelIMAPXServer *is,
                          const gchar *pattern,
                          CamelStoreGetFolderInfoFlags flags,
-                         const gchar *ext,
                          GCancellable *cancellable,
                          GError **error)
 {
        CamelIMAPXJob *job;
-       GPtrArray *folders = NULL;
        ListData *data;
+       gboolean success;
 
-       g_return_val_if_fail (CAMEL_IS_IMAPX_SERVER (is), NULL);
-       g_return_val_if_fail (pattern != NULL, NULL);
+       g_return_val_if_fail (CAMEL_IS_IMAPX_SERVER (is), FALSE);
+       g_return_val_if_fail (pattern != NULL, FALSE);
 
        data = g_slice_new0 (ListData);
        data->pattern = g_strdup (pattern);
-       data->flags = flags;
-       data->ext = g_strdup (ext);
-       data->folders = g_hash_table_new_full (
-               (GHashFunc) camel_imapx_list_response_hash,
-               (GEqualFunc) camel_imapx_list_response_equal,
-               (GDestroyNotify) g_object_unref,
-               (GDestroyNotify) NULL);
 
        job = camel_imapx_job_new (cancellable);
        job->type = IMAPX_JOB_LIST;
@@ -8845,24 +8859,11 @@ camel_imapx_server_list (CamelIMAPXServer *is,
        if (flags & CAMEL_STORE_FOLDER_INFO_SUBSCRIPTION_LIST)
                job->pri += 300;
 
-       if (imapx_submit_job (is, job, error)) {
-               GList *list, *link;
-
-               /* Transfer LIST responses from a GHashTable
-                * to a sorted GPtrArray by way of a GList. */
-               folders = g_ptr_array_new_with_free_func (
-                       (GDestroyNotify) g_object_unref);
-               list = g_list_sort (
-                       g_hash_table_get_keys (data->folders),
-                       (GCompareFunc) camel_imapx_list_response_compare);
-               for (link = list; link != NULL; link = g_list_next (link))
-                       g_ptr_array_add (folders, g_object_ref (link->data));
-               g_list_free (list);
-       }
+       success = imapx_submit_job (is, job, error);
 
        camel_imapx_job_unref (job);
 
-       return folders;
+       return success;
 }
 
 gboolean
diff --git a/camel/camel-imapx-server.h b/camel/camel-imapx-server.h
index 48fe293..b64cf7b 100644
--- a/camel/camel-imapx-server.h
+++ b/camel/camel-imapx-server.h
@@ -178,10 +178,9 @@ CamelAuthenticationResult
                                                 const gchar *mechanism,
                                                 GCancellable *cancellable,
                                                 GError **error);
-GPtrArray *    camel_imapx_server_list         (CamelIMAPXServer *is,
+gboolean       camel_imapx_server_list         (CamelIMAPXServer *is,
                                                 const gchar *pattern,
                                                 CamelStoreGetFolderInfoFlags flags,
-                                                const gchar *ext,
                                                 GCancellable *cancellable,
                                                 GError **error);
 CamelFolderChangeInfo *
diff --git a/camel/camel-imapx-store.c b/camel/camel-imapx-store.c
index 1bbf98e..92ec79f 100644
--- a/camel/camel-imapx-store.c
+++ b/camel/camel-imapx-store.c
@@ -74,6 +74,7 @@ struct _CamelIMAPXStorePrivate {
        /* Used for synchronizing get_folder_info_sync(). */
        GMutex get_finfo_lock;
        time_t last_refresh_time;
+       volatile gint syncing_folders;
 };
 
 enum {
@@ -487,6 +488,13 @@ imapx_store_process_mailbox_attributes (CamelIMAPXStore *store,
                        emit_folder_unsubscribed_deleted = TRUE;
        }
 
+       /* Suppress all signal emissions when synchronizing folders. */
+       if (g_atomic_int_get (&store->priv->syncing_folders) > 0) {
+               emit_folder_created_subscribed = FALSE;
+               emit_folder_unsubscribed_deleted = FALSE;
+               emit_folder_renamed = FALSE;
+       }
+
        /* At most one signal emission flag should be set. */
        g_warn_if_fail (
                (emit_folder_created_subscribed ? 1 : 0) +
@@ -1356,296 +1364,247 @@ get_folder_info_offline (CamelStore *store,
 }
 
 static void
-add_mailbox_to_summary (CamelIMAPXStore *imapx_store,
-                        CamelIMAPXServer *server,
-                        CamelIMAPXListResponse *response,
-                        GHashTable *mailboxes,
-                        gboolean update_for_lsub)
+collect_folder_info_for_list (CamelIMAPXStore *imapx_store,
+                              CamelIMAPXMailbox *mailbox,
+                              GHashTable *folder_info_results)
 {
+       CamelStoreSummary *store_summary;
        CamelIMAPXStoreInfo *si;
        CamelFolderInfo *fi;
+       const gchar *folder_path;
        const gchar *mailbox_name;
-       gchar separator;
-       CamelStoreInfoFlags flags;
-       CamelStoreInfoFlags new_flags;
-
-       mailbox_name = camel_imapx_list_response_get_mailbox_name (response);
-       separator = camel_imapx_list_response_get_separator (response);
-
-       /* XXX The flags type transforms from CamelStoreInfoFlags
-        *     to CamelFolderInfoFlags about half-way through this.
-        *     We should really eliminate the confusing redundancy. */
-       flags = camel_imapx_list_response_get_summary_flags (response);
-
-       if (update_for_lsub) {
-               fi = g_hash_table_lookup (mailboxes, mailbox_name);
-               if (fi != NULL)
-                       fi->flags |= CAMEL_STORE_INFO_FOLDER_SUBSCRIBED;
-               return;
-       }
-
-       si = camel_imapx_store_summary_add_from_mailbox (
-               imapx_store->summary, mailbox_name, separator);
-       if (si == NULL)
-               return;
-
-       new_flags =
-               (si->info.flags & CAMEL_STORE_INFO_FOLDER_SUBSCRIBED) |
-               (flags & ~CAMEL_STORE_INFO_FOLDER_SUBSCRIBED);
 
-       if (CAMEL_IMAPX_LACK_CAPABILITY (server->cinfo, NAMESPACE))
-               imapx_store->dir_sep = separator;
-
-       if (si->info.flags != new_flags) {
-               si->info.flags = new_flags;
-               camel_store_summary_touch (
-                       (CamelStoreSummary *) imapx_store->summary);
-       }
+       store_summary = CAMEL_STORE_SUMMARY (imapx_store->summary);
 
-       fi = camel_folder_info_new ();
-       fi->full_name = g_strdup (camel_store_info_path (
-               CAMEL_STORE_SUMMARY (imapx_store->summary),
-               (CamelStoreInfo *) si));
-       if (g_ascii_strcasecmp (fi->full_name, "inbox") == 0) {
-               flags |= CAMEL_FOLDER_SYSTEM;
-               flags |= CAMEL_FOLDER_TYPE_INBOX;
-               fi->display_name = g_strdup (_("Inbox"));
-       } else {
-               fi->display_name = g_strdup (
-                       camel_store_info_name (
-                       CAMEL_STORE_SUMMARY (imapx_store->summary),
-                       (CamelStoreInfo *) si));
-       }
+       mailbox_name = camel_imapx_mailbox_get_name (mailbox);
 
-       fi->flags |= flags;
+       si = camel_imapx_store_summary_mailbox (
+               imapx_store->summary, mailbox_name);
+       g_return_if_fail (si != NULL);
 
-       fi->total = -1;
-       fi->unread = -1;
+       folder_path = camel_store_info_path (
+               store_summary, (CamelStoreInfo *) si);
+       fi = imapx_store_build_folder_info (imapx_store, folder_path, 0);
 
        /* Takes ownership of the CamelFolderInfo. */
-       g_hash_table_insert (mailboxes, g_strdup (mailbox_name), fi);
+       g_hash_table_insert (folder_info_results, g_strdup (mailbox_name), fi);
 }
 
 static gboolean
-fetch_mailboxes_for_pattern (CamelIMAPXStore *imapx_store,
-                             CamelIMAPXServer *server,
-                             const gchar *pattern,
-                             CamelStoreGetFolderInfoFlags flags,
-                             const gchar *ext,
-                             GHashTable *mailboxes,
-                             GCancellable *cancellable,
-                             GError **error)
+fetch_folder_info_for_pattern (CamelIMAPXServer *server,
+                               CamelIMAPXNamespace *namespace,
+                               const gchar *pattern,
+                               CamelStoreGetFolderInfoFlags flags,
+                               GHashTable *folder_info_results,
+                               GCancellable *cancellable,
+                               GError **error)
 {
-       GPtrArray *folders;
-       gboolean update_for_lsub;
-       guint ii;
+       CamelIMAPXStore *imapx_store;
+       GList *list, *link;
+       gboolean success;
 
-       folders = camel_imapx_server_list (
-               server, pattern, flags, ext, cancellable, error);
-       if (folders == NULL)
+       success = camel_imapx_server_list (
+               server, pattern, flags, cancellable, error);
+       if (!success)
                return FALSE;
 
-       /* Indicates we had to issue a separate LSUB command after the
-        * LIST command and we're just processing subscription results. */
-       if (flags & CAMEL_STORE_FOLDER_INFO_SUBSCRIBED)
-               update_for_lsub = TRUE;
-       else
-               update_for_lsub = FALSE;
+       imapx_store = camel_imapx_server_ref_store (server);
 
-       for (ii = 0; ii < folders->len; ii++) {
-               CamelIMAPXListResponse *response;
+       list = camel_imapx_server_list_mailboxes (server, namespace, pattern);
+
+       for (link = list; link != NULL; link = g_list_next (link)) {
+               CamelIMAPXMailbox *mailbox;
 
-               response = g_ptr_array_index (folders, ii);
+               mailbox = CAMEL_IMAPX_MAILBOX (link->data);
 
-               add_mailbox_to_summary (
-                       imapx_store, server, response,
-                       mailboxes, update_for_lsub);
+               collect_folder_info_for_list (
+                       imapx_store, mailbox, folder_info_results);
        }
 
-       g_ptr_array_unref (folders);
+       g_list_free_full (list, (GDestroyNotify) g_object_unref);
+
+       g_object_unref (imapx_store);
 
        return TRUE;
 }
 
-static GList *
-get_namespaces (CamelIMAPXStore *imapx_store)
+static gboolean
+fetch_folder_info_for_namespace_category (CamelIMAPXServer *server,
+                                          CamelIMAPXNamespaceCategory category,
+                                          CamelStoreGetFolderInfoFlags flags,
+                                          GHashTable *folder_info_results,
+                                          GCancellable *cancellable,
+                                          GError **error)
 {
-       GList *namespaces = NULL;
-       CamelIMAPXNamespaceList *nsl = NULL;
-
-       /* Add code to return the namespaces from preference else all of them */
-       nsl = imapx_store->summary->namespaces;
-       if (nsl->personal != NULL)
-               namespaces = g_list_append (namespaces, nsl->personal);
-       if (nsl->other != NULL)
-               namespaces = g_list_append (namespaces, nsl->other);
-       if (nsl->shared != NULL)
-               namespaces = g_list_append (namespaces, nsl->shared);
-
-       return namespaces;
-}
+       CamelIMAPXNamespaceResponse *namespace_response;
+       GList *list, *link;
+       gboolean success = TRUE;
 
-static GHashTable *
-fetch_mailboxes_for_namespaces (CamelIMAPXStore *imapx_store,
-                                const gchar *pattern,
-                                gboolean sync,
-                                GCancellable *cancellable,
-                                GError **error)
-{
-       CamelIMAPXServer *server;
-       GHashTable *mailboxes = NULL;
-       GList *namespaces = NULL, *l;
-       const gchar *list_ext = NULL;
+       namespace_response = camel_imapx_server_ref_namespaces (server);
+       g_return_val_if_fail (namespace_response != NULL, FALSE);
 
-       server = camel_imapx_store_ref_server (imapx_store, error);
+       list = camel_imapx_namespace_response_list (namespace_response);
 
-       if (server == NULL)
-               return NULL;
+       for (link = list; link != NULL; link = g_list_next (link)) {
+               CamelIMAPXNamespace *namespace;
+               CamelIMAPXNamespaceCategory ns_category;
+               const gchar *ns_prefix;
+               gchar *pattern;
 
-       if (CAMEL_IMAPX_HAVE_CAPABILITY (server->cinfo, LIST_EXTENDED))
-               list_ext = "RETURN (SUBSCRIBED)";
+               namespace = CAMEL_IMAPX_NAMESPACE (link->data);
+               ns_category = camel_imapx_namespace_get_category (namespace);
+               ns_prefix = camel_imapx_namespace_get_prefix (namespace);
 
-       mailboxes = g_hash_table_new_full (
-               (GHashFunc) imapx_name_hash,
-               (GEqualFunc) imapx_name_equal,
-               (GDestroyNotify) g_free,
-               (GDestroyNotify) camel_folder_info_free);
+               if (ns_category != category)
+                       continue;
 
-       namespaces = get_namespaces (imapx_store);
+               pattern = g_strdup_printf ("%s*", ns_prefix);
 
-       for (l = namespaces; l != NULL; l = g_list_next (l)) {
-               CamelIMAPXStoreNamespace *ns = l->data;
+               success = fetch_folder_info_for_pattern (
+                       server, namespace, pattern, flags,
+                       folder_info_results, cancellable, error);
 
-               while (ns != NULL) {
-                       CamelStoreGetFolderInfoFlags flags = 0;
-                       gboolean success;
-                       gchar *pat;
+               g_free (pattern);
 
-                       if (pattern != NULL)
-                               pat = g_strdup_printf ("%s*", pattern);
-                       else if (*ns->prefix != '\0')
-                               pat = g_strdup_printf (
-                                       "%s%c*", ns->prefix, ns->sep);
-                       else
-                               pat = g_strdup ("*");
-
-                       if (sync)
-                               flags |= CAMEL_STORE_FOLDER_INFO_SUBSCRIPTION_LIST;
-
-                       success = fetch_mailboxes_for_pattern (
-                               imapx_store, server, pat, flags, list_ext,
-                               mailboxes, cancellable, error);
-
-                       if (success && list_ext == NULL) {
-                               /* If the server doesn't support LIST-EXTENDED
-                                * then we have to issue the LSUB command to
-                                * list the subscribed mailboxes separately. */
-                               flags |= CAMEL_STORE_FOLDER_INFO_SUBSCRIBED;
-                               success = fetch_mailboxes_for_pattern (
-                                       imapx_store, server, pat, flags, NULL,
-                                       mailboxes, cancellable, error);
-                       }
+               if (!success)
+                       break;
+       }
 
-                       g_free (pat);
+       g_list_free_full (list, (GDestroyNotify) g_object_unref);
 
-                       if (!success) {
-                               g_hash_table_destroy (mailboxes);
-                               mailboxes = NULL;
-                               goto exit;
-                       }
+       g_object_unref (namespace_response);
 
-                       if (pattern != NULL)
-                               goto exit;
+       return success;
+}
 
-                       ns = ns->next;
-               }
+static gboolean
+fetch_folder_info_from_folder_path (CamelIMAPXServer *server,
+                                    const gchar *folder_path,
+                                    CamelStoreGetFolderInfoFlags flags,
+                                    GHashTable *folder_info_results,
+                                    GCancellable *cancellable,
+                                    GError **error)
+{
+       CamelIMAPXNamespaceResponse *namespace_response;
+       CamelIMAPXNamespace *namespace;
+       gchar *mailbox_name;
+       gchar *utf7_mailbox_name;
+       gchar *pattern;
+       gchar separator;
+       gboolean success = FALSE;
+
+       namespace_response = camel_imapx_server_ref_namespaces (server);
+       g_return_val_if_fail (namespace_response != NULL, FALSE);
+
+       /* Find a suitable IMAP namespace for the folder path. */
+       namespace = camel_imapx_namespace_response_lookup_for_path (
+               namespace_response, folder_path);
+       if (namespace == NULL) {
+               g_set_error (
+                       error, CAMEL_STORE_ERROR,
+                       CAMEL_STORE_ERROR_INVALID,
+                       _("No IMAP namespace for folder path '%s'"),
+                       folder_path);
+               goto exit;
        }
 
+       /* Convert the folder path to a mailbox name. */
+       separator = camel_imapx_namespace_get_separator (namespace);
+       mailbox_name = g_strdelimit (g_strdup (folder_path), "/", separator);
+
+       utf7_mailbox_name = camel_utf8_utf7 (mailbox_name);
+       pattern = g_strdup_printf ("%s*", utf7_mailbox_name);
+
+       success = fetch_folder_info_for_pattern (
+               server, namespace, pattern, flags,
+               folder_info_results, cancellable, error);
+
+       g_free (pattern);
+       g_free (utf7_mailbox_name);
+       g_free (mailbox_name);
+
 exit:
-       g_list_free (namespaces);
-       g_object_unref (server);
+       g_clear_object (&namespace);
+       g_clear_object (&namespace_response);
 
-       return mailboxes;
+       return success;
 }
 
 static gboolean
 sync_folders (CamelIMAPXStore *imapx_store,
-              const gchar *pattern,
-              gboolean sync,
+              const gchar *root_folder_path,
+              CamelStoreGetFolderInfoFlags flags,
               GCancellable *cancellable,
               GError **error)
 {
-       CamelSettings *settings;
+       CamelIMAPXServer *server;
        CamelStoreSummary *store_summary;
-       GHashTable *mailboxes;
+       GHashTable *folder_info_results;
        GPtrArray *array;
-       gboolean notify_all;
        guint ii;
+       gboolean success;
+
+       server = camel_imapx_store_ref_server (imapx_store, error);
+       if (server == NULL)
+               return FALSE;
 
        store_summary = CAMEL_STORE_SUMMARY (imapx_store->summary);
 
-       mailboxes = fetch_mailboxes_for_namespaces (
-               imapx_store, pattern, sync, cancellable, error);
+       /* mailbox name -> CamelFolderInfo */
+       folder_info_results = g_hash_table_new_full (
+               (GHashFunc) imapx_name_hash,
+               (GEqualFunc) imapx_name_equal,
+               (GDestroyNotify) g_free,
+               (GDestroyNotify) camel_folder_info_free);
+
+       /* This suppresses CamelStore signal emissions
+        * in imapx_store_process_mailbox_attributes(). */
+       g_atomic_int_inc (&imapx_store->priv->syncing_folders);
 
-       if (mailboxes == NULL)
-               return FALSE;
+       if (root_folder_path != NULL && *root_folder_path != '\0') {
+               success = fetch_folder_info_from_folder_path (
+                       server, root_folder_path, flags,
+                       folder_info_results, cancellable, error);
+       } else {
+               /* XXX We only fetch personal mailboxes at this time. */
+               success = fetch_folder_info_for_namespace_category (
+                       server, CAMEL_IMAPX_NAMESPACE_PERSONAL, flags,
+                       folder_info_results, cancellable, error);
+       }
 
-       settings = camel_service_ref_settings (CAMEL_SERVICE (imapx_store));
-       notify_all = !camel_imapx_settings_get_use_subscriptions (
-               CAMEL_IMAPX_SETTINGS (settings));
-       g_object_unref (settings);
+       /* Don't need to test for zero, just decrement atomically. */
+       g_atomic_int_dec_and_test (&imapx_store->priv->syncing_folders);
+
+       if (!success)
+               goto exit;
 
        array = camel_store_summary_array (store_summary);
 
        for (ii = 0; ii < array->len; ii++) {
                CamelStoreInfo *si;
                CamelFolderInfo *fi;
-               CamelIMAPXStoreNamespace *ns;
                const gchar *mailbox_name;
+               const gchar *si_path;
                gboolean pattern_match;
 
                si = g_ptr_array_index (array, ii);
+               si_path = camel_store_info_path (store_summary, si);
 
                mailbox_name = ((CamelIMAPXStoreInfo *) si)->mailbox_name;
                if (mailbox_name == NULL || *mailbox_name == '\0')
                        continue;
 
-               ns = camel_imapx_store_summary_namespace_find_by_mailbox (
-                       imapx_store->summary, mailbox_name);
-
                pattern_match =
-                       (pattern == NULL) || (*pattern == '\0') ||
-                       imapx_match_pattern (ns, pattern, mailbox_name);
+                       (root_folder_path == NULL) ||
+                       (*root_folder_path == '\0') ||
+                       (g_str_has_prefix (si_path, root_folder_path));
                if (!pattern_match)
                        continue;
 
-               fi = g_hash_table_lookup (mailboxes, mailbox_name);
-
-               if (fi != NULL) {
-                       gboolean do_notify = notify_all;
-
-                       /* Check if the SUBSCRIBED flags in the
-                        * folder info and store info disagree.
-                        * The folder info is authoritative. */
-                       if (((fi->flags ^ si->flags) & CAMEL_STORE_INFO_FOLDER_SUBSCRIBED)) {
-                               si->flags &= ~CAMEL_FOLDER_SUBSCRIBED;
-                               si->flags |= fi->flags & CAMEL_FOLDER_SUBSCRIBED;
-                               camel_store_summary_touch (store_summary);
-                               do_notify = TRUE;
-                       }
-
-                       if (do_notify) {
-                               camel_store_folder_created (
-                                       CAMEL_STORE (imapx_store), fi);
-                               camel_subscribable_folder_subscribed (
-                                       CAMEL_SUBSCRIBABLE (imapx_store), fi);
-                       }
-               } else {
-                       gchar *dup_folder_path;
-                       const gchar *si_path;
+               fi = g_hash_table_lookup (folder_info_results, mailbox_name);
 
-                       si_path = camel_store_info_path (store_summary, si);
-                       dup_folder_path = g_strdup (si_path);
+               if (fi == NULL) {
+                       gchar *dup_folder_path = g_strdup (si_path);
 
                        if (dup_folder_path != NULL) {
                                imapx_unmark_folder_subscribed (
@@ -1661,9 +1620,12 @@ sync_folders (CamelIMAPXStore *imapx_store,
 
        camel_store_summary_array_free (store_summary, array);
 
-       g_hash_table_destroy (mailboxes);
+exit:
+       g_hash_table_destroy (folder_info_results);
 
-       return TRUE;
+       g_object_unref (server);
+
+       return success;
 }
 
 static void
@@ -1689,8 +1651,7 @@ imapx_refresh_finfo (CamelSession *session,
                CAMEL_SERVICE (store), cancellable, error))
                goto exit;
 
-       /* look in all namespaces */
-       sync_folders (store, "", FALSE, cancellable, error);
+       sync_folders (store, NULL, 0, cancellable, error);
 
        camel_store_summary_save (CAMEL_STORE_SUMMARY (store->summary));
 
@@ -1863,7 +1824,6 @@ imapx_store_get_folder_info_sync (CamelStore *store,
        CamelStoreSummary *store_summary;
        gboolean initial_setup = FALSE;
        gboolean use_subscriptions;
-       gchar *pattern = NULL;
 
        service = CAMEL_SERVICE (store);
        imapx_store = CAMEL_IMAPX_STORE (store);
@@ -1891,6 +1851,7 @@ imapx_store_get_folder_info_sync (CamelStore *store,
                initial_setup = TRUE;
        }
 
+       /* XXX I don't know why the SUBSCRIBED flag matters here. */
        if (!initial_setup && flags & CAMEL_STORE_FOLDER_INFO_SUBSCRIBED) {
                time_t time_since_last_refresh;
 
@@ -1912,32 +1873,15 @@ imapx_store_get_folder_info_sync (CamelStore *store,
 
                        g_object_unref (session);
                }
-
-               fi = get_folder_info_offline (store, top, flags, error);
-               goto exit;
        }
 
-       if (*top && flags & CAMEL_STORE_FOLDER_INFO_SUBSCRIPTION_LIST) {
+       /* Avoid server interaction if the FAST flag is set. */
+       if (!initial_setup && flags & CAMEL_STORE_FOLDER_INFO_FAST) {
                fi = get_folder_info_offline (store, top, flags, error);
                goto exit;
        }
 
-       if (*top) {
-               gchar *mailbox;
-
-               mailbox = camel_imapx_store_summary_mailbox_from_path (
-                       imapx_store->summary, top);
-               if (mailbox == NULL)
-                       mailbox = camel_imapx_store_summary_path_to_mailbox (
-                               imapx_store->summary, top,
-                               imapx_store->dir_sep);
-               pattern = camel_utf8_utf7 (mailbox);
-               g_free (mailbox);
-       } else {
-               pattern = g_strdup ("");
-       }
-
-       if (!sync_folders (imapx_store, pattern, TRUE, cancellable, error))
+       if (!sync_folders (imapx_store, top, flags, cancellable, error))
                goto exit;
 
        camel_store_summary_save (store_summary);
@@ -1951,8 +1895,6 @@ imapx_store_get_folder_info_sync (CamelStore *store,
 exit:
        g_mutex_unlock (&imapx_store->priv->get_finfo_lock);
 
-       g_free (pattern);
-
        return fi;
 }
 


[Date Prev][Date Next]   [Thread Prev][Thread Next]   [Thread Index] [Date Index] [Author Index]