2 * @file telepathy-connection.c
6 * Copyright (C) 2012-2016 SIPE Project <http://sipe.sourceforge.net/>
8 * This program is free software; you can redistribute it and/or modify
9 * it under the terms of the GNU General Public License as published by
10 * the Free Software Foundation; either version 2 of the License, or
11 * (at your option) any later version.
13 * This program is distributed in the hope that it will be useful,
14 * but WITHOUT ANY WARRANTY; without even the implied warranty of
15 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
16 * GNU General Public License for more details.
18 * You should have received a copy of the GNU General Public License
19 * along with this program; if not, write to the Free Software
20 * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
30 #include <glib-object.h>
31 #include <glib/gstdio.h>
32 #include <telepathy-glib/base-connection.h>
33 #include <telepathy-glib/base-protocol.h>
34 #include <telepathy-glib/contacts-mixin.h>
35 #include <telepathy-glib/handle-repo-dynamic.h>
36 #include <telepathy-glib/presence-mixin.h>
37 #include <telepathy-glib/simple-password-manager.h>
38 #include <telepathy-glib/telepathy-glib.h>
40 #include "sipe-backend.h"
41 #include "sipe-common.h"
42 #include "sipe-core.h"
44 #include "telepathy-private.h"
48 * Connection class - data structures
50 typedef struct _SipeConnectionClass
{
51 TpBaseConnectionClass parent_class
;
52 TpDBusPropertiesMixinClass properties_mixin
;
53 TpContactsMixinClass contacts_mixin
;
54 TpPresenceMixinClass presence_mixin
;
55 } SipeConnectionClass
;
57 typedef struct _SipeConnection
{
58 TpBaseConnection parent
;
59 TpContactsMixinClass contacts_mixin
;
60 TpPresenceMixin presence_mixin
;
62 /* channel managers */
63 TpSimplePasswordManager
*password_manager
;
64 struct _SipeContactList
*contact_list
;
65 struct _SipeTLSManager
*tls_manager
;
67 struct sipe_backend_private
private;
74 guint authentication_type
;
76 gchar
*authentication
;
78 gboolean dont_publish
;
79 gboolean is_disconnecting
;
81 GPtrArray
*contact_info_fields
;
84 #define SIPE_PUBLIC_TO_CONNECTION sipe_public->backend_private->connection
87 * Connection class - type macros
89 static GType
sipe_connection_get_type(void) G_GNUC_CONST
;
90 #define SIPE_TYPE_CONNECTION \
91 (sipe_connection_get_type())
92 #define SIPE_CONNECTION(obj) \
93 (G_TYPE_CHECK_INSTANCE_CAST((obj), SIPE_TYPE_CONNECTION, \
98 * Connection class - type definition
100 static void init_aliasing (gpointer
, gpointer
);
101 G_DEFINE_TYPE_WITH_CODE(SipeConnection
,
103 TP_TYPE_BASE_CONNECTION
,
104 G_IMPLEMENT_INTERFACE(TP_TYPE_SVC_CONNECTION_INTERFACE_ALIASING
,
106 G_IMPLEMENT_INTERFACE(TP_TYPE_SVC_CONNECTION_INTERFACE_AVATARS
,
107 sipe_telepathy_avatars_iface_init
);
108 G_IMPLEMENT_INTERFACE(TP_TYPE_SVC_CONNECTION_INTERFACE_CONTACTS
,
109 tp_contacts_mixin_iface_init
);
110 G_IMPLEMENT_INTERFACE(TP_TYPE_SVC_CONNECTION_INTERFACE_CONTACT_GROUPS
,
111 tp_base_contact_list_mixin_groups_iface_init
);
112 G_IMPLEMENT_INTERFACE(TP_TYPE_SVC_CONNECTION_INTERFACE_CONTACT_INFO
,
113 sipe_telepathy_contact_info_iface_init
);
114 G_IMPLEMENT_INTERFACE(TP_TYPE_SVC_CONNECTION_INTERFACE_CONTACT_LIST
,
115 tp_base_contact_list_mixin_list_iface_init
);
116 G_IMPLEMENT_INTERFACE(TP_TYPE_SVC_CONNECTION_INTERFACE_PRESENCE
,
117 tp_presence_mixin_iface_init
);
118 G_IMPLEMENT_INTERFACE(TP_TYPE_SVC_CONNECTION_INTERFACE_SIMPLE_PRESENCE
,
119 tp_presence_mixin_simple_presence_iface_init
);
120 G_IMPLEMENT_INTERFACE(TP_TYPE_SVC_DBUS_PROPERTIES
,
121 tp_dbus_properties_mixin_iface_init
);
126 * Connection class - instance methods
128 static gchar
*normalize_contact(SIPE_UNUSED_PARAMETER TpHandleRepoIface
*repo
,
130 SIPE_UNUSED_PARAMETER gpointer context
,
133 return(sipe_telepathy_protocol_normalize_contact(NULL
, id
, error
));
136 static void create_handle_repos(SIPE_UNUSED_PARAMETER TpBaseConnection
*conn
,
137 TpHandleRepoIface
*repos
[NUM_TP_HANDLE_TYPES
])
139 repos
[TP_HANDLE_TYPE_CONTACT
] = tp_dynamic_handle_repo_new(TP_HANDLE_TYPE_CONTACT
,
144 static gboolean
connect_to_core(SipeConnection
*self
,
147 struct sipe_core_public
*sipe_public
;
150 sipe_public
= sipe_core_allocate(self
->account
,
154 NULL
, /* @TODO: email */
155 NULL
, /* @TODO: email_url */
158 SIPE_DEBUG_INFO("connect_to_core: created %p", sipe_public
);
161 struct sipe_backend_private
*telepathy_private
= &self
->private;
163 /* initialize backend private data */
164 sipe_public
->backend_private
= telepathy_private
;
165 telepathy_private
->public = sipe_public
;
166 telepathy_private
->contact_list
= self
->contact_list
;
167 telepathy_private
->connection
= self
;
168 telepathy_private
->activity
= SIPE_ACTIVITY_UNSET
;
169 telepathy_private
->cache_dir
= g_build_path(G_DIR_SEPARATOR_S
,
170 g_get_user_cache_dir(),
175 telepathy_private
->message
= NULL
;
176 telepathy_private
->tls_manager
= self
->tls_manager
;
177 telepathy_private
->transport
= NULL
;
179 /* make sure cache directory exists */
180 if (!g_file_test(telepathy_private
->cache_dir
,
181 G_FILE_TEST_IS_DIR
) &&
182 (g_mkdir_with_parents(telepathy_private
->cache_dir
,
183 S_IRWXU
| S_IRGRP
| S_IXGRP
| S_IROTH
| S_IXOTH
)
185 SIPE_DEBUG_INFO("connect_to_core: created cache directory %s",
186 telepathy_private
->cache_dir
);
188 SIPE_CORE_FLAG_UNSET(DONT_PUBLISH
);
189 if (self
->dont_publish
)
190 SIPE_CORE_FLAG_SET(DONT_PUBLISH
);
192 sipe_core_transport_sip_connect(sipe_public
,
194 self
->authentication_type
,
200 g_set_error_literal(error
, TP_ERROR
, TP_ERROR_INVALID_ARGUMENT
,
206 static void password_manager_cb(GObject
*source
,
207 GAsyncResult
*result
,
210 SipeConnection
*self
= data
;
211 TpBaseConnection
*base
= TP_BASE_CONNECTION(self
);
212 GError
*error
= NULL
;
213 const GString
*password
= tp_simple_password_manager_prompt_finish(
214 TP_SIMPLE_PASSWORD_MANAGER(source
),
218 if (password
== NULL
) {
219 SIPE_DEBUG_ERROR("password_manager_cb: failed: %s",
220 error
? error
->message
: "UNKNOWN");
222 if (base
->status
!= TP_CONNECTION_STATUS_DISCONNECTED
) {
223 tp_base_connection_disconnect_with_dbus_error(base
,
224 error
? tp_error_get_dbus_name(error
->code
) : "",
226 TP_CONNECTION_STATUS_REASON_AUTHENTICATION_FAILED
);
231 g_free(self
->password
);
232 self
->password
= g_strdup(password
->str
);
234 if (!connect_to_core(self
, &error
)) {
235 if (base
->status
!= TP_CONNECTION_STATUS_DISCONNECTED
) {
236 tp_base_connection_disconnect_with_dbus_error(base
,
237 tp_error_get_dbus_name(error
->code
),
239 TP_CONNECTION_STATUS_REASON_AUTHENTICATION_FAILED
);
246 static gboolean
start_connecting(TpBaseConnection
*base
,
249 SipeConnection
*self
= SIPE_CONNECTION(base
);
251 gchar
*uri
= sipe_telepathy_protocol_normalize_contact(NULL
,
255 SIPE_DEBUG_INFO_NOFORMAT("SipeConnection::start_connecting");
257 /* set up mandatory self-handle */
259 base
->self_handle
= tp_handle_ensure(tp_base_connection_get_handles(base
,
260 TP_HANDLE_TYPE_CONTACT
),
265 if (!base
->self_handle
) {
266 SIPE_DEBUG_ERROR("SipeConnection::start_connecting: self handle creation failed: %s",
271 SIPE_DEBUG_ERROR("SipeConnection::start_connecting: %s",
276 tp_base_connection_change_status(base
, TP_CONNECTION_STATUS_CONNECTING
,
277 TP_CONNECTION_STATUS_REASON_REQUESTED
);
279 /* map option list to flags - default is automatic */
280 self
->authentication_type
= SIPE_AUTHENTICATION_TYPE_AUTOMATIC
;
281 if (sipe_strequal(self
->authentication
, "ntlm")) {
282 SIPE_DEBUG_INFO_NOFORMAT("start_connecting: NTLM selected");
283 self
->authentication_type
= SIPE_AUTHENTICATION_TYPE_NTLM
;
285 #ifdef HAVE_GSSAPI_GSSAPI_H
286 if (sipe_strequal(self
->authentication
, "krb5")) {
287 SIPE_DEBUG_INFO_NOFORMAT("start_connecting: KRB5 selected");
288 self
->authentication_type
= SIPE_AUTHENTICATION_TYPE_KERBEROS
;
291 if (sipe_strequal(self
->authentication
, "tls-dsk")) {
292 SIPE_DEBUG_INFO_NOFORMAT("start_connecting: TLS-DSK selected");
293 self
->authentication_type
= SIPE_AUTHENTICATION_TYPE_TLS_DSK
;
296 /* Only ask for a password when required */
297 if (!sipe_core_transport_sip_requires_password(self
->authentication_type
,
299 (self
->password
&& strlen(self
->password
)))
300 rc
= connect_to_core(self
, error
);
302 SIPE_DEBUG_INFO_NOFORMAT("SipeConnection::start_connecting: requesting password from user");
303 tp_simple_password_manager_prompt_async(self
->password_manager
,
311 static gboolean
disconnect_from_core(gpointer data
)
313 TpBaseConnection
*base
= data
;
314 SipeConnection
*self
= SIPE_CONNECTION(base
);
315 struct sipe_backend_private
*telepathy_private
= &self
->private;
316 struct sipe_core_public
*sipe_public
= telepathy_private
->public;
318 SIPE_DEBUG_INFO("disconnect_from_core: %p", sipe_public
);
321 sipe_core_deallocate(sipe_public
);
322 telepathy_private
->public = NULL
;
323 telepathy_private
->transport
= NULL
;
325 g_free(telepathy_private
->message
);
326 telepathy_private
->message
= NULL
;
328 g_free(telepathy_private
->cache_dir
);
329 telepathy_private
->cache_dir
= NULL
;
331 SIPE_DEBUG_INFO_NOFORMAT("disconnect_from_core: core deallocated");
333 /* now it is OK to destroy the connection object */
334 tp_base_connection_finish_shutdown(base
);
339 static void shut_down(TpBaseConnection
*base
)
341 SIPE_DEBUG_INFO_NOFORMAT("SipeConnection::shut_down");
343 /* this can be called synchronously, defer destruction */
344 g_idle_add(disconnect_from_core
, base
);
347 static GPtrArray
*create_channel_managers(TpBaseConnection
*base
)
349 SipeConnection
*self
= SIPE_CONNECTION(base
);
350 GPtrArray
*channel_managers
= g_ptr_array_new();
352 SIPE_DEBUG_INFO_NOFORMAT("SipeConnection::create_channel_managers");
354 self
->contact_list
= sipe_telepathy_contact_list_new(base
);
355 g_ptr_array_add(channel_managers
, self
->contact_list
);
357 self
->password_manager
= tp_simple_password_manager_new(base
);
358 g_ptr_array_add(channel_managers
, self
->password_manager
);
360 g_ptr_array_add(channel_managers
, sipe_telepathy_search_new(base
));
362 self
->tls_manager
= sipe_telepathy_tls_new(base
);
363 g_ptr_array_add(channel_managers
, self
->tls_manager
);
365 return(channel_managers
);
368 static void aliasing_fill_contact_attributes(GObject
*object
,
369 const GArray
*contacts
,
370 GHashTable
*attributes
)
372 SipeConnection
*self
= SIPE_CONNECTION(object
);
375 for (i
= 0; i
< contacts
->len
; i
++) {
376 TpHandle contact
= g_array_index(contacts
, guint
, i
);
378 tp_contacts_mixin_set_contact_attribute(attributes
,
380 TP_TOKEN_CONNECTION_INTERFACE_ALIASING_ALIAS
,
381 tp_g_value_slice_new_string(
382 sipe_telepathy_buddy_get_alias(self
->contact_list
,
387 static void avatars_fill_contact_attributes(GObject
*object
,
388 const GArray
*contacts
,
389 GHashTable
*attributes
)
391 SipeConnection
*self
= SIPE_CONNECTION(object
);
394 for (i
= 0; i
< contacts
->len
; i
++) {
395 TpHandle contact
= g_array_index(contacts
, guint
, i
);
396 const gchar
*hash
= sipe_telepathy_buddy_get_hash(self
->contact_list
,
399 if (!hash
) hash
= "";
400 tp_contacts_mixin_set_contact_attribute(attributes
,
402 TP_IFACE_CONNECTION_INTERFACE_AVATARS
"/token",
403 tp_g_value_slice_new_string(hash
));
407 static void contact_info_properties_getter(GObject
*object
,
408 SIPE_UNUSED_PARAMETER GQuark interface
,
411 gpointer getter_data
)
413 GQuark fields
= g_quark_from_static_string("SupportedFields");
416 g_value_set_boxed(value
,
417 SIPE_CONNECTION(object
)->contact_info_fields
);
419 g_value_set_uint(value
,
420 GPOINTER_TO_UINT(getter_data
));
423 static void sipe_connection_constructed(GObject
*object
)
425 SipeConnection
*self
= SIPE_CONNECTION(object
);
426 TpBaseConnection
*base
= TP_BASE_CONNECTION(object
);
427 void (*chain_up
)(GObject
*) = G_OBJECT_CLASS(sipe_connection_parent_class
)->constructed
;
432 tp_contacts_mixin_init(object
,
433 G_STRUCT_OFFSET(SipeConnection
, contacts_mixin
));
434 tp_base_connection_register_with_contacts_mixin(base
);
436 tp_base_contact_list_mixin_register_with_contacts_mixin(base
);
438 tp_contacts_mixin_add_contact_attributes_iface(object
,
439 TP_IFACE_CONNECTION_INTERFACE_ALIASING
,
440 aliasing_fill_contact_attributes
);
441 tp_contacts_mixin_add_contact_attributes_iface(object
,
442 TP_IFACE_CONNECTION_INTERFACE_AVATARS
,
443 avatars_fill_contact_attributes
);
445 tp_presence_mixin_init(object
,
446 G_STRUCT_OFFSET(SipeConnection
,
448 tp_presence_mixin_simple_presence_register_with_contacts_mixin(object
);
450 self
->contact_info_fields
= sipe_telepathy_contact_info_fields();
453 static void sipe_connection_finalize(GObject
*object
)
455 SipeConnection
*self
= SIPE_CONNECTION(object
);
457 SIPE_DEBUG_INFO_NOFORMAT("SipeConnection::finalize");
459 tp_contacts_mixin_finalize(object
);
460 tp_presence_mixin_finalize(object
);
461 g_boxed_free(TP_ARRAY_TYPE_FIELD_SPECS
, self
->contact_info_fields
);
463 g_free(self
->authentication
);
464 g_free(self
->user_agent
);
466 g_free(self
->server
);
467 g_free(self
->password
);
469 g_free(self
->account
);
471 G_OBJECT_CLASS(sipe_connection_parent_class
)->finalize(object
);
475 * Connection class - type implementation
477 static const gchar
*interfaces_always_present
[] = {
479 TP_IFACE_CONNECTION_INTERFACE_ALIASING
,
480 TP_IFACE_CONNECTION_INTERFACE_AVATARS
,
481 TP_IFACE_CONNECTION_INTERFACE_CONTACT_GROUPS
,
482 TP_IFACE_CONNECTION_INTERFACE_CONTACT_INFO
,
483 TP_IFACE_CONNECTION_INTERFACE_CONTACT_LIST
,
484 TP_IFACE_CONNECTION_INTERFACE_CONTACTS
,
485 TP_IFACE_CONNECTION_INTERFACE_PRESENCE
,
486 TP_IFACE_CONNECTION_INTERFACE_REQUESTS
,
487 TP_IFACE_CONNECTION_INTERFACE_SIMPLE_PRESENCE
,
491 static void sipe_connection_class_init(SipeConnectionClass
*klass
)
493 GObjectClass
*object_class
= G_OBJECT_CLASS(klass
);
494 TpBaseConnectionClass
*base_class
= TP_BASE_CONNECTION_CLASS(klass
);
495 static TpDBusPropertiesMixinIfaceImpl prop_interfaces
[] = {
498 .name
= TP_IFACE_CONNECTION_INTERFACE_CONTACT_INFO
,
499 .getter
= contact_info_properties_getter
,
508 /* initalize non-constant fields */
509 prop_interfaces
[0].props
= sipe_telepathy_contact_info_props();
511 SIPE_DEBUG_INFO_NOFORMAT("SipeConnection::class_init");
513 object_class
->constructed
= sipe_connection_constructed
;
514 object_class
->finalize
= sipe_connection_finalize
;
516 base_class
->create_handle_repos
= create_handle_repos
;
517 base_class
->start_connecting
= start_connecting
;
518 base_class
->shut_down
= shut_down
;
519 base_class
->create_channel_managers
= create_channel_managers
;
521 base_class
->interfaces_always_present
= interfaces_always_present
;
523 klass
->properties_mixin
.interfaces
= prop_interfaces
;
524 tp_dbus_properties_mixin_class_init(object_class
,
525 G_STRUCT_OFFSET(SipeConnectionClass
,
527 tp_contacts_mixin_class_init(object_class
,
528 G_STRUCT_OFFSET(SipeConnectionClass
,
530 sipe_telepathy_status_init(object_class
,
531 G_STRUCT_OFFSET(SipeConnectionClass
,
533 tp_presence_mixin_simple_presence_init_dbus_properties(object_class
);
534 tp_base_contact_list_mixin_class_init(base_class
);
537 static void sipe_connection_init(SIPE_UNUSED_PARAMETER SipeConnection
*self
)
539 SIPE_DEBUG_INFO_NOFORMAT("SipeConnection::init");
543 * Connection class - interface implementation
547 static void get_alias_flags(TpSvcConnectionInterfaceAliasing
*aliasing
,
548 DBusGMethodInvocation
*context
)
550 TpBaseConnection
*base
= TP_BASE_CONNECTION(aliasing
);
552 TP_BASE_CONNECTION_ERROR_IF_NOT_CONNECTED(base
, context
);
553 SIPE_DEBUG_INFO_NOFORMAT("SipeConnection::get_alias_flags called");
555 tp_svc_connection_interface_aliasing_return_from_get_alias_flags(context
,
556 TP_CONNECTION_ALIAS_FLAG_USER_SET
);
559 static void get_aliases(TpSvcConnectionInterfaceAliasing
*aliasing
,
560 const GArray
*contacts
,
561 DBusGMethodInvocation
*context
)
563 SipeConnection
*self
= SIPE_CONNECTION(aliasing
);
564 TpBaseConnection
*base
= TP_BASE_CONNECTION(aliasing
);
565 TpHandleRepoIface
*contact_repo
= tp_base_connection_get_handles(base
,
566 TP_HANDLE_TYPE_CONTACT
);
567 GError
*error
= NULL
;
571 TP_BASE_CONNECTION_ERROR_IF_NOT_CONNECTED(base
, context
);
572 SIPE_DEBUG_INFO_NOFORMAT("SipeConnection::get_aliases called");
574 if (!tp_handles_are_valid(contact_repo
, contacts
, FALSE
, &error
)) {
575 dbus_g_method_return_error(context
, error
);
580 result
= g_hash_table_new(g_direct_hash
, g_direct_equal
);
582 for (i
= 0; i
< contacts
->len
; i
++) {
583 TpHandle contact
= g_array_index(contacts
, TpHandle
, i
);
584 const gchar
*alias
= sipe_telepathy_buddy_get_alias(self
->contact_list
,
586 g_hash_table_insert(result
,
587 GUINT_TO_POINTER(contact
),
591 tp_svc_connection_interface_aliasing_return_from_get_aliases(context
,
593 g_hash_table_unref(result
);
596 static void request_aliases(TpSvcConnectionInterfaceAliasing
*aliasing
,
597 const GArray
*contacts
,
598 DBusGMethodInvocation
*context
)
600 SipeConnection
*self
= SIPE_CONNECTION(aliasing
);
601 TpBaseConnection
*base
= TP_BASE_CONNECTION(aliasing
);
602 TpHandleRepoIface
*contact_repo
= tp_base_connection_get_handles(base
,
603 TP_HANDLE_TYPE_CONTACT
);
604 GError
*error
= NULL
;
609 TP_BASE_CONNECTION_ERROR_IF_NOT_CONNECTED(base
, context
);
610 SIPE_DEBUG_INFO_NOFORMAT("SipeConnection::request_aliases called");
612 if (!tp_handles_are_valid(contact_repo
, contacts
, FALSE
, &error
)) {
613 dbus_g_method_return_error(context
, error
);
618 result
= g_ptr_array_sized_new(contacts
->len
+ 1);
620 for (i
= 0; i
< contacts
->len
; i
++) {
621 TpHandle contact
= g_array_index(contacts
, TpHandle
, i
);
622 const gchar
*alias
= sipe_telepathy_buddy_get_alias(self
->contact_list
,
624 g_ptr_array_add(result
, (gchar
*) alias
);
627 g_ptr_array_add(result
, NULL
);
628 strings
= (gchar
**) g_ptr_array_free(result
, FALSE
);
630 tp_svc_connection_interface_aliasing_return_from_request_aliases(context
,
631 (const gchar
**) strings
);
635 static void set_aliases(TpSvcConnectionInterfaceAliasing
*aliasing
,
637 DBusGMethodInvocation
*context
)
639 SipeConnection
*self
= SIPE_CONNECTION(aliasing
);
640 TpBaseConnection
*base
= TP_BASE_CONNECTION(aliasing
);
641 TpHandleRepoIface
*contact_repo
= tp_base_connection_get_handles(base
,
642 TP_HANDLE_TYPE_CONTACT
);
646 SIPE_DEBUG_INFO_NOFORMAT("SipeConnection::set_aliases called");
648 g_hash_table_iter_init(&iter
, aliases
);
650 while (g_hash_table_iter_next(&iter
, &key
, NULL
)) {
651 GError
*error
= NULL
;
653 if (!tp_handle_is_valid(contact_repo
,
654 GPOINTER_TO_UINT(key
),
656 dbus_g_method_return_error(context
, error
);
662 g_hash_table_iter_init(&iter
, aliases
);
664 while (g_hash_table_iter_next(&iter
, &key
, &value
)) {
665 sipe_telepathy_buddy_set_alias(self
->contact_list
,
666 GPOINTER_TO_UINT(key
),
670 tp_svc_connection_interface_aliasing_return_from_set_aliases(context
);
673 static void init_aliasing(gpointer iface
,
674 SIPE_UNUSED_PARAMETER gpointer iface_data
)
676 TpSvcConnectionInterfaceAliasingClass
*klass
= iface
;
678 SIPE_DEBUG_INFO_NOFORMAT("SipeConnection::init_aliasing called");
680 tp_svc_connection_interface_aliasing_implement_get_alias_flags(klass
, get_alias_flags
);
681 tp_svc_connection_interface_aliasing_implement_request_aliases(klass
, request_aliases
);
682 tp_svc_connection_interface_aliasing_implement_get_aliases(klass
, get_aliases
);
683 tp_svc_connection_interface_aliasing_implement_set_aliases(klass
, set_aliases
);
686 /* create new connection object */
687 TpBaseConnection
*sipe_telepathy_connection_new(TpBaseProtocol
*protocol
,
689 SIPE_UNUSED_PARAMETER GError
**error
)
691 SipeConnection
*conn
= g_object_new(SIPE_TYPE_CONNECTION
,
692 "protocol", tp_base_protocol_get_name(protocol
),
696 gboolean boolean_value
;
699 SIPE_DEBUG_INFO_NOFORMAT("sipe_telepathy_connection_new");
701 /* initialize private fields */
702 conn
->is_disconnecting
= FALSE
;
704 /* account is required field */
705 conn
->account
= g_strdup(tp_asv_get_string(params
, "account"));
707 /* if login is not specified, account value will be used in connect_to_core */
708 value
= tp_asv_get_string(params
, "login");
709 if (value
&& strlen(value
))
710 conn
->login
= g_strdup(value
);
715 value
= tp_asv_get_string(params
, "password");
716 if (value
&& strlen(value
))
717 conn
->password
= g_strdup(value
);
719 conn
->password
= NULL
;
722 value
= tp_asv_get_string(params
, "server");
723 if (value
&& strlen(value
))
724 conn
->server
= g_strdup(value
);
728 /* server port: core expects a string */
729 port
= tp_asv_get_uint32(params
, "port", &valid
);
731 conn
->port
= g_strdup_printf("%d", port
);
736 value
= tp_asv_get_string(params
, "transport");
737 if (sipe_strequal(value
, "auto")) {
738 conn
->transport
= conn
->server
?
739 SIPE_TRANSPORT_TLS
: SIPE_TRANSPORT_AUTO
;
740 } else if (sipe_strequal(value
, "tls")) {
741 conn
->transport
= SIPE_TRANSPORT_TLS
;
743 conn
->transport
= SIPE_TRANSPORT_TCP
;
746 /* User-Agent: override */
747 value
= tp_asv_get_string(params
, "useragent");
748 if (value
&& strlen(value
))
749 conn
->user_agent
= g_strdup(value
);
751 conn
->user_agent
= NULL
;
753 /* authentication type */
754 value
= tp_asv_get_string(params
, "authentication");
755 if (value
&& strlen(value
) && strcmp(value
, "ntlm"))
756 conn
->authentication
= g_strdup(value
);
758 conn
->authentication
= NULL
; /* NTLM is default */
761 boolean_value
= tp_asv_get_boolean(params
, "single-sign-on", &valid
);
763 conn
->sso
= boolean_value
;
767 /* Don't publish my calendar information */
768 boolean_value
= tp_asv_get_boolean(params
, "don't-publish-calendar", &valid
);
770 conn
->dont_publish
= boolean_value
;
772 conn
->dont_publish
= FALSE
;
774 return(TP_BASE_CONNECTION(conn
));
777 void sipe_telepathy_connection_alias_updated(TpBaseConnection
*connection
,
781 GPtrArray
*aliases
= g_ptr_array_sized_new(1);
782 GValueArray
*pair
= g_value_array_new(2);
784 g_value_array_append(pair
, NULL
);
785 g_value_array_append(pair
, NULL
);
786 g_value_init(pair
->values
+ 0, G_TYPE_UINT
);
787 g_value_init(pair
->values
+ 1, G_TYPE_STRING
);
788 g_value_set_uint(pair
->values
+ 0, contact
);
789 g_value_set_string(pair
->values
+ 1, alias
);
790 g_ptr_array_add(aliases
, pair
);
792 tp_svc_connection_interface_aliasing_emit_aliases_changed(SIPE_CONNECTION(connection
),
795 g_ptr_array_unref(aliases
);
796 g_value_array_free(pair
);
799 struct sipe_backend_private
*sipe_telepathy_connection_private(GObject
*object
)
801 SipeConnection
*self
= SIPE_CONNECTION(object
);
802 /* connected to core already? */
803 if (self
->private.public)
804 return(&self
->private);
810 * Backend adaptor functions
812 void sipe_backend_connection_completed(struct sipe_core_public
*sipe_public
)
814 SipeConnection
*self
= SIPE_PUBLIC_TO_CONNECTION
;
815 TpBaseConnection
*base
= TP_BASE_CONNECTION(self
);
817 /* we are only allowed to do this once */
818 if (base
->status
!= TP_CONNECTION_STATUS_CONNECTED
)
819 tp_base_connection_change_status(base
,
820 TP_CONNECTION_STATUS_CONNECTED
,
821 TP_CONNECTION_STATUS_REASON_REQUESTED
);
824 void sipe_backend_connection_error(struct sipe_core_public
*sipe_public
,
825 sipe_connection_error error
,
828 SipeConnection
*self
= SIPE_PUBLIC_TO_CONNECTION
;
829 TpBaseConnection
*base
= TP_BASE_CONNECTION(self
);
830 GHashTable
*details
= tp_asv_new("server-message", G_TYPE_STRING
, msg
,
832 TpConnectionStatusReason reason
;
835 self
->is_disconnecting
= TRUE
;
838 case SIPE_CONNECTION_ERROR_NETWORK
:
839 reason
= TP_CONNECTION_STATUS_REASON_NETWORK_ERROR
;
840 if (base
->status
== TP_CONNECTION_STATUS_CONNECTING
)
841 name
= TP_ERROR_STR_CONNECTION_FAILED
;
843 name
= TP_ERROR_STR_CONNECTION_LOST
;
846 case SIPE_CONNECTION_ERROR_INVALID_USERNAME
:
847 case SIPE_CONNECTION_ERROR_INVALID_SETTINGS
:
848 case SIPE_CONNECTION_ERROR_AUTHENTICATION_FAILED
:
849 case SIPE_CONNECTION_ERROR_AUTHENTICATION_IMPOSSIBLE
:
850 /* copied from haze code. I agree there should be better ones */
851 reason
= TP_CONNECTION_STATUS_REASON_AUTHENTICATION_FAILED
;
852 name
= TP_ERROR_STR_AUTHENTICATION_FAILED
;
856 reason
= TP_CONNECTION_STATUS_REASON_NONE_SPECIFIED
;
857 name
= TP_ERROR_STR_DISCONNECTED
;
861 SIPE_DEBUG_ERROR("sipe_backend_connection_error: %s (%s)", name
, msg
);
862 tp_base_connection_disconnect_with_dbus_error(base
,
866 g_hash_table_unref(details
);
869 gboolean
sipe_backend_connection_is_disconnecting(struct sipe_core_public
*sipe_public
)
871 SipeConnection
*self
= SIPE_PUBLIC_TO_CONNECTION
;
873 /* disconnect was requested or transport was already disconnected */
874 return(self
->is_disconnecting
||
875 self
->private.transport
== NULL
);
878 gboolean
sipe_backend_connection_is_valid(struct sipe_core_public
*sipe_public
)
880 return(!sipe_backend_connection_is_disconnecting(sipe_public
));
883 const gchar
*sipe_backend_setting(struct sipe_core_public
*sipe_public
,
886 SipeConnection
*self
= SIPE_PUBLIC_TO_CONNECTION
;
890 case SIPE_SETTING_USER_AGENT
:
891 value
= self
->user_agent
;
894 /* @TODO: update when settings are implemented */