[evolution-ews] I#163 - GAL: Postpone user photo download
- From: Milan Crha <mcrha src gnome org>
- To: commits-list gnome org
- Cc:
- Subject: [evolution-ews] I#163 - GAL: Postpone user photo download
- Date: Fri, 17 Sep 2021 10:36:53 +0000 (UTC)
commit b72976825fb2f2c81a30df719d21d3cbcd4e0e83
Author: Milan Crha <mcrha redhat com>
Date: Fri Sep 17 12:33:29 2021 +0200
I#163 - GAL: Postpone user photo download
Closes https://gitlab.gnome.org/GNOME/evolution-ews/-/issues/163
src/EWS/addressbook/e-book-backend-ews.c | 205 +++++++++++++++++++++++++------
1 file changed, 165 insertions(+), 40 deletions(-)
---
diff --git a/src/EWS/addressbook/e-book-backend-ews.c b/src/EWS/addressbook/e-book-backend-ews.c
index aacfd71d..d2b8b100 100644
--- a/src/EWS/addressbook/e-book-backend-ews.c
+++ b/src/EWS/addressbook/e-book-backend-ews.c
@@ -95,6 +95,7 @@ struct _EBookBackendEwsPrivate {
gchar *folder_id;
gboolean is_gal;
+ gboolean fetching_gal_photos;
guint subscription_key;
@@ -2743,15 +2744,164 @@ ebb_ews_fetch_gal_photo_sync (EBookBackendEws *bbews,
return success;
}
+static EBookMetaBackendInfo *ebb_ews_contact_to_info (EContact *contact, gboolean is_gal);
+
+static void
+ebb_ews_fetch_gal_photos_thread (EBookBackend *book_backend,
+ gpointer user_data,
+ GCancellable *cancellable,
+ GError **error)
+{
+ EBookBackendEws *bbews = E_BOOK_BACKEND_EWS (book_backend);
+ EBookCache *book_cache;
+ ESourceEwsFolder *ews_folder;
+ GSList *uids = user_data;
+
+ book_cache = e_book_meta_backend_ref_cache (E_BOOK_META_BACKEND (bbews));
+ if (!book_cache)
+ return;
+
+ ews_folder = e_source_get_extension (e_backend_get_source (E_BACKEND (bbews)),
E_SOURCE_EXTENSION_EWS_FOLDER);
+ if (e_source_ews_folder_get_fetch_gal_photos (ews_folder)) {
+ g_rec_mutex_lock (&bbews->priv->cnc_lock);
+
+ if (!bbews->priv->fetching_gal_photos && bbews->priv->cnc &&
+ e_ews_connection_satisfies_server_version (bbews->priv->cnc, E_EWS_EXCHANGE_2013)) {
+ guint counter = 0, percent;
+ gdouble total_count, current_count;
+ GSList *link, *modified = NULL;
+
+ bbews->priv->fetching_gal_photos = TRUE;
+
+ g_rec_mutex_unlock (&bbews->priv->cnc_lock);
+
+ percent = (guint) -1;
+ current_count = 0;
+ total_count = g_slist_length (uids);
+
+ /* Recheck the fetch-gal-photo property every time, in case the user changed it
+ while the fetching had been ongoing. */
+ for (link = uids;
+ link && !g_cancellable_is_cancelled (cancellable) &&
e_source_ews_folder_get_fetch_gal_photos (ews_folder);
+ current_count++, link = g_slist_next (link)) {
+ const gchar *uid = link->data;
+ guint new_percent;
+ EContact *contact = NULL;
+ EBookMetaBackendInfo *nfo;
+ GError *local_error = NULL;
+
+ new_percent = (guint) ((current_count / total_count) * 100.0);
+ if (new_percent != percent) {
+ percent = new_percent;
+ e_book_backend_foreach_view_notify_progress (book_backend, TRUE,
(gint) percent, _("Fetching contact photos…"));
+ }
+
+ if (!e_book_cache_get_contact (book_cache, uid, FALSE, &contact, cancellable,
NULL) ||
+ !contact || e_vcard_get_attribute (E_VCARD (contact), EVC_PHOTO) ||
+ !ebb_ews_can_check_user_photo (contact)) {
+ g_clear_object (&contact);
+ continue;
+ }
+
+ if (!ebb_ews_fetch_gal_photo_sync (bbews, contact, cancellable, &local_error))
+ ebb_ews_store_photo_check_date (contact, NULL);
+
+ nfo = ebb_ews_contact_to_info (contact, bbews->priv->is_gal);
+ if (nfo) {
+ modified = g_slist_prepend (modified, nfo);
+ counter++;
+ }
+
+ g_clear_object (&contact);
+
+ /* Stop immediately when the server is busy */
+ if (g_error_matches (local_error, EWS_CONNECTION_ERROR,
EWS_CONNECTION_ERROR_SERVERBUSY)) {
+ g_clear_error (&local_error);
+ break;
+ }
+
+ g_clear_error (&local_error);
+
+ if (counter == 100) {
+ counter = 0;
+ e_book_meta_backend_process_changes_sync (E_BOOK_META_BACKEND
(bbews), NULL, modified, NULL, cancellable, NULL);
+ g_slist_free_full (modified, e_book_meta_backend_info_free);
+ modified = NULL;
+ }
+
+ /* Give a chance to other threads to work with the connection, because the
photos
+ are not that important like the other operations. */
+ g_thread_yield ();
+ g_usleep (G_USEC_PER_SEC / 4);
+ g_thread_yield ();
+ }
+
+ if (modified) {
+ e_book_meta_backend_process_changes_sync (E_BOOK_META_BACKEND (bbews), NULL,
modified, NULL, cancellable, NULL);
+ g_slist_free_full (modified, e_book_meta_backend_info_free);
+ }
+
+ g_rec_mutex_lock (&bbews->priv->cnc_lock);
+ bbews->priv->fetching_gal_photos = FALSE;
+ g_rec_mutex_unlock (&bbews->priv->cnc_lock);
+
+ if (percent != (guint) -1)
+ e_book_backend_foreach_view_notify_progress (book_backend, TRUE, -1, NULL);
+ } else {
+ g_rec_mutex_unlock (&bbews->priv->cnc_lock);
+ }
+ }
+
+ g_object_unref (book_cache);
+}
+
+static void
+ebb_ews_free_string_slist (gpointer ptr)
+{
+ GSList *list = ptr;
+
+ g_slist_free_full (list, g_free);
+}
+
+static void
+ebb_ews_maybe_schedule_fetch_gal_photos (EBookBackendEws *bbews,
+ GSList **inout_uids) /* gchar * */
+{
+ ESourceEwsFolder *ews_folder;
+
+ if (!inout_uids || !*inout_uids)
+ return;
+
+ ews_folder = e_source_get_extension (e_backend_get_source (E_BACKEND (bbews)),
E_SOURCE_EXTENSION_EWS_FOLDER);
+
+ if (e_source_ews_folder_get_fetch_gal_photos (ews_folder)) {
+ gboolean can_do_it;
+
+ g_rec_mutex_lock (&bbews->priv->cnc_lock);
+ can_do_it = !bbews->priv->fetching_gal_photos && bbews->priv->cnc &&
+ e_ews_connection_satisfies_server_version (bbews->priv->cnc, E_EWS_EXCHANGE_2013);
+ g_rec_mutex_unlock (&bbews->priv->cnc_lock);
+
+ if (can_do_it) {
+ e_book_backend_schedule_custom_operation (E_BOOK_BACKEND (bbews), NULL,
+ ebb_ews_fetch_gal_photos_thread,
+ *inout_uids,
+ ebb_ews_free_string_slist);
+ *inout_uids = NULL;
+ }
+ }
+}
+
struct _db_data {
EBookBackendEws *bbews;
- gboolean fetch_gal_photos;
GHashTable *uids;
GHashTable *sha1s;
gint unchanged;
gint changed;
gint added;
gint percent;
+ gboolean fetch_gal_photos;
+ GSList *fetch_gal_photos_list; /* gchar *uid */
GSList *created_objects;
GSList *modified_objects;
};
@@ -2797,18 +2947,8 @@ ebb_ews_gal_store_contact (EContact *contact,
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;
-
- if (!ebb_ews_fetch_gal_photo_sync (data->bbews, contact, cancellable, &local_error))
- ebb_ews_store_photo_check_date (contact, NULL);
-
- /* The server can kick-off the flood of photo requests, then better stop it */
- if (g_error_matches (local_error, EWS_CONNECTION_ERROR,
EWS_CONNECTION_ERROR_SERVERBUSY))
- data->fetch_gal_photos = FALSE;
-
- g_clear_error (&local_error);
- }
+ if (data->fetch_gal_photos && !g_cancellable_is_cancelled (cancellable))
+ data->fetch_gal_photos_list = g_slist_prepend (data->fetch_gal_photos_list, g_strdup
(uid));
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);
@@ -2896,6 +3036,7 @@ ebb_ews_check_gal_changes (EBookBackendEws *bbews,
data.bbews = bbews;
data.fetch_gal_photos = e_source_ews_folder_get_fetch_gal_photos (ews_folder);
+ data.fetch_gal_photos_list = NULL;
data.created_objects = NULL;
data.modified_objects = NULL;
data.unchanged = data.changed = data.added = 0;
@@ -2926,10 +3067,14 @@ ebb_ews_check_gal_changes (EBookBackendEws *bbews,
*out_removed_objects = g_slist_prepend (*out_removed_objects,
e_book_meta_backend_info_new (uid, NULL, NULL, NULL));
}
+
+ ebb_ews_maybe_schedule_fetch_gal_photos (bbews, &data.fetch_gal_photos_list);
} else {
g_slist_free_full (data.created_objects, e_book_meta_backend_info_free);
g_slist_free_full (data.modified_objects, e_book_meta_backend_info_free);
}
+
+ g_slist_free_full (data.fetch_gal_photos_list, g_free);
} else {
success = FALSE;
}
@@ -4185,42 +4330,22 @@ ebb_ews_search_sync (EBookMetaBackend *meta_backend,
g_rec_mutex_lock (&bbews->priv->cnc_lock);
if (bbews->priv->cnc && e_ews_connection_satisfies_server_version (bbews->priv->cnc,
E_EWS_EXCHANGE_2013)) {
- GSList *link, *modified = NULL;
- gint count = 10;
+ GSList *link, *fetch_gal_photos_list = NULL;
- /* Limit to first 10 without photo, no need to flood the server
- and slow the response too much */
- for (link = *out_contacts; link && count > 0 && !g_cancellable_is_cancelled
(cancellable); link = g_slist_next (link)) {
+ for (link = *out_contacts; link && !g_cancellable_is_cancelled (cancellable);
link = g_slist_next (link)) {
EContact *contact = link->data;
- EBookMetaBackendInfo *nfo;
- GError *local_error = NULL;
if (!contact || e_vcard_get_attribute (E_VCARD (contact), EVC_PHOTO)
||
!ebb_ews_can_check_user_photo (contact))
continue;
- count--;
-
- if (!ebb_ews_fetch_gal_photo_sync (bbews, contact, cancellable,
&local_error))
- ebb_ews_store_photo_check_date (contact, NULL);
-
- nfo = ebb_ews_contact_to_info (contact, bbews->priv->is_gal);
- if (nfo)
- modified = g_slist_prepend (modified, nfo);
-
- /* Stop immediately when the server is busy */
- if (g_error_matches (local_error, EWS_CONNECTION_ERROR,
EWS_CONNECTION_ERROR_SERVERBUSY)) {
- g_clear_error (&local_error);
- break;
- }
-
- g_clear_error (&local_error);
+ fetch_gal_photos_list = g_slist_prepend (fetch_gal_photos_list,
e_contact_get (contact, E_CONTACT_UID));
}
- if (modified) {
- e_book_meta_backend_process_changes_sync (meta_backend, NULL,
modified, NULL, cancellable, NULL);
- g_slist_free_full (modified, e_book_meta_backend_info_free);
- }
+ if (!g_cancellable_is_cancelled (cancellable))
+ ebb_ews_maybe_schedule_fetch_gal_photos (bbews,
&fetch_gal_photos_list);
+
+ g_slist_free_full (fetch_gal_photos_list, g_free);
}
g_rec_mutex_unlock (&bbews->priv->cnc_lock);
[
Date Prev][
Date Next] [
Thread Prev][
Thread Next]
[
Thread Index]
[
Date Index]
[
Author Index]