libpurple/image.c

Fri, 11 Apr 2014 22:53:23 +0200

author
Tomasz Wasilczyk <twasilczyk@pidgin.im>
date
Fri, 11 Apr 2014 22:53:23 +0200
changeset 35830
fb32647ef2f3
parent 35829
268931512478
child 35843
b5d03ca6a680
permissions
-rw-r--r--

PurpleImage: better reference management

/* purple
 *
 * Purple is the legal property of its developers, whose names are too numerous
 * to list here.  Please refer to the COPYRIGHT file distributed with this
 * source distribution.
 *
 * This program is free software; you can redistribute it and/or modify
 * it under the terms of the GNU General Public License as published by
 * the Free Software Foundation; either version 2 of the License, or
 * (at your option) any later version.
 *
 * 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 General Public License
 * along with this program; if not, write to the Free Software
 * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA  02111-1301  USA
 */

#include "internal.h"
#include "glibcompat.h"

#include "debug.h"
#include "image.h"

#define PURPLE_IMAGE_GET_PRIVATE(obj) \
	(G_TYPE_INSTANCE_GET_PRIVATE((obj), PURPLE_TYPE_IMAGE, PurpleImagePrivate))

typedef struct {
	gchar *path;
	GString *contents;

	const gchar *extension;
	const gchar *mime;
	gchar *gen_filename;
	gchar *friendly_filename;

	gboolean is_ready;
	gboolean has_failed;
} PurpleImagePrivate;

enum
{
	PROP_0,
	PROP_IS_READY,
	PROP_HAS_FAILED,
	PROP_LAST
};

enum
{
	SIG_READY,
	SIG_FAILED,
	SIG_LAST
};

static GObjectClass *parent_class;
static guint signals[SIG_LAST];
static GParamSpec *properties[PROP_LAST];

/******************************************************************************
 * Internal logic
 ******************************************************************************/

static void
has_failed(PurpleImage *image)
{
	gboolean ready_changed;
	PurpleImagePrivate *priv;
	priv = PURPLE_IMAGE_GET_PRIVATE(image);

	g_return_if_fail(!priv->has_failed);

	priv->has_failed = TRUE;
	ready_changed = (priv->is_ready != FALSE);
	priv->is_ready = FALSE;

	if (priv->contents) {
		g_string_free(priv->contents, TRUE);
		priv->contents = NULL;
	}

	if (ready_changed) {
		g_object_notify_by_pspec(G_OBJECT(image),
			properties[PROP_IS_READY]);
	}
	g_object_notify_by_pspec(G_OBJECT(image), properties[PROP_HAS_FAILED]);
	g_signal_emit(image, signals[SIG_FAILED], 0);
}

static void
became_ready(PurpleImage *image)
{
	PurpleImagePrivate *priv;
	priv = PURPLE_IMAGE_GET_PRIVATE(image);

	g_return_if_fail(!priv->has_failed);
	g_return_if_fail(!priv->is_ready);

	priv->is_ready = TRUE;

	g_object_notify_by_pspec(G_OBJECT(image), properties[PROP_IS_READY]);
	g_signal_emit(image, signals[SIG_READY], 0);
}

static void
steal_contents(PurpleImagePrivate *priv, gpointer data, gsize length)
{
	g_return_if_fail(priv != NULL);
	g_return_if_fail(priv->contents == NULL);
	g_return_if_fail(data != NULL);
	g_return_if_fail(length > 0);

	priv->contents = g_string_new(NULL);
	g_free(priv->contents->str);
	priv->contents->str = data;
	priv->contents->len = priv->contents->allocated_len = length;
}

static void
fill_data(PurpleImage *image)
{
	PurpleImagePrivate *priv;
	GError *error = NULL;
	gchar *contents;
	gsize length;

	priv = PURPLE_IMAGE_GET_PRIVATE(image);
	if (priv->contents)
		return;

	if (!priv->is_ready)
		return;

	g_return_if_fail(priv->path);
	g_file_get_contents(priv->path, &contents, &length, &error);
	if (error) {
		purple_debug_error("image", "failed to read '%s' image: %s",
			priv->path, error->message);
		g_error_free(error);

		has_failed(image);
		return;
	}

	steal_contents(priv, contents, length);
}


/******************************************************************************
 * API implementation
 ******************************************************************************/

PurpleImage *
purple_image_new_from_file(const gchar *path, gboolean be_eager)
{
	PurpleImagePrivate *priv;
	PurpleImage *img;

	g_return_val_if_fail(path != NULL, NULL);
	g_return_val_if_fail(g_file_test(path, G_FILE_TEST_EXISTS), NULL);

	img = g_object_new(PURPLE_TYPE_IMAGE, NULL);
	purple_image_set_friendly_filename(img, path);
	priv = PURPLE_IMAGE_GET_PRIVATE(img);
	priv->path = g_strdup(path);

	if (be_eager) {
		fill_data(img);
		if (!priv->contents) {
			g_object_unref(img);
			return NULL;
		}

		g_assert(priv->is_ready && !priv->has_failed);
	}

	return img;
}

PurpleImage *
purple_image_new_from_data(gpointer data, gsize length)
{
	PurpleImage *img;
	PurpleImagePrivate *priv;

	g_return_val_if_fail(data != NULL, NULL);
	g_return_val_if_fail(length > 0, NULL);

	img = g_object_new(PURPLE_TYPE_IMAGE, NULL);
	priv = PURPLE_IMAGE_GET_PRIVATE(img);

	steal_contents(priv, data, length);

	return img;
}

gboolean
purple_image_save(PurpleImage *image, const gchar *path)
{
	PurpleImagePrivate *priv = PURPLE_IMAGE_GET_PRIVATE(image);
	gpointer data;
	gsize len;
	gboolean succ;

	g_return_val_if_fail(priv != NULL, FALSE);
	g_return_val_if_fail(path != NULL, FALSE);
	g_return_val_if_fail(path[0] != '\0', FALSE);

	data = purple_image_get_data(image);
	len = purple_image_get_size(image);

	g_return_val_if_fail(data != NULL, FALSE);
	g_return_val_if_fail(len > 0, FALSE);

	succ = purple_util_write_data_to_file_absolute(path, data, len);
	if (succ && priv->path == NULL)
		priv->path = g_strdup(path);

	return succ;
}

const gchar *
purple_image_get_path(PurpleImage *image)
{
	PurpleImagePrivate *priv = PURPLE_IMAGE_GET_PRIVATE(image);

	g_return_val_if_fail(priv != NULL, NULL);

	return priv->path;
}

gboolean
purple_image_is_ready(PurpleImage *image)
{
	PurpleImagePrivate *priv = PURPLE_IMAGE_GET_PRIVATE(image);

	g_return_val_if_fail(priv != NULL, FALSE);

	return priv->is_ready;
}

gboolean
purple_image_has_failed(PurpleImage *image)
{
	PurpleImagePrivate *priv = PURPLE_IMAGE_GET_PRIVATE(image);

	g_return_val_if_fail(priv != NULL, TRUE);

	return priv->has_failed;
}

gsize
purple_image_get_size(PurpleImage *image)
{
	PurpleImagePrivate *priv;
	priv = PURPLE_IMAGE_GET_PRIVATE(image);

	g_return_val_if_fail(priv != NULL, 0);
	g_return_val_if_fail(priv->is_ready, 0);

	fill_data(image);
	g_return_val_if_fail(priv->contents, 0);

	return priv->contents->len;
}

gpointer
purple_image_get_data(PurpleImage *image)
{
	PurpleImagePrivate *priv = PURPLE_IMAGE_GET_PRIVATE(image);

	g_return_val_if_fail(priv != NULL, NULL);
	g_return_val_if_fail(priv->is_ready, NULL);

	fill_data(image);
	g_return_val_if_fail(priv->contents, NULL);

	return priv->contents->str;
}

const gchar *
purple_image_get_extension(PurpleImage *image)
{
	PurpleImagePrivate *priv = PURPLE_IMAGE_GET_PRIVATE(image);
	gpointer data;

	g_return_val_if_fail(priv != NULL, NULL);

	if (priv->extension)
		return priv->extension;

	if (purple_image_get_size(image) < 4)
		return NULL;

	data = purple_image_get_data(image);
	g_assert(data != NULL);

	if (memcmp(data, "GIF8", 4) == 0)
		return priv->extension = "gif";
	if (memcmp(data, "\xff\xd8\xff", 3) == 0) /* 4th may be e0 through ef */
		return priv->extension = "jpg";
	if (memcmp(data, "\x89PNG", 4) == 0)
		return priv->extension = "png";
	if (memcmp(data, "MM", 2) == 0)
		return priv->extension = "tif";
	if (memcmp(data, "II", 2) == 0)
		return priv->extension = "tif";
	if (memcmp(data, "BM", 2) == 0)
		return priv->extension = "bmp";
	if (memcmp(data, "\x00\x00\x01\x00", 4) == 0)
		return priv->extension = "ico";

	return NULL;
}

const gchar *
purple_image_get_mimetype(PurpleImage *image)
{
	PurpleImagePrivate *priv = PURPLE_IMAGE_GET_PRIVATE(image);
	const gchar *ext = purple_image_get_extension(image);

	g_return_val_if_fail(priv != NULL, NULL);

	if (priv->mime)
		return priv->mime;

	g_return_val_if_fail(ext != NULL, NULL);

	if (g_strcmp0(ext, "gif") == 0)
		return priv->mime = "image/gif";
	if (g_strcmp0(ext, "jpg") == 0)
		return priv->mime = "image/jpeg";
	if (g_strcmp0(ext, "png") == 0)
		return priv->mime = "image/png";
	if (g_strcmp0(ext, "tif") == 0)
		return priv->mime = "image/tiff";
	if (g_strcmp0(ext, "bmp") == 0)
		return priv->mime = "image/bmp";
	if (g_strcmp0(ext, "ico") == 0)
		return priv->mime = "image/vnd.microsoft.icon";

	return NULL;
}

const gchar *
purple_image_generate_filename(PurpleImage *image)
{
	PurpleImagePrivate *priv = PURPLE_IMAGE_GET_PRIVATE(image);
	gpointer data;
	gsize len;
	const gchar *ext;
	gchar *checksum;

	g_return_val_if_fail(priv != NULL, NULL);

	if (priv->gen_filename)
		return priv->gen_filename;

	ext = purple_image_get_extension(image);
	data = purple_image_get_data(image);
	len = purple_image_get_size(image);

	g_return_val_if_fail(ext != NULL, NULL);
	g_return_val_if_fail(data != NULL, NULL);
	g_return_val_if_fail(len > 0, NULL);

	checksum = g_compute_checksum_for_data(G_CHECKSUM_SHA1, data, len);
	priv->gen_filename = g_strdup_printf("%s.%s", checksum, ext);
	g_free(checksum);

	return priv->gen_filename;
}

void
purple_image_set_friendly_filename(PurpleImage *image, const gchar *filename)
{
	PurpleImagePrivate *priv = PURPLE_IMAGE_GET_PRIVATE(image);
	gchar *newname;
	const gchar *escaped;

	g_return_if_fail(priv != NULL);

	newname = g_path_get_basename(filename);
	escaped = purple_escape_filename(newname);
	g_free(newname);
	newname = NULL;

	if (g_strcmp0(escaped, "") == 0 || g_strcmp0(escaped, ".") == 0 ||
		g_strcmp0(escaped, G_DIR_SEPARATOR_S) == 0 ||
		g_strcmp0(escaped, "/") == 0 || g_strcmp0(escaped, "\\") == 0)
	{
		escaped = NULL;
	}

	g_free(priv->friendly_filename);
	priv->friendly_filename = g_strdup(escaped);
}

const gchar *
purple_image_get_friendly_filename(PurpleImage *image)
{
	PurpleImagePrivate *priv = PURPLE_IMAGE_GET_PRIVATE(image);

	g_return_val_if_fail(priv != NULL, NULL);

	if (G_UNLIKELY(!priv->friendly_filename)) {
		const gchar *newname = purple_image_generate_filename(image);
		gsize newname_len = strlen(newname);

		if (newname_len < 10)
			return NULL;

		/* let's use last 6 characters from checksum + 4 characters
		 * from file ext */
		newname += newname_len - 10;
		priv->friendly_filename = g_strdup(newname);
	}

	return priv->friendly_filename;
}

PurpleImage *
purple_image_transfer_new(void)
{
	PurpleImage *img;
	PurpleImagePrivate *priv;

	img = g_object_new(PURPLE_TYPE_IMAGE, NULL);
	priv = PURPLE_IMAGE_GET_PRIVATE(img);

	priv->is_ready = FALSE;
	priv->contents = g_string_new(NULL);

	return img;
}

void
purple_image_transfer_write(PurpleImage *image, const gpointer data,
	gsize length)
{
	PurpleImagePrivate *priv =
		PURPLE_IMAGE_GET_PRIVATE(image);

	g_return_if_fail(priv != NULL);
	g_return_if_fail(!priv->has_failed);
	g_return_if_fail(!priv->is_ready);
	g_return_if_fail(priv->contents != NULL);
	g_return_if_fail(data != NULL || length == 0);

	if (length == 0)
		return;

	g_string_append_len(priv->contents, (const gchar*)data, length);
}

void
purple_image_transfer_close(PurpleImage *image)
{
	PurpleImagePrivate *priv =
		PURPLE_IMAGE_GET_PRIVATE(image);

	g_return_if_fail(priv != NULL);
	g_return_if_fail(!priv->has_failed);
	g_return_if_fail(!priv->is_ready);
	g_return_if_fail(priv->contents != NULL);

	if (priv->contents->len == 0) {
		purple_debug_error("image", "image is empty");
		has_failed(image);
		return;
	}

	became_ready(image);
}

void
purple_image_transfer_failed(PurpleImage *image)
{
	PurpleImagePrivate *priv =
		PURPLE_IMAGE_GET_PRIVATE(image);

	g_return_if_fail(priv != NULL);
	g_return_if_fail(!priv->has_failed);
	g_return_if_fail(!priv->is_ready);

	has_failed(image);
}

/******************************************************************************
 * Object stuff
 ******************************************************************************/

static void
purple_image_init(GTypeInstance *instance, gpointer klass)
{
	PurpleImage *image = PURPLE_IMAGE(instance);
	PurpleImagePrivate *priv =
		PURPLE_IMAGE_GET_PRIVATE(image);

	priv->contents = NULL;
	priv->is_ready = TRUE;
	priv->has_failed = FALSE;
}

static void
purple_image_finalize(GObject *obj)
{
	PurpleImage *image = PURPLE_IMAGE(obj);
	PurpleImagePrivate *priv =
		PURPLE_IMAGE_GET_PRIVATE(image);

	if (priv->contents)
		g_string_free(priv->contents, TRUE);
	g_free(priv->path);
	g_free(priv->gen_filename);
	g_free(priv->friendly_filename);

	G_OBJECT_CLASS(parent_class)->finalize(obj);
}

static void
purple_image_get_property(GObject *object, guint par_id, GValue *value,
	GParamSpec *pspec)
{
	PurpleImage *image = PURPLE_IMAGE(object);
	PurpleImagePrivate *priv = PURPLE_IMAGE_GET_PRIVATE(image);

	switch (par_id) {
		case PROP_IS_READY:
			g_value_set_boolean(value, priv->is_ready);
			break;
		case PROP_HAS_FAILED:
			g_value_set_boolean(value, priv->has_failed);
			break;
		default:
			G_OBJECT_WARN_INVALID_PROPERTY_ID(object, par_id, pspec);
			break;
	}
}

static void
purple_image_class_init(PurpleImageClass *klass)
{
	GObjectClass *gobj_class = G_OBJECT_CLASS(klass);

	parent_class = g_type_class_peek_parent(klass);

	g_type_class_add_private(klass, sizeof(PurpleImagePrivate));

	gobj_class->finalize = purple_image_finalize;
	gobj_class->get_property = purple_image_get_property;

	properties[PROP_IS_READY] = g_param_spec_boolean("is-ready",
		"Is ready", "The image is ready to be displayed. Image may "
		"change the state to failed in a single case: if it's backed "
		"by a file and that file fails to load",
		TRUE, G_PARAM_READABLE | G_PARAM_STATIC_STRINGS);

	properties[PROP_HAS_FAILED] = g_param_spec_boolean("has-failed",
		"Has hailed", "The remote host has failed to send the image",
		FALSE, G_PARAM_READABLE | G_PARAM_STATIC_STRINGS);

	g_object_class_install_properties(gobj_class, PROP_LAST, properties);

	/**
	 * PurpleImage::failed:
	 * @image: a image that failed to transfer.
	 *
	 * Called when a @image fails to be transferred. It's guaranteed to be
	 * fired at most once for a particular @image.
	 */
	signals[SIG_FAILED] = g_signal_new("failed", G_OBJECT_CLASS_TYPE(klass),
		0, 0, NULL, NULL,
		g_cclosure_marshal_VOID__VOID, G_TYPE_NONE, 0);

	/**
	 * PurpleImage::ready:
	 * @image: a image that became ready.
	 *
	 * Called when a @image becames ready to be displayed. It's guaranteed to be
	 * fired at most once for a particular @image.
	 */
	signals[SIG_READY] = g_signal_new("ready", G_OBJECT_CLASS_TYPE(klass),
		0, 0, NULL, NULL,
		g_cclosure_marshal_VOID__VOID, G_TYPE_NONE, 0);
}

GType
purple_image_get_type(void)
{
	static GType type = 0;

	if (G_UNLIKELY(type == 0)) {
		static const GTypeInfo info = {
			.class_size = sizeof(PurpleImageClass),
			.class_init = (GClassInitFunc)purple_image_class_init,
			.instance_size = sizeof(PurpleImage),
			.instance_init = purple_image_init,
		};

		type = g_type_register_static(G_TYPE_OBJECT,
			"PurpleImage", &info, 0);
	}

	return type;
}

mercurial