aboutsummaryrefslogtreecommitdiffstats
path: root/libempathy
diff options
context:
space:
mode:
Diffstat (limited to 'libempathy')
-rw-r--r--libempathy/empathy-ft-handler.c75
-rw-r--r--libempathy/empathy-utils.h7
2 files changed, 69 insertions, 13 deletions
diff --git a/libempathy/empathy-ft-handler.c b/libempathy/empathy-ft-handler.c
index 9c5243a91..7c9e24c0b 100644
--- a/libempathy/empathy-ft-handler.c
+++ b/libempathy/empathy-ft-handler.c
@@ -23,6 +23,7 @@
#include <extensions/extensions.h>
#include <glib.h>
+#include <glib/gi18n.h>
#include <telepathy-glib/util.h>
#include "empathy-ft-handler.h"
@@ -57,6 +58,7 @@ typedef struct {
EmpathyFTHandler *handler;
GFile *gfile;
GHashTable *request;
+ goffset total_size;
} RequestData;
typedef struct {
@@ -66,6 +68,7 @@ typedef struct {
GError *error;
guchar *buffer;
GChecksum *checksum;
+ gssize total_read;
} HashingData;
/* private data */
@@ -221,9 +224,9 @@ empathy_ft_handler_class_init (EmpathyFTHandlerClass *klass)
signals[TRANSFER_ERROR] =
g_signal_new ("transfer-error", G_TYPE_FROM_CLASS (klass),
G_SIGNAL_RUN_LAST, 0, NULL, NULL,
- _empathy_marshal_VOID__OBJECT_POINTER,
+ g_cclosure_marshal_VOID__POINTER,
G_TYPE_NONE,
- 2, EMPATHY_TYPE_TP_FILE, G_TYPE_POINTER);
+ 1, G_TYPE_POINTER);
signals[TRANSFER_PROGRESS] =
g_signal_new ("transfer-progress", G_TYPE_FROM_CLASS (klass),
@@ -500,7 +503,9 @@ cleanup:
if (error != NULL)
{
- /* TODO: error handling. */
+ g_signal_emit (req_data->handler, signals[TRANSFER_ERROR], 0, error);
+ g_clear_error (&error);
+ request_data_free (req_data);
}
else
{
@@ -515,6 +520,7 @@ hash_job_async_read_cb (GObject *source,
gpointer user_data)
{
HashingData *hash_data = user_data;
+ RequestData *req_data = hash_data->req_data;
gssize bytes_read;
GError *error = NULL;
@@ -526,7 +532,7 @@ hash_job_async_read_cb (GObject *source,
goto out;
}
- /* TODO: notify progress */
+ hash_data->total_read += bytes_read;
/* we now have the chunk */
if (bytes_read == 0)
@@ -538,6 +544,8 @@ hash_job_async_read_cb (GObject *source,
else
{
g_checksum_update (hash_data->checksum, hash_data->buffer, bytes_read);
+ g_signal_emit (req_data->handler, signals[HASHING_PROGRESS], 0,
+ (guint64) hash_data->total_read, (guint64) req_data->total_size);
}
out:
@@ -550,10 +558,15 @@ out:
static void
schedule_hash_chunk (HashingData *hash_data)
{
+ EmpathyFTHandlerPriv *priv;
+ RequestData *req_data = hash_data->req_data;
+
+ priv = GET_PRIV (req_data->handler);
+
if (hash_data->done_reading)
{
g_input_stream_close_async (hash_data->stream, G_PRIORITY_DEFAULT,
- NULL, hash_job_async_close_stream_cb, hash_data);
+ priv->cancellable, hash_job_async_close_stream_cb, hash_data);
}
else
{
@@ -561,7 +574,7 @@ schedule_hash_chunk (HashingData *hash_data)
hash_data->buffer = g_malloc0 (BUFFER_SIZE);
g_input_stream_read_async (hash_data->stream, hash_data->buffer,
- BUFFER_SIZE, G_PRIORITY_DEFAULT, NULL,
+ BUFFER_SIZE, G_PRIORITY_DEFAULT, priv->cancellable,
hash_job_async_read_cb, hash_data);
}
}
@@ -581,7 +594,11 @@ ft_handler_read_async_cb (GObject *source,
stream = g_file_read_finish (req_data->gfile, res, &error);
if (error != NULL)
{
- /* TODO: error handling. */
+ g_signal_emit (req_data->handler, signals[TRANSFER_ERROR], 0, error);
+
+ request_data_free (req_data);
+ g_clear_error (&error);
+
return;
}
@@ -602,6 +619,8 @@ ft_handler_read_async_cb (GObject *source,
g_hash_table_insert (request,
EMP_IFACE_CHANNEL_TYPE_FILE_TRANSFER ".ContentHashType", value);
+ g_signal_emit (req_data->handler, signals[HASHING_STARTED], 0);
+
schedule_hash_chunk (hash_data);
}
@@ -612,19 +631,26 @@ ft_handler_gfile_ready_cb (GObject *source,
{
GFileInfo *info;
GError *error = NULL;
+ EmpathyFTHandlerPriv *priv = GET_PRIV (req_data->handler);
info = g_file_query_info_finish (req_data->gfile, res, &error);
if (error != NULL)
{
- /* TODO: error handling. */
+ g_signal_emit (req_data->handler, signals[TRANSFER_ERROR], 0, error);
+
+ request_data_free (req_data);
+ g_clear_error (&error);
+
return;
}
ft_handler_populate_outgoing_request (req_data, info);
+ req_data->total_size = g_file_info_get_size (info);
+
/* now start hashing the file */
g_file_read_async (req_data->gfile, G_PRIORITY_DEFAULT,
- NULL, ft_handler_read_async_cb, req_data);
+ priv->cancellable, ft_handler_read_async_cb, req_data);
}
static void
@@ -635,15 +661,38 @@ ft_handler_contact_ready_cb (EmpathyContact *contact,
{
RequestData *req_data = user_data;
EmpathyFTHandlerPriv *priv = GET_PRIV (req_data->handler);
+ GError *myerr = NULL;
g_assert (priv->contact != NULL);
g_assert (priv->gfile != NULL);
- /* check if FT is allowed before firing up the I/O machinery */
- if (!ft_handler_check_if_allowed (req_data->handler))
+ g_cancellable_set_error_if_cancelled (priv->cancellable, &myerr);
+
+ if (myerr == NULL)
{
- /* TODO: error handling. */
+ if (error != NULL)
+ {
+ myerr = g_error_copy (error);
+ }
+ else
+ {
+ /* check if FT is allowed before firing up the I/O machinery */
+ if (!ft_handler_check_if_allowed (req_data->handler))
+ {
+ g_set_error_literal (&myerr, EMPATHY_FT_ERROR_QUARK,
+ EMPATHY_FT_ERROR_NOT_SUPPORTED,
+ _("File transfer not supported by remote contact"));
+ }
+ }
+ }
+
+ if (myerr != NULL)
+ {
+ g_signal_emit (req_data->handler, signals[TRANSFER_ERROR], 0, myerr);
+
request_data_free (req_data);
+ g_clear_error (&myerr);
+
return;
}
@@ -654,7 +703,7 @@ ft_handler_contact_ready_cb (EmpathyContact *contact,
G_FILE_ATTRIBUTE_STANDARD_CONTENT_TYPE ","
G_FILE_ATTRIBUTE_TIME_MODIFIED,
G_FILE_QUERY_INFO_NONE, G_PRIORITY_DEFAULT,
- NULL, (GAsyncReadyCallback) ft_handler_gfile_ready_cb,
+ priv->cancellable, (GAsyncReadyCallback) ft_handler_gfile_ready_cb,
req_data);
}
diff --git a/libempathy/empathy-utils.h b/libempathy/empathy-utils.h
index d36503c7f..8179c90b0 100644
--- a/libempathy/empathy-utils.h
+++ b/libempathy/empathy-utils.h
@@ -40,8 +40,15 @@
#define EMPATHY_GET_PRIV(obj,type) ((type##Priv *) ((type *) obj)->priv)
#define EMP_STR_EMPTY(x) ((x) == NULL || (x)[0] == '\0')
+#define EMPATHY_FT_ERROR_QUARK g_quark_from_static_string ("EmpathyFTError")
+
G_BEGIN_DECLS
+typedef enum {
+ EMPATHY_FT_ERROR_FAILED,
+ EMPATHY_FT_ERROR_NOT_SUPPORTED
+} EmpathyFTErrorEnum;
+
void empathy_init (void);
/* Strings */
gchar * empathy_substring (const gchar *str,