| |
1 /** |
| |
2 * @file buddy_opt.c |
| |
3 * |
| |
4 * purple |
| |
5 * |
| |
6 * Purple is the legal property of its developers, whose names are too numerous |
| |
7 * to list here. Please refer to the COPYRIGHT file distributed with this |
| |
8 * source distribution. |
| |
9 * |
| |
10 * This program is free software; you can redistribute it and/or modify |
| |
11 * it under the terms of the GNU General Public License as published by |
| |
12 * the Free Software Foundation; either version 2 of the License, or |
| |
13 * (at your option) any later version. |
| |
14 * |
| |
15 * This program is distributed in the hope that it will be useful, |
| |
16 * but WITHOUT ANY WARRANTY; without even the implied warranty of |
| |
17 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the |
| |
18 * GNU General Public License for more details. |
| |
19 * |
| |
20 * You should have received a copy of the GNU General Public License |
| |
21 * along with this program; if not, write to the Free Software |
| |
22 * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA |
| |
23 */ |
| |
24 |
| |
25 #include "debug.h" |
| |
26 #include "internal.h" |
| |
27 #include "notify.h" |
| |
28 #include "request.h" |
| |
29 |
| |
30 #include "buddy_info.h" |
| |
31 #include "buddy_list.h" |
| |
32 #include "buddy_opt.h" |
| |
33 #include "char_conv.h" |
| |
34 #include "crypt.h" |
| |
35 #include "header_info.h" |
| |
36 #include "im.h" |
| |
37 #include "keep_alive.h" |
| |
38 #include "packet_parse.h" |
| |
39 #include "send_core.h" |
| |
40 #include "utils.h" |
| |
41 |
| |
42 #define PURPLE_GROUP_QQ_FORMAT "QQ (%s)" |
| |
43 #define PURPLE_GROUP_QQ_UNKNOWN "QQ Unknown" |
| |
44 #define PURPLE_GROUP_QQ_BLOCKED "QQ Blocked" |
| |
45 |
| |
46 #define QQ_REMOVE_BUDDY_REPLY_OK 0x00 |
| |
47 #define QQ_REMOVE_SELF_REPLY_OK 0x00 |
| |
48 #define QQ_ADD_BUDDY_AUTH_REPLY_OK 0x30 /* ASCII value of "0" */ |
| |
49 |
| |
50 enum { |
| |
51 QQ_MY_AUTH_APPROVE = 0x30, /* ASCII value of "0" */ |
| |
52 QQ_MY_AUTH_REJECT = 0x31, /* ASCII value of "1" */ |
| |
53 QQ_MY_AUTH_REQUEST = 0x32, /* ASCII value of "2" */ |
| |
54 }; |
| |
55 |
| |
56 typedef struct _qq_add_buddy_request { |
| |
57 guint32 uid; |
| |
58 guint16 seq; |
| |
59 } qq_add_buddy_request; |
| |
60 |
| |
61 /* send packet to remove a buddy from my buddy list */ |
| |
62 static void _qq_send_packet_remove_buddy(PurpleConnection *gc, guint32 uid) |
| |
63 { |
| |
64 gchar uid_str[11]; |
| |
65 |
| |
66 g_return_if_fail(uid > 0); |
| |
67 |
| |
68 g_snprintf(uid_str, sizeof(uid_str), "%d", uid); |
| |
69 qq_send_cmd(gc, QQ_CMD_DEL_FRIEND, TRUE, 0, |
| |
70 TRUE, (guint8 *) uid_str, strlen(uid_str)); |
| |
71 } |
| |
72 |
| |
73 /* try to remove myself from someone's buddy list */ |
| |
74 static void _qq_send_packet_remove_self_from(PurpleConnection *gc, guint32 uid) |
| |
75 { |
| |
76 guint8 *raw_data, *cursor; |
| |
77 |
| |
78 g_return_if_fail(uid > 0); |
| |
79 |
| |
80 raw_data = g_newa(guint8, 4); |
| |
81 cursor = raw_data; |
| |
82 create_packet_dw(raw_data, &cursor, uid); |
| |
83 |
| |
84 qq_send_cmd(gc, QQ_CMD_REMOVE_SELF, TRUE, 0, TRUE, raw_data, 4); |
| |
85 } |
| |
86 |
| |
87 /* try to add a buddy without authentication */ |
| |
88 static void _qq_send_packet_add_buddy(PurpleConnection *gc, guint32 uid) |
| |
89 { |
| |
90 qq_data *qd; |
| |
91 qq_add_buddy_request *req; |
| |
92 gchar uid_str[11]; |
| |
93 |
| |
94 g_return_if_fail(uid > 0); |
| |
95 |
| |
96 /* we need to send the ascii code of this uid to qq server */ |
| |
97 g_snprintf(uid_str, sizeof(uid_str), "%d", uid); |
| |
98 qq_send_cmd(gc, QQ_CMD_ADD_FRIEND_WO_AUTH, TRUE, 0, |
| |
99 TRUE, (guint8 *) uid_str, strlen(uid_str)); |
| |
100 |
| |
101 /* must be set after sending packet to get the correct send_seq */ |
| |
102 qd = (qq_data *) gc->proto_data; |
| |
103 req = g_new0(qq_add_buddy_request, 1); |
| |
104 req->seq = qd->send_seq; |
| |
105 req->uid = uid; |
| |
106 qd->add_buddy_request = g_list_append(qd->add_buddy_request, req); |
| |
107 } |
| |
108 |
| |
109 /* this buddy needs authentication, text conversion is done at lowest level */ |
| |
110 static void _qq_send_packet_buddy_auth(PurpleConnection *gc, guint32 uid, const gchar response, const gchar *text) |
| |
111 { |
| |
112 gchar *text_qq, uid_str[11]; |
| |
113 guint8 bar, *cursor, *raw_data; |
| |
114 |
| |
115 g_return_if_fail(uid != 0); |
| |
116 |
| |
117 g_snprintf(uid_str, sizeof(uid_str), "%d", uid); |
| |
118 bar = 0x1f; |
| |
119 raw_data = g_newa(guint8, QQ_MSG_IM_MAX); |
| |
120 cursor = raw_data; |
| |
121 |
| |
122 create_packet_data(raw_data, &cursor, (guint8 *) uid_str, strlen(uid_str)); |
| |
123 create_packet_b(raw_data, &cursor, bar); |
| |
124 create_packet_b(raw_data, &cursor, response); |
| |
125 |
| |
126 if (text != NULL) { |
| |
127 text_qq = utf8_to_qq(text, QQ_CHARSET_DEFAULT); |
| |
128 create_packet_b(raw_data, &cursor, bar); |
| |
129 create_packet_data(raw_data, &cursor, (guint8 *) text_qq, strlen(text_qq)); |
| |
130 g_free(text_qq); |
| |
131 } |
| |
132 |
| |
133 qq_send_cmd(gc, QQ_CMD_BUDDY_AUTH, TRUE, 0, TRUE, raw_data, cursor - raw_data); |
| |
134 } |
| |
135 |
| |
136 static void _qq_send_packet_add_buddy_auth_with_gc_and_uid(gc_and_uid *g, const gchar *text) |
| |
137 { |
| |
138 PurpleConnection *gc; |
| |
139 guint32 uid; |
| |
140 g_return_if_fail(g != NULL); |
| |
141 |
| |
142 gc = g->gc; |
| |
143 uid = g->uid; |
| |
144 g_return_if_fail(uid != 0); |
| |
145 |
| |
146 _qq_send_packet_buddy_auth(gc, uid, QQ_MY_AUTH_REQUEST, text); |
| |
147 g_free(g); |
| |
148 } |
| |
149 |
| |
150 /* the real packet to reject and request is sent from here */ |
| |
151 static void _qq_reject_add_request_real(gc_and_uid *g, const gchar *reason) |
| |
152 { |
| |
153 gint uid; |
| |
154 PurpleConnection *gc; |
| |
155 |
| |
156 g_return_if_fail(g != NULL); |
| |
157 |
| |
158 gc = g->gc; |
| |
159 uid = g->uid; |
| |
160 g_return_if_fail(uid != 0); |
| |
161 |
| |
162 _qq_send_packet_buddy_auth(gc, uid, QQ_MY_AUTH_REJECT, reason); |
| |
163 g_free(g); |
| |
164 } |
| |
165 |
| |
166 /* we approve other's request of adding me as friend */ |
| |
167 void qq_approve_add_request_with_gc_and_uid(gc_and_uid *g) |
| |
168 { |
| |
169 gint uid; |
| |
170 PurpleConnection *gc; |
| |
171 |
| |
172 g_return_if_fail(g != NULL); |
| |
173 |
| |
174 gc = g->gc; |
| |
175 uid = g->uid; |
| |
176 g_return_if_fail(uid != 0); |
| |
177 |
| |
178 _qq_send_packet_buddy_auth(gc, uid, QQ_MY_AUTH_APPROVE, NULL); |
| |
179 g_free(g); |
| |
180 } |
| |
181 |
| |
182 void qq_do_nothing_with_gc_and_uid(gc_and_uid *g, const gchar *msg) |
| |
183 { |
| |
184 g_free(g); |
| |
185 } |
| |
186 |
| |
187 /* we reject other's request of adding me as friend */ |
| |
188 void qq_reject_add_request_with_gc_and_uid(gc_and_uid *g) |
| |
189 { |
| |
190 gint uid; |
| |
191 gchar *msg1, *msg2; |
| |
192 PurpleConnection *gc; |
| |
193 gc_and_uid *g2; |
| |
194 |
| |
195 g_return_if_fail(g != NULL); |
| |
196 |
| |
197 gc = g->gc; |
| |
198 uid = g->uid; |
| |
199 g_return_if_fail(uid != 0); |
| |
200 |
| |
201 g_free(g); |
| |
202 |
| |
203 g2 = g_new0(gc_and_uid, 1); |
| |
204 g2->gc = gc; |
| |
205 g2->uid = uid; |
| |
206 |
| |
207 msg1 = g_strdup_printf(_("You rejected %d's request"), uid); |
| |
208 msg2 = g_strdup(_("Input your reason:")); |
| |
209 |
| |
210 purple_request_input(gc, _("Reject request"), msg1, msg2, |
| |
211 _("Sorry, you are not my type..."), TRUE, FALSE, |
| |
212 NULL, _("Reject"), G_CALLBACK(_qq_reject_add_request_real), _("Cancel"), NULL, g2); |
| |
213 } |
| |
214 |
| |
215 void qq_add_buddy_with_gc_and_uid(gc_and_uid *g) |
| |
216 { |
| |
217 gint uid; |
| |
218 PurpleConnection *gc; |
| |
219 |
| |
220 g_return_if_fail(g != NULL); |
| |
221 |
| |
222 gc = g->gc; |
| |
223 uid = g->uid; |
| |
224 g_return_if_fail(uid != 0); |
| |
225 |
| |
226 _qq_send_packet_add_buddy(gc, uid); |
| |
227 g_free(g); |
| |
228 } |
| |
229 |
| |
230 void qq_block_buddy_with_gc_and_uid(gc_and_uid *g) |
| |
231 { |
| |
232 guint32 uid; |
| |
233 PurpleConnection *gc; |
| |
234 PurpleBuddy buddy; |
| |
235 PurpleGroup group; |
| |
236 |
| |
237 g_return_if_fail(g != NULL); |
| |
238 |
| |
239 gc = g->gc; |
| |
240 uid = g->uid; |
| |
241 g_return_if_fail(uid > 0); |
| |
242 |
| |
243 buddy.name = uid_to_purple_name(uid); |
| |
244 group.name = PURPLE_GROUP_QQ_BLOCKED; |
| |
245 |
| |
246 qq_remove_buddy(gc, &buddy, &group); |
| |
247 _qq_send_packet_remove_self_from(gc, uid); |
| |
248 } |
| |
249 |
| |
250 /* process reply to add_buddy_auth request */ |
| |
251 void qq_process_add_buddy_auth_reply(guint8 *buf, gint buf_len, PurpleConnection *gc) |
| |
252 { |
| |
253 qq_data *qd; |
| |
254 gint len; |
| |
255 guint8 *data, *cursor, reply; |
| |
256 gchar **segments, *msg_utf8; |
| |
257 |
| |
258 g_return_if_fail(buf != NULL && buf_len != 0); |
| |
259 |
| |
260 qd = (qq_data *) gc->proto_data; |
| |
261 len = buf_len; |
| |
262 data = g_newa(guint8, len); |
| |
263 cursor = data; |
| |
264 |
| |
265 if (qq_crypt(DECRYPT, buf, buf_len, qd->session_key, data, &len)) { |
| |
266 read_packet_b(data, &cursor, len, &reply); |
| |
267 if (reply != QQ_ADD_BUDDY_AUTH_REPLY_OK) { |
| |
268 purple_debug(PURPLE_DEBUG_WARNING, "QQ", "Add buddy with auth request fails\n"); |
| |
269 if (NULL == (segments = split_data(data, len, "\x1f", 2))) |
| |
270 return; |
| |
271 msg_utf8 = qq_to_utf8(segments[1], QQ_CHARSET_DEFAULT); |
| |
272 purple_notify_error(gc, NULL, _("Add buddy with auth request fails"), msg_utf8); |
| |
273 g_free(msg_utf8); |
| |
274 } else { |
| |
275 purple_debug(PURPLE_DEBUG_INFO, "QQ", "Add buddy with auth request OK\n"); |
| |
276 } |
| |
277 } else { |
| |
278 purple_debug(PURPLE_DEBUG_ERROR, "QQ", "Error decrypt add buddy with auth reply\n"); |
| |
279 } |
| |
280 } |
| |
281 |
| |
282 /* process the server reply for my request to remove a buddy */ |
| |
283 void qq_process_remove_buddy_reply(guint8 *buf, gint buf_len, PurpleConnection *gc) |
| |
284 { |
| |
285 qq_data *qd; |
| |
286 gint len; |
| |
287 guint8 *data, *cursor, reply; |
| |
288 |
| |
289 g_return_if_fail(buf != NULL && buf_len != 0); |
| |
290 |
| |
291 qd = (qq_data *) gc->proto_data; |
| |
292 len = buf_len; |
| |
293 data = g_newa(guint8, len); |
| |
294 |
| |
295 if (qq_crypt(DECRYPT, buf, buf_len, qd->session_key, data, &len)) { |
| |
296 cursor = data; |
| |
297 read_packet_b(data, &cursor, len, &reply); |
| |
298 if (reply != QQ_REMOVE_BUDDY_REPLY_OK) { |
| |
299 /* there is no reason return from server */ |
| |
300 purple_debug(PURPLE_DEBUG_WARNING, "QQ", "Remove buddy fails\n"); |
| |
301 } else { /* if reply */ |
| |
302 purple_debug(PURPLE_DEBUG_INFO, "QQ", "Remove buddy OK\n"); |
| |
303 purple_notify_info(gc, NULL, _("You have successfully removed a buddy"), NULL); |
| |
304 } |
| |
305 } else { |
| |
306 purple_debug(PURPLE_DEBUG_ERROR, "QQ", "Error decrypt remove buddy reply\n"); |
| |
307 } |
| |
308 } |
| |
309 |
| |
310 /* process the server reply for my request to remove myself from a buddy */ |
| |
311 void qq_process_remove_self_reply(guint8 *buf, gint buf_len, PurpleConnection *gc) |
| |
312 { |
| |
313 qq_data *qd; |
| |
314 gint len; |
| |
315 guint8 *data, *cursor, reply; |
| |
316 |
| |
317 g_return_if_fail(buf != NULL && buf_len != 0); |
| |
318 |
| |
319 qd = (qq_data *) gc->proto_data; |
| |
320 len = buf_len; |
| |
321 data = g_newa(guint8, len); |
| |
322 |
| |
323 if (qq_crypt(DECRYPT, buf, buf_len, qd->session_key, data, &len)) { |
| |
324 cursor = data; |
| |
325 read_packet_b(data, &cursor, len, &reply); |
| |
326 if (reply != QQ_REMOVE_SELF_REPLY_OK) |
| |
327 /* there is no reason return from server */ |
| |
328 purple_debug(PURPLE_DEBUG_WARNING, "QQ", "Remove self fails\n"); |
| |
329 else { /* if reply */ |
| |
330 purple_debug(PURPLE_DEBUG_INFO, "QQ", "Remove self from a buddy OK\n"); |
| |
331 purple_notify_info(gc, NULL, _("You have successfully removed yourself from a buddy"), NULL); |
| |
332 } |
| |
333 } else { |
| |
334 purple_debug(PURPLE_DEBUG_ERROR, "QQ", "Error decrypt remove self reply\n"); |
| |
335 } |
| |
336 } |
| |
337 |
| |
338 void qq_process_add_buddy_reply(guint8 *buf, gint buf_len, guint16 seq, PurpleConnection *gc) |
| |
339 { |
| |
340 qq_data *qd; |
| |
341 gint len, for_uid; |
| |
342 gchar *msg, **segments, *uid, *reply; |
| |
343 guint8 *data; |
| |
344 GList *list; |
| |
345 PurpleBuddy *b; |
| |
346 gc_and_uid *g; |
| |
347 qq_add_buddy_request *req; |
| |
348 |
| |
349 g_return_if_fail(buf != NULL && buf_len != 0); |
| |
350 |
| |
351 for_uid = 0; |
| |
352 qd = (qq_data *) gc->proto_data; |
| |
353 len = buf_len; |
| |
354 |
| |
355 list = qd->add_buddy_request; |
| |
356 while (list != NULL) { |
| |
357 req = (qq_add_buddy_request *) list->data; |
| |
358 if (req->seq == seq) { /* reply to this */ |
| |
359 for_uid = req->uid; |
| |
360 qd->add_buddy_request = g_list_remove(qd->add_buddy_request, qd->add_buddy_request->data); |
| |
361 g_free(req); |
| |
362 break; |
| |
363 } |
| |
364 list = list->next; |
| |
365 } |
| |
366 |
| |
367 if (for_uid == 0) { /* we have no record for this */ |
| |
368 purple_debug(PURPLE_DEBUG_ERROR, "QQ", "We have no record for add buddy reply [%d], discard\n", seq); |
| |
369 return; |
| |
370 } else { |
| |
371 purple_debug(PURPLE_DEBUG_INFO, "QQ", "Add buddy reply [%d] is for id [%d]\n", seq, for_uid); |
| |
372 } |
| |
373 |
| |
374 data = g_newa(guint8, len); |
| |
375 |
| |
376 if (qq_crypt(DECRYPT, buf, buf_len, qd->session_key, data, &len)) { |
| |
377 if (NULL == (segments = split_data(data, len, "\x1f", 2))) |
| |
378 return; |
| |
379 uid = segments[0]; |
| |
380 reply = segments[1]; |
| |
381 if (strtol(uid, NULL, 10) != qd->uid) { /* should not happen */ |
| |
382 purple_debug(PURPLE_DEBUG_ERROR, "QQ", "Add buddy reply is to [%s], not me!", uid); |
| |
383 g_strfreev(segments); |
| |
384 return; |
| |
385 } |
| |
386 |
| |
387 if (strtol(reply, NULL, 10) > 0) { /* need auth */ |
| |
388 purple_debug(PURPLE_DEBUG_WARNING, "QQ", "Add buddy attempt fails, need authentication\n"); |
| |
389 b = purple_find_buddy(gc->account, uid_to_purple_name(for_uid)); |
| |
390 if (b != NULL) |
| |
391 purple_blist_remove_buddy(b); |
| |
392 g = g_new0(gc_and_uid, 1); |
| |
393 g->gc = gc; |
| |
394 g->uid = for_uid; |
| |
395 msg = g_strdup_printf(_("User %d needs authentication"), for_uid); |
| |
396 purple_request_input(gc, NULL, msg, |
| |
397 _("Input request here"), |
| |
398 _("Would you be my friend?"), |
| |
399 TRUE, FALSE, NULL, _("Send"), |
| |
400 G_CALLBACK |
| |
401 (_qq_send_packet_add_buddy_auth_with_gc_and_uid), |
| |
402 _("Cancel"), G_CALLBACK(qq_do_nothing_with_gc_and_uid), g); |
| |
403 g_free(msg); |
| |
404 } else { /* add OK */ |
| |
405 qq_add_buddy_by_recv_packet(gc, for_uid, TRUE, TRUE); |
| |
406 msg = g_strdup_printf(_("You have added %d in buddy list"), for_uid); |
| |
407 purple_notify_info(gc, NULL, msg, NULL); |
| |
408 g_free(msg); |
| |
409 } |
| |
410 g_strfreev(segments); |
| |
411 } else { |
| |
412 purple_debug(PURPLE_DEBUG_ERROR, "QQ", "Error decrypt add buddy reply\n"); |
| |
413 } |
| |
414 } |
| |
415 |
| |
416 PurpleGroup *qq_get_purple_group(const gchar *group_name) |
| |
417 { |
| |
418 PurpleGroup *g; |
| |
419 |
| |
420 g_return_val_if_fail(group_name != NULL, NULL); |
| |
421 |
| |
422 g = purple_find_group(group_name); |
| |
423 if (g == NULL) { |
| |
424 g = purple_group_new(group_name); |
| |
425 purple_blist_add_group(g, NULL); |
| |
426 purple_debug(PURPLE_DEBUG_WARNING, "QQ", "Add new group: %s\n", group_name); |
| |
427 } |
| |
428 |
| |
429 return g; |
| |
430 } |
| |
431 |
| |
432 /* we add new buddy, if the received packet is from someone not in my list |
| |
433 * return the PurpleBuddy that is just created */ |
| |
434 PurpleBuddy *qq_add_buddy_by_recv_packet(PurpleConnection *gc, guint32 uid, gboolean is_known, gboolean create) |
| |
435 { |
| |
436 PurpleAccount *a; |
| |
437 PurpleBuddy *b; |
| |
438 PurpleGroup *g; |
| |
439 qq_data *qd; |
| |
440 qq_buddy *q_bud; |
| |
441 gchar *name, *group_name; |
| |
442 |
| |
443 a = gc->account; |
| |
444 qd = (qq_data *) gc->proto_data; |
| |
445 g_return_val_if_fail(a != NULL && uid != 0, NULL); |
| |
446 |
| |
447 group_name = is_known ? |
| |
448 g_strdup_printf(PURPLE_GROUP_QQ_FORMAT, purple_account_get_username(a)) : g_strdup(PURPLE_GROUP_QQ_UNKNOWN); |
| |
449 |
| |
450 g = qq_get_purple_group(group_name); |
| |
451 |
| |
452 name = uid_to_purple_name(uid); |
| |
453 b = purple_find_buddy(gc->account, name); |
| |
454 /* remove old, we can not simply return here |
| |
455 * because there might be old local copy of this buddy */ |
| |
456 if (b != NULL) |
| |
457 purple_blist_remove_buddy(b); |
| |
458 |
| |
459 b = purple_buddy_new(a, name, NULL); |
| |
460 |
| |
461 if (!create) |
| |
462 b->proto_data = NULL; |
| |
463 else { |
| |
464 q_bud = g_new0(qq_buddy, 1); |
| |
465 q_bud->uid = uid; |
| |
466 b->proto_data = q_bud; |
| |
467 qd->buddies = g_list_append(qd->buddies, q_bud); |
| |
468 qq_send_packet_get_info(gc, q_bud->uid, FALSE); |
| |
469 qq_send_packet_get_buddies_online(gc, QQ_FRIENDS_ONLINE_POSITION_START); |
| |
470 } |
| |
471 |
| |
472 purple_blist_add_buddy(b, NULL, g, NULL); |
| |
473 purple_debug(PURPLE_DEBUG_WARNING, "QQ", "Add new buddy: [%s]\n", name); |
| |
474 |
| |
475 g_free(name); |
| |
476 g_free(group_name); |
| |
477 |
| |
478 return b; |
| |
479 } |
| |
480 |
| |
481 /* add a buddy and send packet to QQ server |
| |
482 * note that when purple load local cached buddy list into its blist |
| |
483 * it also calls this funtion, so we have to |
| |
484 * define qd->logged_in=TRUE AFTER serv_finish_login(gc) */ |
| |
485 void qq_add_buddy(PurpleConnection *gc, PurpleBuddy *buddy, PurpleGroup *group) |
| |
486 { |
| |
487 qq_data *qd; |
| |
488 guint32 uid; |
| |
489 PurpleBuddy *b; |
| |
490 |
| |
491 qd = (qq_data *) gc->proto_data; |
| |
492 if (!qd->logged_in) |
| |
493 return; /* IMPORTANT ! */ |
| |
494 |
| |
495 uid = purple_name_to_uid(buddy->name); |
| |
496 if (uid > 0) |
| |
497 _qq_send_packet_add_buddy(gc, uid); |
| |
498 else { |
| |
499 b = purple_find_buddy(gc->account, buddy->name); |
| |
500 if (b != NULL) |
| |
501 purple_blist_remove_buddy(b); |
| |
502 purple_notify_error(gc, NULL, |
| |
503 _("QQid Error"), |
| |
504 _("Invalid QQid")); |
| |
505 } |
| |
506 } |
| |
507 |
| |
508 /* remove a buddy and send packet to QQ server accordingly */ |
| |
509 void qq_remove_buddy(PurpleConnection *gc, PurpleBuddy *buddy, PurpleGroup *group) |
| |
510 { |
| |
511 qq_data *qd; |
| |
512 PurpleBuddy *b; |
| |
513 qq_buddy *q_bud; |
| |
514 guint32 uid; |
| |
515 |
| |
516 qd = (qq_data *) gc->proto_data; |
| |
517 uid = purple_name_to_uid(buddy->name); |
| |
518 |
| |
519 if (!qd->logged_in) |
| |
520 return; |
| |
521 |
| |
522 if (uid > 0) |
| |
523 _qq_send_packet_remove_buddy(gc, uid); |
| |
524 |
| |
525 b = purple_find_buddy(gc->account, buddy->name); |
| |
526 if (b != NULL) { |
| |
527 q_bud = (qq_buddy *) b->proto_data; |
| |
528 if (q_bud != NULL) |
| |
529 qd->buddies = g_list_remove(qd->buddies, q_bud); |
| |
530 else |
| |
531 purple_debug(PURPLE_DEBUG_WARNING, "QQ", "We have no qq_buddy record for %s\n", buddy->name); |
| |
532 /* remove buddy on blist, this does not trigger qq_remove_buddy again |
| |
533 * do this only if the request comes from block request, |
| |
534 * otherwise purple segmentation fault */ |
| |
535 if (g_ascii_strcasecmp(group->name, PURPLE_GROUP_QQ_BLOCKED) == 0) |
| |
536 purple_blist_remove_buddy(b); |
| |
537 } |
| |
538 } |
| |
539 |
| |
540 /* free add buddy request queue */ |
| |
541 void qq_add_buddy_request_free(qq_data *qd) |
| |
542 { |
| |
543 gint i; |
| |
544 qq_add_buddy_request *p; |
| |
545 |
| |
546 i = 0; |
| |
547 while (qd->add_buddy_request) { |
| |
548 p = (qq_add_buddy_request *) (qd->add_buddy_request->data); |
| |
549 qd->add_buddy_request = g_list_remove(qd->add_buddy_request, p); |
| |
550 g_free(p); |
| |
551 i++; |
| |
552 } |
| |
553 purple_debug(PURPLE_DEBUG_INFO, "QQ", "%d add buddy requests are freed!\n", i); |
| |
554 } |
| |
555 |
| |
556 /* free up all qq_buddy */ |
| |
557 void qq_buddies_list_free(PurpleAccount *account, qq_data *qd) |
| |
558 { |
| |
559 gint i; |
| |
560 qq_buddy *p; |
| |
561 gchar *name; |
| |
562 PurpleBuddy *b; |
| |
563 |
| |
564 i = 0; |
| |
565 while (qd->buddies) { |
| |
566 p = (qq_buddy *) (qd->buddies->data); |
| |
567 qd->buddies = g_list_remove(qd->buddies, p); |
| |
568 name = uid_to_purple_name(p->uid); |
| |
569 b = purple_find_buddy(account, name); |
| |
570 if(b != NULL) |
| |
571 b->proto_data = NULL; |
| |
572 else |
| |
573 purple_debug(PURPLE_DEBUG_INFO, "QQ", "qq_buddy %s not found in purple proto_data\n", name); |
| |
574 g_free(name); |
| |
575 |
| |
576 g_free(p); |
| |
577 i++; |
| |
578 } |
| |
579 purple_debug(PURPLE_DEBUG_INFO, "QQ", "%d qq_buddy structures are freed!\n", i); |
| |
580 } |