aboutsummaryrefslogtreecommitdiffstats
path: root/libempathy/empathy-ft-handler.c
diff options
context:
space:
mode:
Diffstat (limited to 'libempathy/empathy-ft-handler.c')
-rw-r--r--libempathy/empathy-ft-handler.c114
1 files changed, 74 insertions, 40 deletions
diff --git a/libempathy/empathy-ft-handler.c b/libempathy/empathy-ft-handler.c
index 9315e6538..be40d5e01 100644
--- a/libempathy/empathy-ft-handler.c
+++ b/libempathy/empathy-ft-handler.c
@@ -1010,9 +1010,7 @@ ft_handler_read_async_cb (GObject *source,
hash_data->stream = G_INPUT_STREAM (stream);
hash_data->total_bytes = priv->total_bytes;
hash_data->handler = g_object_ref (handler);
- /* FIXME: should look at the CM capabilities before setting the
- * checksum type?
- */
+ /* FIXME: MD5 is the only ContentHashType supported right now */
hash_data->checksum = g_checksum_new (G_CHECKSUM_MD5);
/* org.freedesktop.Telepathy.Channel.Type.FileTransfer.ContentHashType */
@@ -1027,19 +1025,31 @@ ft_handler_read_async_cb (GObject *source,
}
static void
-find_channel_class_cb (GStrv channel_class,
+callbacks_data_free (gpointer user_data)
+{
+ CallbacksData *data = user_data;
+
+ if (data->handler != NULL)
+ g_object_unref (data->handler);
+
+ g_slice_free (CallbacksData, data);
+}
+
+static void
+find_ft_channel_class_cb (GStrv channel_class,
gpointer user_data)
{
- EmpathyFTHandler *handler = user_data;
- EmpathyFTHandlerPriv *priv = GET_PRIV (handler);
+ CallbacksData *data = user_data;
+ EmpathyFTHandler *handler = data->handler;
gboolean allowed = TRUE;
GError *myerr = NULL;
+ /* this takes care of channel_class == NULL as well */
if (!tp_strv_contains ((const gchar * const *) channel_class,
TP_IFACE_CHANNEL ".TargetHandle"))
allowed = FALSE;
- DEBUG ("check if FT allowed: %s", allowed ? "True" : "False");
+ DEBUG ("check if FT is allowed: %s", allowed ? "True" : "False");
if (!allowed)
{
@@ -1047,47 +1057,67 @@ find_channel_class_cb (GStrv channel_class,
EMPATHY_FT_ERROR_NOT_SUPPORTED,
_("File transfer not supported by remote contact"));
- emit_error_signal (handler, myerr);
+ data->callback (NULL, myerr, data->user_data);
g_clear_error (&myerr);
-
- return;
}
-
- /* populate the request table with all the known properties */
- ft_handler_populate_outgoing_request (handler);
-
- if (priv->use_hash)
- /* start hashing the file */
- g_file_read_async (priv->gfile, G_PRIORITY_DEFAULT,
- priv->cancellable, ft_handler_read_async_cb, handler);
else
- /* push directly the handler to the dispatcher */
- ft_handler_push_to_dispatcher (handler);
+ {
+ data->callback (handler, NULL, data->user_data);
+ }
+
+ callbacks_data_free (data);
}
static void
-ft_handler_complete_request (EmpathyFTHandler *handler)
+find_hash_channel_class_cb (GStrv channel_class,
+ gpointer user_data)
{
- EmpathyFTHandlerPriv *priv = GET_PRIV (handler);
- TpConnection *connection;
+ CallbacksData *data = user_data;
+ EmpathyFTHandler *handler = data->handler;
+ EmpathyFTHandlerPriv *priv = GET_PRIV (handler);
+ gboolean allowed = TRUE;
- /* check if FT is allowed before firing up the I/O machinery */
- connection = empathy_contact_get_connection (priv->contact);
+ /* this takes care of channel_class == NULL as well */
+ if (!tp_strv_contains ((const gchar * const *) channel_class,
+ TP_IFACE_CHANNEL ".TargetHandle"))
+ allowed = FALSE;
+
+ DEBUG ("check if FT+hash is allowed: %s", allowed ? "True" : "False");
+
+ if (!allowed)
+ {
+ priv->use_hash = FALSE;
+
+ /* see if we support FT without hash instead */
+ empathy_dispatcher_find_requestable_channel_classes_async
+ (priv->dispatcher, empathy_contact_get_connection (priv->contact),
+ TP_IFACE_CHANNEL_TYPE_FILE_TRANSFER, TP_HANDLE_TYPE_CONTACT,
+ find_ft_channel_class_cb, data, NULL);
- empathy_dispatcher_find_channel_class_async (priv->dispatcher, connection,
- TP_IFACE_CHANNEL_TYPE_FILE_TRANSFER, TP_HANDLE_TYPE_CONTACT,
- find_channel_class_cb, handler);
+ return;
+ }
+ else
+ {
+ data->callback (handler, NULL, data->user_data);
+ callbacks_data_free (data);
+ }
}
static void
-callbacks_data_free (gpointer user_data)
+ft_handler_complete_request (EmpathyFTHandler *handler)
{
- CallbacksData *data = user_data;
+ EmpathyFTHandlerPriv *priv = GET_PRIV (handler);
- if (data->handler != NULL)
- g_object_unref (data->handler);
+ /* populate the request table with all the known properties */
+ ft_handler_populate_outgoing_request (handler);
- g_slice_free (CallbacksData, data);
+ if (priv->use_hash)
+ /* start hashing the file */
+ g_file_read_async (priv->gfile, G_PRIORITY_DEFAULT,
+ priv->cancellable, ft_handler_read_async_cb, handler);
+ else
+ /* push directly the handler to the dispatcher */
+ ft_handler_push_to_dispatcher (handler);
}
static void
@@ -1118,11 +1148,7 @@ ft_handler_gfile_ready_cb (GObject *source,
g_object_unref (info);
out:
- if (error == NULL)
- {
- cb_data->callback (cb_data->handler, NULL, cb_data->user_data);
- }
- else
+ if (error != NULL)
{
if (!g_cancellable_is_cancelled (priv->cancellable))
g_cancellable_cancel (priv->cancellable);
@@ -1130,9 +1156,17 @@ out:
cb_data->callback (NULL, error, cb_data->user_data);
g_error_free (error);
g_object_unref (cb_data->handler);
- }
- callbacks_data_free (cb_data);
+ callbacks_data_free (cb_data);
+ }
+ else
+ {
+ /* see if FT/hashing are allowed */
+ empathy_dispatcher_find_requestable_channel_classes_async
+ (priv->dispatcher, empathy_contact_get_connection (priv->contact),
+ TP_IFACE_CHANNEL_TYPE_FILE_TRANSFER, TP_HANDLE_TYPE_CONTACT,
+ find_hash_channel_class_cb, cb_data, "ContentHashType", NULL);
+ }
}
static void