[evolution/webkit: 125/177] Make the image caching more robust
- From: Dan VrÃtil <dvratil src gnome org>
- To: commits-list gnome org
- Cc:
- Subject: [evolution/webkit: 125/177] Make the image caching more robust
- Date: Fri, 2 Mar 2012 11:40:09 +0000 (UTC)
commit e1ce466f6bc6b4176016b1ddf200561f6e95bd2d
Author: Dan VrÃtil <dvratil redhat com>
Date: Mon Jan 23 15:41:19 2012 +0100
Make the image caching more robust
- be more carefull about NULL pointers
- use MD5 sums of the URI as cache filename instead of the URI itself
(the URI is sometimes too long for a filename and thus it couldn't be cached)
mail/e-mail-request.c | 75 ++++++++++++++++++++++++++++++++++++------------
1 files changed, 56 insertions(+), 19 deletions(-)
---
diff --git a/mail/e-mail-request.c b/mail/e-mail-request.c
index 153bea1..c274faf 100644
--- a/mail/e-mail-request.c
+++ b/mail/e-mail-request.c
@@ -134,7 +134,7 @@ struct http_request_async_data {
GMainLoop *loop;
GCancellable *cancellable;
CamelDataCache *cache;
- gchar *uri;
+ gchar *cache_key;
GInputStream *stream;
CamelStream *cache_stream;
@@ -159,25 +159,38 @@ http_request_write_to_cache (GInputStream *stream,
if (len == -1) {
g_message ("Error while reading input stream: %s",
error ? error->message : "Unknown error");
- if (error)
- g_error_free (error);
+ g_clear_error (&error);
g_main_loop_quit(data->loop);
- g_free (data->buff);
+
+ if (data->buff)
+ g_free (data->buff);
/* Don't keep broken data in cache */
- camel_data_cache_remove (data->cache, "http", data->uri, NULL);
+ camel_data_cache_remove (data->cache, "http", data->cache_key, NULL);
return;
}
/* EOF */
if (len == 0) {
camel_stream_close (data->cache_stream, data->cancellable, NULL);
- g_free (data->buff);
- g_main_loop_quit (data->loop);
+
+ if (data->buff)
+ g_free (data->buff);
+
+ g_main_loop_quit (data->loop);
return;
}
+ if (!data->cache_stream) {
+
+ if (data->buff)
+ g_free (data->buff);
+
+ g_main_loop_quit (data->loop);
+ return;
+ }
+
/* Write chunk to cache and read another block of data. */
camel_stream_write (data->cache_stream, data->buff, len,
data->cancellable, NULL);
@@ -199,8 +212,8 @@ http_request_finished (SoupRequest *request,
data->stream = soup_request_send_finish (request, res, &error);
if (!data->stream) {
- g_warning("HTTP request failed: %s", error->message);
- g_error_free (error);
+ g_warning("HTTP request failed: %s", error ? error->message: "Unknown error");
+ g_clear_error (&error);
g_main_loop_quit (data->loop);
return;
}
@@ -218,6 +231,11 @@ http_request_finished (SoupRequest *request,
data->content_length = soup_request_get_content_length (request);
data->content_type = g_strdup (soup_request_get_content_type (request));
+ if (!data->cache_stream) {
+ g_main_loop_quit (data->loop);
+ return;
+ }
+
data->buff = g_malloc (4096);
g_input_stream_read_async (data->stream, data->buff, 4096,
G_PRIORITY_DEFAULT, data->cancellable,
@@ -235,6 +253,7 @@ handle_http_request (GSimpleAsyncResult *res,
gchar *evo_uri, *uri;
GInputStream *stream;
gboolean force_load_images = FALSE;
+ gchar *uri_md5;
const gchar *user_cache_dir;
CamelDataCache *cache;
@@ -267,6 +286,11 @@ handle_http_request (GSimpleAsyncResult *res,
g_return_if_fail (uri && *uri);
+ /* Use MD5 hash of the URI as a filname of the resourec cache file.
+ * We were previously using the URI as a filename but the URI is
+ * sometimes too long for a filename. */
+ uri_md5 = g_compute_checksum_for_string (G_CHECKSUM_MD5, uri, -1);
+
/* Open Evolution's cache */
user_cache_dir = e_get_user_cache_dir ();
cache = camel_data_cache_new (user_cache_dir, NULL);
@@ -276,7 +300,7 @@ handle_http_request (GSimpleAsyncResult *res,
}
/* Found item in cache! */
- cache_stream = camel_data_cache_get (cache, "http", uri, NULL);
+ cache_stream = camel_data_cache_get (cache, "http", uri_md5, NULL);
if (cache_stream) {
stream = g_memory_input_stream_new ();
@@ -304,10 +328,11 @@ handle_http_request (GSimpleAsyncResult *res,
GFileInfo *info;
gchar *path;
- path = camel_data_cache_get_filename (cache, "http", uri, NULL);
+ path = camel_data_cache_get_filename (cache, "http", uri_md5, NULL);
file = g_file_new_for_path (path);
info = g_file_query_info (file, G_FILE_ATTRIBUTE_STANDARD_CONTENT_TYPE,
- 0, cancellable, NULL);
+ 0, cancellable, NULL);
+
request->priv->mime_type = g_strdup (
g_file_info_get_content_type (info));
@@ -336,6 +361,7 @@ handle_http_request (GSimpleAsyncResult *res,
SoupRequest *http_request;
SoupSession *session;
GMainContext *context;
+ GError *error;
struct http_request_async_data data;
@@ -348,11 +374,18 @@ handle_http_request (GSimpleAsyncResult *res,
http_request = soup_requester_request (requester, uri, NULL);
+ error = NULL;
data.loop = g_main_loop_new (context, TRUE);
data.cancellable = cancellable;
data.cache = cache;
- data.uri = uri;
- data.cache_stream = camel_data_cache_add (cache, "http", uri, NULL);
+ data.cache_key = uri_md5;
+ data.cache_stream = camel_data_cache_add (cache, "http", uri_md5, &error);
+
+ if (!data.cache_stream) {
+ g_warning ("Failed to create cache file for '%s': %s",
+ uri, error ? error->message : "Unknown error");
+ g_clear_error (&error);
+ }
/* Send the request and waint in mainloop until it's finished
* and copied to cache */
@@ -409,6 +442,7 @@ handle_http_request (GSimpleAsyncResult *res,
cleanup:
g_free (uri);
+ g_free (uri_md5);
}
static void
@@ -527,10 +561,10 @@ mail_request_send_async (SoupRequest *request,
return;
- /* For http and https requests we have this evo-http(s) protocol. We first
- * try to lookup the data in local cache and when not found, we send standard
- * http(s) request to fetch them. But only when image loading policy allows us.
- */
+ /* For http and https requests we have this evo-http(s) protocol.
+ * We first try to lookup the data in local cache and when not found,
+ * we send standard http(s) request to fetch them. But only when image
+ * loading policy allows us. */
} else if ((g_strcmp0 (uri->scheme, "evo-http") == 0) ||
(g_strcmp0 (uri->scheme, "evo-https") == 0)) {
@@ -565,7 +599,10 @@ mail_request_send_finish (SoupRequest *request,
g_object_unref (result);
/* Reset the stream before passing it back to webkit */
- g_seekable_seek (G_SEEKABLE (stream), 0, G_SEEK_SET, NULL, NULL);
+ if (stream)
+ g_seekable_seek (G_SEEKABLE (stream), 0, G_SEEK_SET, NULL, NULL);
+ else /* We must always return something */
+ stream = g_memory_input_stream_new ();
return stream;
}
[
Date Prev][
Date Next] [
Thread Prev][
Thread Next]
[
Thread Index]
[
Date Index]
[
Author Index]