65
65
/* send packet to get detailed information of one group */
66
66
void qq_send_cmd_group_get_group_info(PurpleConnection *gc, qq_group *group)
68
guint8 *raw_data, *cursor;
68
guint8 raw_data[16] = {0};
71
71
g_return_if_fail(group != NULL);
74
raw_data = g_newa(guint8, data_len);
78
bytes += create_packet_b(raw_data, &cursor, QQ_GROUP_CMD_GET_GROUP_INFO);
79
bytes += create_packet_dw(raw_data, &cursor, group->internal_group_id);
81
if (bytes != data_len)
82
purple_debug(PURPLE_DEBUG_ERROR, "QQ",
83
"Fail create packet for %s\n", qq_group_cmd_get_desc(QQ_GROUP_CMD_GET_GROUP_INFO));
85
qq_send_group_cmd(gc, group, raw_data, data_len);
73
bytes += qq_put8(raw_data + bytes, QQ_GROUP_CMD_GET_GROUP_INFO);
74
bytes += qq_put32(raw_data + bytes, group->internal_group_id);
76
qq_send_group_cmd(gc, group, raw_data, bytes);
88
79
/* send packet to get online group member, called by keep_alive */
89
80
void qq_send_cmd_group_get_online_members(PurpleConnection *gc, qq_group *group)
91
guint8 *raw_data, *cursor;
82
guint8 raw_data[16] = {0};
94
85
g_return_if_fail(group != NULL);
96
87
/* only get online members when conversation window is on */
97
88
if (NULL == purple_find_conversation_with_account(PURPLE_CONV_TYPE_CHAT,group->group_name_utf8, purple_connection_get_account(gc))) {
98
89
purple_debug(PURPLE_DEBUG_WARNING, "QQ",
99
"Conv windows for \"%s\" is not on, do not get online members\n", group->group_name_utf8);
90
"Conv windows for \"%s\" is not on, do not get online members\n", group->group_name_utf8);
104
raw_data = g_newa(guint8, data_len);
108
bytes += create_packet_b(raw_data, &cursor, QQ_GROUP_CMD_GET_ONLINE_MEMBER);
109
bytes += create_packet_dw(raw_data, &cursor, group->internal_group_id);
111
if (bytes != data_len)
112
purple_debug(PURPLE_DEBUG_ERROR, "QQ",
113
"Fail create packet for %s\n", qq_group_cmd_get_desc(QQ_GROUP_CMD_GET_ONLINE_MEMBER));
115
qq_send_group_cmd(gc, group, raw_data, data_len);
94
bytes += qq_put8(raw_data + bytes, QQ_GROUP_CMD_GET_ONLINE_MEMBER);
95
bytes += qq_put32(raw_data + bytes, group->internal_group_id);
97
qq_send_group_cmd(gc, group, raw_data, bytes);
118
100
/* send packet to get info for each group member */
119
101
void qq_send_cmd_group_get_members_info(PurpleConnection *gc, qq_group *group)
121
guint8 *raw_data, *cursor;
122
gint bytes, data_len, i;
104
gint bytes, num, data_len;
124
106
qq_buddy *member;
126
108
g_return_if_fail(group != NULL);
127
for (i = 0, list = group->members; list != NULL; list = list->next) {
109
for (num = 0, list = group->members; list != NULL; list = list->next) {
128
110
member = (qq_buddy *) list->data;
129
111
if (_is_group_member_need_update_info(member))
134
116
purple_debug(PURPLE_DEBUG_INFO, "QQ", "No group member needs to to update info now.\n");
138
data_len = 5 + 4 * i;
120
data_len = 5 + 4 * num;
139
121
raw_data = g_newa(guint8, data_len);
143
bytes += create_packet_b(raw_data, &cursor, QQ_GROUP_CMD_GET_MEMBER_INFO);
144
bytes += create_packet_dw(raw_data, &cursor, group->internal_group_id);
124
bytes += qq_put8(raw_data + bytes, QQ_GROUP_CMD_GET_MEMBER_INFO);
125
bytes += qq_put32(raw_data + bytes, group->internal_group_id);
146
127
list = group->members;
147
128
while (list != NULL) {
148
129
member = (qq_buddy *) list->data;
149
130
if (_is_group_member_need_update_info(member))
150
bytes += create_packet_dw(raw_data, &cursor, member->uid);
131
bytes += qq_put32(raw_data + bytes, member->uid);
151
132
list = list->next;
154
if (bytes != data_len)
135
if (bytes != data_len) {
155
136
purple_debug(PURPLE_DEBUG_ERROR, "QQ",
156
"Fail create packet for %s\n", qq_group_cmd_get_desc(QQ_GROUP_CMD_GET_MEMBER_INFO));
158
qq_send_group_cmd(gc, group, raw_data, data_len);
137
"Fail create packet for %s\n", qq_group_cmd_get_desc(QQ_GROUP_CMD_GET_MEMBER_INFO));
141
qq_send_group_cmd(gc, group, raw_data, bytes);
161
void qq_process_group_cmd_get_group_info(guint8 *data, guint8 **cursor, gint len, PurpleConnection *gc)
145
* @brief å¤ēē¾¤äæ”ęÆ.å½åē¾¤äæ”ęÆēå¤ēčæäøå®å,ē±äŗēę¬ēäøååƼč“åč®®ēč§£čÆ»ęå·®å¼.
147
void qq_process_group_cmd_get_group_info(guint8 *data, gint len, PurpleConnection *gc)
164
150
qq_buddy *member;
168
154
guint16 unknown, max_members;
169
155
guint32 member_uid, internal_group_id, external_group_id;
170
156
GSList *pending_id;
172
157
guint32 unknown4;
175
161
g_return_if_fail(data != NULL && len > 0);
176
162
qd = (qq_data *) gc->proto_data;
178
read_packet_dw(data, cursor, len, &(internal_group_id));
179
g_return_if_fail(internal_group_id > 0);
180
read_packet_dw(data, cursor, len, &(external_group_id));
181
g_return_if_fail(internal_group_id > 0);
165
bytes += qq_get32(&(internal_group_id), data + bytes);
166
g_return_if_fail(internal_group_id > 0);
168
bytes += qq_get32(&(external_group_id), data + bytes);
169
g_return_if_fail(external_group_id > 0);
183
171
pending_id = qq_get_pending_id(qd->adding_groups_from_server, internal_group_id);
184
172
if (pending_id != NULL) {
189
177
group = qq_group_find_by_id(gc, internal_group_id, QQ_INTERNAL_ID);
190
178
g_return_if_fail(group != NULL);
192
read_packet_b(data, cursor, len, &(group->group_type));
193
read_packet_dw(data, cursor, len, &unknown4); /* unknown 4 bytes */
194
read_packet_dw(data, cursor, len, &(group->creator_uid));
195
read_packet_b(data, cursor, len, &(group->auth_type));
196
read_packet_dw(data, cursor, len, &unknown4); /* oldCategory */
197
read_packet_w(data, cursor, len, &unknown);
198
read_packet_dw(data, cursor, len, &(group->group_category));
199
read_packet_w(data, cursor, len, &max_members);
200
read_packet_b(data, cursor, len, &unknown1);
201
read_packet_dw(data, cursor, len, &(unknown4)); /* versionID */
203
pascal_len = convert_as_pascal_string(*cursor, &(group->group_name_utf8), QQ_CHARSET_DEFAULT);
204
*cursor += pascal_len;
205
read_packet_w(data, cursor, len, &(unknown)); /* 0x0000 */
206
pascal_len = convert_as_pascal_string(*cursor, &(group->notice_utf8), QQ_CHARSET_DEFAULT);
207
*cursor += pascal_len;
208
pascal_len = convert_as_pascal_string(*cursor, &(group->group_desc_utf8), QQ_CHARSET_DEFAULT);
209
*cursor += pascal_len;
180
bytes += qq_get8(&(group->group_type), data + bytes);
181
bytes += qq_get32(&unknown4, data + bytes); /* unknown 4 bytes */
182
bytes += qq_get32(&(group->creator_uid), data + bytes);
183
bytes += qq_get8(&(group->auth_type), data + bytes);
184
bytes += qq_get32(&unknown4, data + bytes); /* oldCategory */
185
bytes += qq_get16(&unknown, data + bytes);
186
bytes += qq_get32(&(group->group_category), data + bytes);
187
bytes += qq_get16(&max_members, data + bytes);
188
bytes += qq_get8(&unknown1, data + bytes);
190
* the following, while Eva:
191
* 4(unk), 4(verID), 1(nameLen), nameLen(qunNameContent), 1(0x00),
192
* 2(qunNoticeLen), qunNoticeLen(qunNoticeContent, 1(qunDescLen),
193
* qunDestLen(qunDestcontent)) */
194
bytes += qq_get8(&unknown1, data + bytes);
195
purple_debug(PURPLE_DEBUG_INFO, "QQ", "type=%u creatorid=%u category=%u\n",
196
group->group_type, group->creator_uid, group->group_category);
197
purple_debug(PURPLE_DEBUG_INFO, "QQ", "maxmembers=%u", max_members);
199
/* strlen + <str content> */
200
bytes += convert_as_pascal_string(data + bytes, &(group->group_name_utf8), QQ_CHARSET_DEFAULT);
201
purple_debug(PURPLE_DEBUG_INFO, "QQ", "group \"%s\"\n", group->group_name_utf8);
202
bytes += qq_get16(&unknown, data + bytes); /* 0x0000 */
203
bytes += convert_as_pascal_string(data + bytes, &(group->notice_utf8), QQ_CHARSET_DEFAULT);
204
purple_debug(PURPLE_DEBUG_INFO, "QQ", "notice \"%s\"\n", group->notice_utf8);
205
bytes += convert_as_pascal_string(data + bytes, &(group->group_desc_utf8), QQ_CHARSET_DEFAULT);
206
purple_debug(PURPLE_DEBUG_INFO, "QQ", "group_desc \"%s\"\n", group->group_desc_utf8);
212
209
/* now comes the member list separated by 0x00 */
213
while (*cursor < data + len) {
214
read_packet_dw(data, cursor, len, &member_uid);
216
read_packet_b(data, cursor, len, &organization);
217
read_packet_b(data, cursor, len, &role);
210
while (bytes < len) {
211
bytes += qq_get32(&member_uid, data + bytes);
213
bytes += qq_get8(&organization, data + bytes);
214
bytes += qq_get8(&role, data + bytes);
219
216
if(organization != 0 || role != 0) {
220
217
purple_debug(PURPLE_DEBUG_INFO, "QQ", "group member %d: organization=%d, role=%d\n", member_uid, organization, role);
237
234
purple_conv = purple_find_conversation_with_account(PURPLE_CONV_TYPE_CHAT,
238
235
group->group_name_utf8, purple_connection_get_account(gc));
239
236
if(NULL == purple_conv) {
240
purple_debug(PURPLE_DEBUG_WARNING, "QQ",
241
"Conv windows for \"%s\" is not on, do not set topic\n", group->group_name_utf8);
237
purple_debug(PURPLE_DEBUG_WARNING, "QQ",
238
"Conv windows for \"%s\" is not on, do not set topic\n", group->group_name_utf8);
244
241
purple_conv_chat_set_topic(PURPLE_CONV_CHAT(purple_conv), NULL, group->notice_utf8);
248
void qq_process_group_cmd_get_online_members(guint8 *data, guint8 **cursor, gint len, PurpleConnection *gc)
245
void qq_process_group_cmd_get_online_members(guint8 *data, gint len, PurpleConnection *gc)
250
247
guint32 internal_group_id, member_uid;
254
251
qq_buddy *member;
256
253
g_return_if_fail(data != NULL && len > 0);
258
if (data + len - *cursor < 4) {
259
256
purple_debug(PURPLE_DEBUG_ERROR, "QQ", "Invalid group online member reply, discard it!\n");
265
bytes += read_packet_dw(data, cursor, len, &internal_group_id);
266
bytes += read_packet_b(data, cursor, len, &unknown); /* 0x3c ?? */
261
bytes += qq_get32(&internal_group_id, data + bytes);
262
bytes += qq_get8(&unknown, data + bytes); /* 0x3c ?? */
267
263
g_return_if_fail(internal_group_id > 0);
269
265
group = qq_group_find_by_id(gc, internal_group_id, QQ_INTERNAL_ID);
276
272
/* set all offline first, then update those online */
277
273
_qq_group_set_members_all_offline(group);
278
while (*cursor < data + len) {
279
bytes += read_packet_dw(data, cursor, len, &member_uid);
275
while (bytes < len) {
276
bytes += qq_get32(&member_uid, data + bytes);
281
278
member = qq_group_find_or_add_member(gc, group, member_uid);
282
279
if (member != NULL)
283
280
member->status = QQ_BUDDY_ONLINE_NORMAL;
285
if(*cursor > (data + len)) {
286
purple_debug(PURPLE_DEBUG_ERROR, "QQ",
287
"group_cmd_get_online_members: Dangerous error! maybe protocol changed, notify developers!");
283
purple_debug(PURPLE_DEBUG_ERROR, "QQ",
284
"group_cmd_get_online_members: Dangerous error! maybe protocol changed, notify developers!");
290
purple_debug(PURPLE_DEBUG_INFO, "QQ", "Group \"%s\" has %d online members\n", group->group_name_utf8, i);
287
purple_debug(PURPLE_DEBUG_INFO, "QQ", "Group \"%s\" has %d online members\n", group->group_name_utf8, num);
293
290
/* process the reply to get_members_info packet */
294
void qq_process_group_cmd_get_members_info(guint8 *data, guint8 **cursor, gint len, PurpleConnection *gc)
291
void qq_process_group_cmd_get_members_info(guint8 *data, gint len, PurpleConnection *gc)
296
295
guint32 internal_group_id, member_uid;
300
298
qq_buddy *member;
302
300
g_return_if_fail(data != NULL && len > 0);
304
read_packet_dw(data, cursor, len, &internal_group_id);
303
bytes += qq_get32(&internal_group_id, data + bytes);
305
304
g_return_if_fail(internal_group_id > 0);
307
306
group = qq_group_find_by_id(gc, internal_group_id, QQ_INTERNAL_ID);
308
307
g_return_if_fail(group != NULL);
311
310
/* now starts the member info, as get buddy list reply */
312
while (*cursor < data + len) {
313
read_packet_dw(data, cursor, len, &member_uid);
311
while (bytes < len) {
312
bytes += qq_get32(&member_uid, data + bytes);
314
313
g_return_if_fail(member_uid > 0);
315
314
member = qq_group_find_member_by_uid(group, member_uid);
316
315
g_return_if_fail(member != NULL);
319
read_packet_w(data, cursor, len, &(member->face));
320
read_packet_b(data, cursor, len, &(member->age));
321
read_packet_b(data, cursor, len, &(member->gender));
322
pascal_len = convert_as_pascal_string(*cursor, &(member->nickname), QQ_CHARSET_DEFAULT);
323
*cursor += pascal_len;
324
read_packet_w(data, cursor, len, &unknown);
325
read_packet_b(data, cursor, len, &(member->flag1));
326
read_packet_b(data, cursor, len, &(member->comm_flag));
318
bytes += qq_get16(&(member->face), data + bytes);
319
bytes += qq_get8(&(member->age), data + bytes);
320
bytes += qq_get8(&(member->gender), data + bytes);
321
bytes += convert_as_pascal_string(data + bytes, &(member->nickname), QQ_CHARSET_DEFAULT);
322
bytes += qq_get16(&unknown, data + bytes);
323
bytes += qq_get8(&(member->flag1), data + bytes);
324
bytes += qq_get8(&(member->comm_flag), data + bytes);
328
326
member->last_refresh = time(NULL);
330
if(*cursor > (data + len)) {
331
purple_debug(PURPLE_DEBUG_ERROR, "QQ",
332
"group_cmd_get_members_info: Dangerous error! maybe protocol changed, notify developers!");
334
purple_debug(PURPLE_DEBUG_INFO, "QQ", "Group \"%s\" obtained %d member info\n", group->group_name_utf8, i);
329
purple_debug(PURPLE_DEBUG_ERROR, "QQ",
330
"group_cmd_get_members_info: Dangerous error! maybe protocol changed, notify developers!");
332
purple_debug(PURPLE_DEBUG_INFO, "QQ", "Group \"%s\" obtained %d member info\n", group->group_name_utf8, num);