jabber.c
来自「Linux下的多协议即时通讯程序源代码」· C语言 代码 · 共 855 行 · 第 1/2 页
C
855 行
/* * purple - Bonjour Protocol Plugin * * Purple 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 */#ifndef _WIN32#include <sys/socket.h>#include <netinet/in.h>#include <arpa/inet.h>#else#include "libc_interface.h"#endif#include <sys/types.h>#include <glib.h>#include <unistd.h>#include <fcntl.h>#include "network.h"#include "eventloop.h"#include "connection.h"#include "blist.h"#include "xmlnode.h"#include "debug.h"#include "notify.h"#include "util.h"#include "jabber.h"#include "bonjour.h"#include "buddy.h"#define STREAM_END "</stream:stream>"/* TODO: specify version='1.0' and send stream features */#define DOCTYPE "<?xml version=\"1.0\" encoding=\"UTF-8\" ?>\n" \ "<stream:stream xmlns=\"jabber:client\" xmlns:stream=\"http://etherx.jabber.org/streams\" from=\"%s\" to=\"%s\">"#if 0 /* this isn't used anywhere... */static const char *_font_size_purple_to_ichat(int size){ switch (size) { case 1: return "8"; case 2: return "10"; case 3: return "12"; case 4: return "14"; case 5: return "17"; case 6: return "21"; case 7: return "24"; } return "12";}#endifstatic BonjourJabberConversation *bonjour_jabber_conv_new() { BonjourJabberConversation *bconv = g_new0(BonjourJabberConversation, 1); bconv->socket = -1; bconv->tx_buf = purple_circ_buffer_new(512); bconv->tx_handler = -1; bconv->rx_handler = -1; return bconv;}static const char *_font_size_ichat_to_purple(int size){ if (size > 24) { return "7"; } else if (size >= 21) { return "6"; } else if (size >= 17) { return "5"; } else if (size >= 14) { return "4"; } else if (size >= 12) { return "3"; } else if (size >= 10) { return "2"; } return "1";}static void_jabber_parse_and_write_message_to_ui(xmlnode *message_node, PurpleConnection *connection, PurpleBuddy *pb){ xmlnode *body_node, *html_node, *events_node; char *body, *html_body = NULL; const char *ichat_balloon_color = NULL; const char *ichat_text_color = NULL; const char *font_face = NULL; const char *font_size = NULL; const char *font_color = NULL; gboolean composing_event = FALSE; body_node = xmlnode_get_child(message_node, "body"); if (body_node == NULL) return; body = xmlnode_get_data(body_node); html_node = xmlnode_get_child(message_node, "html"); if (html_node != NULL) { xmlnode *html_body_node; html_body_node = xmlnode_get_child(html_node, "body"); if (html_body_node != NULL) { xmlnode *html_body_font_node; ichat_balloon_color = xmlnode_get_attrib(html_body_node, "ichatballooncolor"); ichat_text_color = xmlnode_get_attrib(html_body_node, "ichattextcolor"); html_body_font_node = xmlnode_get_child(html_body_node, "font"); if (html_body_font_node != NULL) { /* Types of messages sent by iChat */ font_face = xmlnode_get_attrib(html_body_font_node, "face"); /* The absolute iChat font sizes should be converted to 1..7 range */ font_size = xmlnode_get_attrib(html_body_font_node, "ABSZ"); if (font_size != NULL) font_size = _font_size_ichat_to_purple(atoi(font_size)); font_color = xmlnode_get_attrib(html_body_font_node, "color"); html_body = xmlnode_get_data(html_body_font_node); if (html_body == NULL) /* This is the kind of formated messages that Purple creates */ html_body = xmlnode_to_str(html_body_font_node, NULL); } } } events_node = xmlnode_get_child_with_namespace(message_node, "x", "jabber:x:event"); if (events_node != NULL) { if (xmlnode_get_child(events_node, "composing") != NULL) composing_event = TRUE; if (xmlnode_get_child(events_node, "id") != NULL) { /* The user is just typing */ /* TODO: Deal with typing notification */ g_free(body); g_free(html_body); return; } } /* Compose the message */ if (html_body != NULL) { g_free(body); if (font_face == NULL) font_face = "Helvetica"; if (font_size == NULL) font_size = "3"; if (ichat_text_color == NULL) ichat_text_color = "#000000"; if (ichat_balloon_color == NULL) ichat_balloon_color = "#FFFFFF"; body = g_strdup_printf("<font face='%s' size='%s' color='%s' back='%s'>%s</font>", font_face, font_size, ichat_text_color, ichat_balloon_color, html_body); } /* TODO: Should we do something with "composing_event" here? */ /* Send the message to the UI */ serv_got_im(connection, pb->name, body, 0, time(NULL)); g_free(body); g_free(html_body);}struct _check_buddy_by_address_t { const char *address; PurpleBuddy **pb; BonjourJabber *bj;};static void_check_buddy_by_address(gpointer key, gpointer value, gpointer data){ PurpleBuddy *pb = value; BonjourBuddy *bb; struct _check_buddy_by_address_t *cbba = data; /* * If the current PurpleBuddy's data is not null and the PurpleBuddy's account * is the same as the account requesting the check then continue to determine * whether the buddies IP matches the target IP. */ if (cbba->bj->account == pb->account) { bb = pb->proto_data; if ((bb != NULL) && (g_ascii_strcasecmp(bb->ip, cbba->address) == 0)) *(cbba->pb) = pb; }}static gint_read_data(gint socket, char **message){ GString *data = g_string_new(""); char partial_data[512]; gint total_message_length = 0; gint partial_message_length = 0; /* Read chunks of 512 bytes till the end of the data */ while ((partial_message_length = recv(socket, partial_data, 512, 0)) > 0) { g_string_append_len(data, partial_data, partial_message_length); total_message_length += partial_message_length; } if (partial_message_length == -1) { if (errno != EAGAIN) purple_debug_warning("bonjour", "receive error: %s\n", strerror(errno)); if (total_message_length == 0) { return -1; } } *message = g_string_free(data, FALSE); if (total_message_length != 0) purple_debug_info("bonjour", "Receive: -%s- %d bytes\n", *message, total_message_length); return total_message_length;}static void_send_data_write_cb(gpointer data, gint source, PurpleInputCondition cond){ PurpleBuddy *pb = data; BonjourBuddy *bb = pb->proto_data; BonjourJabberConversation *bconv = bb->conversation; int ret, writelen; /* TODO: Make sure that the stream has been established before sending */ writelen = purple_circ_buffer_get_max_read(bconv->tx_buf); if (writelen == 0) { purple_input_remove(bconv->tx_handler); bconv->tx_handler = -1; return; } ret = send(bconv->socket, bconv->tx_buf->outptr, writelen, 0); if (ret < 0 && errno == EAGAIN) return; else if (ret <= 0) { PurpleConversation *conv; const char *error = strerror(errno); purple_debug_error("bonjour", "Error sending message to buddy %s error: %s\n", purple_buddy_get_name(pb), error ? error : "(null)"); conv = purple_find_conversation_with_account(PURPLE_CONV_TYPE_IM, bb->name, pb->account); if (conv != NULL) purple_conversation_write(conv, NULL, _("Unable to send message."), PURPLE_MESSAGE_SYSTEM, time(NULL)); bonjour_jabber_close_conversation(bb->conversation); bb->conversation = NULL; return; } purple_circ_buffer_mark_read(bconv->tx_buf, ret);}static gint_send_data(PurpleBuddy *pb, char *message){ gint ret; int len = strlen(message); BonjourBuddy *bb = pb->proto_data; BonjourJabberConversation *bconv = bb->conversation; /* If we're not ready to actually send, append it to the buffer */ if (bconv->tx_handler != -1 || bconv->connect_data != NULL || !bconv->stream_started || purple_circ_buffer_get_max_read(bconv->tx_buf) > 0) { ret = -1; errno = EAGAIN; } else { ret = send(bconv->socket, message, len, 0); } if (ret == -1 && errno == EAGAIN) ret = 0; else if (ret <= 0) { PurpleConversation *conv; const char *error = strerror(errno); purple_debug_error("bonjour", "Error sending message to buddy %s error: %s\n", purple_buddy_get_name(pb), error ? error : "(null)"); conv = purple_find_conversation_with_account(PURPLE_CONV_TYPE_IM, bb->name, pb->account); if (conv != NULL) purple_conversation_write(conv, NULL, _("Unable to send message."), PURPLE_MESSAGE_SYSTEM, time(NULL)); bonjour_jabber_close_conversation(bb->conversation); bb->conversation = NULL; return -1; } if (ret < len) { if (bconv->tx_handler == -1) bconv->tx_handler = purple_input_add(bconv->socket, PURPLE_INPUT_WRITE, _send_data_write_cb, pb); purple_circ_buffer_append(bconv->tx_buf, message + ret, len - ret); } return ret;}static void_client_socket_handler(gpointer data, gint socket, PurpleInputCondition condition){ char *message = NULL; gint message_length; PurpleBuddy *pb = data; PurpleAccount *account = pb->account; BonjourBuddy *bb = pb->proto_data; gboolean closed_conversation = FALSE; xmlnode *message_node; /* Read the data from the socket */ if ((message_length = _read_data(socket, &message)) == -1) { /* There have been an error reading from the socket */ if (errno != EAGAIN) { bonjour_jabber_close_conversation(bb->conversation); bb->conversation = NULL; /* I guess we really don't need to notify the user. * If they try to send another message it'll reconnect */ } return; } else if (message_length == 0) { /* The other end has closed the socket */ closed_conversation = TRUE; } else { message[message_length] = '\0'; while (g_ascii_iscntrl(message[message_length - 1])) { message[message_length - 1] = '\0'; message_length--; } } /* Parse the message into an XMLnode for analysis */ message_node = xmlnode_from_str(message, strlen(message)); /* * Check that this is not the end of the conversation. This is * using a magic string, but xmlnode won't play nice when just * parsing an end tag */ if (closed_conversation || purple_str_has_prefix(message, STREAM_END)) { PurpleConversation *conv; /* Close the socket, clear the watcher and free memory */ bonjour_jabber_close_conversation(bb->conversation); bb->conversation = NULL; /* Inform the user that the conversation has been closed */ conv = purple_find_conversation_with_account(PURPLE_CONV_TYPE_IM, pb->name, account); if (conv != NULL) { char *tmp = g_strdup_printf(_("%s has closed the conversation."), pb->name); purple_conversation_write(conv, NULL, tmp, PURPLE_MESSAGE_SYSTEM, time(NULL)); g_free(tmp); } } else if (message_node != NULL) { /* Parse the message to get the data and send to the ui */ _jabber_parse_and_write_message_to_ui(message_node, account->gc, pb); } else { /* TODO: Deal with receiving only a partial message */ } g_free(message); if (message_node != NULL) xmlnode_free(message_node);}struct _stream_start_data { char *msg; PurpleInputFunction tx_handler_cb;};static void_start_stream(gpointer data, gint source, PurpleInputCondition condition){ PurpleBuddy *pb = data; BonjourBuddy *bb = pb->proto_data; struct _stream_start_data *ss = bb->conversation->stream_data; int len, ret; len = strlen(ss->msg); /* Start Stream */ ret = send(source, ss->msg, len, 0); if (ret == -1 && errno == EAGAIN)
⌨️ 快捷键说明
复制代码Ctrl + C
搜索代码Ctrl + F
全屏模式F11
增大字号Ctrl + =
减小字号Ctrl + -
显示快捷键?