src/protocols/qq/group_network.c

changeset 13932
4d131bd2b40f
child 14083
2b68bb18a66c
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/src/protocols/qq/group_network.c	Mon Jun 26 02:58:54 2006 +0000
@@ -0,0 +1,241 @@
+/**
+* The QQ2003C protocol plugin
+ *
+ * for gaim
+ *
+ * Copyright (C) 2004 Puzzlebird
+ *
+ * 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
+ */
+
+// START OF FILE
+/*****************************************************************************/
+#include "debug.h"		// gaim_debug
+#include "notify.h"		// gaim_notify_xxx
+
+#include "utils.h"		// hex_dump_to_str
+#include "char_conv.h"		// qq_to_utf8
+#include "crypt.h"		// qq_crypt
+#include "group_conv.h"		// qq_group_conv_refresh_online_member
+#include "group_find.h"		// qq_group_find_internal_group_id_by_seq
+#include "group_hash.h"		// qq_group_refresh
+#include "group_im.h"		// qq_process_group_cmd_im
+#include "group_info.h"		// qq_process_group_cmd_get_online_member
+#include "group_join.h"		// qq_process_group_cmd_join_group
+#include "group_network.h"
+#include "group_opt.h"		// qq_group_process_modify_info_reply
+#include "group_search.h"	// qq_process_group_cmd_search_group
+#include "header_info.h"	// QQ_CMD_GROUP_CMD
+#include "send_core.h"		// qq_send_cmd
+
+enum {
+	QQ_GROUP_CMD_REPLY_OK = 0x00,
+	QQ_GROUP_CMD_REPLY_NOT_MEMBER = 0x0a,
+};
+
+/*****************************************************************************/
+const gchar *qq_group_cmd_get_desc(qq_group_cmd cmd)
+{
+	switch (cmd) {
+	case QQ_GROUP_CMD_CREATE_GROUP:
+		return "QQ_GROUP_CMD_CREATE_GROUP";
+	case QQ_GROUP_CMD_MEMBER_OPT:
+		return "QQ_GROUP_CMD_MEMBER_OPT";
+	case QQ_GROUP_CMD_MODIFY_GROUP_INFO:
+		return "QQ_GROUP_CMD_MODIFY_GROUP_INFO";
+	case QQ_GROUP_CMD_GET_GROUP_INFO:
+		return "QQ_GROUP_CMD_GET_GROUP_INFO";
+	case QQ_GROUP_CMD_ACTIVATE_GROUP:
+		return "QQ_GROUP_CMD_ACTIVATE_GROUP";
+	case QQ_GROUP_CMD_SEARCH_GROUP:
+		return "QQ_GROUP_CMD_SEARCH_GROUP";
+	case QQ_GROUP_CMD_JOIN_GROUP:
+		return "QQ_GROUP_CMD_JOIN_GROUP";
+	case QQ_GROUP_CMD_JOIN_GROUP_AUTH:
+		return "QQ_GROUP_CMD_JOIN_GROUP_AUTH";
+	case QQ_GROUP_CMD_EXIT_GROUP:
+		return "QQ_GROUP_CMD_EXIT_GROUP";
+	case QQ_GROUP_CMD_SEND_MSG:
+		return "QQ_GROUP_CMD_SEND_MSG";
+	case QQ_GROUP_CMD_GET_ONLINE_MEMBER:
+		return "QQ_GROUP_CMD_GET_ONLINE_MEMBER";
+	case QQ_GROUP_CMD_GET_MEMBER_INFO:
+		return "QQ_GROUP_CMD_GET_MEMBER_INFO";
+	default:
+		return "Unknown QQ Group Command";
+	}			// switch
+}				// qq_group_cmd_get_desc
+
+/*****************************************************************************/
+// default process of reply error
+static void _qq_process_group_cmd_reply_error_default(guint8 reply, guint8 * cursor, gint len, GaimConnection * gc) {
+	gchar *msg, *msg_utf8;
+	g_return_if_fail(cursor != NULL && len > 0 && gc != NULL);
+
+	msg = g_strndup(cursor, len);	// it will append 0x00
+	msg_utf8 = qq_to_utf8(msg, QQ_CHARSET_DEFAULT);
+	g_free(msg);
+	msg = g_strdup_printf(_("Code [0x%02X]: %s"), reply, msg_utf8);
+	gaim_notify_error(gc, NULL, _("Group Operation Error"), msg);
+	g_free(msg);
+	g_free(msg_utf8);
+
+}				// _qq_process_group_cmd_reply_error_default
+
+/*****************************************************************************/
+// default process, dump only
+static void _qq_process_group_cmd_reply_default(guint8 * data, guint8 ** cursor, gint len, GaimConnection * gc) {
+	g_return_if_fail(gc != NULL && data != NULL && len > 0);
+	gaim_debug(GAIM_DEBUG_INFO, "QQ", "Dump unprocessed group cmd reply:\n%s", hex_dump_to_str(data, len));
+}				// _qq_process_group_cmd_reply_default
+
+/*****************************************************************************/
+// The lower layer command of send group cmd
+void qq_send_group_cmd(GaimConnection * gc, qq_group * group, guint8 * raw_data, gint data_len) {
+	qq_data *qd;
+	group_packet *p;
+
+	g_return_if_fail(gc != NULL);
+	g_return_if_fail(raw_data != NULL && data_len > 0);
+
+	qd = (qq_data *) gc->proto_data;
+	g_return_if_fail(qd != NULL);
+
+	qq_send_cmd(gc, QQ_CMD_GROUP_CMD, TRUE, 0, TRUE, raw_data, data_len);
+
+	p = g_new0(group_packet, 1);
+
+	p->send_seq = qd->send_seq;
+	if (group == NULL)
+		p->internal_group_id = 0;
+	else
+		p->internal_group_id = group->internal_group_id;
+
+	qd->group_packets = g_list_append(qd->group_packets, p);
+
+}				// qq_send_group_cmd
+
+/*****************************************************************************/
+// the main entry of group cmd processing, called by qq_recv_core.c
+void qq_process_group_cmd_reply(guint8 * buf, gint buf_len, guint16 seq, GaimConnection * gc) {
+	qq_group *group;
+	qq_data *qd;
+	gint len, bytes;
+	guint32 internal_group_id;
+	guint8 *data, *cursor, sub_cmd, reply;
+
+	g_return_if_fail(gc != NULL && gc->proto_data != NULL);
+	g_return_if_fail(buf != NULL && buf_len != 0);
+
+	qd = (qq_data *) gc->proto_data;
+	len = buf_len;
+	data = g_newa(guint8, len);
+
+	if (!qq_group_find_internal_group_id_by_seq(gc, seq, &internal_group_id)) {
+		gaim_debug(GAIM_DEBUG_WARNING, "QQ", "We have no record of group cmd, seq [%d]\n", seq);
+		return;
+	}			// if ! qq_group_find_internal_group_id_by_seq
+
+	if (qq_crypt(DECRYPT, buf, buf_len, qd->session_key, data, &len)) {
+		if (len <= 2) {
+			gaim_debug(GAIM_DEBUG_ERROR, "QQ", "Group cmd reply is too short, only %d bytes\n", len);
+			return;
+		}		// if len
+
+		bytes = 0;
+		cursor = data;
+		bytes += read_packet_b(data, &cursor, len, &sub_cmd);
+		bytes += read_packet_b(data, &cursor, len, &reply);
+
+		group = qq_group_find_by_internal_group_id(gc, internal_group_id);
+
+		if (reply != QQ_GROUP_CMD_REPLY_OK) {
+			gaim_debug(GAIM_DEBUG_WARNING, "QQ",
+				   "Group cmd reply says cmd %s fails\n", qq_group_cmd_get_desc(sub_cmd));
+			switch (reply) {	// this should be all errors
+			case QQ_GROUP_CMD_REPLY_NOT_MEMBER:
+				if (group != NULL) {
+					gaim_debug(GAIM_DEBUG_WARNING,
+						   "QQ",
+						   "You are not a member of group \"%s\"\n", group->group_name_utf8);
+					group->my_status = QQ_GROUP_MEMBER_STATUS_NOT_MEMBER;
+					qq_group_refresh(gc, group);
+				}	// if group
+				break;
+			default:
+				_qq_process_group_cmd_reply_error_default(reply, cursor, len - bytes, gc);
+			}	// switch reply
+			return;
+		}		// if reply != QQ_GROUP_CMD_REPLY_OK
+
+		// seems to ok so far, so we process the reply according to sub_cmd
+		switch (sub_cmd) {
+		case QQ_GROUP_CMD_GET_GROUP_INFO:
+			qq_process_group_cmd_get_group_info(data, &cursor, len, gc);
+			if (group != NULL) {
+				qq_send_cmd_group_get_member_info(gc, group);
+				qq_send_cmd_group_get_online_member(gc, group);
+			}
+			break;
+		case QQ_GROUP_CMD_CREATE_GROUP:
+			qq_group_process_create_group_reply(data, &cursor, len, gc);
+			break;
+		case QQ_GROUP_CMD_MODIFY_GROUP_INFO:
+			qq_group_process_modify_info_reply(data, &cursor, len, gc);
+			break;
+		case QQ_GROUP_CMD_MEMBER_OPT:
+			qq_group_process_modify_members_reply(data, &cursor, len, gc);
+			break;
+		case QQ_GROUP_CMD_ACTIVATE_GROUP:
+			qq_group_process_activate_group_reply(data, &cursor, len, gc);
+			break;
+		case QQ_GROUP_CMD_SEARCH_GROUP:
+			qq_process_group_cmd_search_group(data, &cursor, len, gc);
+			break;
+		case QQ_GROUP_CMD_JOIN_GROUP:
+			qq_process_group_cmd_join_group(data, &cursor, len, gc);
+			break;
+		case QQ_GROUP_CMD_JOIN_GROUP_AUTH:
+			qq_process_group_cmd_join_group_auth(data, &cursor, len, gc);
+			break;
+		case QQ_GROUP_CMD_EXIT_GROUP:
+			qq_process_group_cmd_exit_group(data, &cursor, len, gc);
+			break;
+		case QQ_GROUP_CMD_SEND_MSG:
+			qq_process_group_cmd_im(data, &cursor, len, gc);
+			break;
+		case QQ_GROUP_CMD_GET_ONLINE_MEMBER:
+			qq_process_group_cmd_get_online_member(data, &cursor, len, gc);
+			if (group != NULL)
+				qq_group_conv_refresh_online_member(gc, group);
+			break;
+		case QQ_GROUP_CMD_GET_MEMBER_INFO:
+			qq_process_group_cmd_get_member_info(data, &cursor, len, gc);
+			if (group != NULL)
+				qq_group_conv_refresh_online_member(gc, group);
+			break;
+		default:
+			gaim_debug(GAIM_DEBUG_WARNING, "QQ",
+				   "Group cmd %s is processed by default\n", qq_group_cmd_get_desc(sub_cmd));
+			_qq_process_group_cmd_reply_default(data, &cursor, len, gc);
+		}		// switch sub_cmd
+
+	} else
+		gaim_debug(GAIM_DEBUG_ERROR, "QQ", "Error decrypt group cmd reply\n");
+
+}				// qq_process_group_cmd_reply
+
+/*****************************************************************************/
+// END OF FILE

mercurial