⭐ 欢迎来到虫虫下载站! | 📦 资源下载 📁 资源专辑 ℹ️ 关于我们
⭐ 虫虫下载站

📄 family_chat.c

📁 Linux下的多协议即时通讯程序源代码
💻 C
字号:
/* * Purple's oscar protocol plugin * This file is the legal property of its developers. * Please see the AUTHORS file distributed alongside this file. * * This library is free software; you can redistribute it and/or * modify it under the terms of the GNU Lesser General Public * License as published by the Free Software Foundation; either * version 2 of the License, or (at your option) any later version. * * This library 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 * Lesser General Public License for more details. * * You should have received a copy of the GNU Lesser General Public * License along with this library; if not, write to the Free Software * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA  02111-1307  USA*//* * Family 0x000e - Routines for the Chat service. * */#include "oscar.h"#include <string.h>/* Stored in the ->internal of chat connections */struct chatconnpriv{	guint16 exchange;	char *name;	guint16 instance;};voidflap_connection_destroy_chat(OscarData *od, FlapConnection *conn){	struct chatconnpriv *ccp = (struct chatconnpriv *)conn->internal;	if (ccp)		g_free(ccp->name);	g_free(ccp);	return;}char *aim_chat_getname(FlapConnection *conn){	struct chatconnpriv *ccp;	if (!conn)		return NULL;	if (conn->type != SNAC_FAMILY_CHAT)		return NULL;	ccp = (struct chatconnpriv *)conn->internal;	return ccp->name;}/* XXX get this into conn.c -- evil!! */FlapConnection *aim_chat_getconn(OscarData *od, const char *name){	GSList *cur;	for (cur = od->oscar_connections; cur; cur = cur->next)	{		FlapConnection *conn;		struct chatconnpriv *ccp;		conn = cur->data;		ccp = (struct chatconnpriv *)conn->internal;		if (conn->type != SNAC_FAMILY_CHAT)			continue;		if (!conn->internal) {			purple_debug_misc("oscar", "faim: chat: chat connection with no name! (fd = %d)\n", conn->fd);			continue;		}		if (strcmp(ccp->name, name) == 0)			return conn;;	}	return NULL;}intaim_chat_attachname(FlapConnection *conn, guint16 exchange, const char *roomname, guint16 instance){	struct chatconnpriv *ccp;	if (!conn || !roomname)		return -EINVAL;	if (conn->internal)		g_free(conn->internal);	ccp = g_new(struct chatconnpriv, 1);	ccp->exchange = exchange;	ccp->name = g_strdup(roomname);	ccp->instance = instance;	conn->internal = (void *)ccp;	return 0;}intaim_chat_readroominfo(ByteStream *bs, struct aim_chat_roominfo *outinfo){	if (!bs || !outinfo)		return 0;	outinfo->exchange = byte_stream_get16(bs);	outinfo->namelen = byte_stream_get8(bs);	outinfo->name = (char *)byte_stream_getraw(bs, outinfo->namelen);	outinfo->instance = byte_stream_get16(bs);	return 0;}intaim_chat_leaveroom(OscarData *od, const char *name){	FlapConnection *conn;	if (!(conn = aim_chat_getconn(od, name)))		return -ENOENT;	flap_connection_close(od, conn);	return 0;}/* * Subtype 0x0002 - General room information.  Lots of stuff. * * Values I know are in here but I haven't attached * them to any of the 'Unknown's: *	- Language (English) * */static intinfoupdate(OscarData *od, FlapConnection *conn, aim_module_t *mod, FlapFrame *frame, aim_modsnac_t *snac, ByteStream *bs){	aim_userinfo_t *userinfo = NULL;	aim_rxcallback_t userfunc;	int ret = 0;	int usercount;	guint8 detaillevel = 0;	char *roomname;	struct aim_chat_roominfo roominfo;	guint16 tlvcount = 0;	GSList *tlvlist;	aim_tlv_t *tlv;	char *roomdesc;	guint16 flags;	guint32 creationtime;	guint16 maxmsglen, maxvisiblemsglen;	guint16 unknown_d2, unknown_d5;	aim_chat_readroominfo(bs, &roominfo);	detaillevel = byte_stream_get8(bs);	if (detaillevel != 0x02) {		purple_debug_misc("oscar", "faim: chat_roomupdateinfo: detail level %d not supported\n", detaillevel);		return 1;	}	tlvcount = byte_stream_get16(bs);	/*	 * Everything else are TLVs.	 */	tlvlist = aim_tlvlist_read(bs);	/*	 * TLV type 0x006a is the room name in Human Readable Form.	 */	roomname = aim_tlv_getstr(tlvlist, 0x006a, 1);	/*	 * Type 0x006f: Number of occupants.	 */	usercount = aim_tlv_get16(tlvlist, 0x006f, 1);	/*	 * Type 0x0073:  Occupant list.	 */	tlv = aim_tlv_gettlv(tlvlist, 0x0073, 1);	if (tlv != NULL)	{		int curoccupant = 0;		ByteStream occbs;		/* Allocate enough userinfo structs for all occupants */		userinfo = g_new0(aim_userinfo_t, usercount);		byte_stream_init(&occbs, tlv->value, tlv->length);		while (curoccupant < usercount)			aim_info_extract(od, &occbs, &userinfo[curoccupant++]);	}	/*	 * Type 0x00c9: Flags. (AIM_CHATROOM_FLAG)	 */	flags = aim_tlv_get16(tlvlist, 0x00c9, 1);	/*	 * Type 0x00ca: Creation time (4 bytes)	 */	creationtime = aim_tlv_get32(tlvlist, 0x00ca, 1);	/*	 * Type 0x00d1: Maximum Message Length	 */	maxmsglen = aim_tlv_get16(tlvlist, 0x00d1, 1);	/*	 * Type 0x00d2: Unknown. (2 bytes)	 */	unknown_d2 = aim_tlv_get16(tlvlist, 0x00d2, 1);	/*	 * Type 0x00d3: Room Description	 */	roomdesc = aim_tlv_getstr(tlvlist, 0x00d3, 1);#if 0	/*	 * Type 0x000d4: Unknown (flag only)	 */	if (aim_tlv_gettlv(tlvlist, 0x000d4, 1)) {		/* Unhandled */	}#endif	/*	 * Type 0x00d5: Unknown. (1 byte)	 */	unknown_d5 = aim_tlv_get8(tlvlist, 0x00d5, 1);#if 0	/*	 * Type 0x00d6: Encoding 1 ("us-ascii")	 */	if (aim_tlv_gettlv(tlvlist, 0x000d6, 1)) {		/* Unhandled */	}	/*	 * Type 0x00d7: Language 1 ("en")	 */	if (aim_tlv_gettlv(tlvlist, 0x000d7, 1)) {		/* Unhandled */	}	/*	 * Type 0x00d8: Encoding 2 ("us-ascii")	 */	if (aim_tlv_gettlv(tlvlist, 0x000d8, 1)) {		/* Unhandled */	}	/*	 * Type 0x00d9: Language 2 ("en")	 */	if (aim_tlv_gettlv(tlvlist, 0x000d9, 1)) {		/* Unhandled */	}#endif	/*	 * Type 0x00da: Maximum visible message length	 */	maxvisiblemsglen = aim_tlv_get16(tlvlist, 0x00da, 1);	if ((userfunc = aim_callhandler(od, snac->family, snac->subtype))) {		ret = userfunc(od, conn,				frame,				&roominfo,				roomname,				usercount,				userinfo,				roomdesc,				flags,				creationtime,				maxmsglen,				unknown_d2,				unknown_d5,				maxvisiblemsglen);	}	g_free(roominfo.name);	while (usercount > 0)		aim_info_free(&userinfo[--usercount]);	g_free(userinfo);	g_free(roomname);	g_free(roomdesc);	aim_tlvlist_free(tlvlist);	return ret;}/* Subtypes 0x0003 and 0x0004 */static intuserlistchange(OscarData *od, FlapConnection *conn, aim_module_t *mod, FlapFrame *frame, aim_modsnac_t *snac, ByteStream *bs){	aim_userinfo_t *userinfo = NULL;	aim_rxcallback_t userfunc;	int curcount = 0, ret = 0;	while (byte_stream_empty(bs)) {		curcount++;		userinfo = g_realloc(userinfo, curcount * sizeof(aim_userinfo_t));		aim_info_extract(od, bs, &userinfo[curcount-1]);	}	if ((userfunc = aim_callhandler(od, snac->family, snac->subtype)))		ret = userfunc(od, conn, frame, curcount, userinfo);	aim_info_free(userinfo);	g_free(userinfo);	return ret;}/* * Subtype 0x0005 - Send a Chat Message. * * Possible flags: *   AIM_CHATFLAGS_NOREFLECT   --  Unset the flag that requests messages *                                 should be sent to their sender. *   AIM_CHATFLAGS_AWAY        --  Mark the message as an autoresponse *                                 (Note that WinAIM does not honor this, *                                 and displays the message as normal.) * * XXX convert this to use tlvchains */intaim_chat_send_im(OscarData *od, FlapConnection *conn, guint16 flags, const gchar *msg, int msglen, const char *encoding, const char *language){	int i;	FlapFrame *frame;	IcbmCookie *cookie;	aim_snacid_t snacid;	guint8 ckstr[8];	GSList *tlvlist = NULL, *inner_tlvlist = NULL;	if (!od || !conn || !msg || (msglen <= 0))		return 0;	frame = flap_frame_new(od, 0x02, 1152);	snacid = aim_cachesnac(od, 0x000e, 0x0005, 0x0000, NULL, 0);	aim_putsnac(&frame->data, 0x000e, 0x0005, 0x0000, snacid);	/*	 * Cookie	 *	 * XXX mkcookie should generate the cookie and cache it in one	 * operation to preserve uniqueness.	 */	for (i = 0; i < 8; i++)		ckstr[i] = (guint8)rand();	cookie = aim_mkcookie(ckstr, AIM_COOKIETYPE_CHAT, NULL);	cookie->data = NULL; /* XXX store something useful here */	aim_cachecookie(od, cookie);	/* ICBM Header */	byte_stream_putraw(&frame->data, ckstr, 8); /* Cookie */	byte_stream_put16(&frame->data, 0x0003); /* Channel */	/*	 * Type 1: Flag meaning this message is destined to the room.	 */	aim_tlvlist_add_noval(&tlvlist, 0x0001);	/*	 * Type 6: Reflect	 */	if (!(flags & AIM_CHATFLAGS_NOREFLECT))		aim_tlvlist_add_noval(&tlvlist, 0x0006);	/*	 * Type 7: Autoresponse	 */	if (flags & AIM_CHATFLAGS_AWAY)		aim_tlvlist_add_noval(&tlvlist, 0x0007);	/*	 * SubTLV: Type 1: Message	 */	aim_tlvlist_add_raw(&inner_tlvlist, 0x0001, msglen, (guchar *)msg);	/*	 * SubTLV: Type 2: Encoding	 */	if (encoding != NULL)		aim_tlvlist_add_str(&inner_tlvlist, 0x0002, encoding);	/*	 * SubTLV: Type 3: Language	 */	if (language != NULL)		aim_tlvlist_add_str(&inner_tlvlist, 0x0003, language);	/*	 * Type 5: Message block.  Contains more TLVs.	 *	 * This could include other information... We just	 * put in a message TLV however.	 *	 */	aim_tlvlist_add_frozentlvlist(&tlvlist, 0x0005, &inner_tlvlist);	aim_tlvlist_write(&frame->data, &tlvlist);	aim_tlvlist_free(inner_tlvlist);	aim_tlvlist_free(tlvlist);	flap_connection_send(conn, frame);	return 0;}/* * Subtype 0x0006 * * We could probably include this in the normal ICBM parsing * code as channel 0x0003, however, since only the start * would be the same, we might as well do it here. * * General outline of this SNAC: *   snac *   cookie *   channel id *   tlvlist *     unknown *     source user info *       name *       evility *       userinfo tlvs *         online time *         etc *     message metatlv *       message tlv *         message string *       possibly others * */static intincomingim_ch3(OscarData *od, FlapConnection *conn, aim_module_t *mod, FlapFrame *frame, aim_modsnac_t *snac, ByteStream *bs){	int ret = 0, i;	aim_rxcallback_t userfunc;	aim_userinfo_t userinfo;	guint8 cookie[8];	guint16 channel;	GSList *tlvlist;	char *msg = NULL;	int len = 0;	char *encoding = NULL, *language = NULL;	IcbmCookie *ck;	aim_tlv_t *tlv;	ByteStream tbs;	memset(&userinfo, 0, sizeof(aim_userinfo_t));	/*	 * Read ICBM Cookie.	 */	for (i = 0; i < 8; i++)		cookie[i] = byte_stream_get8(bs);	if ((ck = aim_uncachecookie(od, cookie, AIM_COOKIETYPE_CHAT))) {		g_free(ck->data);		g_free(ck);	}	/*	 * Channel ID	 *	 * Channel 0x0003 is used for chat messages.	 *	 */	channel = byte_stream_get16(bs);	if (channel != 0x0003) {		purple_debug_misc("oscar", "faim: chat_incoming: unknown channel! (0x%04x)\n", channel);		return 0;	}	/*	 * Start parsing TLVs right away.	 */	tlvlist = aim_tlvlist_read(bs);	/*	 * Type 0x0003: Source User Information	 */	tlv = aim_tlv_gettlv(tlvlist, 0x0003, 1);	if (tlv != NULL)	{		byte_stream_init(&tbs, tlv->value, tlv->length);		aim_info_extract(od, &tbs, &userinfo);	}#if 0	/*	 * Type 0x0001: If present, it means it was a message to the	 * room (as opposed to a whisper).	 */	if (aim_tlv_gettlv(tlvlist, 0x0001, 1)) {		/* Unhandled */	}#endif	/*	 * Type 0x0005: Message Block.  Conains more TLVs.	 */	tlv = aim_tlv_gettlv(tlvlist, 0x0005, 1);	if (tlv != NULL)	{		GSList *inner_tlvlist;		aim_tlv_t *inner_tlv;		byte_stream_init(&tbs, tlv->value, tlv->length);		inner_tlvlist = aim_tlvlist_read(&tbs);		/*		 * Type 0x0001: Message.		 */		inner_tlv = aim_tlv_gettlv(inner_tlvlist, 0x0001, 1);		if (inner_tlv != NULL)		{			len = inner_tlv->length;			msg = aim_tlv_getvalue_as_string(inner_tlv);		}		/*		 * Type 0x0002: Encoding.		 */		encoding = aim_tlv_getstr(inner_tlvlist, 0x0002, 1);		/*		 * Type 0x0003: Language.		 */		language = aim_tlv_getstr(inner_tlvlist, 0x0003, 1);		aim_tlvlist_free(inner_tlvlist);	}	if ((userfunc = aim_callhandler(od, snac->family, snac->subtype)))		ret = userfunc(od, conn, frame, &userinfo, len, msg, encoding, language);	aim_info_free(&userinfo);	g_free(msg);	g_free(encoding);	g_free(language);	aim_tlvlist_free(tlvlist);	return ret;}static intsnachandler(OscarData *od, FlapConnection *conn, aim_module_t *mod, FlapFrame *frame, aim_modsnac_t *snac, ByteStream *bs){	if (snac->subtype == 0x0002)		return infoupdate(od, conn, mod, frame, snac, bs);	else if ((snac->subtype == 0x0003) || (snac->subtype == 0x0004))		return userlistchange(od, conn, mod, frame, snac, bs);	else if (snac->subtype == 0x0006)		return incomingim_ch3(od, conn, mod, frame, snac, bs);	return 0;}intchat_modfirst(OscarData *od, aim_module_t *mod){	mod->family = 0x000e;	mod->version = 0x0001;	mod->toolid = 0x0010;	mod->toolversion = 0x0629;	mod->flags = 0;	strncpy(mod->name, "chat", sizeof(mod->name));	mod->snachandler = snachandler;	return 0;}

⌨️ 快捷键说明

复制代码 Ctrl + C
搜索代码 Ctrl + F
全屏模式 F11
切换主题 Ctrl + Shift + D
显示快捷键 ?
增大字号 Ctrl + =
减小字号 Ctrl + -