[evolution] Bug 776162 - "Ignore Thread" does not always ignore thread
- From: Milan Crha <mcrha src gnome org>
- To: commits-list gnome org
- Cc:
- Subject: [evolution] Bug 776162 - "Ignore Thread" does not always ignore thread
- Date: Thu, 14 Jun 2018 09:02:18 +0000 (UTC)
commit 5f6de5af8bd148b9e9bc277246198b32983b0bb0
Author: Milan Crha <mcrha redhat com>
Date: Thu Jun 14 11:03:14 2018 +0200
Bug 776162 - "Ignore Thread" does not always ignore thread
src/libemail-engine/mail-folder-cache.c | 67 ++++++++++++++++++++++++++++++---
1 file changed, 62 insertions(+), 5 deletions(-)
---
diff --git a/src/libemail-engine/mail-folder-cache.c b/src/libemail-engine/mail-folder-cache.c
index 62c01a102a..a9bb31a40f 100644
--- a/src/libemail-engine/mail-folder-cache.c
+++ b/src/libemail-engine/mail-folder-cache.c
@@ -927,9 +927,14 @@ update_1folder (MailFolderCache *cache,
}
}
+#define IGNORE_THREAD_VALUE_TODO GINT_TO_POINTER (1)
+#define IGNORE_THREAD_VALUE_IN_PROGRESS GINT_TO_POINTER (2)
+#define IGNORE_THREAD_VALUE_DONE GINT_TO_POINTER (3)
+
static gboolean
folder_cache_check_ignore_thread (CamelFolder *folder,
CamelMessageInfo *info,
+ GHashTable *added_uids, /* gchar *uid ~> IGNORE_THREAD_VALUE_... */
GCancellable *cancellable,
GError **error)
{
@@ -941,6 +946,11 @@ folder_cache_check_ignore_thread (CamelFolder *folder,
g_return_val_if_fail (CAMEL_IS_FOLDER (folder), FALSE);
g_return_val_if_fail (info != NULL, FALSE);
+ g_return_val_if_fail (added_uids != NULL, FALSE);
+ g_return_val_if_fail (camel_message_info_get_uid (info) != NULL, FALSE);
+
+ if (g_hash_table_lookup (added_uids, camel_message_info_get_uid (info)) == IGNORE_THREAD_VALUE_DONE)
+ return camel_message_info_get_user_flag (info, "ignore-thread");
references = camel_message_info_dup_references (info);
if (!references || references->len <= 0) {
@@ -976,18 +986,45 @@ folder_cache_check_ignore_thread (CamelFolder *folder,
for (ii = 0; ii < uids->len; ii++) {
const gchar *refruid = uids->pdata[ii];
CamelMessageInfo *refrinfo;
+ gpointer cached_value;
refrinfo = camel_folder_get_message_info (folder, refruid);
if (!refrinfo)
continue;
+ /* This is for cases when a subthread is received and the order of UIDs
+ doesn't match the order in the thread (parent before child). */
+ cached_value = g_hash_table_lookup (added_uids, refruid);
+ if (cached_value == IGNORE_THREAD_VALUE_TODO) {
+ GError *local_error = NULL;
+
+ /* To avoid infinite recursion */
+ g_hash_table_insert (added_uids, (gpointer) camel_pstring_strdup
(refruid), IGNORE_THREAD_VALUE_IN_PROGRESS);
+
+ if (folder_cache_check_ignore_thread (folder, refrinfo, added_uids,
cancellable, &local_error))
+ camel_message_info_set_user_flag (refrinfo, "ignore-thread",
TRUE);
+
+ if (local_error) {
+ g_clear_error (&local_error);
+ } else {
+ cached_value = IGNORE_THREAD_VALUE_DONE;
+ g_hash_table_insert (added_uids, (gpointer)
camel_pstring_strdup (refruid), IGNORE_THREAD_VALUE_DONE);
+ }
+ }
+
+ if (!cached_value)
+ cached_value = IGNORE_THREAD_VALUE_DONE;
+
if (first_msgid && camel_message_info_get_message_id (refrinfo) ==
first_msgid) {
- /* The first msgid in the references is In-ReplyTo, which is the
master;
+ /* The first msgid in the references is In-Reply-To, which is the
master;
the rest is just a guess. */
- found_first_msgid = TRUE;
first_ignore_thread = camel_message_info_get_user_flag (refrinfo,
"ignore-thread");
- g_clear_object (&refrinfo);
- break;
+ found_first_msgid = first_ignore_thread || cached_value ==
IGNORE_THREAD_VALUE_DONE;
+
+ if (found_first_msgid) {
+ g_clear_object (&refrinfo);
+ break;
+ }
}
has_ignore_thread = has_ignore_thread || camel_message_info_get_user_flag
(refrinfo, "ignore-thread");
@@ -1061,6 +1098,20 @@ folder_cache_process_folder_changes_thread (CamelFolder *folder,
&& folder != local_outbox
&& folder != local_sent
&& changes && (changes->uid_added->len > 0)) {
+ GHashTable *added_uids; /* gchar *uid ~> IGNORE_THREAD_VALUE_... */
+
+ /* The messages can be received in a wrong order (by UID), the same as the In-Reply-To
+ message can be a new message here, in which case it might not be already updated,
+ thus remember which messages are added and eventually update them when needed. */
+ added_uids = g_hash_table_new_full (g_str_hash, g_str_equal, (GDestroyNotify)
camel_pstring_free, NULL);
+
+ for (i = 0; i < changes->uid_added->len; i++) {
+ const gchar *uid = changes->uid_added->pdata[i];
+
+ if (uid)
+ g_hash_table_insert (added_uids, (gpointer) camel_pstring_strdup (uid),
IGNORE_THREAD_VALUE_TODO);
+ }
+
/* for each added message, check to see that it is
* brand new, not junk and not already deleted */
for (i = 0; i < changes->uid_added->len && !g_cancellable_is_cancelled (cancellable); i++) {
@@ -1072,7 +1123,7 @@ folder_cache_process_folder_changes_thread (CamelFolder *folder,
flags = camel_message_info_get_flags (info);
if (((flags & CAMEL_MESSAGE_SEEN) == 0) &&
((flags & CAMEL_MESSAGE_DELETED) == 0) &&
- folder_cache_check_ignore_thread (folder, info, cancellable,
&local_error)) {
+ folder_cache_check_ignore_thread (folder, info, added_uids, cancellable,
&local_error)) {
camel_message_info_set_flags (info, CAMEL_MESSAGE_SEEN,
CAMEL_MESSAGE_SEEN);
camel_message_info_set_user_flag (info, "ignore-thread", TRUE);
flags = flags | CAMEL_MESSAGE_SEEN;
@@ -1108,6 +1159,8 @@ folder_cache_process_folder_changes_thread (CamelFolder *folder,
}
}
}
+
+ g_hash_table_destroy (added_uids);
}
if (new > 0) {
@@ -1134,6 +1187,10 @@ folder_cache_process_folder_changes_thread (CamelFolder *folder,
g_object_unref (session);
}
+#undef IGNORE_THREAD_VALUE_TODO
+#undef IGNORE_THREAD_VALUE_IN_PROGRESS
+#undef IGNORE_THREAD_VALUE_DONE
+
static void
folder_changed_cb (CamelFolder *folder,
CamelFolderChangeInfo *changes,
[
Date Prev][
Date Next] [
Thread Prev][
Thread Next]
[
Thread Index]
[
Date Index]
[
Author Index]