Blame SOURCES/evolution-ews-3.28.5-sync-category-list.patch

f7d506
diff -up evolution-ews-3.28.5/src/camel/camel-ews-store.c.sync-category-list evolution-ews-3.28.5/src/camel/camel-ews-store.c
f7d506
--- evolution-ews-3.28.5/src/camel/camel-ews-store.c.sync-category-list	2019-10-24 09:39:08.336278207 +0200
f7d506
+++ evolution-ews-3.28.5/src/camel/camel-ews-store.c	2019-10-24 09:39:08.341278207 +0200
f7d506
@@ -673,6 +673,43 @@ ews_update_has_ooo_set (CamelSession *se
f7d506
 	g_clear_object (&oof_settings);
f7d506
 }
f7d506
 
f7d506
+static void
f7d506
+ews_exchange_server_categories_cb (CamelSession *session,
f7d506
+				   GCancellable *cancellable,
f7d506
+				   gpointer user_data,
f7d506
+				   GError **error)
f7d506
+{
f7d506
+	CamelEwsStore *ews_store = user_data;
f7d506
+	EEwsConnection *cnc;
f7d506
+	EwsFolderId fid = { 0 };
f7d506
+	gchar *properties = NULL;
f7d506
+	GError *local_error = NULL;
f7d506
+
f7d506
+	cnc = camel_ews_store_ref_connection (ews_store);
f7d506
+	if (!cnc)
f7d506
+		return;
f7d506
+
f7d506
+	fid.id = (gchar *) "calendar";
f7d506
+	fid.is_distinguished_id = TRUE;
f7d506
+
f7d506
+	if (e_ews_connection_get_user_configuration_sync (cnc, G_PRIORITY_DEFAULT, &fid, "CategoryList",
f7d506
+		E_EWS_USER_CONFIGURATION_PROPERTIES_XMLDATA, &properties, cancellable, &local_error) && properties) {
f7d506
+		guchar *data;
f7d506
+		gsize data_len = 0;
f7d506
+
f7d506
+		data = g_base64_decode (properties, &data_len);
f7d506
+
f7d506
+		if (data && data_len > 0)
f7d506
+			camel_ews_utils_merge_category_list (ews_store, data, data_len);
f7d506
+
f7d506
+		g_free (data);
f7d506
+	}
f7d506
+
f7d506
+	g_clear_error (&local_error);
f7d506
+	g_clear_object (&cnc);
f7d506
+	g_free (properties);
f7d506
+}
f7d506
+
f7d506
 struct ScheduleUpdateData
f7d506
 {
f7d506
 	GCancellable *cancellable;
f7d506
@@ -1252,6 +1289,12 @@ ews_connect_sync (CamelService *service,
f7d506
 				g_object_ref (ews_store),
f7d506
 				g_object_unref);
f7d506
 
f7d506
+		camel_session_submit_job (
f7d506
+			session, _("Look up Exchange server categories"),
f7d506
+			ews_exchange_server_categories_cb,
f7d506
+			g_object_ref (ews_store),
f7d506
+			g_object_unref);
f7d506
+
f7d506
 		if (!priv->updates_cancellable)
f7d506
 			priv->updates_cancellable = g_cancellable_new ();
f7d506
 
f7d506
@@ -2377,6 +2420,17 @@ ews_get_folder_info_sync (CamelStore *st
f7d506
 	ews_store = (CamelEwsStore *) store;
f7d506
 	priv = ews_store->priv;
f7d506
 
f7d506
+	if ((flags & CAMEL_STORE_FOLDER_INFO_REFRESH) != 0 &&
f7d506
+	    camel_offline_store_get_online (CAMEL_OFFLINE_STORE (ews_store))) {
f7d506
+		CamelSession *session;
f7d506
+
f7d506
+		session = camel_service_ref_session (CAMEL_SERVICE (ews_store));
f7d506
+		if (session) {
f7d506
+			ews_exchange_server_categories_cb (session, cancellable, ews_store, NULL);
f7d506
+			g_object_unref (session);
f7d506
+		}
f7d506
+	}
f7d506
+
f7d506
 	if ((flags & CAMEL_STORE_FOLDER_INFO_SUBSCRIPTION_LIST) != 0) {
f7d506
 		gboolean includes_last_folder = TRUE;
f7d506
 		GSList *folders = NULL, *to_check = NULL;
f7d506
diff -up evolution-ews-3.28.5/src/camel/camel-ews-store-summary.c.sync-category-list evolution-ews-3.28.5/src/camel/camel-ews-store-summary.c
f7d506
--- evolution-ews-3.28.5/src/camel/camel-ews-store-summary.c.sync-category-list	2018-07-30 16:01:00.000000000 +0200
f7d506
+++ evolution-ews-3.28.5/src/camel/camel-ews-store-summary.c	2019-10-24 09:39:08.341278207 +0200
f7d506
@@ -31,6 +31,7 @@
f7d506
 #define S_UNLOCK(x) (g_rec_mutex_unlock(&(x)->priv->s_lock))
f7d506
 
f7d506
 #define STORE_GROUP_NAME "##storepriv"
f7d506
+#define CATEGORIES_KEY "Categories"
f7d506
 #define CURRENT_SUMMARY_VERSION 3
f7d506
 
f7d506
 struct _CamelEwsStoreSummaryPrivate {
f7d506
@@ -1047,3 +1048,159 @@ camel_ews_store_summary_has_folder (Came
f7d506
 
f7d506
 	return ret;
f7d506
 }
f7d506
+
f7d506
+static gchar *
f7d506
+camel_ews_category_to_string (const CamelEwsCategory *cat)
f7d506
+{
f7d506
+	gchar *guid, *name, *color_def = NULL, *str;
f7d506
+
f7d506
+	g_return_val_if_fail (cat != NULL, NULL);
f7d506
+
f7d506
+	guid = g_uri_escape_string (cat->guid, NULL, TRUE);
f7d506
+	name = g_uri_escape_string (cat->name, NULL, TRUE);
f7d506
+
f7d506
+	if (cat->color_def)
f7d506
+		color_def = g_uri_escape_string (cat->color_def, NULL, TRUE);
f7d506
+
f7d506
+	str = g_strconcat (
f7d506
+		guid ? guid : "", "\t",
f7d506
+		name ? name : "", "\t",
f7d506
+		color_def ? color_def : "",
f7d506
+		NULL);
f7d506
+
f7d506
+	g_free (guid);
f7d506
+	g_free (name);
f7d506
+	g_free (color_def);
f7d506
+
f7d506
+	return str;
f7d506
+}
f7d506
+
f7d506
+static CamelEwsCategory *
f7d506
+camel_ews_category_from_string (const gchar *str)
f7d506
+{
f7d506
+	CamelEwsCategory *cat;
f7d506
+	gchar **strv, *guid, *name, *color_def;
f7d506
+
f7d506
+	g_return_val_if_fail (str != NULL, NULL);
f7d506
+
f7d506
+	strv = g_strsplit (str, "\t", -1);
f7d506
+	if (!strv || !strv[0] || !strv[1]) {
f7d506
+		g_strfreev (strv);
f7d506
+		return NULL;
f7d506
+	}
f7d506
+
f7d506
+	guid = g_uri_unescape_string (strv[0], NULL);
f7d506
+	name = g_uri_unescape_string (strv[1], NULL);
f7d506
+	color_def = (strv[2] && strv[2][0]) ? g_uri_unescape_string (strv[2], NULL) : NULL;
f7d506
+
f7d506
+	cat = camel_ews_category_new (guid, name, color_def);
f7d506
+
f7d506
+	g_free (guid);
f7d506
+	g_free (name);
f7d506
+	g_free (color_def);
f7d506
+	g_strfreev (strv);
f7d506
+
f7d506
+	return cat;
f7d506
+}
f7d506
+
f7d506
+GHashTable * /* gchar *guid ~> CamelEwsCategory * */
f7d506
+camel_ews_store_summary_get_categories (CamelEwsStoreSummary *ews_summary)
f7d506
+{
f7d506
+	GHashTable *categories;
f7d506
+	gchar **strv;
f7d506
+	g_return_val_if_fail (CAMEL_IS_EWS_STORE_SUMMARY (ews_summary), NULL);
f7d506
+
f7d506
+	S_LOCK (ews_summary);
f7d506
+
f7d506
+	strv = g_key_file_get_string_list (ews_summary->priv->key_file, STORE_GROUP_NAME, CATEGORIES_KEY, NULL, NULL);
f7d506
+
f7d506
+	S_UNLOCK (ews_summary);
f7d506
+
f7d506
+	categories = g_hash_table_new_full (g_str_hash, g_str_equal, NULL, camel_ews_category_free);
f7d506
+
f7d506
+	if (strv) {
f7d506
+		gint ii;
f7d506
+
f7d506
+		for (ii = 0; strv[ii]; ii++) {
f7d506
+			CamelEwsCategory *cat;
f7d506
+
f7d506
+			cat = camel_ews_category_from_string (strv[ii]);
f7d506
+			if (cat)
f7d506
+				g_hash_table_insert (categories, cat->guid, cat);
f7d506
+		}
f7d506
+
f7d506
+		g_strfreev (strv);
f7d506
+	}
f7d506
+
f7d506
+	return categories;
f7d506
+}
f7d506
+
f7d506
+void
f7d506
+camel_ews_store_summary_set_categories (CamelEwsStoreSummary *ews_summary,
f7d506
+					GHashTable *categories) /* gchar *guid ~> CamelEwsCategory * */
f7d506
+{
f7d506
+	GPtrArray *array;
f7d506
+	GHashTableIter iter;
f7d506
+	gpointer value;
f7d506
+
f7d506
+	g_return_if_fail (CAMEL_IS_EWS_STORE_SUMMARY (ews_summary));
f7d506
+	g_return_if_fail (categories != NULL);
f7d506
+
f7d506
+	array = g_ptr_array_new_full (g_hash_table_size (categories), g_free);
f7d506
+
f7d506
+	g_hash_table_iter_init (&iter, categories);
f7d506
+	while (g_hash_table_iter_next (&iter, NULL, &value)) {
f7d506
+		CamelEwsCategory *cat = value;
f7d506
+
f7d506
+		if (cat) {
f7d506
+			gchar *str;
f7d506
+
f7d506
+			str = camel_ews_category_to_string (cat);
f7d506
+
f7d506
+			if (str)
f7d506
+				g_ptr_array_add (array, str);
f7d506
+		}
f7d506
+	}
f7d506
+
f7d506
+	S_LOCK (ews_summary);
f7d506
+
f7d506
+	g_key_file_set_string_list (ews_summary->priv->key_file, STORE_GROUP_NAME, CATEGORIES_KEY,
f7d506
+		(const gchar * const *) array->pdata, array->len);
f7d506
+
f7d506
+	ews_summary->priv->dirty = TRUE;
f7d506
+
f7d506
+	S_UNLOCK (ews_summary);
f7d506
+
f7d506
+	g_ptr_array_free (array, TRUE);
f7d506
+}
f7d506
+
f7d506
+CamelEwsCategory *
f7d506
+camel_ews_category_new (const gchar *guid,
f7d506
+			const gchar *name,
f7d506
+			const gchar *color_def)
f7d506
+{
f7d506
+	CamelEwsCategory *cat;
f7d506
+
f7d506
+	g_return_val_if_fail (guid != NULL, NULL);
f7d506
+	g_return_val_if_fail (name != NULL, NULL);
f7d506
+
f7d506
+	cat = g_new0 (CamelEwsCategory, 1);
f7d506
+	cat->guid = g_strdup (guid);
f7d506
+	cat->name = g_strdup (name);
f7d506
+	cat->color_def = g_strdup (color_def);
f7d506
+
f7d506
+	return cat;
f7d506
+}
f7d506
+
f7d506
+void
f7d506
+camel_ews_category_free (gpointer ptr)
f7d506
+{
f7d506
+	CamelEwsCategory *cat = ptr;
f7d506
+
f7d506
+	if (cat) {
f7d506
+		g_free (cat->guid);
f7d506
+		g_free (cat->name);
f7d506
+		g_free (cat->color_def);
f7d506
+		g_free (cat);
f7d506
+	}
f7d506
+}
f7d506
diff -up evolution-ews-3.28.5/src/camel/camel-ews-store-summary.h.sync-category-list evolution-ews-3.28.5/src/camel/camel-ews-store-summary.h
f7d506
--- evolution-ews-3.28.5/src/camel/camel-ews-store-summary.h.sync-category-list	2018-07-30 16:01:00.000000000 +0200
f7d506
+++ evolution-ews-3.28.5/src/camel/camel-ews-store-summary.h	2019-10-24 09:39:08.341278207 +0200
f7d506
@@ -50,6 +50,12 @@
f7d506
 
f7d506
 G_BEGIN_DECLS
f7d506
 
f7d506
+typedef struct _CamelEwsCategory {
f7d506
+	gchar *guid;
f7d506
+	gchar *name;
f7d506
+	gchar *color_def;
f7d506
+} CamelEwsCategory;
f7d506
+
f7d506
 typedef struct _CamelEwsStoreSummary CamelEwsStoreSummary;
f7d506
 typedef struct _CamelEwsStoreSummaryClass CamelEwsStoreSummaryClass;
f7d506
 typedef struct _CamelEwsStoreSummaryPrivate CamelEwsStoreSummaryPrivate;
f7d506
@@ -215,6 +221,17 @@ gchar *		camel_ews_store_summary_get_fol
f7d506
 gboolean	camel_ews_store_summary_has_folder
f7d506
 						(CamelEwsStoreSummary *ews_summary,
f7d506
 						 const gchar *id);
f7d506
+GHashTable *	camel_ews_store_summary_get_categories /* gchar *guid ~> CamelEwsCategory * */
f7d506
+						(CamelEwsStoreSummary *ews_summary);
f7d506
+void		camel_ews_store_summary_set_categories
f7d506
+						(CamelEwsStoreSummary *ews_summary,
f7d506
+						 GHashTable *categories); /* gchar *guid ~> CamelEwsCategory * */
f7d506
+
f7d506
+CamelEwsCategory *
f7d506
+		camel_ews_category_new		(const gchar *guid,
f7d506
+						 const gchar *name,
f7d506
+						 const gchar *color_def);
f7d506
+void		camel_ews_category_free		(gpointer ptr); /* CamelEwsCategory * */
f7d506
 
f7d506
 G_END_DECLS
f7d506
 
f7d506
diff -up evolution-ews-3.28.5/src/camel/camel-ews-utils.c.sync-category-list evolution-ews-3.28.5/src/camel/camel-ews-utils.c
f7d506
--- evolution-ews-3.28.5/src/camel/camel-ews-utils.c.sync-category-list	2018-07-30 16:01:00.000000000 +0200
f7d506
+++ evolution-ews-3.28.5/src/camel/camel-ews-utils.c	2019-10-24 09:39:08.341278207 +0200
f7d506
@@ -29,6 +29,7 @@
f7d506
 #include <glib/gstdio.h>
f7d506
 
f7d506
 #include <libemail-engine/libemail-engine.h>
f7d506
+#include <e-util/e-util.h>
f7d506
 
f7d506
 #include "server/camel-ews-settings.h"
f7d506
 #include "server/e-ews-camel-common.h"
f7d506
@@ -381,6 +382,43 @@ camel_ews_utils_sync_deleted_items (Came
f7d506
 }
f7d506
 
f7d506
 static const gchar *
f7d506
+ews_utils_outlook_color_index_to_color_def (gint color_index)
f7d506
+{
f7d506
+	const gchar *colors_array[] = {
f7d506
+		"#ff1a36", /* Red */
f7d506
+		"#ff8c00", /* Orange */
f7d506
+		"#f4b10b", /* Peach */
f7d506
+		"#fff100", /* Yellow */
f7d506
+		"#009e48", /* Green */
f7d506
+		"#00b294", /* Teal */
f7d506
+		"#89933f", /* Olive */
f7d506
+		"#00bcf2", /* Blue */
f7d506
+		"#8e69df", /* Purple */
f7d506
+		"#f30092", /* Maroon */
f7d506
+		"#6c7e9a", /* Steel */
f7d506
+		"#425066", /* DarkSteel */
f7d506
+		"#969696", /* Gray */
f7d506
+		"#525552", /* DarkGray */
f7d506
+		"#282828", /* Black */
f7d506
+		"#a00023", /* DarkRed */
f7d506
+		"#c45502", /* DarkOrange */
f7d506
+		"#af7000", /* DarkPeach */
f7d506
+		"#b59b02", /* DarkYellow */
f7d506
+		"#176002", /* DarkGreen */
f7d506
+		"#00725c", /* DarkTeal */
f7d506
+		"#5c6022", /* DarkOlive */
f7d506
+		"#036393", /* DarkBlue */
f7d506
+		"#422f8e", /* DarkPurple */
f7d506
+		"#960269"  /* DarkMaroon */
f7d506
+	};
f7d506
+
f7d506
+	if (color_index >= 0 && color_index < G_N_ELEMENTS (colors_array))
f7d506
+		return colors_array[color_index];
f7d506
+
f7d506
+	return NULL;
f7d506
+}
f7d506
+
f7d506
+static const gchar *
f7d506
 ews_utils_rename_label (const gchar *cat,
f7d506
                         gboolean from_cat)
f7d506
 {
f7d506
@@ -422,6 +460,58 @@ ews_utils_is_system_user_flag (const gch
f7d506
 		g_str_equal (name, "$has-cal");
f7d506
 }
f7d506
 
f7d506
+/* From Exchange name (which allows spaces) to evolution-name */
f7d506
+static gchar *
f7d506
+camel_ews_utils_encode_category_name (const gchar *name)
f7d506
+{
f7d506
+	if (name && strchr (name, ' ')) {
f7d506
+		GString *str;
f7d506
+
f7d506
+		str = g_string_sized_new (strlen (name) + 16);
f7d506
+
f7d506
+		while (*name) {
f7d506
+			if (*name == '_')
f7d506
+				g_string_append_c (str, '_');
f7d506
+
f7d506
+			g_string_append_c (str, *name == ' ' ? '_' : *name);
f7d506
+
f7d506
+			name++;
f7d506
+		}
f7d506
+
f7d506
+		return g_string_free (str, FALSE);
f7d506
+	}
f7d506
+
f7d506
+	return g_strdup (name);
f7d506
+}
f7d506
+
f7d506
+/* From evolution-name to Exchange name (which allows spaces) */
f7d506
+static gchar *
f7d506
+camel_ews_utils_decode_category_name (const gchar *flag)
f7d506
+{
f7d506
+	if (flag && strchr (flag, '_')) {
f7d506
+		GString *str = g_string_sized_new (strlen (flag));
f7d506
+
f7d506
+		while (*flag) {
f7d506
+			if (*flag == '_') {
f7d506
+				if (flag[1] == '_') {
f7d506
+					g_string_append_c (str, '_');
f7d506
+					flag++;
f7d506
+				} else {
f7d506
+					g_string_append_c (str, ' ');
f7d506
+				}
f7d506
+			} else {
f7d506
+				g_string_append_c (str, *flag);
f7d506
+			}
f7d506
+
f7d506
+			flag++;
f7d506
+		}
f7d506
+
f7d506
+		return g_string_free (str, FALSE);
f7d506
+	}
f7d506
+
f7d506
+	return g_strdup (flag);
f7d506
+}
f7d506
+
f7d506
 /* free with g_slist_free_full (flags, g_free);
f7d506
    the lists' members are values for the String xml element. */
f7d506
 GSList *
f7d506
@@ -441,6 +531,7 @@ ews_utils_gather_server_user_flags (ESoa
f7d506
 	 * array of strings */
f7d506
 	for (ii = 0; ii < len; ii++) {
f7d506
 		const gchar *n = ews_utils_rename_label (camel_named_flags_get (user_flags, ii), FALSE);
f7d506
+
f7d506
 		if (*n == '\0')
f7d506
 			continue;
f7d506
 
f7d506
@@ -449,26 +540,7 @@ ews_utils_gather_server_user_flags (ESoa
f7d506
 		if (ews_utils_is_system_user_flag (n))
f7d506
 			continue;
f7d506
 
f7d506
-		if (strchr (n, '_')) {
f7d506
-			GString *str = g_string_sized_new (strlen (n));
f7d506
-
f7d506
-			while (*n) {
f7d506
-				if (*n == '_') {
f7d506
-					if (n[1] == '_')
f7d506
-						g_string_append_c (str, '_');
f7d506
-					else
f7d506
-						g_string_append_c (str, ' ');
f7d506
-				} else {
f7d506
-					g_string_append_c (str, *n);
f7d506
-				}
f7d506
-
f7d506
-				n++;
f7d506
-			}
f7d506
-
f7d506
-			out_user_flags = g_slist_prepend (out_user_flags, g_string_free (str, FALSE));
f7d506
-		} else {
f7d506
-			out_user_flags = g_slist_prepend (out_user_flags, g_strdup (n));
f7d506
-		}
f7d506
+		out_user_flags = g_slist_prepend (out_user_flags, camel_ews_utils_decode_category_name (n));
f7d506
 	}
f7d506
 
f7d506
 	camel_message_info_property_unlock (mi);
f7d506
@@ -512,33 +584,17 @@ ews_utils_merge_server_user_flags (EEwsI
f7d506
 
f7d506
 	/* now transfer over all the categories */
f7d506
 	for (p = e_ews_item_get_categories (item); p; p = p->next) {
f7d506
-		const gchar *flag = ews_utils_rename_label (p->data, 1);
f7d506
-		gchar *underscored = NULL;
f7d506
+		const gchar *name = ews_utils_rename_label (p->data, 1);
f7d506
+		gchar *flag;
f7d506
 
f7d506
-		if (!flag || !*flag)
f7d506
+		if (!name || !*name)
f7d506
 			continue;
f7d506
 
f7d506
-		if (strchr (flag, ' ')) {
f7d506
-			GString *str;
f7d506
-
f7d506
-			str = g_string_sized_new (strlen (flag) + 16);
f7d506
-
f7d506
-			while (*flag) {
f7d506
-				if (*flag == '_')
f7d506
-					g_string_append_c (str, '_');
f7d506
-
f7d506
-				g_string_append_c (str, *flag == ' ' ? '_' : *flag);
f7d506
-
f7d506
-				flag++;
f7d506
-			}
f7d506
-
f7d506
-			underscored = g_string_free (str, FALSE);
f7d506
-			flag = underscored;
f7d506
-		}
f7d506
+		flag = camel_ews_utils_encode_category_name (name);
f7d506
 
f7d506
 		camel_message_info_set_user_flag (mi, flag, TRUE);
f7d506
 
f7d506
-		g_free (underscored);
f7d506
+		g_free (flag);
f7d506
 	}
f7d506
 
f7d506
 	camel_message_info_thaw_notifications (mi);
f7d506
@@ -1281,3 +1337,279 @@ camel_ews_utils_ref_corresponding_source
f7d506
 
f7d506
 	return source;
f7d506
 }
f7d506
+
f7d506
+static gboolean
f7d506
+ews_util_equal_label_tag_cb (gconstpointer ptr1,
f7d506
+			     gconstpointer ptr2)
f7d506
+{
f7d506
+	const gchar *evo_label_def = ptr1;
f7d506
+	const gchar *tag = ptr2;
f7d506
+	const gchar *pos;
f7d506
+
f7d506
+	if (!evo_label_def || !tag || !*tag)
f7d506
+		return FALSE;
f7d506
+
f7d506
+	pos = g_strrstr (evo_label_def, tag);
f7d506
+
f7d506
+	return pos > evo_label_def && pos[-1] == '|' && !pos[strlen (tag)];
f7d506
+}
f7d506
+
f7d506
+static gboolean
f7d506
+ews_utils_find_in_ptr_array (GPtrArray *haystack,
f7d506
+			     gconstpointer needle,
f7d506
+			     GEqualFunc equal_func,
f7d506
+			     guint *out_index)
f7d506
+{
f7d506
+	guint ii;
f7d506
+
f7d506
+	if (!haystack)
f7d506
+		return FALSE;
f7d506
+
f7d506
+	if (!equal_func)
f7d506
+		equal_func = g_direct_equal;
f7d506
+
f7d506
+	for (ii = 0; ii < haystack->len; ii++) {
f7d506
+		if (equal_func (haystack->pdata[ii], needle)) {
f7d506
+			if (out_index)
f7d506
+				*out_index = ii;
f7d506
+
f7d506
+			return TRUE;
f7d506
+		}
f7d506
+	}
f7d506
+
f7d506
+	return FALSE;
f7d506
+}
f7d506
+
f7d506
+/* Returns whether had been done any changes */
f7d506
+static gboolean
f7d506
+ews_utils_save_category_changes (GHashTable *old_categories, /* gchar *guid ~> CamelEwsCategory * */
f7d506
+				 GHashTable *new_categories) /* gchar *guid ~> CamelEwsCategory * */
f7d506
+{
f7d506
+	GHashTableIter iter;
f7d506
+	GSettings *settings;
f7d506
+	GPtrArray *evo_labels; /* gchar * (encoded label definition) */
f7d506
+	gchar **strv;
f7d506
+	gint ii;
f7d506
+	gpointer value;
f7d506
+	gboolean changed = FALSE;
f7d506
+
f7d506
+	if (!old_categories || !new_categories)
f7d506
+		return new_categories != NULL;
f7d506
+
f7d506
+	evo_labels = g_ptr_array_new_full (5, g_free);
f7d506
+
f7d506
+	settings = e_util_ref_settings ("org.gnome.evolution.mail");
f7d506
+	strv = g_settings_get_strv (settings, "labels");
f7d506
+
f7d506
+	for (ii = 0; strv && strv[ii]; ii++) {
f7d506
+		g_ptr_array_add (evo_labels, g_strdup (strv[ii]));
f7d506
+	}
f7d506
+
f7d506
+	g_strfreev (strv);
f7d506
+
f7d506
+	g_hash_table_iter_init (&iter, new_categories);
f7d506
+	while (g_hash_table_iter_next (&iter, NULL, &value)) {
f7d506
+		CamelEwsCategory *new_cat = value, *old_cat;
f7d506
+		gchar *tag = NULL;
f7d506
+
f7d506
+		if (!new_cat)
f7d506
+			continue;
f7d506
+
f7d506
+		old_cat = g_hash_table_lookup (old_categories, new_cat->guid);
f7d506
+		if (old_cat) {
f7d506
+			if (g_strcmp0 (old_cat->name, new_cat->name) != 0 ||
f7d506
+			    g_strcmp0 (old_cat->color_def, new_cat->color_def) != 0) {
f7d506
+				/* Old category changed name or color */
f7d506
+				tag = camel_ews_utils_encode_category_name (new_cat->name);
f7d506
+			}
f7d506
+		} else {
f7d506
+			/* This is a new category */
f7d506
+			tag = camel_ews_utils_encode_category_name (new_cat->name);
f7d506
+		}
f7d506
+
f7d506
+		if (tag && *tag) {
f7d506
+			guint index = (guint) -1;
f7d506
+			gchar *label_def;
f7d506
+
f7d506
+			changed = TRUE;
f7d506
+
f7d506
+			/* Sanitize value */
f7d506
+			for (ii = 0; tag[ii]; ii++) {
f7d506
+				if (tag[ii] == '|')
f7d506
+					tag[ii] = '-';
f7d506
+			}
f7d506
+
f7d506
+			if (old_cat && g_strcmp0 (old_cat->name, new_cat->name) != 0) {
f7d506
+				gchar *old_tag = camel_ews_utils_encode_category_name (old_cat->name);
f7d506
+
f7d506
+				if (old_tag && *old_tag) {
f7d506
+					if (!ews_utils_find_in_ptr_array (evo_labels, old_tag, ews_util_equal_label_tag_cb, &index))
f7d506
+						index = (guint) -1;
f7d506
+				}
f7d506
+
f7d506
+				g_free (old_tag);
f7d506
+			}
f7d506
+
f7d506
+			for (ii = 0; new_cat->name[ii]; ii++) {
f7d506
+				if (new_cat->name[ii] == '|')
f7d506
+					new_cat->name[ii] = '-';
f7d506
+			}
f7d506
+
f7d506
+			if (index == (guint) -1 &&
f7d506
+			    !ews_utils_find_in_ptr_array (evo_labels, tag, ews_util_equal_label_tag_cb, &index))
f7d506
+				index = (guint) -1;
f7d506
+
f7d506
+			label_def = g_strconcat (new_cat->name, "|", new_cat->color_def ? new_cat->color_def : "#FF0000", "|", tag, NULL);
f7d506
+
f7d506
+			if (index == (guint) -1 || index >= (gint) evo_labels->len) {
f7d506
+				g_ptr_array_add (evo_labels, label_def);
f7d506
+			} else {
f7d506
+				g_free (evo_labels->pdata[index]);
f7d506
+				evo_labels->pdata[index] = label_def;
f7d506
+			}
f7d506
+		}
f7d506
+
f7d506
+		g_hash_table_remove (old_categories, new_cat->guid);
f7d506
+
f7d506
+		g_free (tag);
f7d506
+	}
f7d506
+
f7d506
+	if (g_hash_table_size (old_categories) > 0) {
f7d506
+		/* Some categories had been removed */
f7d506
+		changed = TRUE;
f7d506
+
f7d506
+		g_hash_table_iter_init (&iter, old_categories);
f7d506
+		while (g_hash_table_iter_next (&iter, NULL, &value)) {
f7d506
+			CamelEwsCategory *old_cat = value;
f7d506
+			gchar *old_tag;
f7d506
+			guint index;
f7d506
+
f7d506
+			if (!old_cat)
f7d506
+				continue;
f7d506
+
f7d506
+			old_tag = camel_ews_utils_encode_category_name (old_cat->name);
f7d506
+
f7d506
+			for (ii = 0; old_tag && old_tag[ii]; ii++) {
f7d506
+				if (old_tag[ii] == '|')
f7d506
+					old_tag[ii] = '-';
f7d506
+			}
f7d506
+
f7d506
+			if (old_tag &&
f7d506
+			    ews_utils_find_in_ptr_array (evo_labels, old_tag, ews_util_equal_label_tag_cb, &index))
f7d506
+				g_ptr_array_remove_index (evo_labels, index);
f7d506
+
f7d506
+			g_free (old_tag);
f7d506
+		}
f7d506
+	}
f7d506
+
f7d506
+	if (changed) {
f7d506
+		/* NULL-terminated array of strings */
f7d506
+		g_ptr_array_add (evo_labels, NULL);
f7d506
+
f7d506
+		g_settings_set_strv (settings, "labels", (const gchar * const *) evo_labels->pdata);
f7d506
+	}
f7d506
+
f7d506
+	g_ptr_array_free (evo_labels, TRUE);
f7d506
+	g_object_unref (settings);
f7d506
+
f7d506
+	return changed;
f7d506
+}
f7d506
+
f7d506
+void
f7d506
+camel_ews_utils_merge_category_list (CamelEwsStore *ews_store,
f7d506
+				     const guchar *xml_data,
f7d506
+				     gsize xml_data_len)
f7d506
+{
f7d506
+	xmlDocPtr doc;
f7d506
+	xmlXPathContextPtr xpath_ctx;
f7d506
+
f7d506
+	g_return_if_fail (CAMEL_IS_EWS_STORE (ews_store));
f7d506
+	g_return_if_fail (xml_data != NULL);
f7d506
+
f7d506
+	doc = e_xml_parse_data (xml_data, xml_data_len);
f7d506
+	if (!doc)
f7d506
+		return;
f7d506
+
f7d506
+	xpath_ctx = e_xml_new_xpath_context_with_namespaces (doc, "C", "CategoryList.xsd", NULL);
f7d506
+
f7d506
+	if (xpath_ctx) {
f7d506
+		xmlXPathObjectPtr xpath_obj_categories;
f7d506
+
f7d506
+		xpath_obj_categories = e_xml_xpath_eval (xpath_ctx, "%s", "/C:categories/C:category");
f7d506
+
f7d506
+		if (xpath_obj_categories) {
f7d506
+			GHashTable *old_categories, *new_categories;
f7d506
+			gint response_index, response_length;
f7d506
+
f7d506
+			new_categories = g_hash_table_new_full (g_str_hash, g_str_equal, NULL, camel_ews_category_free);
f7d506
+
f7d506
+			response_length = xmlXPathNodeSetGetLength (xpath_obj_categories->nodesetval);
f7d506
+
f7d506
+			for (response_index = 0; response_index < response_length; response_index++) {
f7d506
+				xmlXPathObjectPtr xpath_obj_category;
f7d506
+
f7d506
+				xpath_obj_category = e_xml_xpath_eval (xpath_ctx,
f7d506
+					"/C:categories/C:category[%d]",
f7d506
+					response_index + 1);
f7d506
+
f7d506
+				if (xpath_obj_category) {
f7d506
+					gchar *name;
f7d506
+
f7d506
+					name = e_xml_xpath_eval_as_string (xpath_ctx, "/C:categories/C:category[%d]/@name", response_index + 1);
f7d506
+
f7d506
+					if (name && ews_utils_rename_label (name, 1) == name) {
f7d506
+						const gchar *color_def = NULL;
f7d506
+						gchar *color, *guid;
f7d506
+						gint color_index = -1;
f7d506
+
f7d506
+						color = e_xml_xpath_eval_as_string (xpath_ctx, "/C:categories/C:category[%d]/@color", response_index + 1);
f7d506
+						if (color) {
f7d506
+							gchar *endptr = NULL;
f7d506
+
f7d506
+							color_index = (gint) g_ascii_strtoll (color, &endptr, 10);
f7d506
+
f7d506
+							if (endptr == color)
f7d506
+								color_index = -1;
f7d506
+						}
f7d506
+
f7d506
+						g_free (color);
f7d506
+
f7d506
+						if (color_index >= 0)
f7d506
+							color_def = ews_utils_outlook_color_index_to_color_def (color_index);
f7d506
+
f7d506
+						guid = e_xml_xpath_eval_as_string (xpath_ctx, "/C:categories/C:category[%d]/@guid", response_index + 1);
f7d506
+
f7d506
+						if (guid && *guid) {
f7d506
+							CamelEwsCategory *cat;
f7d506
+
f7d506
+							cat = camel_ews_category_new (guid, name, color_def);
f7d506
+							if (cat)
f7d506
+								g_hash_table_insert (new_categories, cat->guid, cat);
f7d506
+						}
f7d506
+
f7d506
+						g_free (guid);
f7d506
+					}
f7d506
+
f7d506
+					g_free (name);
f7d506
+					xmlXPathFreeObject (xpath_obj_category);
f7d506
+				}
f7d506
+			}
f7d506
+
f7d506
+			xmlXPathFreeObject (xpath_obj_categories);
f7d506
+
f7d506
+			old_categories = camel_ews_store_summary_get_categories (ews_store->summary);
f7d506
+
f7d506
+			if (ews_utils_save_category_changes (old_categories, new_categories)) {
f7d506
+				camel_ews_store_summary_set_categories (ews_store->summary, new_categories);
f7d506
+				camel_ews_store_summary_save (ews_store->summary, NULL);
f7d506
+			}
f7d506
+
f7d506
+			g_hash_table_destroy (new_categories);
f7d506
+			g_hash_table_destroy (old_categories);
f7d506
+		}
f7d506
+	}
f7d506
+
f7d506
+	if (xpath_ctx)
f7d506
+		xmlXPathFreeContext (xpath_ctx);
f7d506
+	xmlFreeDoc (doc);
f7d506
+}
f7d506
diff -up evolution-ews-3.28.5/src/camel/camel-ews-utils.h.sync-category-list evolution-ews-3.28.5/src/camel/camel-ews-utils.h
f7d506
--- evolution-ews-3.28.5/src/camel/camel-ews-utils.h.sync-category-list	2018-07-30 16:01:00.000000000 +0200
f7d506
+++ evolution-ews-3.28.5/src/camel/camel-ews-utils.h	2019-10-24 09:39:08.341278207 +0200
f7d506
@@ -94,6 +94,10 @@ CamelMessageInfo * /* (transfer full) */
f7d506
 						 EEwsConnection *cnc,
f7d506
 						 EEwsItem *item,
f7d506
 						 GCancellable *cancellable);
f7d506
+void		camel_ews_utils_merge_category_list
f7d506
+						(CamelEwsStore *ews_store,
f7d506
+						 const guchar *xml_data,
f7d506
+						 gsize xml_data_len);
f7d506
 
f7d506
 G_END_DECLS
f7d506
 
f7d506
diff -up evolution-ews-3.28.5/src/server/e-ews-connection.c.sync-category-list evolution-ews-3.28.5/src/server/e-ews-connection.c
f7d506
--- evolution-ews-3.28.5/src/server/e-ews-connection.c.sync-category-list	2019-10-24 09:39:08.339278207 +0200
f7d506
+++ evolution-ews-3.28.5/src/server/e-ews-connection.c	2019-10-24 09:46:14.667272312 +0200
f7d506
@@ -155,7 +155,7 @@ struct _EwsAsyncData {
f7d506
 	EwsDelegateDeliver deliver_to;
f7d506
 	EEwsFolderType folder_type;
f7d506
 	EEwsConnection *cnc;
f7d506
-	gchar *user_photo; /* base64-encoded, as GetUserPhoto result */
f7d506
+	gchar *custom_data; /* Can be re-used by operations, will be freed with g_free() */
f7d506
 };
f7d506
 
f7d506
 struct _EwsNode {
f7d506
@@ -200,7 +200,7 @@ ews_connection_error_quark (void)
f7d506
 static void
f7d506
 async_data_free (EwsAsyncData *async_data)
f7d506
 {
f7d506
-	g_free (async_data->user_photo);
f7d506
+	g_free (async_data->custom_data);
f7d506
 	g_free (async_data);
f7d506
 }
f7d506
 
f7d506
@@ -10830,10 +10830,10 @@ get_user_photo_response_cb (ESoapRespons
f7d506
 		return;
f7d506
 	}
f7d506
 
f7d506
-	async_data->user_photo = e_soap_parameter_get_string_value (param);
f7d506
-	if (async_data->user_photo && !*async_data->user_photo) {
f7d506
-		g_free (async_data->user_photo);
f7d506
-		async_data->user_photo = NULL;
f7d506
+	async_data->custom_data = e_soap_parameter_get_string_value (param);
f7d506
+	if (async_data->custom_data && !*async_data->custom_data) {
f7d506
+		g_free (async_data->custom_data);
f7d506
+		async_data->custom_data = NULL;
f7d506
 	}
f7d506
 }
f7d506
 
f7d506
@@ -10918,11 +10918,11 @@ e_ews_connection_get_user_photo_finish (
f7d506
 	if (g_simple_async_result_propagate_error (simple, error))
f7d506
 		return FALSE;
f7d506
 
f7d506
-	if (!async_data->user_photo)
f7d506
+	if (!async_data->custom_data)
f7d506
 		return FALSE;
f7d506
 
f7d506
-	*out_picture_data = async_data->user_photo;
f7d506
-	async_data->user_photo = NULL;
f7d506
+	*out_picture_data = async_data->custom_data;
f7d506
+	async_data->custom_data = NULL;
f7d506
 
f7d506
 	return TRUE;
f7d506
 }
f7d506
@@ -10953,5 +10953,256 @@ e_ews_connection_get_user_photo_sync (EE
f7d506
 
f7d506
 	e_async_closure_free (closure);
f7d506
 
f7d506
+	return success;
f7d506
+}
f7d506
+
f7d506
+static void
f7d506
+get_user_configuration_response_cb (ESoapResponse *response,
f7d506
+				    GSimpleAsyncResult *simple)
f7d506
+{
f7d506
+	EwsAsyncData *async_data;
f7d506
+	ESoapParameter *param, *subparam;
f7d506
+	GError *error = NULL;
f7d506
+
f7d506
+	async_data = g_simple_async_result_get_op_res_gpointer (simple);
f7d506
+
f7d506
+	param = e_soap_response_get_first_parameter_by_name (response, "ResponseMessages", &error);
f7d506
+
f7d506
+	if (param) {
f7d506
+		param = e_soap_parameter_get_first_child_by_name (param, "GetUserConfigurationResponseMessage");
f7d506
+		if (!param) {
f7d506
+			g_set_error (&error,
f7d506
+				SOUP_HTTP_ERROR, SOUP_STATUS_MALFORMED,
f7d506
+				"Missing <%s> in SOAP response", "GetUserConfigurationResponseMessage");
f7d506
+		}
f7d506
+	}
f7d506
+
f7d506
+	if (param) {
f7d506
+		param = e_soap_parameter_get_first_child_by_name (param, "UserConfiguration");
f7d506
+		if (!param) {
f7d506
+			g_set_error (&error,
f7d506
+				SOUP_HTTP_ERROR, SOUP_STATUS_MALFORMED,
f7d506
+				"Missing <%s> in SOAP response", "UserConfiguration");
f7d506
+		}
f7d506
+	}
f7d506
+
f7d506
+	/* Sanity check */
f7d506
+	g_return_if_fail (
f7d506
+		(param != NULL && error == NULL) ||
f7d506
+		(param == NULL && error != NULL));
f7d506
+
f7d506
+	if (error != NULL) {
f7d506
+		g_simple_async_result_take_error (simple, error);
f7d506
+		return;
f7d506
+	}
f7d506
+
f7d506
+	subparam = e_soap_parameter_get_first_child_by_name (param, "ItemId");
f7d506
+	if (subparam) {
f7d506
+		gchar *id, *changekey;
f7d506
+
f7d506
+		id = e_soap_parameter_get_property (subparam, "Id");
f7d506
+		changekey = e_soap_parameter_get_property (subparam, "ChangeKey");
f7d506
+
f7d506
+		/* Encoded as: Id + "\n" + ChangeKey */
f7d506
+		async_data->custom_data = g_strconcat (id ? id : "", "\n", changekey, NULL);
f7d506
+
f7d506
+		g_free (changekey);
f7d506
+		g_free (id);
f7d506
+	}
f7d506
+
f7d506
+	if (!subparam) {
f7d506
+		subparam = e_soap_parameter_get_first_child_by_name (param, "Dictionary");
f7d506
+		if (subparam)
f7d506
+			async_data->custom_data = e_soap_response_dump_parameter (response, subparam);
f7d506
+	}
f7d506
+
f7d506
+	if (!subparam) {
f7d506
+		subparam = e_soap_parameter_get_first_child_by_name (param, "XmlData");
f7d506
+		if (subparam) {
f7d506
+			async_data->custom_data = e_soap_parameter_get_string_value (subparam);
f7d506
+		}
f7d506
+	}
f7d506
+
f7d506
+	if (!subparam) {
f7d506
+		subparam = e_soap_parameter_get_first_child_by_name (param, "BinaryData");
f7d506
+		if (subparam) {
f7d506
+			async_data->custom_data = e_soap_parameter_get_string_value (subparam);
f7d506
+		}
f7d506
+	}
f7d506
+
f7d506
+	if (async_data->custom_data && !*async_data->custom_data) {
f7d506
+		g_free (async_data->custom_data);
f7d506
+		async_data->custom_data = NULL;
f7d506
+	}
f7d506
+}
f7d506
+
f7d506
+static void
f7d506
+e_ews_folder_id_append_to_msg (ESoapMessage *msg,
f7d506
+			       const gchar *email,
f7d506
+			       const EwsFolderId *fid)
f7d506
+{
f7d506
+	g_return_if_fail (msg != NULL);
f7d506
+	g_return_if_fail (fid != NULL);
f7d506
+
f7d506
+	if (fid->is_distinguished_id)
f7d506
+		e_soap_message_start_element (msg, "DistinguishedFolderId", NULL, NULL);
f7d506
+	else
f7d506
+		e_soap_message_start_element (msg, "FolderId", NULL, NULL);
f7d506
+
f7d506
+	e_soap_message_add_attribute (msg, "Id", fid->id, NULL, NULL);
f7d506
+	if (fid->change_key)
f7d506
+		e_soap_message_add_attribute (msg, "ChangeKey", fid->change_key, NULL, NULL);
f7d506
+
f7d506
+	if (fid->is_distinguished_id && email) {
f7d506
+		e_soap_message_start_element (msg, "Mailbox", NULL, NULL);
f7d506
+		e_ews_message_write_string_parameter (msg, "EmailAddress", NULL, email);
f7d506
+		e_soap_message_end_element (msg);
f7d506
+	}
f7d506
+
f7d506
+	e_soap_message_end_element (msg);
f7d506
+}
f7d506
+
f7d506
+void
f7d506
+e_ews_connection_get_user_configuration (EEwsConnection *cnc,
f7d506
+					 gint pri,
f7d506
+					 const EwsFolderId *fid,
f7d506
+					 const gchar *config_name,
f7d506
+					 EEwsUserConfigurationProperties props,
f7d506
+					 GCancellable *cancellable,
f7d506
+					 GAsyncReadyCallback callback,
f7d506
+					 gpointer user_data)
f7d506
+{
f7d506
+	ESoapMessage *msg;
f7d506
+	GSimpleAsyncResult *simple;
f7d506
+	EwsAsyncData *async_data;
f7d506
+	EwsFolderId local_fid;
f7d506
+
f7d506
+	g_return_if_fail (cnc != NULL);
f7d506
+	g_return_if_fail (cnc->priv != NULL);
f7d506
+	g_return_if_fail (fid != NULL);
f7d506
+	g_return_if_fail (config_name != NULL);
f7d506
+
f7d506
+	simple = g_simple_async_result_new (G_OBJECT (cnc), callback, user_data, e_ews_connection_get_user_configuration);
f7d506
+	async_data = g_new0 (EwsAsyncData, 1);
f7d506
+	g_simple_async_result_set_op_res_gpointer (simple, async_data, (GDestroyNotify) async_data_free);
f7d506
+
f7d506
+	/* EWS server version earlier than 2010 doesn't support it. */
f7d506
+	if (!e_ews_connection_satisfies_server_version (cnc, E_EWS_EXCHANGE_2010)) {
f7d506
+		g_simple_async_result_complete_in_idle (simple);
f7d506
+		g_object_unref (simple);
f7d506
+		return;
f7d506
+	}
f7d506
+
f7d506
+	local_fid = *fid;
f7d506
+	local_fid.change_key = NULL;
f7d506
+
f7d506
+	msg = e_ews_message_new_with_header (
f7d506
+		cnc->priv->settings,
f7d506
+		cnc->priv->uri,
f7d506
+		cnc->priv->impersonate_user,
f7d506
+		"GetUserConfiguration",
f7d506
+		NULL,
f7d506
+		NULL,
f7d506
+		cnc->priv->version,
f7d506
+		E_EWS_EXCHANGE_2010,
f7d506
+		FALSE,
f7d506
+		TRUE);
f7d506
+
f7d506
+	e_soap_message_start_element (msg, "UserConfigurationName", "messages", NULL);
f7d506
+	e_soap_message_add_attribute (msg, "Name", config_name, NULL, NULL);
f7d506
+
f7d506
+	e_ews_folder_id_append_to_msg (msg, cnc->priv->email, &local_fid);
f7d506
+
f7d506
+	e_soap_message_end_element (msg); /* UserConfigurationName */
f7d506
+
f7d506
+	e_soap_message_start_element (msg, "UserConfigurationProperties", "messages", NULL);
f7d506
+
f7d506
+	switch (props) {
f7d506
+	case E_EWS_USER_CONFIGURATION_PROPERTIES_ID:
f7d506
+		e_soap_message_write_string (msg, "Id");
f7d506
+		break;
f7d506
+	case E_EWS_USER_CONFIGURATION_PROPERTIES_DICTIONARY:
f7d506
+		e_soap_message_write_string (msg, "Dictionary");
f7d506
+		break;
f7d506
+	case E_EWS_USER_CONFIGURATION_PROPERTIES_XMLDATA:
f7d506
+		e_soap_message_write_string (msg, "XmlData");
f7d506
+		break;
f7d506
+	case E_EWS_USER_CONFIGURATION_PROPERTIES_BINARYDATA:
f7d506
+		e_soap_message_write_string (msg, "BinaryData");
f7d506
+		break;
f7d506
+	/* case E_EWS_USER_CONFIGURATION_PROPERTIES_ALL:
f7d506
+		e_soap_message_write_string (msg, "All");
f7d506
+		break; */
f7d506
+	default:
f7d506
+		e_soap_message_write_string (msg, "Unknown");
f7d506
+		break;
f7d506
+	}
f7d506
+
f7d506
+	e_soap_message_end_element (msg); /* UserConfigurationProperties */
f7d506
+
f7d506
+	e_ews_message_write_footer (msg);
f7d506
+
f7d506
+	e_ews_connection_queue_request (cnc, msg, get_user_configuration_response_cb, pri, cancellable, simple);
f7d506
+
f7d506
+	g_object_unref (simple);
f7d506
+}
f7d506
+
f7d506
+gboolean
f7d506
+e_ews_connection_get_user_configuration_finish (EEwsConnection *cnc,
f7d506
+						GAsyncResult *result,
f7d506
+						gchar **out_properties,
f7d506
+						GError **error)
f7d506
+{
f7d506
+	GSimpleAsyncResult *simple;
f7d506
+	EwsAsyncData *async_data;
f7d506
+
f7d506
+	g_return_val_if_fail (cnc != NULL, FALSE);
f7d506
+	g_return_val_if_fail (
f7d506
+		g_simple_async_result_is_valid (result, G_OBJECT (cnc), e_ews_connection_get_user_configuration),
f7d506
+		FALSE);
f7d506
+	g_return_val_if_fail (out_properties != NULL, FALSE);
f7d506
+
f7d506
+	simple = G_SIMPLE_ASYNC_RESULT (result);
f7d506
+	async_data = g_simple_async_result_get_op_res_gpointer (simple);
f7d506
+
f7d506
+	if (g_simple_async_result_propagate_error (simple, error))
f7d506
+		return FALSE;
f7d506
+
f7d506
+	if (!async_data->custom_data)
f7d506
+		return FALSE;
f7d506
+
f7d506
+	*out_properties = async_data->custom_data;
f7d506
+	async_data->custom_data = NULL;
f7d506
+
f7d506
+	return TRUE;
f7d506
+}
f7d506
+
f7d506
+gboolean
f7d506
+e_ews_connection_get_user_configuration_sync (EEwsConnection *cnc,
f7d506
+					      gint pri,
f7d506
+					      const EwsFolderId *fid,
f7d506
+					      const gchar *config_name,
f7d506
+					      EEwsUserConfigurationProperties props,
f7d506
+					      gchar **out_properties,
f7d506
+					      GCancellable *cancellable,
f7d506
+					      GError **error)
f7d506
+{
f7d506
+	EAsyncClosure *closure;
f7d506
+	GAsyncResult *result;
f7d506
+	gboolean success;
f7d506
+
f7d506
+	g_return_val_if_fail (cnc != NULL, FALSE);
f7d506
+
f7d506
+	closure = e_async_closure_new ();
f7d506
+
f7d506
+	e_ews_connection_get_user_configuration (
f7d506
+		cnc, pri, fid, config_name, props, cancellable, e_async_closure_callback, closure);
f7d506
+
f7d506
+	result = e_async_closure_wait (closure);
f7d506
+
f7d506
+	success = e_ews_connection_get_user_configuration_finish (cnc, result, out_properties, error);
f7d506
+
f7d506
+	e_async_closure_free (closure);
f7d506
+
f7d506
 	return success;
f7d506
 }
f7d506
diff -up evolution-ews-3.28.5/src/server/e-ews-connection.h.sync-category-list evolution-ews-3.28.5/src/server/e-ews-connection.h
f7d506
--- evolution-ews-3.28.5/src/server/e-ews-connection.h.sync-category-list	2019-10-24 09:39:08.339278207 +0200
f7d506
+++ evolution-ews-3.28.5/src/server/e-ews-connection.h	2019-10-24 09:39:08.342278207 +0200
f7d506
@@ -132,6 +132,15 @@ typedef enum {
f7d506
 	E_EWS_SIZE_REQUESTED_648X648 = 648
f7d506
 } EEwsSizeRequested;
f7d506
 
f7d506
+typedef enum {
f7d506
+	E_EWS_USER_CONFIGURATION_PROPERTIES_UNKNOWN = -1,
f7d506
+	E_EWS_USER_CONFIGURATION_PROPERTIES_ID,
f7d506
+	E_EWS_USER_CONFIGURATION_PROPERTIES_DICTIONARY,
f7d506
+	E_EWS_USER_CONFIGURATION_PROPERTIES_XMLDATA,
f7d506
+	E_EWS_USER_CONFIGURATION_PROPERTIES_BINARYDATA /*,
f7d506
+	E_EWS_USER_CONFIGURATION_PROPERTIES_ALL - skip it, be specific */
f7d506
+} EEwsUserConfigurationProperties;
f7d506
+
f7d506
 typedef struct {
f7d506
 	gchar *id;
f7d506
 	gchar *dn;
f7d506
@@ -1377,6 +1386,29 @@ gboolean	e_ews_connection_get_user_photo
f7d506
 						 gchar **out_picture_data, /* base64-encoded */
f7d506
 						 GCancellable *cancellable,
f7d506
 						 GError **error);
f7d506
+void		e_ews_connection_get_user_configuration
f7d506
+						(EEwsConnection *cnc,
f7d506
+						 gint pri,
f7d506
+						 const EwsFolderId *fid,
f7d506
+						 const gchar *config_name,
f7d506
+						 EEwsUserConfigurationProperties props,
f7d506
+						 GCancellable *cancellable,
f7d506
+						 GAsyncReadyCallback callback,
f7d506
+						 gpointer user_data);
f7d506
+gboolean	e_ews_connection_get_user_configuration_finish
f7d506
+						(EEwsConnection *cnc,
f7d506
+						 GAsyncResult *result,
f7d506
+						 gchar **out_properties,
f7d506
+						 GError **error);
f7d506
+gboolean	e_ews_connection_get_user_configuration_sync
f7d506
+						(EEwsConnection *cnc,
f7d506
+						 gint pri,
f7d506
+						 const EwsFolderId *fid,
f7d506
+						 const gchar *config_name,
f7d506
+						 EEwsUserConfigurationProperties props,
f7d506
+						 gchar **out_properties,
f7d506
+						 GCancellable *cancellable,
f7d506
+						 GError **error);
f7d506
 
f7d506
 G_END_DECLS
f7d506
 
f7d506
diff -up evolution-ews-3.28.5/src/server/e-soap-response.c.sync-category-list evolution-ews-3.28.5/src/server/e-soap-response.c
f7d506
--- evolution-ews-3.28.5/src/server/e-soap-response.c.sync-category-list	2018-07-30 16:01:00.000000000 +0200
f7d506
+++ evolution-ews-3.28.5/src/server/e-soap-response.c	2019-10-24 09:39:08.342278207 +0200
f7d506
@@ -685,3 +685,29 @@ e_soap_response_dump_response (ESoapResp
f7d506
 
f7d506
 	return ret;
f7d506
 }
f7d506
+
f7d506
+gchar *
f7d506
+e_soap_response_dump_parameter (ESoapResponse *response,
f7d506
+				ESoapParameter *param)
f7d506
+{
f7d506
+	xmlBuffer *buffer;
f7d506
+	gint len;
f7d506
+	gchar *data;
f7d506
+
f7d506
+	g_return_val_if_fail (E_IS_SOAP_RESPONSE (response), NULL);
f7d506
+	g_return_val_if_fail (param != NULL, NULL);
f7d506
+
f7d506
+	buffer = xmlBufferCreate ();
f7d506
+	len = xmlNodeDump (buffer, response->priv->xmldoc, param, 0, 0);
f7d506
+
f7d506
+	if (len <= 0) {
f7d506
+		xmlBufferFree (buffer);
f7d506
+		return NULL;
f7d506
+	}
f7d506
+
f7d506
+	data = g_strndup ((const gchar *) buffer->content, len);
f7d506
+
f7d506
+	xmlBufferFree (buffer);
f7d506
+
f7d506
+	return data;
f7d506
+}
f7d506
diff -up evolution-ews-3.28.5/src/server/e-soap-response.h.sync-category-list evolution-ews-3.28.5/src/server/e-soap-response.h
f7d506
--- evolution-ews-3.28.5/src/server/e-soap-response.h.sync-category-list	2018-07-30 16:01:00.000000000 +0200
f7d506
+++ evolution-ews-3.28.5/src/server/e-soap-response.h	2019-10-24 09:39:08.343278207 +0200
f7d506
@@ -101,6 +101,8 @@ ESoapParameter *
f7d506
 						 const gchar *name);
f7d506
 gint		e_soap_response_dump_response	(ESoapResponse *response,
f7d506
 						 FILE *buffer);
f7d506
+gchar *		e_soap_response_dump_parameter	(ESoapResponse *response,
f7d506
+						 ESoapParameter *param);
f7d506
 
f7d506
 G_END_DECLS
f7d506