/* -*- Mode: C; tab-width: 8; indent-tabs-mode: t; c-basic-offset: 8 -*- */ /* * Copyright (C) 2003-2007 Imendio AB * Copyright (C) 2007-2008 Collabora Ltd. * * 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. * * Authors: Richard Hult * Martyn Russell * Xavier Claessens * Jonny Lamb */ #include "config.h" #include #include #include #include #include #include #include #include #include #include #include #include "empathy-utils.h" #include "empathy-contact-factory.h" #include "empathy-contact-manager.h" #define DEBUG_FLAG EMPATHY_DEBUG_OTHER #include "empathy-debug.h" static void regex_init (void); gchar * empathy_substring (const gchar *str, gint start, gint end) { return g_strndup (str + start, end - start); } /* * Regular Expression code to match urls. */ #define APTCHARS "-A-Za-z0-9,-." #define USERCHARS "-A-Za-z0-9" #define PASSCHARS "-A-Za-z0-9,?;.:/!%$^*&~\"#'" #define HOSTCHARS "-A-Za-z0-9_" #define PATHCHARS "-A-Za-z0-9_$.+!*(),;:@&=?/~#%" #define SCHEME "(news:|telnet:|nntp:|file:/|https?:|ftps?:|webcal:)" #define USER "[" USERCHARS "]+(:["PASSCHARS "]+)?" #define URLPATH "/[" PATHCHARS "]*[^]'.}>) \t\r\n,\\\"]" static regex_t dingus[EMPATHY_REGEX_ALL]; static void regex_init (void) { static gboolean inited = FALSE; const gchar *expression; gint i; if (inited) { return; } for (i = 0; i < EMPATHY_REGEX_ALL; i++) { switch (i) { case EMPATHY_REGEX_AS_IS: expression = SCHEME "//(" USER "@)?[" HOSTCHARS ".]+" "(:[0-9]+)?(" URLPATH ")?"; break; case EMPATHY_REGEX_BROWSER: expression = "(www|ftp)[" HOSTCHARS "]*\\.[" HOSTCHARS ".]+" "(:[0-9]+)?(" URLPATH ")?"; break; case EMPATHY_REGEX_APT: expression = "apt://[" APTCHARS "]*"; break; case EMPATHY_REGEX_EMAIL: expression = "(mailto:)?[a-z0-9][a-z0-9._-]*@[a-z0-9]" "[a-z0-9-]*(\\.[a-z0-9][a-z0-9-]*)+"; break; case EMPATHY_REGEX_OTHER: expression = "news:[-A-Z\\^_a-z{|}~!\"#$%&'()*+,./0-9;:=?`]+" "@[" HOSTCHARS ".]+(:[0-9]+)?"; break; default: /* Silence the compiler. */ expression = NULL; continue; } memset (&dingus[i], 0, sizeof (regex_t)); regcomp (&dingus[i], expression, REG_EXTENDED | REG_ICASE); } inited = TRUE; } gint empathy_regex_match (EmpathyRegExType type, const gchar *msg, GArray *start, GArray *end) { regmatch_t matches[1]; gint ret = 0; gint num_matches = 0; gint offset = 0; gint i; g_return_val_if_fail (type >= 0 || type <= EMPATHY_REGEX_ALL, 0); regex_init (); while (!ret && type != EMPATHY_REGEX_ALL) { ret = regexec (&dingus[type], msg + offset, 1, matches, 0); if (ret == 0) { gint s; num_matches++; s = matches[0].rm_so + offset; offset = matches[0].rm_eo + offset; g_array_append_val (start, s); g_array_append_val (end, offset); } } if (type != EMPATHY_REGEX_ALL) { DEBUG ("Found %d matches for regex type:%d", num_matches, type); return num_matches; } /* If EMPATHY_REGEX_ALL then we run ALL regex's on the string. */ for (i = 0; i < EMPATHY_REGEX_ALL; i++, ret = 0) { while (!ret) { ret = regexec (&dingus[i], msg + offset, 1, matches, 0); if (ret == 0) { gint s; num_matches++; s = matches[0].rm_so + offset; offset = matches[0].rm_eo + offset; g_array_append_val (start, s); g_array_append_val (end, offset); } } } DEBUG ("Found %d matches for ALL regex types", num_matches); return num_matches; } gint empathy_strcasecmp (const gchar *s1, const gchar *s2) { return empathy_strncasecmp (s1, s2, -1); } gint empathy_strncasecmp (const gchar *s1, const gchar *s2, gsize n) { gchar *u1, *u2; gint ret_val; u1 = g_utf8_casefold (s1, n); u2 = g_utf8_casefold (s2, n); ret_val = g_utf8_collate (u1, u2); g_free (u1); g_free (u2); return ret_val; } gboolean empathy_xml_validate (xmlDoc *doc, const gchar *dtd_filename) { gchar *path, *escaped; xmlValidCtxt cvp; xmlDtd *dtd; gboolean ret; path = g_build_filename (g_getenv ("EMPATHY_SRCDIR"), "libempathy", dtd_filename, NULL); if (!g_file_test (path, G_FILE_TEST_EXISTS)) { g_free (path); path = g_build_filename (DATADIR, "empathy", dtd_filename, NULL); } DEBUG ("Loading dtd file %s", path); /* The list of valid chars is taken from libxml. */ escaped = xmlURIEscapeStr (path, ":@&=+$,/?;"); g_free (path); memset (&cvp, 0, sizeof (cvp)); dtd = xmlParseDTD (NULL, escaped); ret = xmlValidateDtd (&cvp, doc, dtd); xmlFree (escaped); xmlFreeDtd (dtd); return ret; } xmlNodePtr empathy_xml_node_get_child (xmlNodePtr node, const gchar *child_name) { xmlNodePtr l; g_return_val_if_fail (node != NULL, NULL); g_return_val_if_fail (child_name != NULL, NULL); for (l = node->children; l; l = l->next) { if (l->name && strcmp (l->name, child_name) == 0) { return l; } } return NULL; } xmlChar * empathy_xml_node_get_child_content (xmlNodePtr node, const gchar *child_name) { xmlNodePtr l; g_return_val_if_fail (node != NULL, NULL); g_return_val_if_fail (child_name != NULL, NULL); l = empathy_xml_node_get_child (node, child_name); if (l) { return xmlNodeGetContent (l); } return NULL; } xmlNodePtr empathy_xml_node_find_child_prop_value (xmlNodePtr node, const gchar *prop_name, const gchar *prop_value) { xmlNodePtr l; xmlNodePtr found = NULL; g_return_val_if_fail (node != NULL, NULL); g_return_val_if_fail (prop_name != NULL, NULL); g_return_val_if_fail (prop_value != NULL, NULL); for (l = node->children; l && !found; l = l->next) { xmlChar *prop; if (!xmlHasProp (l, prop_name)) { continue; } prop = xmlGetProp (l, prop_name); if (prop && strcmp (prop, prop_value) == 0) { found = l; } xmlFree (prop); } return found; } guint empathy_account_hash (gconstpointer key) { g_return_val_if_fail (MC_IS_ACCOUNT (key), 0); return g_str_hash (mc_account_get_unique_name (MC_ACCOUNT (key))); } gboolean empathy_account_equal (gconstpointer a, gconstpointer b) { const gchar *name_a; const gchar *name_b; g_return_val_if_fail (MC_IS_ACCOUNT (a), FALSE); g_return_val_if_fail (MC_IS_ACCOUNT (b), FALSE); name_a = mc_account_get_unique_name (MC_ACCOUNT (a)); name_b = mc_account_get_unique_name (MC_ACCOUNT (b)); return g_str_equal (name_a, name_b); } MissionControl * empathy_mission_control_new (void) { static MissionControl *mc = NULL; if (!mc) { mc = mission_control_new (tp_get_bus ()); g_object_add_weak_pointer (G_OBJECT (mc), (gpointer) &mc); } else { g_object_ref (mc); } return mc; } const gchar * empathy_presence_get_default_message (McPresence presence) { switch (presence) { case MC_PRESENCE_AVAILABLE: return _("Available"); case MC_PRESENCE_DO_NOT_DISTURB: return _("Busy"); case MC_PRESENCE_AWAY: case MC_PRESENCE_EXTENDED_AWAY: return _("Away"); case MC_PRESENCE_HIDDEN: return _("Hidden"); case MC_PRESENCE_OFFLINE: case MC_PRESENCE_UNSET: return _("Offline"); default: g_assert_not_reached (); } return NULL; } const gchar * empathy_presence_to_str (McPresence presence) { switch (presence) { case MC_PRESENCE_AVAILABLE: return "available"; case MC_PRESENCE_DO_NOT_DISTURB: return "busy"; case MC_PRESENCE_AWAY: return "away"; case MC_PRESENCE_EXTENDED_AWAY: return "ext_away"; case MC_PRESENCE_HIDDEN: return "hidden"; case MC_PRESENCE_OFFLINE: return "offline"; case MC_PRESENCE_UNSET: return "unset"; default: g_assert_not_reached (); } return NULL; } McPresence empathy_presence_from_str (const gchar *str) { if (strcmp (str, "available") == 0) { return MC_PRESENCE_AVAILABLE; } else if ((strcmp (str, "dnd") == 0) || (strcmp (str, "busy") == 0)) { return MC_PRESENCE_DO_NOT_DISTURB; } else if ((strcmp (str, "away") == 0) || (strcmp (str, "brb") == 0)) { return MC_PRESENCE_AWAY; } else if ((strcmp (str, "xa") == 0) || (strcmp (str, "ext_away") == 0)) { return MC_PRESENCE_EXTENDED_AWAY; } else if (strcmp (str, "hidden") == 0) { return MC_PRESENCE_HIDDEN; } else if (strcmp (str, "offline") == 0) { return MC_PRESENCE_OFFLINE; } else if (strcmp (str, "unset") == 0) { return MC_PRESENCE_UNSET; } return MC_PRESENCE_UNSET; } gchar * empathy_file_lookup (const gchar *filename, const gchar *subdir) { gchar *path; if (!subdir) { subdir = "."; } path = g_build_filename (g_getenv ("EMPATHY_SRCDIR"), subdir, filename, NULL); if (!g_file_test (path, G_FILE_TEST_EXISTS)) { g_free (path); path = g_build_filename (DATADIR, "empathy", filename, NULL); } return path; } typedef struct { EmpathyRunUntilReadyFunc func; gpointer user_data; GObject *object; GMainLoop *loop; } RunUntilReadyData; static void run_until_ready_cb (RunUntilReadyData *data) { if (!data->func || data->func (data->object, data->user_data)) { DEBUG ("Object %p is ready", data->object); g_main_loop_quit (data->loop); } } static gboolean object_is_ready (GObject *object, gpointer user_data) { gboolean ready; g_object_get (object, "ready", &ready, NULL); return ready; } void empathy_run_until_ready_full (gpointer object, const gchar *signal, EmpathyRunUntilReadyFunc func, gpointer user_data, GMainLoop **loop) { RunUntilReadyData data; gulong signal_id; g_return_if_fail (G_IS_OBJECT (object)); g_return_if_fail (signal != NULL); if (func && func (object, user_data)) { return; } DEBUG ("Starting run until ready for object %p", object); data.func = func; data.user_data = user_data; data.object = object; data.loop = g_main_loop_new (NULL, FALSE); signal_id = g_signal_connect_swapped (object, signal, G_CALLBACK (run_until_ready_cb), &data); if (loop != NULL) { *loop = data.loop; } g_main_loop_run (data.loop); if (loop != NULL) { *loop = NULL; } g_signal_handler_disconnect (object, signal_id); g_main_loop_unref (data.loop); } void empathy_run_until_ready (gpointer object) { empathy_run_until_ready_full (object, "notify::ready", object_is_ready, NULL, NULL); } McAccount * empathy_channel_get_account (TpChannel *channel) { TpConnection *connection; McAccount *account; MissionControl *mc; g_object_get (channel, "connection", &connection, NULL); mc = empathy_mission_control_new (); account = mission_control_get_account_for_tpconnection (mc, connection, NULL); g_object_unref (connection); g_object_unref (mc); return account; } typedef void (*AccountStatusChangedFunc) (MissionControl *mc, TpConnectionStatus status, McPresence presence, TpConnectionStatusReason reason, const gchar *unique_name, gpointer user_data); typedef struct { AccountStatusChangedFunc handler; gpointer user_data; GClosureNotify free_func; MissionControl *mc; } AccountStatusChangedData; typedef struct { TpConnectionStatus status; McPresence presence; TpConnectionStatusReason reason; gchar *unique_name; AccountStatusChangedData *data; } InvocationData; static void account_status_changed_data_free (gpointer ptr, GClosure *closure) { AccountStatusChangedData *data = ptr; if (data->free_func) { data->free_func (data->user_data, closure); } g_object_unref (data->mc); g_slice_free (AccountStatusChangedData, data); } static gboolean account_status_changed_invoke_callback (gpointer data) { InvocationData *invocation_data = data; invocation_data->data->handler (invocation_data->data->mc, invocation_data->status, invocation_data->presence, invocation_data->reason, invocation_data->unique_name, invocation_data->data->user_data); g_free (invocation_data->unique_name); g_slice_free (InvocationData, invocation_data); return FALSE; } static void account_status_changed_cb (MissionControl *mc, TpConnectionStatus status, McPresence presence, TpConnectionStatusReason reason, const gchar *unique_name, AccountStatusChangedData *data) { InvocationData *invocation_data; invocation_data = g_slice_new (InvocationData); invocation_data->status = status; invocation_data->presence = presence; invocation_data->reason = reason; invocation_data->unique_name = g_strdup (unique_name); invocation_data->data = data; g_idle_add_full (G_PRIORITY_HIGH, account_status_changed_invoke_callback, invocation_data, NULL); } gpointer empathy_connect_to_account_status_changed (MissionControl *mc, GCallback handler, gpointer user_data, GClosureNotify free_func) { AccountStatusChangedData *data; g_return_val_if_fail (IS_MISSIONCONTROL (mc), NULL); g_return_val_if_fail (handler != NULL, NULL); data = g_slice_new (AccountStatusChangedData); data->handler = (AccountStatusChangedFunc) handler; data->user_data = user_data; data->free_func = free_func; data->mc = g_object_ref (mc); dbus_g_proxy_connect_signal (DBUS_G_PROXY (mc), "AccountStatusChanged", G_CALLBACK (account_status_changed_cb), data, account_status_changed_data_free); return data; } void empathy_disconnect_account_status_changed (gpointer token) { AccountStatusChangedData *data = token; dbus_g_proxy_disconnect_signal (DBUS_G_PROXY (data->mc), "AccountStatusChanged", G_CALLBACK (account_status_changed_cb), data); } guint empathy_proxy_hash (gconstpointer key) { TpProxy *proxy = TP_PROXY (key); TpProxyClass *proxy_class = TP_PROXY_GET_CLASS (key); g_return_val_if_fail (TP_IS_PROXY (proxy), 0); g_return_val_if_fail (proxy_class->must_have_unique_name, 0); return g_str_hash (proxy->object_path) ^ g_str_hash (proxy->bus_name); } gboolean empathy_proxy_equal (gconstpointer a, gconstpointer b) { TpProxy *proxy_a = TP_PROXY (a); TpProxy *proxy_b = TP_PROXY (b); TpProxyClass *proxy_a_class = TP_PROXY_GET_CLASS (a); TpProxyClass *proxy_b_class = TP_PROXY_GET_CLASS (b); g_return_val_if_fail (TP_IS_PROXY (proxy_a), FALSE); g_return_val_if_fail (TP_IS_PROXY (proxy_b), FALSE); g_return_val_if_fail (proxy_a_class->must_have_unique_name, 0); g_return_val_if_fail (proxy_b_class->must_have_unique_name, 0); return g_str_equal (proxy_a->object_path, proxy_b->object_path) && g_str_equal (proxy_a->bus_name, proxy_b->bus_name); } typedef struct { gint timeout_ms; gchar *channel_type; guint handle_type; empathy_connection_callback_for_request_channel callback; gpointer user_data; GDestroyNotify destroy; TpHandle handle; gboolean suppress_handler; guint ref_count; } ConnectionRequestChannelData; static void connection_request_channel_data_unref (gpointer user_data) { ConnectionRequestChannelData *data = (ConnectionRequestChannelData*) user_data; if (--data->ref_count == 0) { g_free (data->channel_type); if (data->destroy) { data->destroy (data->user_data); } g_slice_free (ConnectionRequestChannelData, data); } } static void connection_request_channel_cb (TpConnection *connection, const gchar *object_path, const GError *error, gpointer user_data, GObject *weak_object) { ConnectionRequestChannelData *data = (ConnectionRequestChannelData*) user_data; TpChannel *channel; if (!data->callback) { return; } if (error) { data->callback (connection, NULL, error, data->user_data, weak_object); return; } channel = tp_channel_new (connection, object_path, data->channel_type, data->handle_type, data->handle, NULL); data->callback (connection, channel, NULL, data->user_data, weak_object); g_object_unref (channel); } static void connection_request_handles_cb (TpConnection *connection, const GArray *handles, const GError *error, gpointer user_data, GObject *weak_object) { ConnectionRequestChannelData *data = (ConnectionRequestChannelData*) user_data; if (error) { if (data->callback) { data->callback (connection, NULL, error, data->user_data, weak_object); } return; } data->handle = g_array_index (handles, guint, 0); data->ref_count++; tp_cli_connection_call_request_channel (connection, data->timeout_ms, data->channel_type, data->handle_type, data->handle, data->suppress_handler, connection_request_channel_cb, data, (GDestroyNotify) connection_request_channel_data_unref, weak_object); } void empathy_connection_request_channel (TpConnection *connection, gint timeout_ms, const gchar *channel_type, guint handle_type, const gchar *name, gboolean suppress_handler, empathy_connection_callback_for_request_channel callback, gpointer user_data, GDestroyNotify destroy, GObject *weak_object) { const gchar *names[] = {name, NULL}; ConnectionRequestChannelData *data; data = g_slice_new (ConnectionRequestChannelData); data->timeout_ms = timeout_ms; data->channel_type = g_strdup (channel_type); data->handle_type = handle_type; data->callback = callback; data->user_data = user_data; data->destroy = destroy; data->handle = 0; data->suppress_handler = suppress_handler; data->ref_count = 1; tp_cli_connection_call_request_handles (connection, timeout_ms, handle_type, names, connection_request_handles_cb, data, (GDestroyNotify) connection_request_channel_data_unref, weak_object); } EmpathyTpFile * empathy_send_file (EmpathyContact *contact, GFile *gfile) { GFileInfo *info; guint64 size; GInputStream *in_stream = NULL; MissionControl *mc; McAccount *account; TpConnection *connection; guint handle; gchar *object_path; TpChannel *channel; EmpathyTpFile *tp_file; GError *error = NULL; GValue value = { 0 }; gchar *filename; g_return_val_if_fail (EMPATHY_IS_CONTACT (contact), NULL); g_return_val_if_fail (G_IS_FILE (gfile), NULL); info = g_file_query_info (gfile, G_FILE_ATTRIBUTE_STANDARD_SIZE "," G_FILE_ATTRIBUTE_STANDARD_CONTENT_TYPE, 0, NULL, NULL); size = info ? g_file_info_get_size (info) : EMPATHY_TP_FILE_UNKNOWN_SIZE; filename = g_file_get_basename (gfile); in_stream = G_INPUT_STREAM (g_file_read (gfile, NULL, NULL)); mc = empathy_mission_control_new (); account = empathy_contact_get_account (contact); connection = mission_control_get_tpconnection (mc, account, NULL); tp_connection_run_until_ready (connection, FALSE, NULL, NULL); handle = empathy_contact_get_handle (contact); DEBUG ("Sending %s from a stream to %s (size %llu, content-type %s)", filename, empathy_contact_get_name (contact), size, g_file_info_get_content_type (info)); if (!tp_cli_connection_run_request_channel (connection, -1, EMP_IFACE_CHANNEL_TYPE_FILE, TP_HANDLE_TYPE_CONTACT, handle, FALSE, &object_path, &error, NULL)) { DEBUG ("Couldn't request channel: %s", error ? error->message : "No error given"); g_clear_error (&error); g_object_unref (mc); g_object_unref (connection); return NULL; } channel = tp_channel_new (connection, object_path, EMP_IFACE_CHANNEL_TYPE_FILE, TP_HANDLE_TYPE_CONTACT, handle, NULL); /* FIXME: this should go in CreateChannel in the new requests API */ g_value_init (&value, G_TYPE_STRING); g_value_set_string (&value, g_filename_display_basename (filename)); tp_cli_dbus_properties_call_set (TP_PROXY (channel), -1, EMP_IFACE_CHANNEL_TYPE_FILE, "Filename", &value, NULL, NULL, NULL, NULL); g_value_reset (&value); g_value_set_string (&value, g_file_info_get_content_type (info)); tp_cli_dbus_properties_call_set (TP_PROXY (channel), -1, EMP_IFACE_CHANNEL_TYPE_FILE, "ContentType", &value, NULL, NULL, NULL, NULL); g_value_unset (&value); g_value_init (&value, G_TYPE_UINT64); g_value_set_uint64 (&value, size); tp_cli_dbus_properties_call_set (TP_PROXY (channel), -1, EMP_IFACE_CHANNEL_TYPE_FILE, "Size", &value, NULL, NULL, NULL, NULL); g_value_unset (&value); tp_file = empathy_tp_file_new (account, channel); if (tp_file) { empathy_tp_file_set_input_stream (tp_file, in_stream); } empathy_tp_file_offer (tp_file); g_object_unref (mc); g_object_unref (connection); g_object_unref (channel); g_free (object_path); return tp_file; }