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 + -
显示快捷键?