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

📄 ycht.c

📁 yahoo message protocol stack
💻 C
字号:
/** * @file ycht.c The Yahoo! protocol plugin, YCHT protocol stuff. * * gaim * * Copyright (C) 2004 Timothy Ringenbach <omarvo@hotmail.com> * Liberal amounts of code borrowed from the rest of the Yahoo! prpl. * * 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 <string.h>#include "internal.h"#include "prpl.h"#include "notify.h"#include "account.h"#include "proxy.h"#include "debug.h"#include "conversation.h"#include "util.h"#include "yahoo.h"#include "ycht.h"#include "yahoochat.h"/* * dword: YCHT * dword: 0x000000AE * dword: service * word:  status * word:  size */#define YAHOO_CHAT_ID (1)/************************************************************************************ * Functions to process various kinds of packets. ************************************************************************************/static void ycht_process_login(YchtConn *ycht, YchtPkt *pkt){	GaimConnection *gc = ycht->gc;	struct yahoo_data *yd = gc->proto_data;	if (ycht->logged_in)		return;	yd->chat_online = TRUE;	ycht->logged_in = TRUE;	if (ycht->room)		ycht_chat_join(ycht, ycht->room);}static void ycht_process_logout(YchtConn *ycht, YchtPkt *pkt){	GaimConnection *gc = ycht->gc;	struct yahoo_data *yd = gc->proto_data;	yd->chat_online = FALSE;	ycht->logged_in = FALSE;}static void ycht_process_chatjoin(YchtConn *ycht, YchtPkt *pkt){	char *room, *topic;	GaimConnection *gc = ycht->gc;	GaimConversation *c = NULL;	gboolean new_room = FALSE;	char **members;	int i;	room = g_list_nth_data(pkt->data, 0);	topic = g_list_nth_data(pkt->data, 1);	if (!g_list_nth_data(pkt->data, 4))		return;	if (!room)		return;	members = g_strsplit(g_list_nth_data(pkt->data, 4), "\001", 0);	for (i = 0; members[i]; i++) {		char *tmp = strchr(members[i], '\002');		if (tmp)			*tmp = '\0';	}	if (g_list_length(pkt->data) > 5)		new_room = TRUE;	if (new_room && ycht->changing_rooms) {		serv_got_chat_left(gc, YAHOO_CHAT_ID);		ycht->changing_rooms = FALSE;		c = serv_got_joined_chat(gc, YAHOO_CHAT_ID, room);	} else {		c = gaim_find_chat(gc, YAHOO_CHAT_ID);	}	if (topic)		gaim_conv_chat_set_topic(GAIM_CONV_CHAT(c), NULL, topic);	for (i = 0; members[i]; i++) {		if (new_room) {			/*if (!strcmp(members[i], gaim_connection_get_display_name(ycht->gc)))				continue;*/			gaim_conv_chat_add_user(GAIM_CONV_CHAT(c), members[i], NULL, GAIM_CBFLAGS_NONE, TRUE);		} else {			yahoo_chat_add_user(GAIM_CONV_CHAT(c), members[i], NULL);		}	}	g_strfreev(members);}static void ycht_process_chatpart(YchtConn *ycht, YchtPkt *pkt){	char *room, *who;	room = g_list_nth_data(pkt->data, 0);	who = g_list_nth_data(pkt->data, 1);	if (who && room) {		GaimConversation *c = gaim_find_chat(ycht->gc, YAHOO_CHAT_ID);		if (c && !gaim_utf8_strcasecmp(gaim_conversation_get_name(c), room))			gaim_conv_chat_remove_user(GAIM_CONV_CHAT(c), who, NULL);	}}static void ycht_progress_chatmsg(YchtConn *ycht, YchtPkt *pkt){	char *who, *what, *msg;	GaimConversation *c;	GaimConnection *gc = ycht->gc;	who = g_list_nth_data(pkt->data, 1);	what = g_list_nth_data(pkt->data, 2);	if (!who || !what)		return;	c = gaim_find_chat(gc, YAHOO_CHAT_ID);	if (!c)		return;	msg = yahoo_string_decode(gc, what, 1);	what = yahoo_codes_to_html(msg);	g_free(msg);	if (pkt->service == YCHT_SERVICE_CHATMSG_EMOTE) {		char *tmp = g_strdup_printf("/me %s", what);		g_free(what);		what = tmp;	}	serv_got_chat_in(gc, YAHOO_CHAT_ID, who, 0, what, time(NULL));	g_free(what);}static void ycht_progress_online_friends(YchtConn *ycht, YchtPkt *pkt){#if 0	GaimConnection *gc = ycht->gc;	struct yahoo_data *yd = gc->proto_data;	if (ycht->logged_in)		return;	yd->chat_online = TRUE;	ycht->logged_in = TRUE;	if (ycht->room)		ycht_chat_join(ycht, ycht->room);#endif}/***************************************************************************** * Functions dealing with YCHT packets and their contents directly. *****************************************************************************/static void ycht_packet_dump(const char *data, int len){#ifdef YAHOO_YCHT_DEBUG	int i;	gaim_debug(GAIM_DEBUG_MISC, "yahoo", "");	for (i = 0; i + 1 < len; i += 2) {		if ((i % 16 == 0) && i) {			gaim_debug(GAIM_DEBUG_MISC, NULL, "\n");			gaim_debug(GAIM_DEBUG_MISC, "yahoo", "");		}		gaim_debug(GAIM_DEBUG_MISC, NULL, "%02hhx%02hhx ", data[i], data[i + 1]);	}	if (i < len)		gaim_debug(GAIM_DEBUG_MISC, NULL, "%02hhx", data[i]);	gaim_debug(GAIM_DEBUG_MISC, NULL, "\n");	gaim_debug(GAIM_DEBUG_MISC, "yahoo", "");	for (i = 0; i < len; i++) {		if ((i % 16 == 0) && i) {			gaim_debug(GAIM_DEBUG_MISC, NULL, "\n");			gaim_debug(GAIM_DEBUG_MISC, "yahoo", "");		}		if (g_ascii_isprint(data[i]))			gaim_debug(GAIM_DEBUG_MISC, NULL, "%c ", data[i]);		else			gaim_debug(GAIM_DEBUG_MISC, NULL, ". ");	}	gaim_debug(GAIM_DEBUG_MISC, NULL, "\n");#endif}static YchtPkt *ycht_packet_new(guint version, guint service, int status){	YchtPkt *ret;	ret = g_new0(YchtPkt, 1);	ret->version = version;	ret->service = service;	ret->status = status;	return ret;}static void ycht_packet_append(YchtPkt *pkt, const char *str){	g_return_if_fail(pkt != NULL);	g_return_if_fail(str != NULL);	pkt->data = g_list_append(pkt->data, g_strdup(str));}static int ycht_packet_length(YchtPkt *pkt){	int ret;	GList *l;	ret = YCHT_HEADER_LEN;	for (l = pkt->data; l; l = l->next) {		ret += strlen(l->data);		if (l->next)			ret += strlen(YCHT_SEP);	}	return ret;}static void ycht_packet_send(YchtConn *ycht, YchtPkt *pkt){	int len, pos;	char *buf;	GList *l;	g_return_if_fail(ycht != NULL);	g_return_if_fail(pkt != NULL);	g_return_if_fail(ycht->fd != -1);	pos = 0;	len = ycht_packet_length(pkt);	buf = g_malloc(len);	memcpy(buf + pos, "YCHT", 4); pos += 4;	pos += yahoo_put32(buf + pos, pkt->version);	pos += yahoo_put32(buf + pos, pkt->service);	pos += yahoo_put16(buf + pos, pkt->status);	pos += yahoo_put16(buf + pos, len - YCHT_HEADER_LEN);	for (l = pkt->data; l; l = l->next) {		int slen = strlen(l->data);		memcpy(buf + pos, l->data, slen); pos += slen;		if (l->next) {			memcpy(buf + pos, YCHT_SEP, strlen(YCHT_SEP));			pos += strlen(YCHT_SEP);		}	}	write(ycht->fd, buf, len);	g_free(buf);}static void ycht_packet_read(YchtPkt *pkt, const char *buf, int len){	const char *pos = buf;	const char *needle;	char *tmp, *tmp2;	int i = 0;	while (len > 0 && (needle = g_strstr_len(pos, len, YCHT_SEP))) {		tmp = g_strndup(pos, needle - pos);		pkt->data = g_list_append(pkt->data, tmp);		len -= needle - pos + strlen(YCHT_SEP);		pos = needle + strlen(YCHT_SEP);		tmp2 = g_strescape(tmp, NULL);		gaim_debug_misc("yahoo", "Data[%d]:\t%s\n", i++, tmp2);		g_free(tmp2);	}	if (len) {		tmp = g_strndup(pos, len);		pkt->data = g_list_append(pkt->data, tmp);		tmp2 = g_strescape(tmp, NULL);		gaim_debug_misc("yahoo", "Data[%d]:\t%s\n", i, tmp2);		g_free(tmp2);	};	gaim_debug_misc("yahoo", "--==End of incoming YCHT packet==--\n");}static void ycht_packet_process(YchtConn *ycht, YchtPkt *pkt){	if (pkt->data && !strncmp(pkt->data->data, "*** Danger Will Robinson!!!", strlen("*** Danger Will Robinson!!!")))		return;	switch (pkt->service) {	case YCHT_SERVICE_LOGIN:		ycht_process_login(ycht, pkt);		break;	case YCHT_SERVICE_LOGOUT:		ycht_process_logout(ycht, pkt);		break;	case YCHT_SERVICE_CHATJOIN:		ycht_process_chatjoin(ycht, pkt);		break;	case YCHT_SERVICE_CHATPART:		ycht_process_chatpart(ycht, pkt);		break;	case YCHT_SERVICE_CHATMSG:	case YCHT_SERVICE_CHATMSG_EMOTE:		ycht_progress_chatmsg(ycht, pkt);		break;	case YCHT_SERVICE_ONLINE_FRIENDS:		ycht_progress_online_friends(ycht, pkt);		break;	default:		gaim_debug_warning("yahoo", "YCHT: warning, unhandled service 0x%02x\n", pkt->service);	}}static void ycht_packet_free(YchtPkt *pkt){	GList *l;	g_return_if_fail(pkt != NULL);	for (l = pkt->data; l; l = l->next)		g_free(l->data);	g_list_free(pkt->data);	g_free(pkt);}/************************************************************************************ * Functions dealing with connecting and disconnecting and reading data into YchtPkt * structs, and all that stuff. ************************************************************************************/void ycht_connection_close(YchtConn *ycht){	struct yahoo_data *yd = ycht->gc->proto_data;	if (yd) {		yd->ycht = NULL;		yd->chat_online = FALSE;	}	if (ycht->fd > 0)		close(ycht->fd);	if (ycht->inpa)		gaim_input_remove(ycht->inpa);	if (ycht->rxqueue)		g_free(ycht->rxqueue);	g_free(ycht);}static void ycht_connection_error(YchtConn *ycht, const gchar *error){	gaim_notify_info(ycht->gc, NULL, _("Connection problem with the YCHT server."), error);	ycht_connection_close(ycht);}static void ycht_pending(gpointer data, gint source, GaimInputCondition cond){	YchtConn *ycht = data;	char buf[1024];	int len;	len = read(ycht->fd, buf, sizeof(buf));	if (len <= 0) {		ycht_connection_error(ycht, _("Unable to read"));		return;	}	ycht->rxqueue = g_realloc(ycht->rxqueue, len + ycht->rxlen);	memcpy(ycht->rxqueue + ycht->rxlen, buf, len);	ycht->rxlen += len;	while (1) {		YchtPkt *pkt;		int pos = 0;		int pktlen;		guint service;		guint version;		gint status;		if (ycht->rxlen < YCHT_HEADER_LEN)			return;		if (strncmp("YCHT", ycht->rxqueue, 4) != 0)			gaim_debug_error("yahoo", "YCHT: protocol error.\n");		pos += 4; /* YCHT */		version = yahoo_get32(ycht->rxqueue + pos); pos += 4;		service = yahoo_get32(ycht->rxqueue + pos); pos += 4;		status = yahoo_get16(ycht->rxqueue + pos); pos += 2;		pktlen  = yahoo_get16(ycht->rxqueue + pos); pos += 2;		gaim_debug(GAIM_DEBUG_MISC, "yahoo",				   "ycht: %d bytes to read, rxlen is %d\n", pktlen, ycht->rxlen);		if (ycht->rxlen < (YCHT_HEADER_LEN + pktlen))			return;		gaim_debug_misc("yahoo", "--==Incoming YCHT packet==--\n");		gaim_debug(GAIM_DEBUG_MISC, "yahoo",			   "YCHT Service: 0x%02x Version: 0x%02x Status: 0x%02x\n",			   service, version, status);		ycht_packet_dump(ycht->rxqueue, YCHT_HEADER_LEN + pktlen);		pkt = ycht_packet_new(version, service, status);		ycht_packet_read(pkt, ycht->rxqueue + pos, pktlen);		ycht->rxlen -= YCHT_HEADER_LEN + pktlen;		if (ycht->rxlen) {			char *tmp = g_memdup(ycht->rxqueue + YCHT_HEADER_LEN + pktlen, ycht->rxlen);			g_free(ycht->rxqueue);			ycht->rxqueue = tmp;		} else {			g_free(ycht->rxqueue);			ycht->rxqueue = NULL;		}		ycht_packet_process(ycht, pkt);		ycht_packet_free(pkt);	}}static void ycht_got_connected(gpointer data, gint source, GaimInputCondition cond){	YchtConn *ycht = data;	GaimConnection *gc = ycht->gc;	struct yahoo_data *yd = gc->proto_data;	YchtPkt *pkt;	char *buf;	if (source < 0) {		ycht_connection_error(ycht, _("Unable to connect."));		return;	}	ycht->fd = source;	pkt = ycht_packet_new(YCHT_VERSION, YCHT_SERVICE_LOGIN, 0);	buf = g_strdup_printf("%s\001Y=%s; T=%s", gaim_connection_get_display_name(gc), yd->cookie_y, yd->cookie_t);	ycht_packet_append(pkt, buf);	g_free(buf);	ycht_packet_send(ycht, pkt);	ycht_packet_free(pkt);	ycht->inpa = gaim_input_add(ycht->fd, GAIM_INPUT_READ, ycht_pending, ycht);}void ycht_connection_open(GaimConnection *gc){	YchtConn *ycht;	struct yahoo_data *yd = gc->proto_data;	GaimAccount *account = gaim_connection_get_account(gc);	ycht = g_new0(YchtConn, 1);	ycht->gc = gc;	ycht->fd = -1;	yd->ycht = ycht;	if (gaim_proxy_connect(account,	                       gaim_account_get_string(account, "ycht-server",  YAHOO_YCHT_HOST),	                       gaim_account_get_int(account, "ycht-port", YAHOO_YCHT_PORT),	                       ycht_got_connected, ycht) != 0)	{		ycht_connection_error(ycht, _("Connection problem"));		return;	}}/******************************************************************************************* * These are functions called because the user did something. *******************************************************************************************/void ycht_chat_join(YchtConn *ycht, const char *room){	YchtPkt *pkt;	char *tmp;	tmp = g_strdup(room);	if (ycht->room)		g_free(ycht->room);	ycht->room = tmp;	if (!ycht->logged_in)		return;	ycht->changing_rooms = TRUE;	pkt = ycht_packet_new(YCHT_VERSION, YCHT_SERVICE_CHATJOIN, 0);	ycht_packet_append(pkt, ycht->room);	ycht_packet_send(ycht, pkt);	ycht_packet_free(pkt);}int ycht_chat_send(YchtConn *ycht, const char *room, const char *what){	YchtPkt *pkt;	char *msg1, *msg2, *buf;	if (strcmp(room, ycht->room))		gaim_debug_warning("yahoo", "uhoh, sending to the wrong room!\n");	pkt = ycht_packet_new(YCHT_VERSION, YCHT_SERVICE_CHATMSG, 0);	msg1 = yahoo_html_to_codes(what);	msg2 = yahoo_string_encode(ycht->gc, msg1, NULL);	g_free(msg1);	buf = g_strdup_printf("%s\001%s", ycht->room, msg2);	ycht_packet_append(pkt, buf);	g_free(msg2);	g_free(buf);	ycht_packet_send(ycht, pkt);	ycht_packet_free(pkt);	return 1;}void ycht_chat_leave(YchtConn *ycht, const char *room, gboolean logout){	if (logout)		ycht_connection_close(ycht);}void ycht_chat_send_invite(YchtConn *ycht, const char *room, const char *buddy, const char *msg){}void ycht_chat_goto_user(YchtConn *ycht, const char *name){}void ycht_chat_send_keepalive(YchtConn *ycht){	YchtPkt *pkt;	pkt = ycht_packet_new(YCHT_VERSION, YCHT_SERVICE_PING, 0);	ycht_packet_send(ycht, pkt);	ycht_packet_free(pkt);}

⌨️ 快捷键说明

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