2 * Copyright (C) 2008 Collabora Ltd.
4 * This library is free software; you can redistribute it and/or
5 * modify it under the terms of the GNU Lesser General Public
6 * License as published by the Free Software Foundation; either
7 * version 2.1 of the License, or (at your option) any later version.
9 * This library is distributed in the hope that it will be useful,
10 * but WITHOUT ANY WARRANTY; without even the implied warranty of
11 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
12 * Lesser General Public License for more details.
14 * You should have received a copy of the GNU Lesser General Public
15 * License along with this library; if not, write to the Free Software
16 * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA
18 * Authors: Guillaume Desmottes <guillaume.desmottes@collabora.co.uk>
19 * Elliot Fairweather <elliot.fairweather@collabora.co.uk>
24 #include <telepathy-glib/connection.h>
25 #include <telepathy-glib/proxy.h>
26 #include <telepathy-glib/util.h>
27 #include <extensions/extensions.h>
29 #include "empathy-enum-types.h"
30 #include "empathy-tp-tube.h"
31 #include "empathy-utils.h"
33 #define DEBUG_FLAG EMPATHY_DEBUG_TP
34 #include "empathy-debug.h"
37 TpSocketAddressType type;
38 EmpatyTpTubeAcceptStreamTubeCb *callback;
40 } EmpathyTpTubeAcceptData;
42 static EmpathyTpTubeAcceptData *
43 new_empathy_tp_tube_accept_data (TpSocketAddressType type,
44 EmpatyTpTubeAcceptStreamTubeCb *callback, gpointer user_data)
46 EmpathyTpTubeAcceptData *r;
48 r = g_slice_new0 (EmpathyTpTubeAcceptData);
50 r->callback = callback;
51 r->user_data = user_data;
57 free_empathy_tp_tube_accept_data (gpointer data)
59 g_slice_free (EmpathyTpTubeAcceptData, data);
63 #define GET_PRIV(obj) EMPATHY_GET_PRIV (obj, EmpathyTpTube)
67 EmpTubeChannelState state;
84 static guint signals[LAST_SIGNAL];
86 G_DEFINE_TYPE (EmpathyTpTube, empathy_tp_tube, G_TYPE_OBJECT)
89 tp_tube_state_changed_cb (TpProxy *proxy,
90 EmpTubeChannelState state,
94 EmpathyTpTubePriv *priv = GET_PRIV (tube);
97 /* We didn't get the state yet */
100 DEBUG ("Tube state changed");
103 g_object_notify (tube, "state");
107 tp_tube_invalidated_cb (TpChannel *channel,
113 DEBUG ("Channel invalidated: %s", message);
114 g_signal_emit (tube, signals[DESTROY], 0);
118 tp_tube_async_cb (TpChannel *channel,
124 DEBUG ("Error %s: %s", (gchar*) user_data, error->message);
128 tp_tube_set_property (GObject *object,
133 EmpathyTpTubePriv *priv = GET_PRIV (object);
138 priv->channel = g_value_dup_object (value);
141 G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
147 tp_tube_get_property (GObject *object,
152 EmpathyTpTubePriv *priv = GET_PRIV (object);
157 g_value_set_object (value, priv->channel);
160 g_value_set_uint (value, priv->state);
163 G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
169 got_tube_state_cb (TpProxy *proxy,
170 const GValue *out_value,
173 GObject *weak_object)
175 EmpathyTpTube *self = EMPATHY_TP_TUBE (user_data);
176 EmpathyTpTubePriv *priv = GET_PRIV (self);
182 DEBUG ("Error getting State property: %s", error->message);
186 priv->state = g_value_get_uint (out_value);
187 g_object_notify (G_OBJECT (self), "state");
191 tp_tube_constructor (GType type,
193 GObjectConstructParam *props)
196 EmpathyTpTubePriv *priv;
198 self = G_OBJECT_CLASS (empathy_tp_tube_parent_class)->constructor (
199 type, n_props, props);
200 priv = GET_PRIV (self);
202 g_signal_connect (priv->channel, "invalidated",
203 G_CALLBACK (tp_tube_invalidated_cb), self);
207 emp_cli_channel_interface_tube_connect_to_tube_channel_state_changed (
208 TP_PROXY (priv->channel), tp_tube_state_changed_cb, NULL, NULL,
211 tp_cli_dbus_properties_call_get (priv->channel, -1,
212 EMP_IFACE_CHANNEL_INTERFACE_TUBE, "State", got_tube_state_cb,
213 self, NULL, G_OBJECT (self));
219 tp_tube_finalize (GObject *object)
221 EmpathyTpTubePriv *priv = GET_PRIV (object);
223 DEBUG ("Finalizing: %p", object);
227 g_signal_handlers_disconnect_by_func (priv->channel,
228 tp_tube_invalidated_cb, object);
229 tp_cli_channel_call_close (priv->channel, -1, tp_tube_async_cb,
230 "closing tube", NULL, NULL);
231 g_object_unref (priv->channel);
234 G_OBJECT_CLASS (empathy_tp_tube_parent_class)->finalize (object);
238 empathy_tp_tube_class_init (EmpathyTpTubeClass *klass)
240 GObjectClass *object_class = G_OBJECT_CLASS (klass);
242 object_class->constructor = tp_tube_constructor;
243 object_class->finalize = tp_tube_finalize;
244 object_class->set_property = tp_tube_set_property;
245 object_class->get_property = tp_tube_get_property;
247 g_object_class_install_property (object_class, PROP_CHANNEL,
248 g_param_spec_object ("channel", "channel", "channel", TP_TYPE_CHANNEL,
249 G_PARAM_READWRITE | G_PARAM_CONSTRUCT_ONLY | G_PARAM_STATIC_STRINGS));
251 g_object_class_install_property (object_class, PROP_STATE,
252 g_param_spec_uint ("state", "state", "state",
253 0, NUM_EMP_TUBE_CHANNEL_STATES, 0,
254 G_PARAM_READABLE | G_PARAM_STATIC_NAME | G_PARAM_STATIC_STRINGS));
256 signals[DESTROY] = g_signal_new ("destroy",
257 G_TYPE_FROM_CLASS (klass),
260 g_cclosure_marshal_VOID__VOID,
263 g_type_class_add_private (klass, sizeof (EmpathyTpTubePriv));
267 empathy_tp_tube_init (EmpathyTpTube *tube)
269 EmpathyTpTubePriv *priv = G_TYPE_INSTANCE_GET_PRIVATE (tube,
270 EMPATHY_TYPE_TP_TUBE, EmpathyTpTubePriv);
276 empathy_tp_tube_new (TpChannel *channel)
278 g_return_val_if_fail (TP_IS_CHANNEL (channel), NULL);
280 return g_object_new (EMPATHY_TYPE_TP_TUBE, "channel", channel, NULL);
284 empathy_tp_tube_new_stream_tube (EmpathyContact *contact,
285 TpSocketAddressType type,
286 const gchar *hostname,
288 const gchar *service,
289 GHashTable *parameters)
293 TpConnection *connection;
298 GValue *control_param;
299 EmpathyTpTube *tube = NULL;
300 GError *error = NULL;
302 GHashTable *channel_properties;
305 g_return_val_if_fail (EMPATHY_IS_CONTACT (contact), NULL);
306 g_return_val_if_fail (hostname != NULL, NULL);
307 g_return_val_if_fail (service != NULL, NULL);
309 mc = empathy_mission_control_dup_singleton ();
310 account = empathy_contact_get_account (contact);
311 connection = mission_control_get_tpconnection (mc, account, NULL);
314 tp_connection_run_until_ready (connection, FALSE, NULL, NULL);
316 request = g_hash_table_new_full (g_str_hash, g_str_equal, NULL,
317 (GDestroyNotify) tp_g_value_slice_free);
319 /* org.freedesktop.Telepathy.Channel.ChannelType */
320 value = tp_g_value_slice_new (G_TYPE_STRING);
321 g_value_set_string (value, EMP_IFACE_CHANNEL_TYPE_STREAM_TUBE);
322 g_hash_table_insert (request, TP_IFACE_CHANNEL ".ChannelType", value);
324 /* org.freedesktop.Telepathy.Channel.TargetHandleType */
325 value = tp_g_value_slice_new (G_TYPE_UINT);
326 g_value_set_uint (value, TP_HANDLE_TYPE_CONTACT);
327 g_hash_table_insert (request, TP_IFACE_CHANNEL ".TargetHandleType", value);
329 /* org.freedesktop.Telepathy.Channel.TargetHandleType */
330 value = tp_g_value_slice_new (G_TYPE_UINT);
331 g_value_set_uint (value, empathy_contact_get_handle (contact));
332 g_hash_table_insert (request, TP_IFACE_CHANNEL ".TargetHandle", value);
334 /* org.freedesktop.Telepathy.Channel.Type.StreamTube.Service */
335 value = tp_g_value_slice_new (G_TYPE_STRING);
336 g_value_set_string (value, service);
337 g_hash_table_insert (request,
338 EMP_IFACE_CHANNEL_TYPE_STREAM_TUBE ".Service", value);
340 if (!tp_cli_connection_interface_requests_run_create_channel (connection, -1,
341 request, &object_path, &channel_properties, &error, NULL))
343 DEBUG ("Error requesting channel: %s", error->message);
344 g_clear_error (&error);
345 g_object_unref (connection);
349 DEBUG ("Offering a new stream tube");
351 channel = tp_channel_new_from_properties (connection, object_path,
352 channel_properties, NULL);
354 tp_channel_run_until_ready (channel, NULL, NULL);
356 #define ADDRESS_TYPE dbus_g_type_get_struct ("GValueArray",\
357 G_TYPE_STRING, G_TYPE_UINT, G_TYPE_INVALID)
358 params = g_hash_table_new (g_str_hash, g_str_equal);
359 address = tp_g_value_slice_new (ADDRESS_TYPE);
360 g_value_take_boxed (address, dbus_g_type_specialized_construct (ADDRESS_TYPE));
361 dbus_g_type_struct_set (address, 0, hostname, 1, port, G_MAXUINT);
362 control_param = tp_g_value_slice_new (G_TYPE_STRING);
364 if (parameters == NULL)
365 /* Pass an empty dict as parameters */
366 parameters = g_hash_table_new (g_str_hash, g_str_equal);
368 g_hash_table_ref (parameters);
370 if (!emp_cli_channel_type_stream_tube_run_offer_stream_tube (
371 TP_PROXY(channel), -1, type, address,
372 TP_SOCKET_ACCESS_CONTROL_LOCALHOST, control_param, parameters,
375 DEBUG ("Couldn't offer tube: %s", error->message);
376 g_clear_error (&error);
380 DEBUG ("Stream tube offered");
382 tube = empathy_tp_tube_new (channel);
385 g_object_unref (channel);
386 g_free (object_path);
387 g_hash_table_destroy (request);
388 g_hash_table_destroy (channel_properties);
389 tp_g_value_slice_free (address);
390 tp_g_value_slice_free (control_param);
391 g_object_unref (connection);
392 g_hash_table_unref (parameters);
398 tp_tube_accept_stream_cb (TpProxy *proxy,
399 const GValue *address,
402 GObject *weak_object)
404 EmpathyTpTube *tube = EMPATHY_TP_TUBE (weak_object);
405 EmpathyTpTubeAcceptData *data = (EmpathyTpTubeAcceptData *)user_data;
406 EmpathyTpTubeAddress eaddress;
408 eaddress.type = data->type;
412 DEBUG ("Error accepting tube: %s", error->message);
413 data->callback (tube, NULL, error, data->user_data);
417 switch (eaddress.type)
419 case TP_SOCKET_ADDRESS_TYPE_UNIX:
420 case TP_SOCKET_ADDRESS_TYPE_ABSTRACT_UNIX:
421 eaddress.a.socket.path = g_value_get_boxed (address);
423 case TP_SOCKET_ADDRESS_TYPE_IPV4:
424 case TP_SOCKET_ADDRESS_TYPE_IPV6:
425 dbus_g_type_struct_get (address,
426 0, &eaddress.a.inet.hostname,
427 1, &eaddress.a.inet.port, G_MAXUINT);
431 data->callback (tube, &eaddress, NULL, data->user_data);
435 empathy_tp_tube_accept_stream_tube (EmpathyTpTube *tube,
436 TpSocketAddressType type, EmpatyTpTubeAcceptStreamTubeCb *callback,
439 EmpathyTpTubePriv *priv = GET_PRIV (tube);
440 GValue *control_param;
441 EmpathyTpTubeAcceptData *data;
443 g_return_if_fail (EMPATHY_IS_TP_TUBE (tube));
445 DEBUG ("Accepting stream tube");
446 /* FIXME allow other acls */
447 control_param = tp_g_value_slice_new (G_TYPE_STRING);
449 data = new_empathy_tp_tube_accept_data (type, callback, user_data);
451 emp_cli_channel_type_stream_tube_call_accept_stream_tube (
452 TP_PROXY (priv->channel), -1, type, TP_SOCKET_ACCESS_CONTROL_LOCALHOST,
453 control_param, tp_tube_accept_stream_cb, data,
454 free_empathy_tp_tube_accept_data, G_OBJECT (tube));
456 tp_g_value_slice_free (control_param);