/*
* e-gravatar-photo-source.c
*
* This program is free software; you can redistribute it and/or modify it
* under the terms of the GNU Lesser General Public License as published by
* the Free Software Foundation.
*
* This program is distributed in the hope that it will be useful, but
* WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY
* or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License
* for more details.
*
* You should have received a copy of the GNU Lesser General Public License
* along with this program; if not, see .
*
*/
#include "e-gravatar-photo-source.h"
#include
#include
#define E_GRAVATAR_PHOTO_SOURCE_GET_PRIVATE(obj) \
(G_TYPE_INSTANCE_GET_PRIVATE \
((obj), E_TYPE_GRAVATAR_PHOTO_SOURCE, EGravatarPhotoSourcePrivate))
#define AVATAR_BASE_URI "http://www.gravatar.com/avatar/"
struct _EGravatarPhotoSourcePrivate
{
gboolean enabled;
};
enum {
PROP_0,
PROP_ENABLED
};
typedef struct _AsyncContext AsyncContext;
struct _AsyncContext {
gchar *email_address;
GInputStream *stream;
};
/* Forward Declarations */
static void e_gravatar_photo_source_interface_init
(EPhotoSourceInterface *iface);
G_DEFINE_DYNAMIC_TYPE_EXTENDED (
EGravatarPhotoSource,
e_gravatar_photo_source,
G_TYPE_OBJECT,
0,
G_IMPLEMENT_INTERFACE_DYNAMIC (
E_TYPE_PHOTO_SOURCE,
e_gravatar_photo_source_interface_init))
static void
async_context_free (AsyncContext *async_context)
{
g_free (async_context->email_address);
g_clear_object (&async_context->stream);
g_slice_free (AsyncContext, async_context);
}
static void
gravatar_photo_source_get_photo_thread (GSimpleAsyncResult *simple,
GObject *source_object,
GCancellable *cancellable)
{
AsyncContext *async_context;
SoupRequest *request;
SoupSession *session;
GInputStream *stream = NULL;
gchar *hash;
gchar *uri;
GError *local_error = NULL;
g_return_if_fail (E_IS_GRAVATAR_PHOTO_SOURCE (source_object));
if (!e_gravatar_photo_source_get_enabled (E_GRAVATAR_PHOTO_SOURCE (source_object)))
return;
async_context = g_simple_async_result_get_op_res_gpointer (simple);
hash = e_gravatar_get_hash (async_context->email_address);
uri = g_strdup_printf ("%s%s?d=404", AVATAR_BASE_URI, hash);
g_debug ("Requesting avatar for %s", async_context->email_address);
g_debug ("%s", uri);
session = soup_session_new ();
/* We control the URI so there should be no error. */
request = soup_session_request (session, uri, NULL);
g_return_if_fail (request != NULL);
stream = soup_request_send (request, cancellable, &local_error);
/* Sanity check. */
g_return_if_fail (
((stream != NULL) && (local_error == NULL)) ||
((stream == NULL) && (local_error != NULL)));
/* XXX soup_request_send() returns a stream on HTTP errors.
* We need to check the status code on the SoupMessage
* to make sure the we're not getting an error message. */
if (stream != NULL) {
SoupMessage *message;
message = soup_request_http_get_message (
SOUP_REQUEST_HTTP (request));
if (SOUP_STATUS_IS_SUCCESSFUL (message->status_code)) {
async_context->stream = g_object_ref (stream);
} else if (message->status_code != SOUP_STATUS_NOT_FOUND) {
local_error = g_error_new_literal (
SOUP_HTTP_ERROR,
message->status_code,
message->reason_phrase);
}
g_object_unref (message);
g_object_unref (stream);
}
if (local_error != NULL) {
const gchar *domain;
domain = g_quark_to_string (local_error->domain);
g_debug ("Error: %s (%s)", local_error->message, domain);
g_simple_async_result_take_error (simple, local_error);
}
g_debug ("Request complete");
g_clear_object (&request);
g_clear_object (&session);
g_free (hash);
g_free (uri);
}
static void
gravatar_photo_source_get_photo (EPhotoSource *photo_source,
const gchar *email_address,
GCancellable *cancellable,
GAsyncReadyCallback callback,
gpointer user_data)
{
GSimpleAsyncResult *simple;
AsyncContext *async_context;
async_context = g_slice_new0 (AsyncContext);
async_context->email_address = g_strdup (email_address);
simple = g_simple_async_result_new (
G_OBJECT (photo_source), callback,
user_data, gravatar_photo_source_get_photo);
g_simple_async_result_set_check_cancellable (simple, cancellable);
g_simple_async_result_set_op_res_gpointer (
simple, async_context, (GDestroyNotify) async_context_free);
g_simple_async_result_run_in_thread (
simple, gravatar_photo_source_get_photo_thread,
G_PRIORITY_DEFAULT, cancellable);
g_object_unref (simple);
}
static gboolean
gravatar_photo_source_get_photo_finish (EPhotoSource *photo_source,
GAsyncResult *result,
GInputStream **out_stream,
gint *out_priority,
GError **error)
{
GSimpleAsyncResult *simple;
AsyncContext *async_context;
g_return_val_if_fail (
g_simple_async_result_is_valid (
result, G_OBJECT (photo_source),
gravatar_photo_source_get_photo), FALSE);
simple = G_SIMPLE_ASYNC_RESULT (result);
async_context = g_simple_async_result_get_op_res_gpointer (simple);
if (g_simple_async_result_propagate_error (simple, error))
return FALSE;
if (async_context->stream != NULL) {
*out_stream = g_object_ref (async_context->stream);
if (out_priority != NULL)
*out_priority = G_PRIORITY_DEFAULT;
} else {
*out_stream = NULL;
}
return TRUE;
}
static void
gravatar_photo_source_set_property (GObject *object,
guint property_id,
const GValue *value,
GParamSpec *pspec)
{
switch (property_id) {
case PROP_ENABLED:
e_gravatar_photo_source_set_enabled (
E_GRAVATAR_PHOTO_SOURCE (object),
g_value_get_boolean (value));
return;
}
G_OBJECT_WARN_INVALID_PROPERTY_ID (object, property_id, pspec);
}
static void
gravatar_photo_source_get_property (GObject *object,
guint property_id,
GValue *value,
GParamSpec *pspec)
{
switch (property_id) {
case PROP_ENABLED:
g_value_set_boolean (
value,
e_gravatar_photo_source_get_enabled (
E_GRAVATAR_PHOTO_SOURCE (object)));
return;
}
G_OBJECT_WARN_INVALID_PROPERTY_ID (object, property_id, pspec);
}
static void
e_gravatar_photo_source_class_init (EGravatarPhotoSourceClass *class)
{
GObjectClass *object_class;
g_type_class_add_private (class, sizeof (EGravatarPhotoSourcePrivate));
object_class = G_OBJECT_CLASS (class);
object_class->set_property = gravatar_photo_source_set_property;
object_class->get_property = gravatar_photo_source_get_property;
g_object_class_install_property (
object_class,
PROP_ENABLED,
g_param_spec_boolean (
"enabled",
"Enabled",
"Whether can search for contact photos",
FALSE,
G_PARAM_READWRITE |
G_PARAM_STATIC_STRINGS));
}
static void
e_gravatar_photo_source_class_finalize (EGravatarPhotoSourceClass *class)
{
}
static void
e_gravatar_photo_source_interface_init (EPhotoSourceInterface *iface)
{
iface->get_photo = gravatar_photo_source_get_photo;
iface->get_photo_finish = gravatar_photo_source_get_photo_finish;
}
static void
e_gravatar_photo_source_init (EGravatarPhotoSource *photo_source)
{
GSettings *settings;
photo_source->priv = E_GRAVATAR_PHOTO_SOURCE_GET_PRIVATE (photo_source);
settings = g_settings_new ("org.gnome.evolution.mail");
g_settings_bind (settings, "search-gravatar-for-photo",
photo_source, "enabled",
G_SETTINGS_BIND_DEFAULT | G_SETTINGS_BIND_NO_SENSITIVITY);
g_object_unref (settings);
}
void
e_gravatar_photo_source_type_register (GTypeModule *type_module)
{
/* XXX G_DEFINE_DYNAMIC_TYPE declares a static type registration
* function, so we have to wrap it with a public function in
* order to register types from a separate compilation unit. */
e_gravatar_photo_source_register_type (type_module);
}
EPhotoSource *
e_gravatar_photo_source_new (void)
{
return g_object_new (E_TYPE_GRAVATAR_PHOTO_SOURCE, NULL);
}
gchar *
e_gravatar_get_hash (const gchar *email_address)
{
gchar *string;
gchar *hash;
g_return_val_if_fail (email_address != NULL, NULL);
g_return_val_if_fail (g_utf8_validate (email_address, -1, NULL), NULL);
string = g_strstrip (g_utf8_strdown (email_address, -1));
hash = g_compute_checksum_for_string (G_CHECKSUM_MD5, string, -1);
g_free (string);
return hash;
}
gboolean
e_gravatar_photo_source_get_enabled (EGravatarPhotoSource *photo_source)
{
g_return_val_if_fail (E_IS_GRAVATAR_PHOTO_SOURCE (photo_source), FALSE);
return photo_source->priv->enabled;
}
void
e_gravatar_photo_source_set_enabled (EGravatarPhotoSource *photo_source,
gboolean enabled)
{
g_return_if_fail (E_IS_GRAVATAR_PHOTO_SOURCE (photo_source));
if ((photo_source->priv->enabled ? 1 : 0) == (enabled ? 1 : 0))
return;
photo_source->priv->enabled = enabled;
g_object_notify (G_OBJECT (photo_source), "enabled");
}