[evolution-ews] eds-I#12 - vCard REV attribute should be a timestamp
- From: Milan Crha <mcrha src gnome org>
- To: commits-list gnome org
- Cc:
- Subject: [evolution-ews] eds-I#12 - vCard REV attribute should be a timestamp
- Date: Thu, 21 Jun 2018 14:02:41 +0000 (UTC)
commit 4c2891e77dc05ac6c1508640a47367361927db64
Author: Milan Crha <mcrha redhat com>
Date: Thu Jun 21 16:04:31 2018 +0200
eds-I#12 - vCard REV attribute should be a timestamp
Closes https://gitlab.gnome.org/GNOME/evolution-data-server/issues/12
src/addressbook/e-book-backend-ews.c | 123 ++++++++++++++++++++++++++++-------
src/server/e-ews-item.c | 11 ++++
src/server/e-ews-item.h | 2 +
3 files changed, 113 insertions(+), 23 deletions(-)
---
diff --git a/src/addressbook/e-book-backend-ews.c b/src/addressbook/e-book-backend-ews.c
index ffb06306..ff4c96d1 100644
--- a/src/addressbook/e-book-backend-ews.c
+++ b/src/addressbook/e-book-backend-ews.c
@@ -51,12 +51,23 @@
#include "ews-oab-decoder.h"
#include "ews-oab-decompress.h"
+#ifdef G_OS_WIN32
+#ifdef gmtime_r
+#undef gmtime_r
+#endif
+
+/* The gmtime() in Microsoft's C library is MT-safe */
+#define gmtime_r(tp,tmp) (gmtime(tp)?(*(tmp)=*gmtime(tp),(tmp)):0)
+#endif
+
#define d(x)
#define EDB_ERROR(_code) e_data_book_create_error (E_DATA_BOOK_STATUS_ ## _code, NULL)
#define EDB_ERROR_EX(_code,_msg) e_data_book_create_error (E_DATA_BOOK_STATUS_ ## _code, _msg)
#define X_EWS_ORIGINAL_VCARD "X-EWS-ORIGINAL-VCARD"
+#define X_EWS_CHANGEKEY "X-EWS-CHANGEKEY"
+#define X_EWS_GAL_SHA1 "X-EWS-GAL-SHA1"
#define X_EWS_PHOTO_CHECK_DATE "X-EWS-PHOTO-CHECK-DATE" /* YYYYMMDD of the last check for photo */
#define EWS_MAX_FETCH_COUNT 500
@@ -67,7 +78,7 @@
/* passing field uris for PhysicalAddress, PhoneNumbers causes error, so we
* use Default view to fetch them. Thus the summary props just have attachments
* and some additional properties that are not return with Default view */
-#define CONTACT_ITEM_PROPS "item:Attachments item:HasAttachments item:Body contacts:Manager
contacts:Department contacts:SpouseName contacts:AssistantName contacts:BusinessHomePage contacts:Birthday"
+#define CONTACT_ITEM_PROPS "item:Attachments item:HasAttachments item:Body item:LastModifiedTime
contacts:Manager contacts:Department contacts:SpouseName contacts:AssistantName contacts:BusinessHomePage
contacts:Birthday"
struct _EBookBackendEwsPrivate {
GRecMutex cnc_lock;
@@ -325,6 +336,31 @@ static const struct phone_field_mapping {
{E_CONTACT_PHONE_TTYTDD, "TtyTddPhone"}
};
+static void
+ebews_populate_rev (EContact *contact,
+ EEwsItem *item)
+{
+ struct tm stm;
+ time_t tt = 0;
+ gchar time_string[100] = { 0 };
+
+ g_return_if_fail (E_IS_CONTACT (contact));
+
+ if (item) {
+ g_return_if_fail (E_IS_EWS_ITEM (item));
+
+ tt = e_ews_item_get_last_modified_time (item);
+ }
+
+ if (tt <= 0)
+ tt = time (NULL);
+
+ gmtime_r (&tt, &stm);
+ strftime (time_string, 100, "%Y-%m-%dT%H:%M:%SZ", &stm);
+
+ e_contact_set (contact, E_CONTACT_REV, time_string);
+}
+
static void
ebews_populate_uid (EBookBackendEws *bbews,
EContact *contact,
@@ -337,7 +373,8 @@ ebews_populate_uid (EBookBackendEws *bbews,
id = e_ews_item_get_id (item);
if (id) {
e_contact_set (contact, E_CONTACT_UID, id->id);
- e_contact_set (contact, E_CONTACT_REV, id->change_key);
+ ebews_populate_rev (contact, item);
+ e_vcard_util_set_x_attribute (E_VCARD (contact), X_EWS_CHANGEKEY, id->change_key);
}
}
@@ -1009,7 +1046,9 @@ set_photo (EBookBackendEws *bbews,
if (!item_id) {
id = g_new0 (EwsId, 1);
id->id = e_contact_get (contact, E_CONTACT_UID);
- id->change_key = e_contact_get (contact, E_CONTACT_REV);
+ id->change_key = e_vcard_util_dup_x_attribute (E_VCARD (contact), X_EWS_CHANGEKEY);
+ if (!id->change_key)
+ id->change_key = e_contact_get (contact, E_CONTACT_REV);
item_id = id;
}
@@ -1598,16 +1637,25 @@ ebb_ews_convert_dl_to_updatexml_cb (ESoapMessage *msg,
ConvertData *cd = user_data;
EContact *old_contact = cd->old_contact;
EContact *new_contact = cd->new_contact;
+ gchar *change_key = NULL;
+
+ if (!cd->change_key) {
+ change_key = e_vcard_util_dup_x_attribute (E_VCARD (old_contact), X_EWS_CHANGEKEY);
+ if (!change_key)
+ change_key = e_contact_get (old_contact, E_CONTACT_REV);
+ }
e_ews_message_start_item_change (msg, E_EWS_ITEMCHANGE_TYPE_ITEM,
e_contact_get_const (old_contact, E_CONTACT_UID),
- cd->change_key ? cd->change_key : e_contact_get_const (old_contact, E_CONTACT_REV),
+ cd->change_key ? cd->change_key : change_key,
0);
e_ews_message_start_set_item_field (msg, "Members", "distributionlist", "DistributionList");
ebb_ews_write_dl_members (msg, new_contact);
e_ews_message_end_set_item_field (msg);
e_ews_message_end_item_change (msg);
+ g_free (change_key);
+
return TRUE;
}
@@ -1620,6 +1668,7 @@ ebb_ews_convert_contact_to_updatexml_cb (ESoapMessage *msg,
EContact *old_contact = cd->old_contact;
EContact *new_contact = cd->new_contact;
gchar *value = NULL, *old_value = NULL;
+ gchar *change_key = NULL;
gint i, element_type;
/* Pre-flight, to update the ChangeKey if needed */
@@ -1641,9 +1690,15 @@ ebb_ews_convert_contact_to_updatexml_cb (ESoapMessage *msg,
}
}
+ if (!cd->change_key) {
+ change_key = e_vcard_util_dup_x_attribute (E_VCARD (old_contact), X_EWS_CHANGEKEY);
+ if (!change_key)
+ change_key = e_contact_get (old_contact, E_CONTACT_REV);
+ }
+
e_ews_message_start_item_change (msg, E_EWS_ITEMCHANGE_TYPE_ITEM,
e_contact_get_const (old_contact, E_CONTACT_UID),
- cd->change_key ? cd->change_key : e_contact_get_const (old_contact, E_CONTACT_REV),
+ cd->change_key ? cd->change_key : change_key,
0);
/* Iterate for each field in contact */
@@ -1676,6 +1731,8 @@ ebb_ews_convert_contact_to_updatexml_cb (ESoapMessage *msg,
e_ews_message_end_item_change (msg);
+ g_free (change_key);
+
return TRUE;
}
@@ -1841,6 +1898,7 @@ ebb_ews_traverse_dl (EBookBackendEws *bbews,
static EContact *
ebb_ews_get_dl_info (EBookBackendEws *bbews,
+ EEwsItem *item,
const EwsId *id,
const gchar *d_name,
GSList *members,
@@ -1853,7 +1911,8 @@ ebb_ews_get_dl_info (EBookBackendEws *bbews,
contact = e_contact_new ();
e_contact_set (contact, E_CONTACT_UID, id->id);
- e_contact_set (contact, E_CONTACT_REV, id->change_key);
+ e_vcard_util_set_x_attribute (E_VCARD (contact), X_EWS_CHANGEKEY, id->change_key);
+ ebews_populate_rev (contact, item);
e_contact_set (contact, E_CONTACT_IS_LIST, GINT_TO_POINTER (TRUE));
e_contact_set (contact, E_CONTACT_LIST_SHOW_ADDRESSES, GINT_TO_POINTER (TRUE));
@@ -1905,6 +1964,7 @@ ebb_ews_get_dl_info_gal (EBookBackendEws *bbews,
static gboolean
ebb_ews_contacts_append_dl (EBookBackendEws *bbews,
+ EEwsItem *item,
const EwsId *id,
const gchar *d_name,
GSList *members,
@@ -1915,7 +1975,7 @@ ebb_ews_contacts_append_dl (EBookBackendEws *bbews,
EContact *contact;
EVCardAttribute *attr;
- contact = ebb_ews_get_dl_info (bbews, id, d_name, members, cancellable, error);
+ contact = ebb_ews_get_dl_info (bbews, item, id, d_name, members, cancellable, error);
if (contact == NULL)
return FALSE;
@@ -2003,7 +2063,7 @@ ebb_ews_fetch_items_sync (EBookBackendEws *bbews,
if (e_ews_connection_expand_dl_sync (
bbews->priv->cnc, EWS_PRIORITY_MEDIUM, mb, &members,
&includes_last, cancellable, &local_error)) {
- ret = ebb_ews_contacts_append_dl (bbews, id, d_name, members, contacts, cancellable,
error);
+ ret = ebb_ews_contacts_append_dl (bbews, item, id, d_name, members, contacts,
cancellable, error);
g_slist_free_full (members, (GDestroyNotify) e_ews_mailbox_free);
} else {
ret = g_error_matches (local_error, EWS_CONNECTION_ERROR,
EWS_CONNECTION_ERROR_NAMERESOLUTIONNORESULTS);
@@ -2389,7 +2449,8 @@ ebb_ews_gal_store_contact (EContact *contact,
const gchar *uid = e_contact_get_const (contact, E_CONTACT_UID);
EBookMetaBackendInfo *nfo;
- e_contact_set (contact, E_CONTACT_REV, sha1);
+ ebews_populate_rev (contact, NULL);
+ e_vcard_util_set_x_attribute (E_VCARD (contact), X_EWS_GAL_SHA1, sha1);
if (data->fetch_gal_photos && !g_cancellable_is_cancelled (cancellable)) {
GError *local_error = NULL;
@@ -2404,7 +2465,7 @@ ebb_ews_gal_store_contact (EContact *contact,
g_clear_error (&local_error);
}
- nfo = e_book_meta_backend_info_new (uid, sha1, NULL, NULL);
+ nfo = e_book_meta_backend_info_new (uid, e_contact_get_const (contact, E_CONTACT_REV), NULL,
NULL);
nfo->object = e_vcard_to_string (E_VCARD (contact), EVC_FORMAT_VCARD_30);
if (g_hash_table_remove (data->uids, uid)) {
@@ -2434,18 +2495,27 @@ ebb_ews_gather_existing_uids_cb (EBookCache *book_cache,
gpointer user_data)
{
struct _db_data *data = user_data;
- gchar *dup_uid, *dup_rev;
+ EVCard *vcard;
+ gchar *dup_uid, *dup_sha1 = NULL;
g_return_val_if_fail (data != NULL, FALSE);
g_return_val_if_fail (data->uids != NULL, FALSE);
g_return_val_if_fail (data->sha1s != NULL, FALSE);
+ g_return_val_if_fail (object != NULL, FALSE);
+
+ vcard = e_vcard_new_from_string (object);
+ if (vcard) {
+ dup_sha1 = e_vcard_util_dup_x_attribute (vcard, X_EWS_GAL_SHA1);
+ g_object_unref (vcard);
+ }
dup_uid = g_strdup (uid);
- dup_rev = g_strdup (revision);
+ if (!dup_sha1)
+ dup_sha1 = g_strdup (revision);
- g_hash_table_insert (data->uids, dup_uid, dup_rev);
- if (dup_rev)
- g_hash_table_insert (data->sha1s, dup_rev, dup_uid);
+ g_hash_table_insert (data->uids, dup_uid, dup_sha1);
+ if (dup_sha1)
+ g_hash_table_insert (data->sha1s, dup_sha1, dup_uid);
return TRUE;
}
@@ -2797,7 +2867,6 @@ ebb_ews_update_cache_for_expression (EBookBackendEws *bbews,
EContact *contact = NULL, *old_contact = NULL;
gboolean is_public_dl = FALSE, mailbox_address_set = FALSE;
const gchar *str;
- gchar *fake_rev;
if (g_strcmp0 (mb->mailbox_type, "PublicDL") == 0) {
contact = e_contact_new ();
@@ -2820,11 +2889,7 @@ ebb_ews_update_cache_for_expression (EBookBackendEws *bbews,
/* There is no ChangeKey provided either, thus make up some revision,
to have the contact always updated in the local cache. */
- fake_rev = e_util_generate_uid ();
-
- e_contact_set (contact, E_CONTACT_REV, fake_rev);
-
- g_free (fake_rev);
+ ebews_populate_rev (contact, NULL);
if (use_primary_address && !is_public_dl && mb->email &&
(!mb->routing_type || g_ascii_strcasecmp (mb->routing_type, "SMTP") ==
0)) {
@@ -2962,8 +3027,20 @@ ebb_ews_verify_changes (EBookCache *book_cache,
EContact *existing = NULL;
if (e_book_cache_get_contact (book_cache, id->id, TRUE, &existing, cancellable, NULL)
&&
- existing && g_strcmp0 (e_contact_get_const (existing, E_CONTACT_REV),
id->change_key) == 0) {
- g_object_unref (item);
+ existing) {
+ gchar *change_key;
+
+ change_key = e_vcard_util_dup_x_attribute (E_VCARD (existing),
X_EWS_CHANGEKEY);
+ if (!change_key)
+ change_key = e_contact_get (existing, E_CONTACT_REV);
+
+ if (g_strcmp0 (change_key, id->change_key) == 0) {
+ g_object_unref (item);
+ } else {
+ items = g_slist_prepend (items, item);
+ }
+
+ g_free (change_key);
} else {
items = g_slist_prepend (items, item);
}
diff --git a/src/server/e-ews-item.c b/src/server/e-ews-item.c
index 3caedbe8..5b322780 100644
--- a/src/server/e-ews-item.c
+++ b/src/server/e-ews-item.c
@@ -114,6 +114,7 @@ struct _EEwsItemPrivate {
time_t date_received;
time_t date_sent;
time_t date_created;
+ time_t last_modified_time;
gsize size;
gchar *msg_id;
@@ -1553,6 +1554,8 @@ e_ews_item_set_from_soap_parameter (EEwsItem *item,
priv->date_sent = ews_item_parse_date (subparam);
} else if (!g_ascii_strcasecmp (name, "DateTimeCreated")) {
priv->date_created = ews_item_parse_date (subparam);
+ } else if (!g_ascii_strcasecmp (name, "LastModifiedTime")) {
+ priv->last_modified_time = ews_item_parse_date (subparam);
} else if (!g_ascii_strcasecmp (name, "HasAttachments")) {
value = e_soap_parameter_get_string_value (subparam);
priv->has_attachments = (!g_ascii_strcasecmp (value, "true"));
@@ -1853,6 +1856,14 @@ e_ews_item_get_date_created (EEwsItem *item)
return item->priv->date_created;
}
+time_t
+e_ews_item_get_last_modified_time (EEwsItem *item)
+{
+ g_return_val_if_fail (E_IS_EWS_ITEM (item), -1);
+
+ return item->priv->last_modified_time;
+}
+
gboolean
e_ews_item_has_attachments (EEwsItem *item,
gboolean *has_attachments)
diff --git a/src/server/e-ews-item.h b/src/server/e-ews-item.h
index 9c4f8c85..8c4b126a 100644
--- a/src/server/e-ews-item.h
+++ b/src/server/e-ews-item.h
@@ -277,6 +277,8 @@ const gchar * e_ews_item_get_date_header (EEwsItem *item);
time_t e_ews_item_get_date_received (EEwsItem *item);
time_t e_ews_item_get_date_sent (EEwsItem *item);
time_t e_ews_item_get_date_created (EEwsItem *item);
+time_t e_ews_item_get_last_modified_time
+ (EEwsItem *item);
gboolean e_ews_item_has_attachments (EEwsItem *item,
gboolean *has_attachments);
gboolean e_ews_item_is_read (EEwsItem *item,
[
Date Prev][
Date Next] [
Thread Prev][
Thread Next]
[
Thread Index]
[
Date Index]
[
Author Index]