Mercurial > pidgin
view src/dbus-server.c @ 11083:8faabf4081ca
[gaim-migrate @ 13101]
sf patch #1225691, from Levi Bard
Implements gc->keepalive for irc
committer: Tailor Script <tailor@pidgin.im>
author | Mark Doliner <mark@kingant.net> |
---|---|
date | Mon, 11 Jul 2005 00:36:28 +0000 |
parents | f54740547c95 |
children | 1c5398ccbeb0 |
line wrap: on
line source
/* * gaim * * Gaim 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., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA * */ #include <dbus/dbus-glib.h> #include <dbus/dbus-glib-bindings.h> #include <stdio.h> #include <stdlib.h> #include <string.h> #include <glib/gi18n.h> #include <glib-object.h> #include <glib/gquark.h> #include <glib.h> #include "account.h" #include "blist.h" #include "conversation.h" #include "dbus-gaim.h" #include "dbus-server.h" #include "debug.h" #include "core.h" #include "value.h" static gint gaim_dbus_pointer_to_id(gpointer node); static gpointer gaim_dbus_id_to_pointer(gint id, GaimDBusPointerType type); /**************************************************************************/ /** @name Lots of GObject stuff I don't really understand */ /**************************************************************************/ GType gaim_object_get_type(void); struct _GaimObject { GObject parent; int ping_signal_id; }; typedef struct { GObjectClass parent; } GaimObjectClass; #define GAIM_DBUS_TYPE_OBJECT (gaim_object_get_type ()) #define GAIM_DBUS_OBJECT(object) (G_TYPE_CHECK_INSTANCE_CAST ((object), GAIM_DBUS_TYPE_OBJECT, GaimObject)) #define GAIM_DBUS_OBJECT_CLASS(klass) (G_TYPE_CHECK_CLASS_CAST ((klass), GAIM_DBUS_TYPE_OBJECT, GaimObjectClass)) #define GAIM_DBUS_IS_OBJECT(object) (G_TYPE_CHECK_INSTANCE_TYPE ((object), GAIM_DBUS_TYPE_OBJECT)) #define GAIM_DBUS_IS_OBJECT_CLASS(klass) (G_TYPE_CHECK_CLASS_TYPE ((klass), GAIM_DBUS_TYPE_OBJECT)) #define GAIM_DBUS_OBJECT_GET_CLASS(obj) (G_TYPE_INSTANCE_GET_CLASS ((obj), GAIM_DBUS_TYPE_OBJECT, GaimObjectClass)) G_DEFINE_TYPE(GaimObject, gaim_object, G_TYPE_OBJECT) GaimObject *gaim_dbus_object; static GQuark gaim_object_error_quark; static const char* null_to_empty(const char *s) { if (s) return s; else return ""; } static void gaim_object_class_init(GaimObjectClass *klass) { } /**************************************************************************/ /** @name Signals */ /**************************************************************************/ /* used in #gaim_values_to_gvalues, undefined afterwards */ #define my_arg(type) (ptr != NULL ? * ((type *)ptr) : va_arg(data, type)) /** Converts from a list of data into an GValue array. @param gvalus Array of empty gvalues to be filled. @param number The number of data items. @param gaim_value Array of #number pointers to GaimValues. The types of of these GaimValues determine the type of data items. The values do not matter. @param mainptr A pointer to a single data item. If this pointer is not #NULL, then #number must be 1. @param data A va_list containing data items. If Exactly one of #mainptr and #data must be not #NULL. If #mainptr is not #NULL, then there is a single piece of data at the address pointed at by #mainptr. If #data is not #NULL, then there are #number data items in the #va_list #data. */ static void gaim_values_to_gvalues(GValue *gvalue, int number, GaimValue **gaim_values, gpointer mainptr, va_list data) { int i; gpointer ptr; g_assert(mainptr == NULL || data == NULL); g_assert(mainptr != NULL || data != NULL); g_assert(number == 1 || data != NULL); for(i=0; i<number; i++, gvalue++) { ptr = mainptr; if (gaim_value_is_outgoing(gaim_values[i])) { ptr = my_arg(gpointer); g_assert(ptr); } switch(gaim_values[i]->type) { case GAIM_TYPE_CHAR: g_value_init(gvalue, G_TYPE_CHAR); g_value_set_char(gvalue, (char) my_arg(int)); break; case GAIM_TYPE_INT: g_value_init(gvalue, G_TYPE_INT); g_value_set_int(gvalue, my_arg(gint)); break; case GAIM_TYPE_UINT: g_value_init(gvalue, G_TYPE_UINT); g_value_set_uint(gvalue, my_arg(guint)); break; case GAIM_TYPE_BOOLEAN: g_value_init(gvalue, G_TYPE_BOOLEAN); g_value_set_boolean(gvalue, my_arg(gboolean)); break; case GAIM_TYPE_STRING: g_value_init(gvalue, G_TYPE_STRING); g_value_set_string(gvalue, null_to_empty(my_arg(char*))); break; case GAIM_TYPE_SUBTYPE: /* registered pointers only! */ g_value_init(gvalue, G_TYPE_INT); g_value_set_int(gvalue, gaim_dbus_pointer_to_id(my_arg(gpointer))); break; case GAIM_TYPE_POINTER: case GAIM_TYPE_OBJECT: case GAIM_TYPE_BOXED: my_arg(gpointer); /* cannot pass general pointers */ g_value_init(gvalue, G_TYPE_INT); g_value_set_int(gvalue, 0); break; default: /* no conversion implemented */ g_assert_not_reached(); } } if (data) va_end(data); } #undef my_arg /* my_arg was only used in gaim_values_to_gvalues */ /** Converts from GaimTypes to GTypes. @param type A GaimType to be converted. @result The result of the conversion (GType). */ static GType gaim_type_to_g_type(GaimType type) { switch(type) { case GAIM_TYPE_CHAR: return G_TYPE_CHAR; case GAIM_TYPE_INT: return G_TYPE_INT; case GAIM_TYPE_UINT: return G_TYPE_UINT; case GAIM_TYPE_BOOLEAN: return G_TYPE_BOOLEAN; case GAIM_TYPE_STRING: return G_TYPE_STRING; case GAIM_TYPE_SUBTYPE: /* registered pointers only! */ return G_TYPE_INT; case GAIM_TYPE_POINTER: case GAIM_TYPE_BOXED: case GAIM_TYPE_OBJECT: return G_TYPE_INT; /* always 0 */ default: /* no conversion implemented */ g_assert_not_reached(); } } static const char *gaim_dbus_convert_signal_name(const char *gaim_name) { int gaim_index, g_index; char *g_name = g_new(char, strlen(gaim_name)+1); gboolean capitalize_next = TRUE; for(gaim_index = g_index = 0; gaim_name[gaim_index]; gaim_index++) if (gaim_name[gaim_index] != '-' && gaim_name[gaim_index] != '_') { if (capitalize_next) g_name[g_index++] = g_ascii_toupper(gaim_name[gaim_index]); else g_name[g_index++] = gaim_name[gaim_index]; capitalize_next = FALSE; } else capitalize_next = TRUE; g_name[g_index] = 0; return g_name; } /* Public signal-related functions */ void gaim_dbus_invalid_marshaller(GClosure *closure, GValue *return_value, guint n_param_values, const GValue *param_values, gpointer invocation_hint, gpointer marshal_data) { g_assert_not_reached(); } int gaim_dbus_signal_register(GaimObject *object, const char *name, GSignalCMarshaller marshaller, int num_values, ...) { va_list args; va_start(args, num_values); return g_signal_new_valist(name, G_OBJECT_TYPE(object), G_SIGNAL_RUN_LAST | G_SIGNAL_DETAILED, NULL, NULL, NULL, marshaller, G_TYPE_NONE, num_values, args); } void gaim_dbus_signal_emit(GaimObject *object, int dbus_id, ...) { va_list args; va_start(args, dbus_id); gaim_dbus_signal_emit_valist(object, dbus_id, args); } void gaim_dbus_signal_emit_valist(GaimObject *object, int dbus_id, va_list args) { g_signal_emit_valist(object, dbus_id, 0, args); } int gaim_dbus_signal_register_gaim(GaimObject *object, const char *name, GSignalCMarshaller marshaller, int num_values, GaimValue **values) { int i; int dbus_id; GType *types; types = g_new0(GType, num_values); for(i=0; i<num_values; i++) types[i] = gaim_type_to_g_type(values[i]->type); dbus_id = g_signal_newv(gaim_dbus_convert_signal_name(name), G_OBJECT_TYPE(object), G_SIGNAL_RUN_LAST | G_SIGNAL_DETAILED, NULL, NULL, NULL, marshaller, G_TYPE_NONE, num_values, types); g_free(types); return dbus_id; } void gaim_dbus_signal_emit_gaim(GaimObject *object, int dbus_id, int num_values, GaimValue **values, va_list vargs) { GValue *args; int i; g_return_if_fail(dbus_id); args = g_new0(GValue, num_values + 1); g_value_init(args + 0, G_OBJECT_TYPE(object)); g_value_set_object(args + 0, object); gaim_values_to_gvalues(args + 1, num_values, values, NULL, vargs); g_signal_emitv(args, dbus_id, 0, NULL); for(i = 1; i <= num_values; i++) g_value_unset(args + i); g_free(args); } /**************************************************************************/ /** @name Utility functions */ /**************************************************************************/ #define error_unless_1(condition, str, parameter) \ if (!(condition)) { \ g_set_error(error, gaim_object_error_quark, \ DBUS_ERROR_NOT_FOUND, \ str, parameter); \ return FALSE; \ } #define error_unless_2(condition, str, a,b) \ if (!(condition)) { \ g_set_error(error, gaim_object_error_quark, \ DBUS_ERROR_NOT_FOUND, \ str, a,b); \ return FALSE; \ } typedef gboolean (*GaimNodeFilter)(GaimBlistNode *node, gpointer *user_data); static gboolean filter_is_buddy(GaimBlistNode *node, gpointer *user_data) { return GAIM_BLIST_NODE_IS_BUDDY(node); } static gboolean filter_is_online_buddy(GaimBlistNode *node, gpointer *user_data) { return GAIM_BLIST_NODE_IS_BUDDY(node) && GAIM_BUDDY_IS_ONLINE((GaimBuddy *)node); } static GList* get_buddy_list (GList *created_list, /**< can be NULL */ GaimBlistNode *gaim_buddy_list, /**< can be NULL */ GaimNodeFilter filter, gpointer user_data) { GaimBlistNode *node; for(node = gaim_buddy_list; node; node = node->next) { if ((*filter)(node, user_data)) created_list = g_list_prepend(created_list, node); created_list = get_buddy_list(created_list, node->child, filter, user_data); } return created_list; } /**************************************************************************/ /** @name Implementations of remote DBUS calls */ /**************************************************************************/ static gboolean gaim_object_ping(GaimObject *object, GError **error) { gaim_dbus_signal_emit(object, object->ping_signal_id, "Ping Pong!"); return TRUE; } static gboolean gaim_object_quit(GaimObject *obj, GError **error) { g_timeout_add(0, gaim_core_quit_cb, NULL); return TRUE; } static gboolean gaim_object_connect_all(GaimObject *obj, GError **error) { GList *cur; for (cur = gaim_accounts_get_all(); cur != NULL; cur = cur->next) gaim_account_connect((GaimAccount*) cur->data); return TRUE; } static gboolean gaim_object_get_buddy_list (GaimObject *obj, GArray **out_buddy_ids, GError **error) { GList *node; GList *buddy_list = get_buddy_list(NULL, gaim_get_blist()->root, &filter_is_buddy, NULL); GArray *buddy_ids = g_array_new(FALSE, TRUE, sizeof(gint32));; buddy_list = g_list_reverse(buddy_list); for (node = buddy_list; node; node = node->next) { gint32 id = gaim_dbus_pointer_to_id(node->data); g_array_append_val(buddy_ids, id); } g_list_free(buddy_list); *out_buddy_ids = buddy_ids; return TRUE; } static gboolean gaim_object_find_account(GaimObject *object, const char *account_name, const char *protocol_name, gint *account_id, GError **error) { GaimAccount *account = gaim_accounts_find(account_name, protocol_name); error_unless_2(account, "Account '%s' with protocol '%s' not found", account_name, protocol_name); *account_id = gaim_dbus_pointer_to_id(account); return TRUE; } static gboolean gaim_object_find_buddy(GaimObject *object, gint account_id, const char *buddy_name, gint *buddy_id, GError **error) { GaimAccount *account; GaimBuddy *buddy; account = gaim_dbus_id_to_pointer(account_id, DBUS_POINTER_ACCOUNT); error_unless_1(account, "Invalid account id: %i", account_id); buddy = gaim_find_buddy(account, buddy_name); error_unless_1(account, "Buddy '%s' not found.", buddy_name); *buddy_id = gaim_dbus_pointer_to_id(buddy); return TRUE; } static gboolean gaim_object_start_im_conversation (GaimObject *object, gint buddy_id, GError **error) { GaimBuddy *buddy = (GaimBuddy*) gaim_dbus_id_to_pointer(buddy_id, DBUS_POINTER_BUDDY); error_unless_1(buddy, "Invalid buddy id: %i", buddy_id); gaim_conversation_new(GAIM_CONV_IM, buddy->account, buddy->name); return TRUE; } /**************************************************************************/ /** @name Gaim DBUS property handling functions */ /**************************************************************************/ typedef struct _GaimDBusProperty { char *name; gint offset; GaimType type; } GaimDBusProperty; /* change GAIM_TYPE into G_TYPE */ static gboolean gaim_dbus_get_property(GaimDBusProperty *list, int list_len, gpointer data, const char *name, GValue *value, GError **error) { int i; for(i=0; i<list_len; i++) { if (!strcmp(list[i].name, name)) { gpointer ptr; GaimValue gaim_value, *gaim_value_ptr; ptr = G_STRUCT_MEMBER_P(data, list[i].offset); gaim_value.type = list[i].type; gaim_value.flags = 0; gaim_value_ptr = &gaim_value; gaim_values_to_gvalues(value, 1, &gaim_value_ptr, ptr, NULL); return TRUE; } } g_value_init(value, G_TYPE_INT); g_value_set_int(value, 0); error_unless_1(FALSE, "Invalid property '%s'", name); } #define DECLARE_PROPERTY(maintype, name, type) {#name, G_STRUCT_OFFSET(maintype, name), type} GaimDBusProperty buddy_properties [] = { DECLARE_PROPERTY(GaimBuddy, name, GAIM_TYPE_STRING), DECLARE_PROPERTY(GaimBuddy, alias, GAIM_TYPE_STRING), DECLARE_PROPERTY(GaimBuddy, server_alias, GAIM_TYPE_STRING), DECLARE_PROPERTY(GaimBuddy, account, GAIM_TYPE_SUBTYPE) }; GaimDBusProperty account_properties [] = { DECLARE_PROPERTY(GaimAccount, username, GAIM_TYPE_STRING), DECLARE_PROPERTY(GaimAccount, alias, GAIM_TYPE_STRING), DECLARE_PROPERTY(GaimAccount, user_info, GAIM_TYPE_STRING), DECLARE_PROPERTY(GaimAccount, protocol_id, GAIM_TYPE_STRING), }; GaimDBusProperty contact_properties [] = { DECLARE_PROPERTY(GaimContact, alias, GAIM_TYPE_STRING), DECLARE_PROPERTY(GaimContact, totalsize, GAIM_TYPE_INT), DECLARE_PROPERTY(GaimContact, currentsize, GAIM_TYPE_INT), DECLARE_PROPERTY(GaimContact, online, GAIM_TYPE_INT), DECLARE_PROPERTY(GaimContact, priority, GAIM_TYPE_SUBTYPE), DECLARE_PROPERTY(GaimContact, priority_valid, GAIM_TYPE_BOOLEAN), }; GaimDBusProperty group_properties [] = { DECLARE_PROPERTY(GaimGroup, name, GAIM_TYPE_STRING), DECLARE_PROPERTY(GaimGroup, totalsize, GAIM_TYPE_INT), DECLARE_PROPERTY(GaimGroup, currentsize, GAIM_TYPE_INT), DECLARE_PROPERTY(GaimGroup, online, GAIM_TYPE_INT), }; GaimDBusProperty chat_properties [] = { DECLARE_PROPERTY(GaimChat, alias, GAIM_TYPE_STRING), DECLARE_PROPERTY(GaimChat, account, GAIM_TYPE_SUBTYPE), }; #define DECLARE_PROPERTY_HANDLER(type, gaim_type) \ static gboolean \ gaim_object_get_##type##_property (GaimObject *object, \ gint id, const char *property_name, \ GValue *value, GError **error) \ { \ gpointer ptr = gaim_dbus_id_to_pointer(id, gaim_type); \ \ error_unless_1(ptr, "Invalid " #type " id: %i", id); \ \ return gaim_dbus_get_property(type##_properties, \ G_N_ELEMENTS(type##_properties), \ ptr, property_name, value, error); \ } DECLARE_PROPERTY_HANDLER(buddy, DBUS_POINTER_BUDDY) DECLARE_PROPERTY_HANDLER(account, DBUS_POINTER_ACCOUNT) DECLARE_PROPERTY_HANDLER(contact, DBUS_POINTER_CONTACT) DECLARE_PROPERTY_HANDLER(group, DBUS_POINTER_GROUP) DECLARE_PROPERTY_HANDLER(chat, DBUS_POINTER_CHAT) #include "dbus-server-bindings.c" /**************************************************************************/ /** @name Gaim DBUS pointer registration mechanism */ /**************************************************************************/ static GHashTable *map_id_node; static GHashTable *map_id_type; static GHashTable *map_node_id; void gaim_dbus_init_ids(void) { map_id_node = g_hash_table_new (g_direct_hash, g_direct_equal); map_id_type = g_hash_table_new (g_direct_hash, g_direct_equal); map_node_id = g_hash_table_new (g_direct_hash, g_direct_equal); } void gaim_dbus_register_pointer(gpointer node, GaimDBusPointerType type) { static gint last_id = 0; g_assert(map_node_id); g_assert(g_hash_table_lookup(map_node_id, node) == NULL); last_id++; g_hash_table_insert(map_node_id, node, GINT_TO_POINTER(last_id)); g_hash_table_insert(map_id_node, GINT_TO_POINTER(last_id), node); g_hash_table_insert(map_id_type, GINT_TO_POINTER(last_id), GINT_TO_POINTER(type)); } void gaim_dbus_unregister_pointer(gpointer node) { gpointer id = g_hash_table_lookup(map_node_id, node); g_hash_table_remove(map_node_id, node); g_hash_table_remove(map_id_node, GINT_TO_POINTER(id)); g_hash_table_remove(map_id_node, GINT_TO_POINTER(id)); } static gint gaim_dbus_pointer_to_id(gpointer node) { g_assert(map_node_id); gint id = GPOINTER_TO_INT(g_hash_table_lookup(map_node_id, node)); g_return_val_if_fail(id, 0); return id; } static gpointer gaim_dbus_id_to_pointer(gint id, GaimDBusPointerType type) { if (type != GPOINTER_TO_INT(g_hash_table_lookup(map_id_type, GINT_TO_POINTER(id)))) return NULL; return g_hash_table_lookup(map_id_node, GINT_TO_POINTER(id)); } /**************************************************************************/ /** @name Gaim DBUS init function */ /**************************************************************************/ gboolean gaim_dbus_connect(GaimObject *object) { DBusGConnection *connection; GError *error = NULL; DBusGProxy *driver_proxy; guint32 request_name_ret; gaim_debug_misc("dbus", "launching dbus server\n"); /* Connect to the bus */ error = NULL; connection = dbus_g_bus_get(DBUS_BUS_STARTER, &error); if (connection == NULL) { g_assert(error); gaim_debug_error("dbus", "Failed to open connection to bus: %s\n", error->message); g_error_free (error); return FALSE; } /* Instantiate the gaim dbus object and register it */ dbus_g_object_type_install_info (GAIM_DBUS_TYPE_OBJECT, &dbus_glib_gaim_object_object_info); dbus_g_connection_register_g_object (connection, DBUS_PATH_GAIM, (GObject*) object); /* Obtain a proxy for the DBus object */ driver_proxy = dbus_g_proxy_new_for_name (connection, DBUS_SERVICE_DBUS, DBUS_PATH_DBUS, DBUS_INTERFACE_DBUS); g_assert(driver_proxy); /* Test whether the registration was successful */ org_freedesktop_DBus_request_name(driver_proxy, DBUS_SERVICE_GAIM, DBUS_NAME_FLAG_PROHIBIT_REPLACEMENT, &request_name_ret, &error); if (error) { gaim_debug_error("dbus", "Failed to get name: %s\n", error->message); g_error_free (error); return FALSE; } if (request_name_ret != DBUS_REQUEST_NAME_REPLY_PRIMARY_OWNER) { gaim_debug_error ("dbus", "Got result code %u from requesting name\n", request_name_ret); return FALSE; } gaim_debug_misc ("dbus", "GLib test service has name '%s'\n", DBUS_SERVICE_GAIM); gaim_debug_misc ("dbus", "GLib test service entering main loop\n"); return TRUE; } static void gaim_object_init(GaimObject *object) { object->ping_signal_id = gaim_dbus_signal_register(object, "PingSignal", g_cclosure_marshal_VOID__STRING, 1, G_TYPE_STRING); } gboolean gaim_dbus_init(void) { gaim_dbus_init_ids(); gaim_object_error_quark = g_quark_from_static_string("org.gaim.GaimError"); gaim_dbus_object = GAIM_DBUS_OBJECT(g_object_new (GAIM_DBUS_TYPE_OBJECT, NULL)); return TRUE; }