Re: Some IDLE improvements



On Tue, 2007-10-23 at 15:13 +0200, Philip Van Hoof wrote:

And another version :)

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



Index: libtinymail-camel/camel-lite/camel/providers/imap/camel-imap-store.c
===================================================================
--- libtinymail-camel/camel-lite/camel/providers/imap/camel-imap-store.c	(revision 2872)
+++ libtinymail-camel/camel-lite/camel/providers/imap/camel-imap-store.c	(working copy)
@@ -195,49 +195,48 @@
 }
 
 static void 
-let_idle_die (CamelImapStore *imap_store, gboolean connect_buz)
+let_idle_die (CamelImapStore *store, gboolean connect_buz)
 {
-	imap_store->idle_cont = FALSE;
 
-	g_static_rec_mutex_lock (imap_store->idle_prefix_lock);
-	g_static_rec_mutex_lock (imap_store->idle_lock);
+	idle_debug ("let_idle_die starts\n");
 
-	imap_store->idle_cont = FALSE;
+	g_static_rec_mutex_lock (store->idle_prefix_lock);
+	g_static_rec_mutex_lock (store->idle_lock);
 
-	/* This one can get called from within the thread! This would deadlock
+	store->idle_kill = TRUE;
+	store->idle_cont = FALSE;
 
-	if (imap_store->in_idle && imap_store->idle_thread) {
-		g_thread_join (imap_store->idle_thread);
-		imap_store->idle_thread = NULL;
-	} */
-
-	if (imap_store->idle_prefix)
+	if (store->idle_prefix)
 	{
 		gchar *resp = NULL;
 		int nwritten = 0;
 		CamelException ex = CAMEL_EXCEPTION_INITIALISER;
 
-		g_free (imap_store->idle_prefix); 
-		imap_store->idle_prefix=NULL;
-
 		idle_debug ("Sending DONE in let_idle_die\n");
-		nwritten = camel_stream_printf (imap_store->ostream, "DONE\r\n");
-		if (nwritten != -1) 
-		{
+		CAMEL_SERVICE_REC_LOCK (store, connect_lock);
+		nwritten = camel_stream_printf (store->ostream, "DONE\r\n");
+		store->in_idle = FALSE;
+		if (nwritten != -1) {
 			resp = NULL;
-			while ((camel_imap_command_response_idle (imap_store, &resp, &ex)) == CAMEL_IMAP_RESPONSE_UNTAGGED) 
-			{
-				idle_debug ("(.., ..) <- %s | in idle_deal_with_stuff in let_idle_die\n", resp); 
+			while ((camel_imap_command_response_idle (store, &resp, &ex)) == CAMEL_IMAP_RESPONSE_UNTAGGED) {
+				idle_debug ("(.., ..) <- %s | in let_idle_die\n", resp); 
 				g_free (resp); resp=NULL; 
 			}
-			if (resp)
+			if (resp) {
+				idle_debug ("(.., ..) <- %s\n", resp);
 				g_free (resp);
+			}
 		}
+		CAMEL_SERVICE_REC_UNLOCK (store, connect_lock);
+		g_free (store->idle_prefix); 
+		store->idle_prefix=NULL;
 	}
 
-	g_static_rec_mutex_unlock (imap_store->idle_lock);
-	g_static_rec_mutex_unlock (imap_store->idle_prefix_lock);
+	g_static_rec_mutex_unlock (store->idle_lock);
+	g_static_rec_mutex_unlock (store->idle_prefix_lock);
 
+	idle_debug ("let_idle_die finished\n");
+
 	return;
 }
 
@@ -247,41 +246,33 @@
 	if (store->current_folder && CAMEL_IS_IMAP_FOLDER (store->current_folder))
 		camel_imap_folder_stop_idle (store->current_folder);
 	else {
-		store->idle_cont = FALSE;
-
 		g_static_rec_mutex_lock (store->idle_prefix_lock);
 		g_static_rec_mutex_lock (store->idle_lock);
 
+		store->idle_kill = TRUE;
 		store->idle_cont = FALSE;
-		if (store->in_idle && store->idle_thread) {
-			g_thread_join (store->idle_thread);
-			store->idle_thread = NULL;
-		}
 
-		if (store->idle_prefix) 
-		{
+		if (store->idle_prefix) {
 			gchar *resp = NULL;
 			int nwritten = 0;
 			CamelException ex = CAMEL_EXCEPTION_INITIALISER;
 
-			idle_debug ("Sending DONE in camel_imap_store_stop_idle (no current folder?)\n");
+			idle_debug ("Sending DONE in camel_imap_store_stop_idle\n");
 			CAMEL_SERVICE_REC_LOCK (store, connect_lock);
 			nwritten = camel_stream_printf (store->ostream, "DONE\r\n");
-
-			if (nwritten != -1) 
-			{
+			store->in_idle = FALSE;
+			if (nwritten != -1) {
 				resp = NULL;
-				while ((camel_imap_command_response_idle (store, &resp, &ex)) == CAMEL_IMAP_RESPONSE_UNTAGGED) 
-				{
+				while ((camel_imap_command_response_idle (store, &resp, &ex)) == CAMEL_IMAP_RESPONSE_UNTAGGED) {
 					idle_debug ("(.., ..) <- %s | in idle_deal_with_stuff (no current folder?)\n", resp); 
 					g_free (resp); resp=NULL; 
 				}
-				if (resp)
+				if (resp) {
+					idle_debug ("(.., ..) <- %s\n", resp);
 					g_free (resp);
+				}
 			}
-
 			CAMEL_SERVICE_REC_UNLOCK (store, connect_lock);
-
 			g_free (store->idle_prefix);
 			store->idle_prefix = NULL;
 		}
@@ -312,6 +303,7 @@
 			camel_imap_folder_selected (folder, response2, &nex, TRUE);
 			camel_imap_response_free (imap_store, response2);
 		}
+		camel_imap_store_start_idle (imap_store);
 	}
 
 	return;
@@ -441,7 +433,7 @@
 
 	imap_store->dontdistridlehack = FALSE;
 
-	imap_store->idle_sleep = 20; /* default of 20s */
+	imap_store->idle_sleep = 600; /* default of 10m */
 	imap_store->getsrv_sleep = 100; /* default of 100s */
 
 	imap_store->in_idle = FALSE;
@@ -2588,8 +2580,8 @@
 	}
 	imap_status_item_free (items);
 
-	camel_imap_store_start_idle (imap_store);
 	CAMEL_SERVICE_REC_UNLOCK (imap_store, connect_lock);
+	camel_imap_store_start_idle (imap_store);
 
 	return;
 }
@@ -2630,16 +2622,18 @@
 		const char *c;
 		
 		if (camel_exception_get_id(ex) == CAMEL_EXCEPTION_USER_CANCEL) {
-			camel_imap_store_start_idle (imap_store);
 			CAMEL_SERVICE_REC_UNLOCK (imap_store, connect_lock);
+			camel_imap_store_start_idle (imap_store);
+
 			return NULL;
 		}
 		
 		camel_exception_clear (ex);
 		
 		if (!(flags & CAMEL_STORE_FOLDER_CREATE)) {
-			camel_imap_store_start_idle (imap_store);
 			CAMEL_SERVICE_REC_UNLOCK (imap_store, connect_lock);
+			camel_imap_store_start_idle (imap_store);
+
 			camel_exception_setv (ex, CAMEL_EXCEPTION_STORE_NO_FOLDER,
 					      _("No such folder %s"), folder_name);
 			return NULL;
@@ -2651,11 +2645,11 @@
 			c++;
 		
 		if (*c != '\0') {
-			camel_imap_store_start_idle (imap_store);
 			CAMEL_SERVICE_REC_UNLOCK (imap_store, connect_lock);
 			camel_exception_setv (ex, CAMEL_EXCEPTION_FOLDER_INVALID_PATH,
 					      _("The folder name \"%s\" is invalid because it contains the character \"%c\""),
 					      folder_name, *c);
+			camel_imap_store_start_idle (imap_store);
 			return NULL;
 		}
 
@@ -2673,10 +2667,12 @@
 			int i;
 			
 			if (!(response = camel_imap_command (imap_store, NULL, ex, "LIST \"\" %G", parent_real))) {
-				camel_imap_store_start_idle (imap_store);
-				CAMEL_SERVICE_REC_UNLOCK (imap_store, connect_lock);
 				g_free (parent_name);
 				g_free (parent_real);
+
+				CAMEL_SERVICE_REC_UNLOCK (imap_store, connect_lock);
+				camel_imap_store_start_idle (imap_store);
+
 				return NULL;
 			}
 			
@@ -2717,12 +2713,13 @@
 				imap_status_item_free (items);
 				
 				if (messages > 0) {
-					camel_imap_store_start_idle (imap_store);
 					camel_exception_set (ex, CAMEL_EXCEPTION_FOLDER_INVALID_STATE,
 							     _("The parent folder is not allowed to contain subfolders"));
-					CAMEL_SERVICE_REC_UNLOCK (imap_store, connect_lock);
 					g_free (parent_name);
 					g_free (parent_real);
+					CAMEL_SERVICE_REC_UNLOCK (imap_store, connect_lock);
+					camel_imap_store_start_idle (imap_store);
+
 					return NULL;
 				}
 				
@@ -2730,11 +2727,11 @@
 				camel_exception_init (&lex);
 				delete_folder (store, parent_name, &lex);
 				if (camel_exception_is_set (&lex)) {
-					camel_imap_store_start_idle (imap_store);
-					CAMEL_SERVICE_REC_UNLOCK (imap_store, connect_lock);
-					camel_exception_xfer (ex, &lex);
 					g_free (parent_name);
 					g_free (parent_real);
+					CAMEL_SERVICE_REC_UNLOCK (imap_store, connect_lock);
+					camel_imap_store_start_idle (imap_store);
+					camel_exception_xfer (ex, &lex);
 					return NULL;
 				}
 				
@@ -2745,10 +2742,11 @@
 				g_free (name);
 				
 				if (!response) {
-					camel_imap_store_start_idle (imap_store);
-					CAMEL_SERVICE_REC_UNLOCK (imap_store, connect_lock);
 					g_free (parent_name);
 					g_free (parent_real);
+					CAMEL_SERVICE_REC_UNLOCK (imap_store, connect_lock);
+					camel_imap_store_start_idle (imap_store);
+
 					return NULL;
 				} else
 					camel_imap_response_free (imap_store, response);
@@ -2770,8 +2768,9 @@
 		}
 		g_free(folder_real);
 		if (!response) {
-			camel_imap_store_start_idle (imap_store);
 			CAMEL_SERVICE_REC_UNLOCK (imap_store, connect_lock);
+			camel_imap_store_start_idle (imap_store);
+
 			return NULL;
 		}
 	} else if (flags & CAMEL_STORE_FOLDER_EXCL) {
@@ -2780,9 +2779,9 @@
 				      folder_name);
 		
 		camel_imap_response_free_without_processing (imap_store, response);
-		camel_imap_store_start_idle (imap_store);
 		CAMEL_SERVICE_REC_UNLOCK (imap_store, connect_lock);
-		
+		camel_imap_store_start_idle (imap_store);
+
 		return NULL;
 	}
 
@@ -2809,9 +2808,9 @@
 	}
 	camel_imap_response_free_without_processing (imap_store, response);
 
-	camel_imap_store_start_idle (imap_store);
 	CAMEL_SERVICE_REC_UNLOCK (imap_store, connect_lock);
-	
+	camel_imap_store_start_idle (imap_store);
+
 	return new_folder;
 }
 
@@ -2850,7 +2849,6 @@
 {
 	CamelImapStore *imap_store = CAMEL_IMAP_STORE (store);
 	CamelImapResponse *response;
-	gboolean did_start_idle = FALSE;
 	CamelFolder *old_in_case;
 
 	CAMEL_SERVICE_REC_LOCK (imap_store, connect_lock);
@@ -2881,7 +2879,6 @@
 		if (response2) {
 			camel_imap_folder_selected (old_in_case, response2, &nex, TRUE);
 			camel_imap_response_free (imap_store, response2);
-			did_start_idle = TRUE;
 		}
 	}
 
@@ -2890,9 +2887,10 @@
 		imap_forget_folder (imap_store, folder_name, ex);
 	}
 fail:
-	if (!did_start_idle)
-		camel_imap_store_start_idle (imap_store);
 	CAMEL_SERVICE_REC_UNLOCK(imap_store, connect_lock);
+
+	camel_imap_store_start_idle (imap_store);
+
 }
 
 static void
@@ -3033,7 +3031,6 @@
 	char *oldpath, *newpath, *storage_path;
 	char *tpath, *lslash;
 	CamelFolder *old_in_case = NULL;
-	gboolean did_start_idle = FALSE;
 
 	CAMEL_SERVICE_REC_LOCK (imap_store, connect_lock);
 
@@ -3078,7 +3075,6 @@
 		if (response2) {
 			camel_imap_folder_selected (old_in_case, response2, &nex, TRUE);
 			camel_imap_response_free (imap_store, response2);
-			did_start_idle = TRUE;
 		}
 	}
 
@@ -3126,10 +3122,11 @@
 	g_free (newpath);
 fail:
 	imap_store->renaming = FALSE;
-	if (!did_start_idle)
-		camel_imap_store_start_idle (imap_store);
 	CAMEL_SERVICE_REC_UNLOCK(imap_store, connect_lock);
 	camel_operation_end (NULL);
+
+	camel_imap_store_start_idle (imap_store);
+
 }
 
 static CamelFolderInfo *
@@ -3955,8 +3952,10 @@
 
 	imap_folder_effectively_unsubscribed (imap_store, folder_name, ex);
 done:
-	camel_imap_store_start_idle (imap_store);
 	CAMEL_SERVICE_REC_UNLOCK(store, connect_lock);
+
+	camel_imap_store_start_idle (imap_store);
+
 }
 
 #if 0
Index: libtinymail-camel/camel-lite/camel/providers/imap/camel-imap-store.h
===================================================================
--- libtinymail-camel/camel-lite/camel/providers/imap/camel-imap-store.h	(revision 2872)
+++ libtinymail-camel/camel-lite/camel/providers/imap/camel-imap-store.h	(working copy)
@@ -165,7 +165,7 @@
 
 	GStaticRecMutex *idle_prefix_lock, *idle_lock, *sum_lock;
 	GThread *idle_thread;
-	gboolean idle_cont, in_idle;
+	gboolean idle_cont, in_idle, idle_kill;
 	guint idle_sleep, getsrv_sleep;
 	gboolean courier_crap;
 	gboolean going_online, got_online, clean_exit;
Index: libtinymail-camel/camel-lite/camel/providers/imap/camel-imap-folder.c
===================================================================
--- libtinymail-camel/camel-lite/camel/providers/imap/camel-imap-folder.c	(revision 2872)
+++ libtinymail-camel/camel-lite/camel/providers/imap/camel-imap-folder.c	(working copy)
@@ -38,6 +38,8 @@
 
 #include <config.h> 
 
+#include <sched.h>
+
 #include <ctype.h>
 #include <errno.h>
 #include <fcntl.h>
@@ -760,8 +762,6 @@
 		camel_folder_change_info_free(changes);
 	}
 
-	if (idle)
-		camel_imap_folder_start_idle (folder);
 }
 
 
@@ -907,18 +907,17 @@
 		 * messages. */
 		imap_rescan (folder, camel_folder_summary_count (folder->summary), ex);
 	} else {
+
 #if 0
 		/* on some servers need to CHECKpoint INBOX to recieve new messages?? */
 		/* rfc2060 suggests this, but havent seen a server that requires it */
 		if (g_ascii_strcasecmp(folder->full_name, "INBOX") == 0) {
 			response = camel_imap_command (imap_store, folder, ex, "CHECK");
 			camel_imap_response_free (imap_store, response);
-			camel_imap_folder_start_idle (folder);
 		}
 #endif
 		response = camel_imap_command (imap_store, folder, ex, "NOOP");
 		camel_imap_response_free (imap_store, response);
-		camel_imap_folder_start_idle (folder);
 	}
 
 	si = camel_store_summary_path((CamelStoreSummary *)((CamelImapStore *)folder->parent_store)->summary, folder->full_name);
@@ -939,8 +938,10 @@
 	CAMEL_FOLDER_REC_UNLOCK(folder, lock);
 
 	camel_folder_summary_save(folder->summary, ex);
+	camel_store_summary_save((CamelStoreSummary *)((CamelImapStore *)folder->parent_store)->summary, ex);
 
-	camel_store_summary_save((CamelStoreSummary *)((CamelImapStore *)folder->parent_store)->summary, ex);
+	camel_imap_folder_start_idle (folder);
+
 }
 
 #if 0
@@ -1522,19 +1523,17 @@
 		}
 		g_ptr_array_free (matches, TRUE);
 
-		if (camel_exception_is_set (&local_ex))
-			camel_imap_folder_start_idle (folder);
-
 		/* We unlock here so that other threads can have a chance to grab the connect_lock */
 		CAMEL_SERVICE_REC_UNLOCK (store, connect_lock);
-		
+
 		/* check for an exception */
 		if (camel_exception_is_set (&local_ex)) {
-			/* Look up */
+			
 			camel_exception_xfer (ex, &local_ex);
+			camel_imap_folder_start_idle (folder);
 			return;
 		}
-		
+
 		/* Re-lock the connect_lock */
 		CAMEL_SERVICE_REC_LOCK (store, connect_lock);
 	}
@@ -1542,9 +1541,9 @@
 	/* Save the summary */
 	imap_sync_offline (folder, ex);
 
+	CAMEL_SERVICE_REC_UNLOCK (store, connect_lock);
 	camel_imap_folder_start_idle (folder);
-	CAMEL_SERVICE_REC_UNLOCK (store, connect_lock);
-	/* camel_imap_folder_start_idle (folder); */
+
 }
 
 static int
@@ -1643,7 +1642,6 @@
 	
 	CAMEL_SERVICE_REC_UNLOCK (store, connect_lock);
 
-
 	camel_imap_folder_start_idle (folder);
 
 }
@@ -3591,11 +3589,8 @@
 idle_response_new (CamelFolder *folder)
 {
 	IdleResponse *idle_resp = g_slice_new0 (IdleResponse);
-
+	idle_debug ("idle_response_new\n");
 	idle_resp->vanished = NULL;
-
-	idle_debug ("idle_response_new\n");
-
 	idle_resp->folder = folder;
 	camel_object_ref (CAMEL_OBJECT (folder));
 	return idle_resp;
@@ -3604,27 +3599,23 @@
 static void
 idle_response_free (IdleResponse *idle_resp)
 {
-	guint i=0;
-
 	idle_debug ("idle_response_free\n");
-
 	if (idle_resp->expunged)
 		g_array_free (idle_resp->expunged, TRUE);
-
 	if (idle_resp->vanished) {
-		g_list_foreach (idle_resp->vanished, (GFunc)g_free, NULL);
+		g_list_foreach (idle_resp->vanished, (GFunc) g_free, NULL);
 		g_list_free (idle_resp->vanished);
 		idle_resp->vanished = NULL;
 	}
-
-	if (idle_resp->fetch) 
+	if (idle_resp->fetch) {
+		guint i=0;
 		for (i=0 ;i < idle_resp->fetch->len; i++)
 			g_slice_free (FetchIdleResponse, idle_resp->fetch->pdata[i]);
-
+	}
 	if (idle_resp->folder)
 		camel_object_unref (CAMEL_OBJECT (idle_resp->folder));
-
 	g_slice_free (IdleResponse, idle_resp);
+	return;
 }
 
 
@@ -3633,18 +3624,18 @@
 {
 	char *resp;
 	CamelException ex = CAMEL_EXCEPTION_INITIALISER;
-
+	gboolean l = CAMEL_SERVICE_REC_TRYLOCK (store, connect_lock);
 	if (store->ostream && store->istream && CAMEL_IS_STREAM (store->ostream))
 	{
+		int nwritten = 0;
 		if (store->idle_prefix)
 			g_free (store->idle_prefix);
 		store->idle_prefix = g_strdup_printf ("%c%.5u", 
 			store->tag_prefix, store->command++);
-
-		idle_debug ("(.., ..) -> %s IDLE | in idle_real_start\n", store->idle_prefix);
-
-		camel_stream_printf (store->ostream, "%s IDLE\r\n",
+		nwritten = camel_stream_printf (store->ostream, "%s IDLE\r\n",
 			store->idle_prefix);
+		idle_debug ("(%d, %d) -> %s IDLE\n", strlen (store->idle_prefix)+5, 
+			nwritten, store->idle_prefix);
 	} else {
 		idle_debug ("idle_real_start connection lost\n");
 		goto errh;
@@ -3658,7 +3649,6 @@
 	 * active, the server is now free to send untagged EXISTS, EXPUNGE, and
 	 *  other messages at any time. */
 
-
 	/* So according to the RFC, we will wait for the server for its + 
 	 * continuation. If the server doesn't do this, it's an incorrect 
 	 * IDLE implementation at the server. Right? */
@@ -3680,248 +3670,236 @@
 	}
 	if (resp)
 		g_free (resp);
-
 errh:
+	if (l)
+		CAMEL_SERVICE_REC_UNLOCK (store, connect_lock);
+	return;
+}
 
+
+static void
+consume_idle_line (CamelImapStore *store, CamelFolder *folder, char *resp, IdleResponse *idle_resp)
+{
+	if (strchr (resp, '*') != NULL && (camel_strstrcase (resp, "EXISTS") || 
+		camel_strstrcase (resp, "FETCH")|| camel_strstrcase (resp, "EXPUNGE") || 
+		camel_strstrcase (resp, "VANISHED") || camel_strstrcase (resp, "RECENT")))
+	{
+		if (!idle_resp) 
+			idle_resp = idle_response_new (folder);
+		read_idle_response (folder, resp, idle_resp);
+	}
+	idle_debug ("(%d, ..) <- %s\n", 
+		strlen (resp), resp);
 	return;
 }
 
-static IdleResponse*
-idle_deal_with_stuff (CamelFolder *folder, CamelImapStore *store, gboolean *had_err, gboolean *had_lock)
+typedef struct {
+	CamelFolder *folder;
+	GCond *condition;
+	GMutex *mutex;
+	gboolean had_cond;
+} IdleThreadInfo;
+
+static gpointer 
+idle_thread (gpointer data)
 {
-  IdleResponse *idle_resp = NULL;
-  CamelException ex = CAMEL_EXCEPTION_INITIALISER;
-  char *resp = NULL;
-  int nwritten=0;
-  CamelImapResponseType type;
-  gboolean hlock = FALSE;
+	IdleThreadInfo *info = (IdleThreadInfo *) data;
+	CamelFolder *folder = (CamelFolder *) info->folder;
+	CamelImapFolder *imap_folder;
+	CamelImapStore *store;
+	gboolean tfirst = TRUE, first = TRUE, my_cont;
+	int cnt = 0;
+	int nwritten=0;
 
-  idle_debug ("idle_deal_with_stuff\n");
+	idle_debug ("idle_thread starts\n");
 
-  if (!camel_disco_store_check_online ((CamelDiscoStore*)store, &ex))
-	return NULL;
+	if (!folder || !folder->parent_store) { 
+		g_thread_exit (NULL); 
+		return NULL; 
+	}
 
-  hlock = CAMEL_SERVICE_REC_TRYLOCK (store, connect_lock);
+	if (!CAMEL_IS_FOLDER (folder) || !CAMEL_IS_STORE (folder->parent_store)) { 
+		g_thread_exit (NULL); 
+		return NULL; 
+	}
 
-  if (hlock)
-  {
-	if (store->current_folder)
-	{
-		/* We read-away everything non-blocking and process it */
-		resp = NULL;
-		while (camel_imap_store_readline_nb (store, &resp, &ex) > 0)
-		{
-			if (strchr (resp, '*') != NULL && (camel_strstrcase (resp, "EXISTS") || 
-				camel_strstrcase (resp, "FETCH")|| camel_strstrcase (resp, "EXPUNGE") || 
-				camel_strstrcase (resp, "VANISHED") || camel_strstrcase (resp, "RECENT")))
-			{
-				if (!idle_resp) 
-					idle_resp = idle_response_new (folder);
-				read_idle_response (folder, resp, idle_resp);
-			}
-			idle_debug ("(%d, ..) <- %s | in idle_deal_with_stuff at nb\n", 
-				strlen (resp), resp);
-			g_free (resp); resp=NULL;
-		}
-		if (resp)
-			g_free (resp);
+	store = CAMEL_IMAP_STORE (folder->parent_store);
+	imap_folder = CAMEL_IMAP_FOLDER (folder);
 
+	if (!imap_folder->do_push_email) {
+		idle_debug ("Folder set not to do idle\n");
+		g_thread_exit (NULL);
+		return NULL;
+	}
 
-		/* Here we force the server to tell us about the changes: */
+	if (!(store->capabilities & IMAP_CAPABILITY_IDLE)) {
+		idle_debug ("Server has no IDLE capabilities\n");
+		g_thread_exit (NULL);
+		return NULL;
+	}
 
-		/* The IDLE command is terminated by the receipt of a "DONE"
-		 * continuation from the client; such response satisfies the server's
-		 * continuation request.  At that point, the server MAY send any
-		 * remaining queued untagged responses and then MUST immediately send
-		 * the tagged response to the IDLE command and prepare to process other
-		 * commands. */
 
-		if (store->ostream && CAMEL_IS_STREAM (store->ostream)) {
-			nwritten = camel_stream_printf (store->ostream, "DONE\r\n");
-			idle_debug ("(%d, 8) -> DONE | Sending DONE in idle_deal_with_stuff (nb)\n",
-				nwritten);
+	camel_object_ref (folder);
+
+	my_cont = store->idle_cont;
+
+	if (my_cont) {
+		idle_debug ("idle_thread starting\n");
+	} else {
+		idle_debug ("idle_thread not starting\n");
+	}
+
+	while (my_cont && !store->idle_kill) 
+	{
+		CamelException ex = CAMEL_EXCEPTION_INITIALISER;
+		char *resp = NULL;
+		IdleResponse *idle_resp = NULL;
+
+		if (store->idle_cont && first) {
+			gboolean l = g_static_rec_mutex_trylock (store->idle_lock);
+			if (!store->idle_kill)
+				idle_real_start (store);
+			if (l)
+				g_static_rec_mutex_unlock (store->idle_lock);
+			first = FALSE;
 		}
 
-		if (nwritten == -1) 
-			goto outofhere;
+		if (tfirst) {
+			if (info->condition) {
+				g_mutex_lock (info->mutex);
+				g_cond_broadcast (info->condition);
+				info->had_cond = TRUE;
+				g_mutex_unlock (info->mutex);
+			}
+			tfirst = FALSE;
+		}
 
-		resp = NULL;
-		while ((type = camel_imap_command_response_idle (store, &resp, &ex)) == CAMEL_IMAP_RESPONSE_UNTAGGED) 
-		{
-			/* printf ("D resp: %s\n", resp); */
-			if (strchr (resp, '*') != NULL && (camel_strstrcase (resp, "EXISTS") ||
-				camel_strstrcase (resp, "FETCH") || camel_strstrcase (resp, "EXPUNGE") || 
-				camel_strstrcase (resp, "RECENT")))
-			{
+		if (g_static_rec_mutex_trylock (store->idle_lock)) {
+			if (!store->idle_kill)
+			  while (camel_imap_store_readline_nb (store, &resp, &ex) > 0)
+			  {
 				if (!idle_resp)
 					idle_resp = idle_response_new (folder);
-				read_idle_response (folder, resp, idle_resp);
-			}
-			idle_debug ("(%d, ..) <- %s | in idle_deal_with_stuff at idle\n", 
-				strlen (resp), resp);
-			g_free (resp); resp=NULL;
+				consume_idle_line (store, folder, resp, idle_resp);
+				g_free (resp);
+				resp = NULL;
+			  }
+			g_static_rec_mutex_unlock (store->idle_lock);
 		}
 
-		if (type == CAMEL_IMAP_RESPONSE_ERROR)
-			*had_err = TRUE;
-
 		if (resp)
 			g_free (resp);
 
-	} else {
-		/* Trying to deal while the current folder is gone: just read away everything */
-		if (store->ostream && CAMEL_IS_STREAM (store->ostream)) {
-			nwritten = camel_stream_printf (store->ostream, "DONE\r\n");
-			idle_debug ("(%d, 8) -> DONE | Sending DONE in idle_deal_with_stuff (b)\n",
-				nwritten);
+		if (idle_resp) {
+			process_idle_response (idle_resp);
+			idle_response_free (idle_resp);
 		}
-		if (nwritten == -1) 
-			goto outofhere;
-		resp = NULL;
-		while ((type = camel_imap_command_response_idle (store, &resp, &ex)) == CAMEL_IMAP_RESPONSE_UNTAGGED) 
+
+		idle_resp = NULL;
+
+		if ((cnt > store->idle_sleep) || (!store->idle_cont)) 
 		{
-			idle_debug ("(.., ..) <- %s | in idle_deal_with_stuff in else\n", resp); 
-			g_free (resp); resp=NULL; 
+			if (store->idle_prefix) 
+			{
+				CamelImapResponseType type;
+				gboolean l = g_static_rec_mutex_trylock (store->idle_lock);
+
+				if (!store->idle_kill) {
+				  nwritten = camel_stream_printf (store->ostream, "DONE\r\n");
+				  idle_debug ("(%d, 8) -> DONE\n", nwritten);
+				  while ((type = camel_imap_command_response_idle (store, &resp, &ex)) == CAMEL_IMAP_RESPONSE_UNTAGGED) 
+				  {
+					if (!idle_resp)
+						idle_resp = idle_response_new (folder);
+					consume_idle_line (store, folder, resp, idle_resp);
+					g_free (resp);
+					resp = NULL;
+				  }
+				}
+
+				if (l)
+					g_static_rec_mutex_unlock (store->idle_lock);
+				if (resp) {
+					idle_debug ("(.., ..) <- %s\n", resp);
+					g_free (resp);
+				}
+
+				if (idle_resp) {
+					process_idle_response (idle_resp);
+					idle_response_free (idle_resp);
+				}
+			}
+			if (store->idle_cont)
+				first = TRUE; 
+			else
+				my_cont = FALSE;
+
+			cnt = 0;
 		}
-		if (resp)
-			g_free (resp);
+
+		if (my_cont)
+			usleep (500000);
+		cnt++;
+
 	}
 
-outofhere:
+	if (!info->had_cond && info->condition) {
+		g_mutex_lock (info->mutex);
+		g_cond_broadcast (info->condition);
+		info->had_cond = TRUE;
+		g_mutex_unlock (info->mutex);
+	}
 
-	CAMEL_SERVICE_REC_UNLOCK (store, connect_lock);
-  } 
+	camel_object_unref (folder);
 
-  *had_lock = hlock;
+	g_thread_exit (NULL);
 
-  return idle_resp;
+	return NULL;
 }
 
+
 void
 camel_imap_folder_stop_idle (CamelFolder *folder)
 {
 	CamelImapStore *store;
-	IdleResponse *idle_resp = NULL;
-	gboolean had_err = FALSE;
 	CamelException ex = CAMEL_EXCEPTION_INITIALISER;
 
 	idle_debug ("camel_imap_folder_stop_idle\n");
 
 	store = CAMEL_IMAP_STORE (folder->parent_store);
 
+	store->idle_cont = FALSE;
+
 	if (!camel_disco_store_check_online ((CamelDiscoStore*)store, &ex))
 		return;
 
-	store->idle_cont = FALSE;
-
-	g_static_rec_mutex_lock (store->idle_lock);
-	g_static_rec_mutex_lock (store->idle_prefix_lock);
-
-	if ((store->capabilities & IMAP_CAPABILITY_IDLE) && store->idle_prefix)
+	if ((store->capabilities & IMAP_CAPABILITY_IDLE))
 	{
-		gboolean had_lock = FALSE;
-
-		store->idle_cont = FALSE;
 		if (store->in_idle && store->idle_thread) {
+			store->idle_cont = FALSE;
 			g_thread_join (store->idle_thread);
+			g_static_rec_mutex_lock (store->idle_prefix_lock);
+			g_static_rec_mutex_lock (store->idle_lock);
+			store->in_idle = FALSE;
 			store->idle_thread = NULL;
+			if (store->idle_prefix)
+				g_free (store->idle_prefix);
+			g_static_rec_mutex_unlock (store->idle_lock);
+			g_static_rec_mutex_unlock (store->idle_prefix_lock);
 		}
-
-		if (store->idle_prefix)
-			g_free (store->idle_prefix);
 		store->idle_prefix = NULL;
-
-		idle_resp = idle_deal_with_stuff (folder, store, &had_err, &had_lock);
-
-		if (idle_resp && !had_err)
-			process_idle_response (idle_resp);
-
-		if (idle_resp) 
-			idle_response_free (idle_resp);
 	}
 
-	g_static_rec_mutex_unlock (store->idle_prefix_lock);
-	g_static_rec_mutex_unlock (store->idle_lock);
-
+	return;
 }
 
 
-static gpointer 
-idle_thread (gpointer data)
-{
-	CamelFolder *folder = (CamelFolder *) data;
-	CamelImapFolder *imap_folder;
-	CamelImapStore *store;
-	gboolean had_err = FALSE, hadlock = FALSE;
-
-	idle_debug ("idle_thread\n");
-
-	if (!folder || !folder->parent_store)
-		{ g_thread_exit (NULL); return NULL; }
-
-	if (!CAMEL_IS_FOLDER (folder) || !CAMEL_IS_STORE (folder->parent_store))
-		{ g_thread_exit (NULL); return NULL; }
-
-	store = CAMEL_IMAP_STORE (folder->parent_store);
-	imap_folder = CAMEL_IMAP_FOLDER (folder);
-
-	if (!imap_folder->do_push_email) {
-		idle_debug ("Folder set not to do idle\n");
-		return NULL;
-	}
-
-	if (!(store->capabilities & IMAP_CAPABILITY_IDLE)) {
-		idle_debug ("Server has no IDLE capabilities\n");
-		return NULL;
-	}
-
-	idle_debug ("idle_thread starting (%s)\n", store->idle_prefix?store->idle_prefix:"(none)");
-
-	store->idle_cont = TRUE;
-
-	while (store->idle_cont && store->idle_prefix != NULL)
-	{
-		int x = 0;
-
-		g_static_rec_mutex_lock (store->idle_lock);
-		g_static_rec_mutex_lock (store->idle_prefix_lock);
-
-		store->in_idle = TRUE;
-		if (store->idle_prefix != NULL)
-		{
-			IdleResponse *idle_resp = idle_deal_with_stuff (folder, store, &had_err,&hadlock);
-
-			if (idle_resp && !had_err)
-				process_idle_response (idle_resp);
-
-			if (hadlock)
-				idle_real_start (store);
-
-			if (idle_resp)
-				idle_response_free (idle_resp);
-		}
-
-		g_static_rec_mutex_unlock (store->idle_prefix_lock);
-		g_static_rec_mutex_unlock (store->idle_lock);
-
-		idle_debug ("idle checked in idle_thread, waiting %ds for new check\n", store->idle_sleep);
-
-		for (x=0; x<1000 && store->idle_cont; x++)
-			usleep (store->idle_sleep * 1000);
-	}
-
-	store->in_idle = FALSE;
-
-	g_thread_exit (NULL);
-	return NULL;
-}
-
-
 void
 camel_imap_folder_start_idle (CamelFolder *folder)
 {
 	CamelImapStore *store;
 	CamelException ex = CAMEL_EXCEPTION_INITIALISER;
+	CamelImapFolder *imap_folder = (CamelImapFolder *) folder;
 
 	idle_debug ("camel_imap_folder_start_idle\n");
 
@@ -3930,6 +3908,9 @@
 	if (!camel_disco_store_check_online ((CamelDiscoStore*)store, &ex))
 		return;
 
+	if (!imap_folder->do_push_email)
+		return;
+
 	g_static_rec_mutex_lock (store->idle_lock);
 
 	if (store->capabilities & IMAP_CAPABILITY_IDLE)
@@ -3939,17 +3920,36 @@
 		{
 			folder->folder_flags |= CAMEL_FOLDER_HAS_PUSHEMAIL_CAPABILITY;
 
-			if (!store->in_idle && store->idle_thread)
-			{
+			if (!store->in_idle && store->idle_thread) {
 				store->idle_cont = FALSE;
 				g_thread_join (store->idle_thread);
 				store->idle_thread = NULL;
 			}
 
 			if (!store->in_idle) {
-				idle_real_start (store);
+				IdleThreadInfo *info = g_slice_new0 (IdleThreadInfo);
+				store->idle_kill = FALSE;
+				store->in_idle = TRUE;
+				store->idle_cont = TRUE;
+
+				info->mutex = g_mutex_new ();
+				info->condition = g_cond_new ();
+				info->had_cond = FALSE;
+				info->folder = folder;
+				camel_object_ref (info->folder);
+
 				store->idle_thread = g_thread_create (idle_thread, 
-					folder, TRUE, NULL);
+					info, TRUE, NULL);
+				g_mutex_lock (info->mutex);
+				if (!info->had_cond)
+					g_cond_wait (info->condition, info->mutex);
+				g_mutex_unlock (info->mutex);
+
+				camel_object_unref (info->folder);
+				g_mutex_free (info->mutex);
+				g_cond_free (info->condition);
+				g_slice_free (IdleThreadInfo, info);
+
 			}
 
 		}
@@ -3977,6 +3977,8 @@
 
 	return;
 }
+
+
 /* Called with the store's connect_lock locked */
 void
 camel_imap_folder_changed (CamelFolder *folder, int exists,
@@ -4018,9 +4020,9 @@
 	len = camel_folder_summary_count (folder->summary);
 
 	if (exists > len) {
+		/* TNY Question: is this stop really necessary? Why? */
 		camel_imap_folder_stop_idle (folder);
 		imap_update_summary (folder, exists, changes, ex);
-		/* camel_imap_folder_start_idle (folder); will happen later? */
 	}
 
 	if (camel_folder_change_info_changed (changes))
Index: ChangeLog
===================================================================
--- ChangeLog	(revision 2874)
+++ ChangeLog	(working copy)
@@ -3,6 +3,7 @@
 	* Reference count problem in TnyCamelHeader
 	* priv->folder_name in TnyCamelFolder sometimes is NULL, which doesn't
 	seem right (and is racy).
+	* Improvements for the IDLE support
 
 2007-10-19  Philip Van Hoof  <pvanhoof gnome org>
 


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