Latest version namespace support



FYI

-- 
Philip Van Hoof, freelance software developer
home: me at pvanhoof dot be 
gnome: pvanhoof at gnome dot org 
http://pvanhoof.be/blog
http://codeminded.be



Index: libtinymail-camel/camel-lite/camel/camel-store.c
===================================================================
--- libtinymail-camel/camel-lite/camel/camel-store.c	(revision 3174)
+++ libtinymail-camel/camel-lite/camel/camel-store.c	(working copy)
@@ -1044,6 +1044,7 @@
 				camel_url_set_param (url, "noselect", "yes");
 				pfi->uri = camel_url_to_string (url, 0);
 				camel_url_free (url);
+				pfi->flags |= CAMEL_FOLDER_SUBSCRIBED | CAMEL_FOLDER_NOSELECT;
 
 				g_hash_table_insert (hash, pname, pfi);
 				g_ptr_array_add (folders, pfi);
Index: libtinymail-camel/camel-lite/camel/providers/imap/camel-imap-store.c
===================================================================
--- libtinymail-camel/camel-lite/camel/providers/imap/camel-imap-store.c	(revision 3174)
+++ libtinymail-camel/camel-lite/camel/providers/imap/camel-imap-store.c	(working copy)
@@ -415,6 +415,9 @@
 		disco->diary = NULL;
 	}
 
+	if (imap_store->namespaces)
+		imap_namespaces_destroy (imap_store->namespaces);
+
 	/* g_static_rec_mutex_free (imap_store->idle_prefix_lock); */
 	g_free (imap_store->idle_prefix_lock);
 	imap_store->idle_prefix_lock = NULL;
@@ -1863,6 +1866,8 @@
 	if ((store->capabilities & IMAP_CAPABILITY_NAMESPACE) &&
 		!(store->parameters & IMAP_PARAM_OVERRIDE_NAMESPACE))
 	{
+		struct _namespaces *namespaces;
+
 		response = camel_imap_command (store, NULL, ex, "NAMESPACE");
 		if (!response)
 			goto done;
@@ -1871,34 +1876,43 @@
 		if (!result)
 			goto done;
 
-#if 0
-		/* new code... */
-		namespaces = imap_parse_namespace_response (result);
-		imap_namespaces_destroy (namespaces);
-		/* end new code */
-#endif
+		store->namespaces = imap_parse_namespace_response (result);
+		namespaces = store->namespaces;
 
-		name = camel_strstrcase (result, "NAMESPACE ((");
-		if (name) {
-			char *sep;
+		if (namespaces && namespaces->personal) {
+			store->namespace = namespaces->personal->prefix;
+			store->dir_sep = namespaces->personal->delim;
+		} else {
+			store->namespace = NULL;
+			store->dir_sep = 0;
+		}
 
-			name += 12;
-			store->namespace = imap_parse_string ((const char **) &name, &len);
-			if (name && *name++ == ' ') {
-				sep = imap_parse_string ((const char **) &name, &len);
-				if (sep) {
-					store->dir_sep = *sep;
-					g_free (sep);
-				}
-			}
+		if (namespaces && namespaces->personal) {
+			ns = camel_imap_store_summary_namespace_new(store->summary, namespaces->personal->prefix, namespaces->personal->delim);
+			camel_imap_store_summary_namespace_add(store->summary,ns);
+			camel_imap_store_summary_namespace_set(store->summary, ns);
 		}
+
+		if (namespaces && namespaces->other) {
+			ns = camel_imap_store_summary_namespace_new(store->summary, namespaces->other->prefix, namespaces->other->delim);
+			camel_imap_store_summary_namespace_add(store->summary,ns);
+		}
+
+		if (namespaces && namespaces->shared) {
+			ns = camel_imap_store_summary_namespace_new(store->summary, namespaces->shared->prefix, namespaces->shared->delim);
+			camel_imap_store_summary_namespace_add(store->summary,ns);
+		}
+
+
 		g_free (result);
+	} else {
+		ns = camel_imap_store_summary_namespace_new(store->summary, store->namespace, store->dir_sep);
+		camel_imap_store_summary_namespace_add(store->summary,ns);
+		camel_imap_store_summary_namespace_set(store->summary, ns);
 	}
 
-	if (store->namespace && strlen (store->namespace) == 0) {
-		g_free (store->namespace);
+	if (store->namespace && strlen (store->namespace) == 0)
 		store->namespace = NULL;
-	}
 
 	if (store->namespace && !store->dir_sep) {
 		if (FALSE && store->server_level >= IMAP_LEVEL_IMAP4REV1) {
@@ -1938,7 +1952,6 @@
 	len = strlen (store->namespace);
 	if (len && store->namespace[len - 1] != store->dir_sep) {
 		gchar *tmp;
-
 		tmp = g_strdup_printf ("%s%c", store->namespace, store->dir_sep);
 		g_free (store->namespace);
 		store->namespace = tmp;
@@ -1946,8 +1959,6 @@
 
 	g_static_rec_mutex_lock (store->sum_lock);
 
-	ns = camel_imap_store_summary_namespace_new(store->summary, store->namespace, store->dir_sep);
-	camel_imap_store_summary_namespace_set(store->summary, ns);
 
 	if ((store->parameters & IMAP_PARAM_SUBSCRIPTIONS)
 	    && camel_store_summary_count((CamelStoreSummary *)store->summary) == 0)
@@ -2054,10 +2065,8 @@
 		store->authtypes = NULL;
 	}
 
-	if (store->namespace && !(store->parameters & IMAP_PARAM_OVERRIDE_NAMESPACE)) {
-		g_free (store->namespace);
+	if (store->namespace && !(store->parameters & IMAP_PARAM_OVERRIDE_NAMESPACE))
 		store->namespace = NULL;
-	}
 
 	return TRUE;
 }
@@ -2452,106 +2461,7 @@
 	return (CamelMessageInfo *) mi;
 }
 
-GPtrArray*
-_camel_imap_store_get_recent_messages (CamelImapStore *imap_store, const char *folder_name, int *unseen, int *messages, gboolean withthem)
-{
-	guint32 uidnext=-1;
-	struct imap_status_item *items, *item;
-	guint ounseen, omessages, ouidnext;
-	GPtrArray *retval = NULL;
-	CamelException ex = CAMEL_EXCEPTION_INITIALISER;
-	CamelImapResponse *response;
-	CamelException tex = CAMEL_EXCEPTION_INITIALISER;
 
-	if (!camel_disco_store_check_online (CAMEL_DISCO_STORE (imap_store), &ex))
-		return NULL;
-
-/*
-      Example:    C: A042 STATUS blurdybloop (UIDNEXT MESSAGES)
-                  S: * STATUS blurdybloop (MESSAGES 231 UIDNEXT 44292)
-                  S: A042 OK STATUS completed
-*/
-
-	camel_operation_uncancel (NULL);
-
-
-	/* On for example courier, the selected's STATUS is cached (kill that cache) */
-	if (withthem)
-	{
-	    response = camel_imap_command (imap_store, NULL, &tex,
-			    "SELECT"/*, folder_name*/);
-
-printf ("SELECT %s\n", folder_name);
-
-	    if (response)
-		    camel_imap_response_free (imap_store, response);
-	}
-
-	item = items = get_folder_status (imap_store, folder_name, "MESSAGES UNSEEN UIDNEXT", FALSE);
-	while (item != NULL) {
-		if (!g_ascii_strcasecmp (item->name, "MESSAGES"))
-			*messages = item->value;
-		if (!g_ascii_strcasecmp (item->name, "UNSEEN"))
-			*unseen = item->value;
-		if (!g_ascii_strcasecmp (item->name, "UIDNEXT"))
-			uidnext = item->value;
-		item = item->next;
-	}
-	imap_status_item_free (items);
-printf ("%d %d %d\n", *messages, *unseen, uidnext);
-
-	if (withthem)
-	{
-		camel_imap_store_get_status_for (imap_store, folder_name, &omessages, &ounseen, &ouidnext);
-
-		if (ouidnext != uidnext)
-		{
-			CamelImapResponseType type;
-			char *resp;
-
-			camel_exception_clear (&tex);
-
-			if (!camel_imap_command_start (imap_store, NULL, &tex,
-				"UID FETCH %d:* (FLAGS RFC822.SIZE INTERNALDATE BODY.PEEK[HEADER])", ouidnext-1))
-				goto done;
-
-			while ((type = camel_imap_command_response (imap_store, &resp, &tex))
-				== CAMEL_IMAP_RESPONSE_UNTAGGED)
-			{
-			   if (resp)
-			   {
-				CamelMessageInfo *mi = parse_message_header (resp);
-				g_free (resp);
-
-				if (mi)
-				{
-					if (retval == NULL)
-						retval = g_ptr_array_new ();
-					g_ptr_array_add (retval, mi);
-				}
-			   }
-			}
-
-			/* Restore the original folder selection */
-			if (imap_store->current_folder != NULL && imap_store->current_folder->full_name != NULL)
-			{
-				response = camel_imap_command (imap_store, NULL, &tex,
-						"SELECT %F", imap_store->current_folder->full_name);
-				if (response)
-					camel_imap_response_free (imap_store, response);
-			}
-		}
-	}
-
-done:
-
-	camel_imap_store_set_status_for (imap_store, folder_name, *messages, *unseen, uidnext);
-
-	camel_imap_store_start_idle (imap_store);
-
-	return retval;
-}
-
 static void
 imap_get_folder_status (CamelStore *store, const char *folder_name, int *unseen, int *messages, int *uidnext)
 {
@@ -3445,12 +3355,6 @@
 	if (imap_store->capabilities & IMAP_CAPABILITY_LISTEXT)
 		loops = 1;
 
-	/* We do a LIST followed by LSUB, and merge the results.  LSUB may not be a strict
-	   subset of LIST for some servers, so we can't use either or separately */
-
-	/* TNY TODO! It used to loop until 2, but I think it's just wrong to
-	   merge with LSUB?! (It doesn't make any sense) */
-
 	present = g_hash_table_new(folder_hash, folder_eq);
 
 	for (j = 0; j < loops; j++) {
@@ -3465,6 +3369,7 @@
 			response = camel_imap_command (imap_store, NULL, ex,
 				"%s \"\" %G", j==1 ? "LSUB" : "LIST",
 				pattern);
+
 		if (!response)
 			goto fail;
 
@@ -3473,19 +3378,6 @@
 			fi = parse_list_response_as_folder_info (imap_store, list);
 
 			if (fi) {
-				if (FALSE && j == 0) {
-					struct imap_status_item *item, *items;
-					item = items = get_folder_status (imap_store, fi->full_name, "MESSAGES UNSEEN", TRUE);
-					while (item != NULL) {
-						if (!g_ascii_strcasecmp (item->name, "MESSAGES"))
-							fi->total = item->value;
-						if (!g_ascii_strcasecmp (item->name, "UNSEEN"))
-							fi->unread = item->value;
-						item = item->next;
-					}
-					imap_status_item_free (items);
-				}
-
 				if (fi->flags & CAMEL_FOLDER_NONEXISTENT) {
 					camel_folder_info_free(fi);
 					continue;
@@ -3497,7 +3389,6 @@
 
 					if (j == 1) {
 						/* It's in LSUB but not in LIST? */
-
 						fi->flags |= CAMEL_FOLDER_SUBSCRIBED;
 						if ((fi->flags & (CAMEL_IMAP_FOLDER_MARKED | CAMEL_IMAP_FOLDER_UNMARKED)))
 							imap_store->capabilities |= IMAP_CAPABILITY_useful_lsub;
@@ -3514,12 +3405,10 @@
 						fi->flags |= CAMEL_FOLDER_SUBSCRIBED;
 						hfi->flags |= CAMEL_STORE_INFO_FOLDER_SUBSCRIBED;
 					}
-					if (j == 0)
-					{
+					if (j == 0) {
 						hfi->unread = fi->unread;
 						hfi->total = fi->total;
 					}
-					camel_folder_info_free(fi);
 				}
 			}
 		}
@@ -3560,32 +3449,181 @@
 	camel_imap_store_start_idle (imap_store);
 }
 
-#if 0
+
 static void
-dumpfi(CamelFolderInfo *fi)
+get_folders_sync_ns(CamelImapStore *imap_store, struct _namespace *namespace, CamelException *ex)
 {
-	int depth;
-	CamelFolderInfo *n = fi;
+	CamelImapResponse *response;
+	CamelFolderInfo *fi, *hfi;
+	char *list;
+	int i, count, j;
+	GHashTable *present;
+	CamelStoreInfo *si;
+	int loops = 2;
 
-	if (fi == NULL)
-		return;
+	if (imap_store->capabilities & IMAP_CAPABILITY_LISTEXT)
+		loops = 1;
 
-	depth = 0;
-	while (n->parent) {
-		depth++;
-		n = n->parent;
+	present = g_hash_table_new(folder_hash, folder_eq);
+
+	for (j = 0; j < loops; j++) {
+
+		camel_imap_store_stop_idle (imap_store);
+
+		if (imap_store->capabilities & IMAP_CAPABILITY_LISTEXT)
+			response = camel_imap_command (imap_store, NULL, ex,
+				"%s \"%s\" %G", "LIST (SUBSCRIBED)",
+				(strlen (namespace->prefix) > 0)?"*":"%");
+		else
+			response = camel_imap_command (imap_store, NULL, ex,
+				"%s \"%s\" %G", j==1 ? "LSUB" : "LIST",
+				namespace->prefix, 
+				(strlen (namespace->prefix) > 0)?"*":"%");
+
+		if (!response)
+			goto fail;
+
+		for (i = 0; i < response->untagged->len; i++) {
+			list = response->untagged->pdata[i];
+			fi = parse_list_response_as_folder_info (imap_store, list);
+
+			if (fi) {
+				if (fi->flags & CAMEL_FOLDER_NONEXISTENT) {
+					camel_folder_info_free(fi);
+					continue;
+				}
+
+				hfi = g_hash_table_lookup(present, fi->full_name);
+
+				if (hfi == NULL) {
+
+					if (j == 1) {
+						/* It's in LSUB but not in LIST? */
+						fi->flags |= CAMEL_FOLDER_SUBSCRIBED;
+						if ((fi->flags & (CAMEL_IMAP_FOLDER_MARKED | CAMEL_IMAP_FOLDER_UNMARKED)))
+							imap_store->capabilities |= IMAP_CAPABILITY_useful_lsub;
+					}
+
+					if (j == 0) /* From the LSUB we don't add folders */
+						g_hash_table_insert(present, fi->full_name, fi);
+					else {
+						fi->flags |= CAMEL_FOLDER_NONEXISTENT;
+						camel_folder_info_free(fi);
+					}
+				} else {
+					if (j == 1) {
+						fi->flags |= CAMEL_FOLDER_SUBSCRIBED;
+						hfi->flags |= CAMEL_STORE_INFO_FOLDER_SUBSCRIBED;
+					}
+					if (j == 0) {
+						hfi->unread = fi->unread;
+						hfi->total = fi->total;
+					}
+				}
+			}
+		}
+		camel_imap_response_free (imap_store, response);
 	}
 
-	while (fi) {
-		printf("%-40s %-30s %*s\n", fi->path, fi->full_name, depth*2+strlen(fi->url), fi->url);
-		if (fi->child)
-			dumpfi(fi->child);
-		fi = fi->sibling;
+	/* Sync summary to match */
+
+	/* FIXME: we need to emit folder_create/subscribed/etc events for any new folders */
+	count = camel_store_summary_count((CamelStoreSummary *)imap_store->summary);
+
+	for (i=0;i<count;i++) {
+		si = camel_store_summary_index((CamelStoreSummary *)imap_store->summary, i);
+		if (si == NULL)
+			continue;
+
+		if ((fi = g_hash_table_lookup(present, camel_store_info_path(imap_store->summary, si))) != NULL) {
+			if (((fi->flags ^ si->flags) & CAMEL_STORE_INFO_FOLDER_SUBSCRIBED))
+				si->flags = (si->flags & ~CAMEL_FOLDER_SUBSCRIBED) | (fi->flags & CAMEL_FOLDER_SUBSCRIBED);
+			si->unread = fi->unread;
+			si->total = fi->total;
+			camel_store_summary_touch((CamelStoreSummary *)imap_store->summary);
+			camel_store_summary_save((CamelStoreSummary *)imap_store->summary, ex);
+		} else {
+			camel_store_summary_remove((CamelStoreSummary *)imap_store->summary, si);
+			camel_store_summary_save((CamelStoreSummary *)imap_store->summary, ex);
+			count--;
+			i--;
+		}
+		camel_store_summary_info_free((CamelStoreSummary *)imap_store->summary, si);
 	}
+fail:
+	g_hash_table_foreach(present, get_folders_free, NULL);
+	g_hash_table_destroy(present);
+
+	camel_imap_store_start_idle (imap_store);
 }
-#endif
 
+
+
+
 static void
+get_folders_sync_ns_only_lsub (CamelImapStore *imap_store, struct _namespace *namespace, CamelException *ex)
+{
+	CamelImapResponse *response;
+	CamelFolderInfo *fi, *hfi;
+	char *list;
+	int i, count, j;
+	GHashTable *present;
+	CamelStoreInfo *si;
+	present = g_hash_table_new(folder_hash, folder_eq);
+
+	camel_imap_store_stop_idle (imap_store);
+
+	response = camel_imap_command (imap_store, NULL, ex,
+		"LSUB \"%s\" %G", namespace->prefix, 
+		(strlen (namespace->prefix) > 0)?"*":"%");
+
+	if (!response)
+		goto fail;
+
+	for (i = 0; i < response->untagged->len; i++) {
+		list = response->untagged->pdata[i];
+		fi = parse_list_response_as_folder_info (imap_store, list);
+
+		hfi = g_hash_table_lookup(present, fi->full_name);
+
+		if (hfi == NULL) {
+			fi->flags |= CAMEL_FOLDER_SUBSCRIBED;
+			g_hash_table_insert(present, fi->full_name, fi);
+		}
+	}
+
+	camel_imap_response_free (imap_store, response);
+
+	count = camel_store_summary_count((CamelStoreSummary *)imap_store->summary);
+
+	for (i=0;i<count;i++) {
+		si = camel_store_summary_index((CamelStoreSummary *)imap_store->summary, i);
+		if (si == NULL)
+			continue;
+
+		if ((fi = g_hash_table_lookup(present, camel_store_info_path(imap_store->summary, si))) != NULL) {
+			if (((fi->flags ^ si->flags) & CAMEL_STORE_INFO_FOLDER_SUBSCRIBED))
+				si->flags = (si->flags & ~CAMEL_FOLDER_SUBSCRIBED) | (fi->flags & CAMEL_FOLDER_SUBSCRIBED);
+			si->unread = fi->unread;
+			si->total = fi->total;
+			camel_store_summary_touch((CamelStoreSummary *)imap_store->summary);
+			camel_store_summary_save((CamelStoreSummary *)imap_store->summary, ex);
+		} else {
+			/* camel_store_summary_remove((CamelStoreSummary *)imap_store->summary, si);
+			camel_store_summary_save((CamelStoreSummary *)imap_store->summary, ex);
+			count--;
+			i--;*/
+		}
+		camel_store_summary_info_free((CamelStoreSummary *)imap_store->summary, si);
+	}
+fail:
+	g_hash_table_foreach(present, get_folders_free, NULL);
+	g_hash_table_destroy(present);
+
+	camel_imap_store_start_idle (imap_store);
+}
+
+static void
 fill_fi(CamelStore *store, CamelFolderInfo *fi, guint32 flags)
 {
 	CamelFolder *folder;
@@ -3613,60 +3651,7 @@
 	g_free (folder_dir);
 }
 
-struct _refresh_msg {
-	CamelSessionThreadMsg msg;
 
-	CamelStore *store;
-	CamelException ex;
-};
-
-static void
-refresh_refresh(CamelSession *session, CamelSessionThreadMsg *msg)
-{
-	struct _refresh_msg *m = (struct _refresh_msg *)msg;
-	CamelImapStore *store = (CamelImapStore *)m->store;
-	CamelException nex = CAMEL_EXCEPTION_INITIALISER;
-
-	CAMEL_SERVICE_REC_LOCK(m->store, connect_lock);
-
-	if (!camel_disco_store_check_online((CamelDiscoStore *)m->store, &m->ex))
-		goto done;
-
-	if (store->namespace && store->namespace[0]) {
-		char *pattern;
-
-		get_folders_sync(store, "INBOX", &m->ex);
-		if (camel_exception_is_set(&m->ex))
-			goto done;
-		get_folders_sync(store, store->namespace, &m->ex);
-		if (camel_exception_is_set(&m->ex))
-			goto done;
-		pattern = imap_concat(store, store->namespace, "*");
-		get_folders_sync(store, pattern, &m->ex);
-		g_free(pattern);
-	} else {
-		get_folders_sync((CamelImapStore *)m->store, "*", &m->ex);
-	}
-	camel_store_summary_touch((CamelStoreSummary *)((CamelImapStore *)m->store)->summary);
-	camel_store_summary_save((CamelStoreSummary *)((CamelImapStore *)m->store)->summary, &nex);
-done:
-	CAMEL_SERVICE_REC_UNLOCK(m->store, connect_lock);
-}
-
-static void
-refresh_free(CamelSession *session, CamelSessionThreadMsg *msg)
-{
-	struct _refresh_msg *m = (struct _refresh_msg *)msg;
-
-	camel_object_unref(m->store);
-	camel_exception_clear(&m->ex);
-}
-
-static CamelSessionThreadOps refresh_ops = {
-	refresh_refresh,
-	refresh_free,
-};
-
 static CamelFolderInfo *
 get_folder_info_online (CamelStore *store, const char *top, guint32 flags, CamelException *ex)
 {
@@ -3675,10 +3660,6 @@
 
 	g_static_rec_mutex_lock (imap_store->sum_lock);
 
-	/* If we have a list of folders already, use that, but if we haven't
-	   updated for a while, then trigger an asynchronous rescan.  Otherwise
-	   we update the list first, and then build it from that */
-
 	if (top == NULL)
 		top = "";
 
@@ -3688,71 +3669,41 @@
 	if (camel_debug("imap:folder_info"))
 		printf("get folder info online\n");
 
-	if (FALSE && (flags & CAMEL_STORE_FOLDER_INFO_SUBSCRIBED)
-	    && camel_store_summary_count((CamelStoreSummary *)imap_store->summary) > 0) {
-		time_t now;
-		int ref;
+	CAMEL_SERVICE_REC_LOCK(store, connect_lock);
 
-		now = time(NULL);
-		ref = now > imap_store->refresh_stamp+60*60*1;
-		if (ref) {
-			CAMEL_SERVICE_REC_LOCK(store, connect_lock);
-			ref = now > imap_store->refresh_stamp+60*60*1;
-			if (ref) {
-				struct _refresh_msg *m;
+	if (!camel_disco_store_check_online((CamelDiscoStore *)imap_store, ex))
+		goto fail;
 
-				imap_store->refresh_stamp = now;
+	if (top[0] == 0) {
+		struct _namespaces *ns = imap_store->namespaces;
 
-				m = camel_session_thread_msg_new(((CamelService *)store)->session, &refresh_ops, sizeof(*m));
-				m->store = store;
-				camel_object_ref(store);
-				camel_exception_init(&m->ex);
-				camel_session_thread_queue(((CamelService *)store)->session, &m->msg, 0);
-			}
-			CAMEL_SERVICE_REC_UNLOCK(store, connect_lock);
-		}
-	} else {
-		char *pattern;
-		int i;
+		if (ns->personal)
+			get_folders_sync_ns (imap_store, ns->personal, ex);
+		if (camel_exception_is_set(ex))
+			goto fail;
 
-		CAMEL_SERVICE_REC_LOCK(store, connect_lock);
+		if (ns->other)
+			get_folders_sync_ns_only_lsub (imap_store, ns->other, ex);
+		if (camel_exception_is_set(ex))
+			goto fail;
 
-		if (!camel_disco_store_check_online((CamelDiscoStore *)imap_store, ex))
+		if (ns->shared)
+			get_folders_sync_ns_only_lsub (imap_store, ns->shared, ex);
+		if (camel_exception_is_set(ex))
 			goto fail;
 
-		if (top[0] == 0) {
-			if (imap_store->namespace && imap_store->namespace[0]) {
-				get_folders_sync (imap_store, "INBOX", ex);
-				if (camel_exception_is_set(ex))
-					goto fail;
-
-				i = strlen(imap_store->namespace)-1;
-				pattern = g_alloca(i+5);
-				strcpy(pattern, imap_store->namespace);
-				while (i>0 && pattern[i] == imap_store->dir_sep)
-					pattern[i--] = 0;
-				i++;
-			} else {
-				pattern = g_alloca(2);
-				pattern[0] = '*';
-				pattern[1] = 0;
-				i=0;
-			}
-		} else {
-			char *name;
-
-			name = camel_imap_store_summary_full_from_path(imap_store->summary, top);
-			if (name == NULL)
-				name = camel_imap_store_summary_path_to_full(imap_store->summary, top, imap_store->dir_sep);
-
-			i = strlen(name);
-			pattern = g_alloca(i+5);
-			strcpy(pattern, name);
-			g_free(name);
-		}
-
-		
-		if (pattern[0] != '*' && imap_store->dir_sep) {
+	} else {
+		char *pattern;
+		int i;
+		char *name;
+		name = camel_imap_store_summary_full_from_path(imap_store->summary, top);
+		if (name == NULL)
+			name = camel_imap_store_summary_path_to_full(imap_store->summary, top, imap_store->dir_sep);
+		i = strlen(name);
+		pattern = g_alloca(i+5);
+		strcpy(pattern, name);
+		g_free(name);
+		if (imap_store->dir_sep) {
 			pattern[i] = imap_store->dir_sep;
 			pattern[i+1] = (flags & CAMEL_STORE_FOLDER_INFO_RECURSIVE)?'*':'%';
 			pattern[i+2] = 0;
@@ -3762,12 +3713,12 @@
 
 		if (camel_exception_is_set(ex))
 			goto fail;
+	}
 
-		camel_store_summary_touch((CamelStoreSummary *)imap_store->summary);
-		camel_store_summary_save((CamelStoreSummary *)imap_store->summary, ex);
+	camel_store_summary_touch((CamelStoreSummary *)imap_store->summary);
+	camel_store_summary_save((CamelStoreSummary *)imap_store->summary, ex);
 
-		CAMEL_SERVICE_REC_UNLOCK(store, connect_lock);
-	}
+	CAMEL_SERVICE_REC_UNLOCK(store, connect_lock);
 
 	tree = get_folder_info_offline(store, top, flags, ex);
 
@@ -3810,11 +3761,7 @@
 
 	/* get starting point */
 	if (top[0] == 0) {
-		if (imap_store->namespace && imap_store->namespace[0]) {
-			name = g_strdup(imap_store->summary->namespace->full_name);
-			top = imap_store->summary->namespace->path;
-		} else
-			name = g_strdup("");
+		name = g_strdup("");
 	} else {
 		name = camel_imap_store_summary_full_from_path(imap_store->summary, top);
 		if (name == NULL)
Index: libtinymail-camel/camel-lite/camel/providers/imap/camel-imap-store-summary.h
===================================================================
--- libtinymail-camel/camel-lite/camel/providers/imap/camel-imap-store-summary.h	(revision 3174)
+++ libtinymail-camel/camel-lite/camel/providers/imap/camel-imap-store-summary.h	(working copy)
@@ -63,7 +63,8 @@
 	/* header info */
 	guint32 version;	/* version of base part of file */
 	guint32 capabilities;
-	CamelImapStoreNamespace *namespace; /* eventually to be a list */
+	GList *namespaces; /* eventually to be a list */
+	CamelImapStoreNamespace *namespace;
 };
 
 struct _CamelImapStoreSummaryClass {
@@ -76,6 +77,7 @@
 /* TODO: this api needs some more work, needs to support lists */
 CamelImapStoreNamespace *camel_imap_store_summary_namespace_new(CamelImapStoreSummary *s, const char *full_name, char dir_sep);
 void camel_imap_store_summary_namespace_set(CamelImapStoreSummary *s, CamelImapStoreNamespace *ns);
+void camel_imap_store_summary_namespace_add(CamelImapStoreSummary *s, CamelImapStoreNamespace *ns);
 CamelImapStoreNamespace *camel_imap_store_summary_namespace_find_path(CamelImapStoreSummary *s, const char *path);
 CamelImapStoreNamespace *camel_imap_store_summary_namespace_find_full(CamelImapStoreSummary *s, const char *full_name);
 
Index: libtinymail-camel/camel-lite/camel/providers/imap/camel-imap-store.h
===================================================================
--- libtinymail-camel/camel-lite/camel/providers/imap/camel-imap-store.h	(revision 3174)
+++ libtinymail-camel/camel-lite/camel/providers/imap/camel-imap-store.h	(working copy)
@@ -160,6 +160,7 @@
 	/* NB: namespace should be handled by summary->namespace */
 	char *namespace, dir_sep, *base_url, *storage_path;
 	GHashTable *authtypes;
+	struct _namespaces *namespaces;
 
 	time_t refresh_stamp;
 	gchar *idle_prefix;
Index: libtinymail-camel/camel-lite/camel/providers/imap/camel-imap-store-summary.c
===================================================================
--- libtinymail-camel/camel-lite/camel/providers/imap/camel-imap-store-summary.c	(revision 3174)
+++ libtinymail-camel/camel-lite/camel/providers/imap/camel-imap-store-summary.c	(working copy)
@@ -42,9 +42,9 @@
 #define d(x)
 #define io(x)			/* io debug */
 
-#define CAMEL_IMAP_STORE_SUMMARY_VERSION_0 (0)
+#define CAMEL_IMAP_STORE_SUMMARY_VERSION_0 (1)
 
-#define CAMEL_IMAP_STORE_SUMMARY_VERSION (0)
+#define CAMEL_IMAP_STORE_SUMMARY_VERSION (1)
 
 #define _PRIVATE(o) (((CamelImapStoreSummary *)(o))->priv)
 
@@ -93,14 +93,20 @@
 
 	((CamelStoreSummary *)s)->store_info_size = sizeof(CamelImapStoreInfo);
 	s->version = CAMEL_IMAP_STORE_SUMMARY_VERSION;
+	s->namespaces = NULL;
+	s->namespace = NULL;
 }
 
 static void
 camel_imap_store_summary_finalise (CamelObject *obj)
 {
 	/*struct _CamelImapStoreSummaryPrivate *p;*/
-	/*CamelImapStoreSummary *s = (CamelImapStoreSummary *)obj;*/
+	CamelImapStoreSummary *is = (CamelImapStoreSummary *)obj;
+	CamelStoreSummary *s = (CamelStoreSummary *) s;
 
+	if (is->namespaces)
+		namespace_clear (s);
+
 	/*p = _PRIVATE(obj);
 	  g_free(p);*/
 }
@@ -395,57 +401,67 @@
 	return ns;
 }
 
+void camel_imap_store_summary_namespace_add(CamelImapStoreSummary *s, CamelImapStoreNamespace *ns)
+{
+	s->namespaces = g_list_prepend (s->namespaces, ns);
+}
+
 void camel_imap_store_summary_namespace_set(CamelImapStoreSummary *s, CamelImapStoreNamespace *ns)
 {
-	d(printf("Setting namesapce to '%s' '%c' -> '%s'\n", ns->full_name, ns->sep, ns->path));
-	namespace_clear((CamelStoreSummary *)s);
 	s->namespace = ns;
-	camel_store_summary_touch((CamelStoreSummary *)s);
+	return;
 }
 
 CamelImapStoreNamespace *
 camel_imap_store_summary_namespace_find_path(CamelImapStoreSummary *s, const char *path)
 {
 	int len;
-	CamelImapStoreNamespace *ns;
+	GList *list;
+	CamelImapStoreNamespace *nsf = NULL;
 
-	/* NB: this currently only compares against 1 namespace, in future compare against others */
-	ns = s->namespace;
-	while (ns) {
+	list = s->namespaces;
+	while (list) {
+		CamelImapStoreNamespace *ns = list->data;
 		len = strlen(ns->path);
 		if (len == 0
 		    || (strncmp(ns->path, path, len) == 0
-			&& (path[len] == '/' || path[len] == 0)))
+			&& (path[len] == '/' || path[len] == 0))) {
+			nsf = ns;
 			break;
-		ns = NULL;
+		}
+		list = list->next;
 	}
 
-	/* have a default? */
-	return ns;
+	return nsf;
 }
 
 CamelImapStoreNamespace *
 camel_imap_store_summary_namespace_find_full(CamelImapStoreSummary *s, const char *full)
 {
 	int len;
-	CamelImapStoreNamespace *ns;
+	GList *list;
+	CamelImapStoreNamespace *nsf = NULL;
 
 	/* NB: this currently only compares against 1 namespace, in future compare against others */
-	ns = s->namespace;
-	while (ns) {
+	list = s->namespaces;
+	while (list) {
+		CamelImapStoreNamespace *ns = list->data;
 		len = strlen(ns->full_name);
 		d(printf("find_full: comparing namespace '%s' to name '%s'\n", ns->full_name, full));
 		if (len == 0
 		    || (strncmp(ns->full_name, full, len) == 0
-			&& (full[len] == ns->sep || full[len] == 0)))
+			&& (full[len] == ns->sep || full[len] == 0))) {
+			nsf = ns;
 			break;
-		ns = NULL;
+		}
+		list = list->next;
 	}
 
 	/* have a default? */
-	return ns;
+	return nsf;
 }
 
+
 static void
 namespace_free(CamelStoreSummary *s, CamelImapStoreNamespace *ns)
 {
@@ -458,51 +474,84 @@
 namespace_clear(CamelStoreSummary *s)
 {
 	CamelImapStoreSummary *is = (CamelImapStoreSummary *)s;
+	GList *list = is->namespaces;
 
-	if (is->namespace)
-		namespace_free(s, is->namespace);
-	is->namespace = NULL;
+	while (list) {
+		CamelImapStoreNamespace *ns = list->data;
+		namespace_free(s, ns);
+		list=list->next;
+	}
+	g_list_free (is->namespaces);
+	is->namespaces = NULL;
 }
 
-static CamelImapStoreNamespace *
+static int
 namespace_load(CamelStoreSummary *s, FILE *in)
 {
+	CamelImapStoreSummary *is = (CamelImapStoreSummary *) s;
+
 	CamelImapStoreNamespace *ns;
-	guint32 sep = '/';
+	int count = 0, i = 0;
 
-	ns = g_malloc0(sizeof(*ns));
+	if (is->namespaces)
+		namespace_clear (s);
 
-	if (camel_file_util_decode_string(in, &ns->path) == -1)
+	if (camel_file_util_decode_uint32(in, &count) == -1)
 		goto nserror;
 
-	if (camel_file_util_decode_string(in, &ns->full_name) == -1)
-		goto nserror;
+	for (i=0 ; i< count; i++) {
+		guint32 sep = '/';
+		ns = g_malloc0(sizeof(*ns));
+		if (camel_file_util_decode_string(in, &ns->path) == -1) {
+			namespace_free(s, ns);
+			goto nserror;
+		}
+		if (camel_file_util_decode_string(in, &ns->full_name) == -1) {
+			namespace_free(s, ns);
+			goto nserror;
+		}
+		if (camel_file_util_decode_uint32(in, &sep) == -1) {
+			namespace_free(s, ns);
+			goto nserror;
+		}
+		ns->sep = sep;
+		is->namespaces = g_list_prepend (is->namespaces, ns);
+	}
 
-	if (camel_file_util_decode_uint32(in, &sep) == -1)
-		goto nserror;
+	return 0;
 
-	ns->sep = sep;
-
-	return ns;
-
  nserror:
 
-	namespace_free(s, ns);
-	return NULL;
+	return -1;
 }
 
 static int
-namespace_save(CamelStoreSummary *s, FILE *in, CamelImapStoreNamespace *ns)
+namespace_save(CamelStoreSummary *s, FILE *in)
 {
-	if (camel_file_util_encode_string(in, ns->path) == -1)
-		goto serr;
+	CamelImapStoreSummary *is = (CamelImapStoreSummary *) s;
 
-	if (camel_file_util_encode_string(in, ns->full_name) == -1)
-		goto serr;
+	GList *list;
 
-	if (camel_file_util_encode_uint32(in, (guint32)ns->sep) == -1)
+	if (camel_file_util_encode_uint32(in, (guint32) g_list_length (is->namespaces)) == -1)
 		goto serr;
 
+	list = is->namespaces;
+
+	while (list) {
+		CamelImapStoreNamespace *ns = list->data;
+
+		if (camel_file_util_encode_string(in, ns->path) == -1)
+			goto serr;
+
+		if (camel_file_util_encode_string(in, ns->full_name) == -1)
+			goto serr;
+
+		if (camel_file_util_encode_uint32(in, (guint32)ns->sep) == -1)
+			goto serr;
+
+		list = list->next;
+	}
+
 	return 0;
 
  serr:
@@ -531,18 +580,12 @@
 	}
 
 	/* note file format can be expanded to contain more namespaces, but only 1 at the moment */
-	if (camel_file_util_decode_fixed_int32(in, &capabilities) == -1
-	    || camel_file_util_decode_fixed_int32(in, &count) == -1
-	    || count > 1)
+	if (camel_file_util_decode_fixed_int32(in, &capabilities) == -1)
 		return -1;
 
 	is->capabilities = capabilities;
-	if (count == 1) {
-		if ((is->namespace = namespace_load(s, in)) == NULL)
-			return -1;
-	}
 
-	return 0;
+	return namespace_load (s, in);
 }
 
 static int
@@ -551,16 +594,13 @@
 	CamelImapStoreSummary *is = (CamelImapStoreSummary *)s;
 	guint32 count;
 
-	count = is->namespace?1:0;
-
 	/* always write as latest version */
 	if (camel_imap_store_summary_parent->summary_header_save((CamelStoreSummary *)s, out) == -1
 	    || camel_file_util_encode_fixed_int32(out, CAMEL_IMAP_STORE_SUMMARY_VERSION) == -1
-	    || camel_file_util_encode_fixed_int32(out, is->capabilities) == -1
-	    || camel_file_util_encode_fixed_int32(out, count) == -1)
+	    || camel_file_util_encode_fixed_int32(out, is->capabilities) == -1)
 		return -1;
 
-	if (is->namespace && namespace_save(s, out, is->namespace) == -1)
+	if (namespace_save(s, out) == -1)
 		return -1;
 
 	return 0;


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