Remove useless comparison
[pidgin-git.git] / libpurple / protocols / oscar / oscar.c
blob6c11b0145115c2d7faf12f985e23225c98c1fa07
1 /*
2 * purple
4 * Some code copyright (C) 1998-1999, Mark Spencer <markster@marko.net>
5 * Some code copyright (C) 1999-2001, Eric Warmenhoven
6 * Some code copyright (C) 2001-2003, Sean Egan
7 * Some code copyright (C) 2001-2007, Mark Doliner <thekingant@users.sourceforge.net>
8 * Some code copyright (C) 2005, Jonathan Clark <ardentlygnarly@users.sourceforge.net>
9 * Some code copyright (C) 2007, ComBOTS Product GmbH (htfv) <foss@combots.com>
10 * Some code copyright (C) 2008, Aman Gupta
12 * Most libfaim code copyright (C) 1998-2001 Adam Fritzler <afritz@auk.cx>
13 * Some libfaim code copyright (C) 2001-2004 Mark Doliner <thekingant@users.sourceforge.net>
15 * This program is free software; you can redistribute it and/or modify
16 * it under the terms of the GNU General Public License as published by
17 * the Free Software Foundation; either version 2 of the License, or
18 * (at your option) any later version.
20 * This program is distributed in the hope that it will be useful,
21 * but WITHOUT ANY WARRANTY; without even the implied warranty of
22 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
23 * GNU General Public License for more details.
25 * You should have received a copy of the GNU General Public License
26 * along with this program; if not, write to the Free Software
27 * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02111-1301 USA
31 #include "internal.h"
33 #include "account.h"
34 #include "accountopt.h"
35 #include "buddyicon.h"
36 #include "cipher.h"
37 #include "conversation.h"
38 #include "core.h"
39 #include "debug.h"
40 #include "encoding.h"
41 #include "imgstore.h"
42 #include "network.h"
43 #include "notify.h"
44 #include "privacy.h"
45 #include "prpl.h"
46 #include "proxy.h"
47 #include "request.h"
48 #include "util.h"
49 #include "version.h"
50 #include "visibility.h"
52 #include "oscarcommon.h"
53 #include "oscar.h"
54 #include "peer.h"
56 #define AIMHASHDATA "http://pidgin.im/aim_data.php3"
58 #define OSCAR_CONNECT_STEPS 6
60 static guint64 purple_caps =
61 OSCAR_CAPABILITY_CHAT
62 | OSCAR_CAPABILITY_BUDDYICON
63 | OSCAR_CAPABILITY_DIRECTIM
64 | OSCAR_CAPABILITY_SENDFILE
65 | OSCAR_CAPABILITY_UNICODE
66 | OSCAR_CAPABILITY_INTEROPERATE
67 | OSCAR_CAPABILITY_SHORTCAPS
68 | OSCAR_CAPABILITY_TYPING
69 | OSCAR_CAPABILITY_ICQSERVERRELAY
70 | OSCAR_CAPABILITY_NEWCAPS
71 | OSCAR_CAPABILITY_XTRAZ
72 | OSCAR_CAPABILITY_HTML_MSGS;
74 static guint8 features_aim[] = {0x01, 0x01, 0x01, 0x02};
75 static guint8 features_icq[] = {0x01};
77 struct create_room {
78 char *name;
79 int exchange;
82 struct oscar_ask_directim_data
84 OscarData *od;
85 char *who;
88 /* All the libfaim->purple callback functions */
90 /* Only used when connecting with the old-style BUCP login */
91 static int purple_parse_auth_resp (OscarData *, FlapConnection *, FlapFrame *, ...);
92 static int purple_parse_login (OscarData *, FlapConnection *, FlapFrame *, ...);
93 static int purple_parse_auth_securid_request(OscarData *, FlapConnection *, FlapFrame *, ...);
95 static int purple_handle_redirect (OscarData *, FlapConnection *, FlapFrame *, ...);
96 static int purple_info_change (OscarData *, FlapConnection *, FlapFrame *, ...);
97 static int purple_account_confirm (OscarData *, FlapConnection *, FlapFrame *, ...);
98 static int purple_parse_oncoming (OscarData *, FlapConnection *, FlapFrame *, ...);
99 static int purple_parse_offgoing (OscarData *, FlapConnection *, FlapFrame *, ...);
100 static int purple_parse_incoming_im(OscarData *, FlapConnection *, FlapFrame *, ...);
101 static int purple_parse_misses (OscarData *, FlapConnection *, FlapFrame *, ...);
102 static int purple_parse_clientauto (OscarData *, FlapConnection *, FlapFrame *, ...);
103 static int purple_parse_motd (OscarData *, FlapConnection *, FlapFrame *, ...);
104 static int purple_chatnav_info (OscarData *, FlapConnection *, FlapFrame *, ...);
105 static int purple_conv_chat_join (OscarData *, FlapConnection *, FlapFrame *, ...);
106 static int purple_conv_chat_leave (OscarData *, FlapConnection *, FlapFrame *, ...);
107 static int purple_conv_chat_info_update (OscarData *, FlapConnection *, FlapFrame *, ...);
108 static int purple_conv_chat_incoming_msg(OscarData *, FlapConnection *, FlapFrame *, ...);
109 static int purple_email_parseupdate(OscarData *, FlapConnection *, FlapFrame *, ...);
110 static int purple_icon_parseicon (OscarData *, FlapConnection *, FlapFrame *, ...);
111 static int purple_parse_searcherror(OscarData *, FlapConnection *, FlapFrame *, ...);
112 static int purple_parse_searchreply(OscarData *, FlapConnection *, FlapFrame *, ...);
113 static int purple_bosrights (OscarData *, FlapConnection *, FlapFrame *, ...);
114 static int purple_connerr (OscarData *, FlapConnection *, FlapFrame *, ...);
115 static int purple_parse_mtn (OscarData *, FlapConnection *, FlapFrame *, ...);
116 static int purple_parse_locaterights(OscarData *, FlapConnection *, FlapFrame *, ...);
117 static int purple_parse_buddyrights(OscarData *, FlapConnection *, FlapFrame *, ...);
118 static int purple_parse_genericerr (OscarData *, FlapConnection *, FlapFrame *, ...);
119 static int purple_memrequest (OscarData *, FlapConnection *, FlapFrame *, ...);
120 static int purple_selfinfo (OscarData *, FlapConnection *, FlapFrame *, ...);
121 static int purple_popup (OscarData *, FlapConnection *, FlapFrame *, ...);
122 static int purple_ssi_parseerr (OscarData *, FlapConnection *, FlapFrame *, ...);
123 static int purple_ssi_parserights (OscarData *, FlapConnection *, FlapFrame *, ...);
124 static int purple_ssi_parselist (OscarData *, FlapConnection *, FlapFrame *, ...);
125 static int purple_ssi_parseack (OscarData *, FlapConnection *, FlapFrame *, ...);
126 static int purple_ssi_parseaddmod (OscarData *, FlapConnection *, FlapFrame *, ...);
127 static int purple_ssi_authgiven (OscarData *, FlapConnection *, FlapFrame *, ...);
128 static int purple_ssi_authrequest (OscarData *, FlapConnection *, FlapFrame *, ...);
129 static int purple_ssi_authreply (OscarData *, FlapConnection *, FlapFrame *, ...);
130 static int purple_ssi_gotadded (OscarData *, FlapConnection *, FlapFrame *, ...);
132 static void purple_icons_fetch(PurpleConnection *gc);
134 void oscar_set_info(PurpleConnection *gc, const char *info);
135 static void oscar_set_info_and_status(PurpleAccount *account, gboolean setinfo, const char *rawinfo, gboolean setstatus, PurpleStatus *status);
136 static void oscar_set_extended_status(PurpleConnection *gc);
137 static gboolean purple_ssi_rerequestdata(gpointer data);
139 void oscar_free_name_data(struct name_data *data) {
140 g_free(data->name);
141 g_free(data->nick);
142 g_free(data);
145 #ifdef _WIN32
146 const char *oscar_get_locale_charset(void) {
147 static const char *charset = NULL;
148 if (charset == NULL)
149 g_get_charset(&charset);
150 return charset;
152 #endif
154 static char *oscar_icqstatus(int state) {
155 /* Make a cute little string that shows the status of the dude or dudet */
156 if (state & AIM_ICQ_STATE_CHAT)
157 return g_strdup(_("Free For Chat"));
158 else if (state & AIM_ICQ_STATE_DND)
159 return g_strdup(_("Do Not Disturb"));
160 else if (state & AIM_ICQ_STATE_OUT)
161 return g_strdup(_("Not Available"));
162 else if (state & AIM_ICQ_STATE_BUSY)
163 return g_strdup(_("Occupied"));
164 else if (state & AIM_ICQ_STATE_AWAY)
165 return g_strdup(_("Away"));
166 else if (state & AIM_ICQ_STATE_WEBAWARE)
167 return g_strdup(_("Web Aware"));
168 else if (state & AIM_ICQ_STATE_INVISIBLE)
169 return g_strdup(_("Invisible"));
170 else if (state & AIM_ICQ_STATE_EVIL)
171 return g_strdup(_("Evil"));
172 else if (state & AIM_ICQ_STATE_DEPRESSION)
173 return g_strdup(_("Depression"));
174 else if (state & AIM_ICQ_STATE_ATHOME)
175 return g_strdup(_("At home"));
176 else if (state & AIM_ICQ_STATE_ATWORK)
177 return g_strdup(_("At work"));
178 else if (state & AIM_ICQ_STATE_LUNCH)
179 return g_strdup(_("At lunch"));
180 else
181 return g_strdup(_("Online"));
184 static char *extract_name(const char *name) {
185 char *tmp, *x;
186 int i, j;
188 if (!name)
189 return NULL;
191 x = strchr(name, '-');
192 if (!x)
193 return NULL;
195 x = strchr(x + 1, '-');
196 if (!x)
197 return NULL;
199 tmp = g_strdup(++x);
201 for (i = 0, j = 0; x[i]; i++) {
202 char hex[3];
203 if (x[i] != '%') {
204 tmp[j++] = x[i];
205 continue;
207 strncpy(hex, x + ++i, 2);
208 hex[2] = 0;
209 i++;
210 tmp[j++] = strtol(hex, NULL, 16);
213 tmp[j] = 0;
214 return tmp;
217 static struct chat_connection *
218 find_oscar_chat(PurpleConnection *gc, int id)
220 OscarData *od = purple_connection_get_protocol_data(gc);
221 GSList *cur;
222 struct chat_connection *cc;
224 for (cur = od->oscar_chats; cur != NULL; cur = cur->next)
226 cc = (struct chat_connection *)cur->data;
227 if (cc->id == id)
228 return cc;
231 return NULL;
234 static struct chat_connection *
235 find_oscar_chat_by_conn(PurpleConnection *gc, FlapConnection *conn)
237 OscarData *od = purple_connection_get_protocol_data(gc);
238 GSList *cur;
239 struct chat_connection *cc;
241 for (cur = od->oscar_chats; cur != NULL; cur = cur->next)
243 cc = (struct chat_connection *)cur->data;
244 if (cc->conn == conn)
245 return cc;
248 return NULL;
251 static struct chat_connection *
252 find_oscar_chat_by_conv(PurpleConnection *gc, PurpleConversation *conv)
254 OscarData *od = purple_connection_get_protocol_data(gc);
255 GSList *cur;
256 struct chat_connection *cc;
258 for (cur = od->oscar_chats; cur != NULL; cur = cur->next)
260 cc = (struct chat_connection *)cur->data;
261 if (cc->conv == conv)
262 return cc;
265 return NULL;
268 void
269 oscar_chat_destroy(struct chat_connection *cc)
271 g_free(cc->name);
272 g_free(cc->show);
273 g_free(cc);
276 static void
277 oscar_chat_kill(PurpleConnection *gc, struct chat_connection *cc)
279 OscarData *od = purple_connection_get_protocol_data(gc);
281 /* Notify the conversation window that we've left the chat */
282 serv_got_chat_left(gc, purple_conv_chat_get_id(PURPLE_CONV_CHAT(cc->conv)));
284 /* Destroy the chat_connection */
285 od->oscar_chats = g_slist_remove(od->oscar_chats, cc);
286 oscar_chat_destroy(cc);
290 * This is called from the callback functions for establishing
291 * a TCP connection with an oscar host if an error occurred.
293 static void
294 connection_common_error_cb(FlapConnection *conn, const gchar *error_message)
296 OscarData *od;
297 PurpleConnection *gc;
299 od = conn->od;
300 gc = od->gc;
302 purple_debug_error("oscar", "unable to connect to FLAP "
303 "server of type 0x%04hx\n", conn->type);
305 if (conn->type == SNAC_FAMILY_AUTH)
307 /* This only happens when connecting with the old-style BUCP login */
308 gchar *msg;
309 msg = g_strdup_printf(_("Unable to connect to authentication server: %s"),
310 error_message);
311 purple_connection_error_reason(gc, PURPLE_CONNECTION_ERROR_NETWORK_ERROR, msg);
312 g_free(msg);
314 else if (conn->type == SNAC_FAMILY_LOCATE)
316 gchar *msg;
317 msg = g_strdup_printf(_("Unable to connect to BOS server: %s"),
318 error_message);
319 purple_connection_error_reason(gc, PURPLE_CONNECTION_ERROR_NETWORK_ERROR, msg);
320 g_free(msg);
322 else
324 /* Maybe we should call this for BOS connections, too? */
325 flap_connection_schedule_destroy(conn,
326 OSCAR_DISCONNECT_COULD_NOT_CONNECT, error_message);
331 * This is called from the callback functions for establishing
332 * a TCP connection with an oscar host. Depending on the type
333 * of host, we do a few different things here.
335 static void
336 connection_common_established_cb(FlapConnection *conn)
338 OscarData *od;
339 PurpleConnection *gc;
340 PurpleAccount *account;
342 od = conn->od;
343 gc = od->gc;
344 account = purple_connection_get_account(gc);
346 purple_debug_info("oscar", "connected to FLAP server of type 0x%04hx\n",
347 conn->type);
349 if (conn->cookie == NULL)
350 flap_connection_send_version(od, conn);
351 else
353 const gchar *login_type = purple_account_get_string(account, "login_type", OSCAR_DEFAULT_LOGIN);
355 if (!purple_strequal(login_type, OSCAR_MD5_LOGIN))
357 ClientInfo aiminfo = CLIENTINFO_PURPLE_AIM;
358 ClientInfo icqinfo = CLIENTINFO_PURPLE_ICQ;
359 flap_connection_send_version_with_cookie_and_clientinfo(od,
360 conn, conn->cookielen, conn->cookie,
361 od->icq ? &icqinfo : &aiminfo,
362 purple_account_get_bool(account, "allow_multiple_logins", OSCAR_DEFAULT_ALLOW_MULTIPLE_LOGINS));
363 } else {
364 flap_connection_send_version_with_cookie(od, conn,
365 conn->cookielen, conn->cookie);
369 g_free(conn->cookie);
370 conn->cookie = NULL;
373 if (conn->type == SNAC_FAMILY_AUTH)
375 /* This only happens when connecting with the old-style BUCP login */
376 aim_request_login(od, conn, purple_account_get_username(account));
377 purple_debug_info("oscar", "Username sent, waiting for response\n");
378 purple_connection_update_progress(gc, _("Username sent"), 1, OSCAR_CONNECT_STEPS);
380 else if (conn->type == SNAC_FAMILY_LOCATE)
382 purple_connection_update_progress(gc, _("Connection established, cookie sent"), 4, OSCAR_CONNECT_STEPS);
384 else if (conn->type == SNAC_FAMILY_CHAT)
386 od->oscar_chats = g_slist_prepend(od->oscar_chats, conn->new_conn_data);
387 conn->new_conn_data = NULL;
391 static void
392 connection_established_cb(gpointer data, gint source, const gchar *error_message)
394 FlapConnection *conn;
396 conn = data;
398 conn->connect_data = NULL;
399 conn->fd = source;
401 if (source < 0)
403 connection_common_error_cb(conn, error_message);
404 return;
407 conn->watcher_incoming = purple_input_add(conn->fd,
408 PURPLE_INPUT_READ, flap_connection_recv_cb, conn);
409 connection_common_established_cb(conn);
412 static void
413 ssl_connection_established_cb(gpointer data, PurpleSslConnection *gsc,
414 PurpleInputCondition cond)
416 FlapConnection *conn;
418 conn = data;
420 purple_ssl_input_add(gsc, flap_connection_recv_cb_ssl, conn);
421 connection_common_established_cb(conn);
424 static void
425 ssl_connection_error_cb(PurpleSslConnection *gsc, PurpleSslErrorType error,
426 gpointer data)
428 FlapConnection *conn;
430 conn = data;
432 if (conn->watcher_outgoing)
434 purple_input_remove(conn->watcher_outgoing);
435 conn->watcher_outgoing = 0;
438 /* sslconn frees the connection on error */
439 conn->gsc = NULL;
441 connection_common_error_cb(conn, purple_ssl_strerror(error));
444 static void
445 flap_connection_established_bos(OscarData *od, FlapConnection *conn)
447 PurpleConnection *gc = od->gc;
449 aim_srv_reqpersonalinfo(od, conn);
451 purple_debug_info("oscar", "ssi: requesting rights and list\n");
452 aim_ssi_reqrights(od);
453 aim_ssi_reqdata(od);
454 if (od->getblisttimer > 0)
455 purple_timeout_remove(od->getblisttimer);
456 od->getblisttimer = purple_timeout_add_seconds(30, purple_ssi_rerequestdata, od);
458 aim_locate_reqrights(od);
459 aim_buddylist_reqrights(od, conn);
460 aim_im_reqparams(od);
461 aim_bos_reqrights(od, conn); /* TODO: Don't call this with ssi */
463 purple_connection_update_progress(gc, _("Finalizing connection"), 5, OSCAR_CONNECT_STEPS);
466 static void
467 flap_connection_established_admin(OscarData *od, FlapConnection *conn)
469 aim_srv_clientready(od, conn);
470 purple_debug_info("oscar", "connected to admin\n");
472 if (od->chpass) {
473 purple_debug_info("oscar", "changing password\n");
474 aim_admin_changepasswd(od, conn, od->newp, od->oldp);
475 g_free(od->oldp);
476 od->oldp = NULL;
477 g_free(od->newp);
478 od->newp = NULL;
479 od->chpass = FALSE;
481 if (od->setnick) {
482 purple_debug_info("oscar", "formatting username\n");
483 aim_admin_setnick(od, conn, od->newformatting);
484 g_free(od->newformatting);
485 od->newformatting = NULL;
486 od->setnick = FALSE;
488 if (od->conf) {
489 purple_debug_info("oscar", "confirming account\n");
490 aim_admin_reqconfirm(od, conn);
491 od->conf = FALSE;
493 if (od->reqemail) {
494 purple_debug_info("oscar", "requesting email address\n");
495 aim_admin_getinfo(od, conn, 0x0011);
496 od->reqemail = FALSE;
498 if (od->setemail) {
499 purple_debug_info("oscar", "setting email address\n");
500 aim_admin_setemail(od, conn, od->email);
501 g_free(od->email);
502 od->email = NULL;
503 od->setemail = FALSE;
507 static void
508 flap_connection_established_chat(OscarData *od, FlapConnection *conn)
510 PurpleConnection *gc = od->gc;
511 struct chat_connection *chatcon;
512 static int id = 1;
514 aim_srv_clientready(od, conn);
516 chatcon = find_oscar_chat_by_conn(gc, conn);
517 if (chatcon) {
518 chatcon->id = id;
519 chatcon->conv = serv_got_joined_chat(gc, id++, chatcon->show);
523 static void
524 flap_connection_established_chatnav(OscarData *od, FlapConnection *conn)
526 aim_srv_clientready(od, conn);
527 aim_chatnav_reqrights(od, conn);
530 static void
531 flap_connection_established_alert(OscarData *od, FlapConnection *conn)
533 aim_email_sendcookies(od);
534 aim_email_activate(od);
535 aim_srv_clientready(od, conn);
538 static void
539 flap_connection_established_bart(OscarData *od, FlapConnection *conn)
541 PurpleConnection *gc = od->gc;
543 aim_srv_clientready(od, conn);
545 od->iconconnecting = FALSE;
547 purple_icons_fetch(gc);
550 static int
551 flap_connection_established(OscarData *od, FlapConnection *conn, FlapFrame *fr, ...)
553 conn->connected = TRUE;
554 purple_debug_info("oscar", "FLAP connection of type 0x%04hx is "
555 "now fully connected\n", conn->type);
556 if (conn->type == SNAC_FAMILY_LOCATE)
557 flap_connection_established_bos(od, conn);
558 else if (conn->type == SNAC_FAMILY_ADMIN)
559 flap_connection_established_admin(od, conn);
560 else if (conn->type == SNAC_FAMILY_CHAT)
561 flap_connection_established_chat(od, conn);
562 else if (conn->type == SNAC_FAMILY_CHATNAV)
563 flap_connection_established_chatnav(od, conn);
564 else if (conn->type == SNAC_FAMILY_ALERT)
565 flap_connection_established_alert(od, conn);
566 else if (conn->type == SNAC_FAMILY_BART)
567 flap_connection_established_bart(od, conn);
569 return 1;
572 static void
573 idle_reporting_pref_cb(const char *name, PurplePrefType type,
574 gconstpointer value, gpointer data)
576 PurpleConnection *gc;
577 OscarData *od;
578 gboolean report_idle;
579 guint32 presence;
581 gc = data;
582 od = purple_connection_get_protocol_data(gc);
583 report_idle = !purple_strequal((const char *)value, "none");
584 presence = aim_ssi_getpresence(od->ssi.local);
586 if (report_idle)
587 aim_ssi_setpresence(od, presence | AIM_SSI_PRESENCE_FLAG_SHOWIDLE);
588 else
589 aim_ssi_setpresence(od, presence & ~AIM_SSI_PRESENCE_FLAG_SHOWIDLE);
593 * Should probably make a "Use recent buddies group" account preference
594 * so that this option is surfaced to the user.
596 static void
597 recent_buddies_pref_cb(const char *name, PurplePrefType type,
598 gconstpointer value, gpointer data)
600 PurpleConnection *gc;
601 OscarData *od;
602 guint32 presence;
604 gc = data;
605 od = purple_connection_get_protocol_data(gc);
606 presence = aim_ssi_getpresence(od->ssi.local);
608 if (value)
609 aim_ssi_setpresence(od, presence & ~AIM_SSI_PRESENCE_FLAG_NORECENTBUDDIES);
610 else
611 aim_ssi_setpresence(od, presence | AIM_SSI_PRESENCE_FLAG_NORECENTBUDDIES);
614 static const gchar *login_servers[] = {
615 AIM_DEFAULT_LOGIN_SERVER,
616 AIM_DEFAULT_SSL_LOGIN_SERVER,
617 ICQ_DEFAULT_LOGIN_SERVER,
618 ICQ_DEFAULT_SSL_LOGIN_SERVER,
621 static const gchar *
622 get_login_server(gboolean is_icq, gboolean use_ssl)
624 return login_servers[(is_icq ? 2 : 0) + (use_ssl ? 1 : 0)];
627 static gint
628 compare_handlers(gconstpointer a, gconstpointer b)
630 guint aa = GPOINTER_TO_UINT(a);
631 guint bb = GPOINTER_TO_UINT(b);
632 guint family1 = aa >> 16;
633 guint family2 = bb >> 16;
634 guint subtype1 = aa & 0xFFFF;
635 guint subtype2 = bb & 0xFFFF;
636 if (family1 != family2) {
637 return family1 - family2;
639 return subtype1 - subtype2;
642 #if !GLIB_CHECK_VERSION(2,14,0)
643 static void hash_table_get_list_of_keys(gpointer key, gpointer value, gpointer user_data)
645 GList **handlers = (GList **)user_data;
647 *handlers = g_list_prepend(*handlers, key);
649 #endif /* GLIB < 2.14.0 */
651 void
652 oscar_login(PurpleAccount *account)
654 PurpleConnection *gc;
655 OscarData *od;
656 const gchar *encryption_type;
657 const gchar *login_type;
658 GList *handlers;
659 GList *sorted_handlers;
660 GList *cur;
661 GString *msg = g_string_new("");
663 gc = purple_account_get_connection(account);
664 od = oscar_data_new();
665 od->gc = gc;
666 purple_connection_set_protocol_data(gc, od);
668 oscar_data_addhandler(od, AIM_CB_FAM_SPECIAL, AIM_CB_SPECIAL_CONNERR, purple_connerr, 0);
669 oscar_data_addhandler(od, AIM_CB_FAM_SPECIAL, AIM_CB_SPECIAL_CONNINITDONE, flap_connection_established, 0);
671 oscar_data_addhandler(od, SNAC_FAMILY_ADMIN, 0x0003, purple_info_change, 0);
672 oscar_data_addhandler(od, SNAC_FAMILY_ADMIN, 0x0005, purple_info_change, 0);
673 oscar_data_addhandler(od, SNAC_FAMILY_ADMIN, 0x0007, purple_account_confirm, 0);
674 oscar_data_addhandler(od, SNAC_FAMILY_ALERT, 0x0001, purple_parse_genericerr, 0);
675 oscar_data_addhandler(od, SNAC_FAMILY_ALERT, SNAC_SUBTYPE_ALERT_MAILSTATUS, purple_email_parseupdate, 0);
677 /* These are only needed when connecting with the old-style BUCP login */
678 oscar_data_addhandler(od, SNAC_FAMILY_AUTH, 0x0003, purple_parse_auth_resp, 0);
679 oscar_data_addhandler(od, SNAC_FAMILY_AUTH, 0x0007, purple_parse_login, 0);
680 oscar_data_addhandler(od, SNAC_FAMILY_AUTH, SNAC_SUBTYPE_AUTH_SECURID_REQUEST, purple_parse_auth_securid_request, 0);
682 oscar_data_addhandler(od, SNAC_FAMILY_BART, SNAC_SUBTYPE_BART_RESPONSE, purple_icon_parseicon, 0);
683 oscar_data_addhandler(od, SNAC_FAMILY_BOS, 0x0001, purple_parse_genericerr, 0);
684 oscar_data_addhandler(od, SNAC_FAMILY_BOS, 0x0003, purple_bosrights, 0);
685 oscar_data_addhandler(od, SNAC_FAMILY_BUDDY, 0x0001, purple_parse_genericerr, 0);
686 oscar_data_addhandler(od, SNAC_FAMILY_BUDDY, SNAC_SUBTYPE_BUDDY_RIGHTSINFO, purple_parse_buddyrights, 0);
687 oscar_data_addhandler(od, SNAC_FAMILY_BUDDY, SNAC_SUBTYPE_BUDDY_ONCOMING, purple_parse_oncoming, 0);
688 oscar_data_addhandler(od, SNAC_FAMILY_BUDDY, SNAC_SUBTYPE_BUDDY_OFFGOING, purple_parse_offgoing, 0);
689 oscar_data_addhandler(od, SNAC_FAMILY_CHAT, 0x0001, purple_parse_genericerr, 0);
690 oscar_data_addhandler(od, SNAC_FAMILY_CHAT, SNAC_SUBTYPE_CHAT_USERJOIN, purple_conv_chat_join, 0);
691 oscar_data_addhandler(od, SNAC_FAMILY_CHAT, SNAC_SUBTYPE_CHAT_USERLEAVE, purple_conv_chat_leave, 0);
692 oscar_data_addhandler(od, SNAC_FAMILY_CHAT, SNAC_SUBTYPE_CHAT_ROOMINFOUPDATE, purple_conv_chat_info_update, 0);
693 oscar_data_addhandler(od, SNAC_FAMILY_CHAT, SNAC_SUBTYPE_CHAT_INCOMINGMSG, purple_conv_chat_incoming_msg, 0);
694 oscar_data_addhandler(od, SNAC_FAMILY_CHATNAV, 0x0001, purple_parse_genericerr, 0);
695 oscar_data_addhandler(od, SNAC_FAMILY_CHATNAV, SNAC_SUBTYPE_CHATNAV_INFO, purple_chatnav_info, 0);
696 oscar_data_addhandler(od, SNAC_FAMILY_FEEDBAG, SNAC_SUBTYPE_FEEDBAG_ERROR, purple_ssi_parseerr, 0);
697 oscar_data_addhandler(od, SNAC_FAMILY_FEEDBAG, SNAC_SUBTYPE_FEEDBAG_RIGHTSINFO, purple_ssi_parserights, 0);
698 oscar_data_addhandler(od, SNAC_FAMILY_FEEDBAG, SNAC_SUBTYPE_FEEDBAG_LIST, purple_ssi_parselist, 0);
699 oscar_data_addhandler(od, SNAC_FAMILY_FEEDBAG, SNAC_SUBTYPE_FEEDBAG_SRVACK, purple_ssi_parseack, 0);
700 oscar_data_addhandler(od, SNAC_FAMILY_FEEDBAG, SNAC_SUBTYPE_FEEDBAG_ADD, purple_ssi_parseaddmod, 0);
701 oscar_data_addhandler(od, SNAC_FAMILY_FEEDBAG, SNAC_SUBTYPE_FEEDBAG_MOD, purple_ssi_parseaddmod, 0);
702 oscar_data_addhandler(od, SNAC_FAMILY_FEEDBAG, SNAC_SUBTYPE_FEEDBAG_RECVAUTH, purple_ssi_authgiven, 0);
703 oscar_data_addhandler(od, SNAC_FAMILY_FEEDBAG, SNAC_SUBTYPE_FEEDBAG_RECVAUTHREQ, purple_ssi_authrequest, 0);
704 oscar_data_addhandler(od, SNAC_FAMILY_FEEDBAG, SNAC_SUBTYPE_FEEDBAG_RECVAUTHREP, purple_ssi_authreply, 0);
705 oscar_data_addhandler(od, SNAC_FAMILY_FEEDBAG, SNAC_SUBTYPE_FEEDBAG_ADDED, purple_ssi_gotadded, 0);
706 oscar_data_addhandler(od, SNAC_FAMILY_ICBM, SNAC_SUBTYPE_ICBM_INCOMING, purple_parse_incoming_im, 0);
707 oscar_data_addhandler(od, SNAC_FAMILY_ICBM, SNAC_SUBTYPE_ICBM_MISSEDCALL, purple_parse_misses, 0);
708 oscar_data_addhandler(od, SNAC_FAMILY_ICBM, SNAC_SUBTYPE_ICBM_CLIENTAUTORESP, purple_parse_clientauto, 0);
709 oscar_data_addhandler(od, SNAC_FAMILY_ICBM, SNAC_SUBTYPE_ICBM_MTN, purple_parse_mtn, 0);
710 oscar_data_addhandler(od, SNAC_FAMILY_LOCATE, SNAC_SUBTYPE_LOCATE_RIGHTSINFO, purple_parse_locaterights, 0);
711 oscar_data_addhandler(od, SNAC_FAMILY_OSERVICE, 0x0001, purple_parse_genericerr, 0);
712 oscar_data_addhandler(od, SNAC_FAMILY_OSERVICE, 0x000f, purple_selfinfo, 0);
713 oscar_data_addhandler(od, SNAC_FAMILY_OSERVICE, 0x001f, purple_memrequest, 0);
714 oscar_data_addhandler(od, SNAC_FAMILY_OSERVICE, SNAC_SUBTYPE_OSERVICE_REDIRECT, purple_handle_redirect, 0);
715 oscar_data_addhandler(od, SNAC_FAMILY_OSERVICE, SNAC_SUBTYPE_OSERVICE_MOTD, purple_parse_motd, 0);
716 oscar_data_addhandler(od, SNAC_FAMILY_POPUP, 0x0002, purple_popup, 0);
717 oscar_data_addhandler(od, SNAC_FAMILY_USERLOOKUP, SNAC_SUBTYPE_USERLOOKUP_ERROR, purple_parse_searcherror, 0);
718 oscar_data_addhandler(od, SNAC_FAMILY_USERLOOKUP, 0x0003, purple_parse_searchreply, 0);
720 g_string_append(msg, "Registered handlers: ");
721 #if GLIB_CHECK_VERSION(2,14,0)
722 handlers = g_hash_table_get_keys(od->handlerlist);
723 #else
724 handlers = NULL;
725 g_hash_table_foreach(od->handlerlist, hash_table_get_list_of_keys, &handlers);
726 #endif /* GLIB < 2.14.0 */
727 sorted_handlers = g_list_sort(g_list_copy(handlers), compare_handlers);
728 for (cur = sorted_handlers; cur; cur = cur->next) {
729 guint x = GPOINTER_TO_UINT(cur->data);
730 g_string_append_printf(msg, "%04x/%04x, ", x >> 16, x & 0xFFFF);
732 g_list_free(sorted_handlers);
733 g_list_free(handlers);
734 purple_debug_misc("oscar", "%s\n", msg->str);
735 g_string_free(msg, TRUE);
737 purple_debug_misc("oscar", "oscar_login: gc = %p\n", gc);
739 if (!oscar_util_valid_name(purple_account_get_username(account))) {
740 gchar *buf;
741 buf = g_strdup_printf(_("Unable to sign on as %s because the username is invalid. Usernames must be a valid email address, or start with a letter and contain only letters, numbers and spaces, or contain only numbers."), purple_account_get_username(account));
742 purple_connection_error_reason(gc, PURPLE_CONNECTION_ERROR_INVALID_SETTINGS, buf);
743 g_free(buf);
744 return;
747 gc->flags |= PURPLE_CONNECTION_HTML;
748 if (purple_strequal(purple_account_get_protocol_id(account), "prpl-icq")) {
749 od->icq = TRUE;
750 } else {
751 gc->flags |= PURPLE_CONNECTION_AUTO_RESP;
754 /* Set this flag based on the protocol_id rather than the username,
755 because that is what's tied to the get_moods prpl callback. */
756 if (purple_strequal(purple_account_get_protocol_id(account), "prpl-icq"))
757 gc->flags |= PURPLE_CONNECTION_SUPPORT_MOODS;
759 od->default_port = purple_account_get_int(account, "port", OSCAR_DEFAULT_LOGIN_PORT);
761 login_type = purple_account_get_string(account, "login_type", OSCAR_DEFAULT_LOGIN);
762 encryption_type = purple_account_get_string(account, "encryption", OSCAR_DEFAULT_ENCRYPTION);
763 if (!purple_ssl_is_supported() && purple_strequal(encryption_type, OSCAR_REQUIRE_ENCRYPTION)) {
764 purple_connection_error_reason(
766 PURPLE_CONNECTION_ERROR_NO_SSL_SUPPORT,
767 _("You required encryption in your account settings, but encryption is not supported by your system."));
768 return;
770 od->use_ssl = purple_ssl_is_supported() && !purple_strequal(encryption_type, OSCAR_NO_ENCRYPTION);
772 /* Connect to core Purple signals */
773 purple_prefs_connect_callback(gc, "/purple/away/idle_reporting", idle_reporting_pref_cb, gc);
774 purple_prefs_connect_callback(gc, "/plugins/prpl/oscar/recent_buddies", recent_buddies_pref_cb, gc);
777 * On 2008-03-05 AOL released some documentation on the OSCAR protocol
778 * which includes a new login method called clientLogin. It is similar
779 * (though not the same?) as what the AIM 6.0 series uses to
780 * authenticate.
782 * AIM 5.9 and lower use an MD5-based login procedure called "BUCP".
783 * This authentication method is used for both ICQ and AIM when
784 * clientLogin is not enabled.
786 if (purple_strequal(login_type, OSCAR_CLIENT_LOGIN)) {
787 /* Note: Actual server/port configuration is ignored here */
788 send_client_login(od, purple_account_get_username(account));
789 } else if (purple_strequal(login_type, OSCAR_KERBEROS_LOGIN)) {
790 const char *server;
792 if (!od->use_ssl) {
793 purple_connection_error_reason(
795 PURPLE_CONNECTION_ERROR_NO_SSL_SUPPORT,
796 _("You required Kerberos authentication but encryption is disabled in your account settings."));
797 return;
799 server = purple_account_get_string(account, "server", AIM_DEFAULT_KDC_SERVER);
801 * If the account's server is what the oscar protocol has offered as
802 * the default login server through the vast eons (all two of
803 * said default options, AFAIK) and the user wants KDC, we'll
804 * do what we know is best for them and change the setting out
805 * from under them to the KDC login server.
807 if (purple_strequal(server, get_login_server(od->icq, FALSE)) ||
808 purple_strequal(server, get_login_server(od->icq, TRUE)) ||
809 purple_strequal(server, AIM_ALT_LOGIN_SERVER) ||
810 purple_strequal(server, "")) {
811 purple_debug_info("oscar", "Account uses Kerberos auth, so changing server to default KDC server\n");
812 purple_account_set_string(account, "server", AIM_DEFAULT_KDC_SERVER);
813 purple_account_set_int(account, "port", AIM_DEFAULT_KDC_PORT);
815 send_kerberos_login(od, purple_account_get_username(account));
816 } else {
817 FlapConnection *newconn;
818 const char *server;
820 newconn = flap_connection_new(od, SNAC_FAMILY_AUTH);
822 if (od->use_ssl) {
823 server = purple_account_get_string(account, "server", get_login_server(od->icq, TRUE));
826 * If the account's server is what the oscar prpl has offered as
827 * the default login server through the vast eons (all two of
828 * said default options, AFAIK) and the user wants SSL, we'll
829 * do what we know is best for them and change the setting out
830 * from under them to the SSL login server.
832 if (purple_strequal(server, get_login_server(od->icq, FALSE)) ||
833 purple_strequal(server, AIM_ALT_LOGIN_SERVER) ||
834 purple_strequal(server, AIM_DEFAULT_KDC_SERVER) ||
835 purple_strequal(server, "")) {
836 purple_debug_info("oscar", "Account uses SSL, so changing server to default SSL server\n");
837 purple_account_set_string(account, "server", get_login_server(od->icq, TRUE));
838 purple_account_set_int(account, "port", OSCAR_DEFAULT_LOGIN_PORT),
839 server = get_login_server(od->icq, TRUE);
842 newconn->gsc = purple_ssl_connect(account, server,
843 purple_account_get_int(account, "port", OSCAR_DEFAULT_LOGIN_PORT),
844 ssl_connection_established_cb, ssl_connection_error_cb, newconn);
845 } else {
846 server = purple_account_get_string(account, "server", get_login_server(od->icq, FALSE));
849 * See the comment above. We do the reverse here. If they don't want
850 * SSL but their server is set to OSCAR_DEFAULT_SSL_LOGIN_SERVER,
851 * set it back to the default.
853 if (purple_strequal(server, get_login_server(od->icq, TRUE)) ||
854 purple_strequal(server, AIM_DEFAULT_KDC_SERVER) ||
855 purple_strequal(server, "")) {
856 purple_debug_info("oscar", "Account does not use SSL, so changing server back to non-SSL\n");
857 purple_account_set_string(account, "server", get_login_server(od->icq, FALSE));
858 purple_account_set_int(account, "port", OSCAR_DEFAULT_LOGIN_PORT),
859 server = get_login_server(od->icq, FALSE);
862 newconn->connect_data = purple_proxy_connect(NULL, account, server,
863 purple_account_get_int(account, "port", OSCAR_DEFAULT_LOGIN_PORT),
864 connection_established_cb, newconn);
867 if (newconn->gsc == NULL && newconn->connect_data == NULL) {
868 purple_connection_error_reason(gc, PURPLE_CONNECTION_ERROR_NETWORK_ERROR,
869 _("Unable to connect"));
870 return;
874 purple_connection_update_progress(gc, _("Connecting"), 0, OSCAR_CONNECT_STEPS);
877 void
878 oscar_close(PurpleConnection *gc)
880 OscarData *od;
882 od = purple_connection_get_protocol_data(gc);
884 while (od->oscar_chats)
886 struct chat_connection *cc = od->oscar_chats->data;
887 od->oscar_chats = g_slist_remove(od->oscar_chats, cc);
888 oscar_chat_destroy(cc);
890 while (od->create_rooms)
892 struct create_room *cr = od->create_rooms->data;
893 g_free(cr->name);
894 od->create_rooms = g_slist_remove(od->create_rooms, cr);
895 g_free(cr);
897 oscar_data_destroy(od);
898 purple_connection_set_protocol_data(gc, NULL);
900 purple_prefs_disconnect_by_handle(gc);
902 purple_debug_info("oscar", "Signed off.\n");
905 /* XXX - Should use purple_util_fetch_url for the below stuff */
906 struct pieceofcrap {
907 PurpleConnection *gc;
908 unsigned long offset;
909 unsigned long len;
910 char *modname;
911 int fd;
912 FlapConnection *conn;
913 unsigned int inpa;
916 static void damn_you(gpointer data, gint source, PurpleInputCondition c)
918 struct pieceofcrap *pos = data;
919 OscarData *od = purple_connection_get_protocol_data(pos->gc);
920 char in = '\0';
921 int x = 0;
922 unsigned char m[17];
923 GString *msg;
925 while (read(pos->fd, &in, 1) == 1) {
926 if (in == '\n')
927 x++;
928 else if (in != '\r')
929 x = 0;
930 if (x == 2)
931 break;
932 in = '\0';
934 if (in != '\n') {
935 char buf[256];
936 g_snprintf(buf, sizeof(buf), _("You may be disconnected shortly. "
937 "If so, check %s for updates."),
938 oscar_get_ui_info_string("website", PURPLE_WEBSITE));
939 purple_notify_warning(pos->gc, NULL,
940 _("Unable to get a valid AIM login hash."),
941 buf);
942 purple_input_remove(pos->inpa);
943 close(pos->fd);
944 g_free(pos);
945 return;
947 if (read(pos->fd, m, 16) != 16)
949 purple_debug_warning("oscar", "Could not read full AIM login hash "
950 "from " AIMHASHDATA "--that's bad.\n");
952 m[16] = '\0';
954 msg = g_string_new("Sending hash: ");
955 for (x = 0; x < 16; x++)
956 g_string_append_printf(msg, "%02hhx ", (unsigned char)m[x]);
957 g_string_append(msg, "\n");
958 purple_debug_misc("oscar", "%s", msg->str);
959 g_string_free(msg, TRUE);
961 purple_input_remove(pos->inpa);
962 close(pos->fd);
963 aim_sendmemblock(od, pos->conn, 0, 16, m, AIM_SENDMEMBLOCK_FLAG_ISHASH);
964 g_free(pos);
967 static void
968 straight_to_hell(gpointer data, gint source, const gchar *error_message)
970 struct pieceofcrap *pos = data;
971 gchar *buf;
972 gssize result;
974 pos->fd = source;
976 if (source < 0) {
977 buf = g_strdup_printf(_("You may be disconnected shortly. "
978 "If so, check %s for updates."),
979 oscar_get_ui_info_string("website", PURPLE_WEBSITE));
980 purple_notify_warning(pos->gc, NULL,
981 _("Unable to get a valid AIM login hash."),
982 buf);
983 g_free(buf);
984 g_free(pos->modname);
985 g_free(pos);
986 return;
989 buf = g_strdup_printf("GET " AIMHASHDATA "?offset=%ld&len=%ld&modname=%s HTTP/1.0\n\n",
990 pos->offset, pos->len, pos->modname ? pos->modname : "");
991 result = send(pos->fd, buf, strlen(buf), 0);
992 if (result < 0)
993 purple_debug_error("oscar", "Error writing %" G_GSIZE_FORMAT
994 " bytes to fetch AIM hash data: %s\n",
995 strlen(buf), g_strerror(errno));
996 else if ((gsize)result != strlen(buf))
997 purple_debug_error("oscar", "Tried to write %"
998 G_GSIZE_FORMAT " bytes to fetch AIM hash data but "
999 "instead wrote %" G_GSSIZE_FORMAT " bytes\n",
1000 strlen(buf), result);
1001 g_free(buf);
1002 g_free(pos->modname);
1003 pos->inpa = purple_input_add(pos->fd, PURPLE_INPUT_READ, damn_you, pos);
1004 return;
1007 /* size of icbmui.ocm, the largest module in AIM 3.5 */
1008 #define AIM_MAX_FILE_SIZE 98304
1010 static int purple_memrequest(OscarData *od, FlapConnection *conn, FlapFrame *fr, ...)
1012 va_list ap;
1013 struct pieceofcrap *pos;
1014 guint32 offset, len;
1015 char *modname;
1017 va_start(ap, fr);
1018 offset = va_arg(ap, guint32);
1019 len = va_arg(ap, guint32);
1020 modname = va_arg(ap, char *);
1021 va_end(ap);
1023 purple_debug_misc("oscar", "offset: %u, len: %u, file: %s\n",
1024 offset, len, (modname ? modname : "aim.exe"));
1026 if (len == 0) {
1027 purple_debug_misc("oscar", "len is 0, hashing NULL\n");
1028 aim_sendmemblock(od, conn, offset, len, NULL,
1029 AIM_SENDMEMBLOCK_FLAG_ISREQUEST);
1030 return 1;
1033 pos = g_new0(struct pieceofcrap, 1);
1034 pos->gc = od->gc;
1035 pos->conn = conn;
1037 pos->offset = offset;
1038 pos->len = len;
1039 pos->modname = g_strdup(modname);
1041 if (purple_proxy_connect(pos->gc, pos->gc->account, "pidgin.im", 80,
1042 straight_to_hell, pos) == NULL)
1044 char buf[256];
1045 g_free(pos->modname);
1046 g_free(pos);
1048 g_snprintf(buf, sizeof(buf), _("You may be disconnected shortly. "
1049 "If so, check %s for updates."),
1050 oscar_get_ui_info_string("website", PURPLE_WEBSITE));
1051 purple_notify_warning(pos->gc, NULL,
1052 _("Unable to get a valid login hash."),
1053 buf);
1056 return 1;
1059 int oscar_connect_to_bos(PurpleConnection *gc, OscarData *od, const char *host, guint16 port, guint8 *cookie, guint16 cookielen, const char *tls_certname)
1061 PurpleAccount *account;
1062 FlapConnection *conn;
1064 account = purple_connection_get_account(gc);
1066 conn = flap_connection_new(od, SNAC_FAMILY_LOCATE);
1067 conn->cookielen = cookielen;
1068 conn->cookie = g_memdup(cookie, cookielen);
1071 * Use TLS only if the server provided us with a tls_certname. The server might not specify a tls_certname even if we requested to use TLS,
1072 * and that is something we should be prepared to.
1074 if (tls_certname)
1076 conn->gsc = purple_ssl_connect_with_ssl_cn(account, host, port,
1077 ssl_connection_established_cb, ssl_connection_error_cb,
1078 tls_certname, conn);
1080 else
1082 conn->connect_data = purple_proxy_connect(NULL,
1083 account, host, port,
1084 connection_established_cb, conn);
1087 if (conn->gsc == NULL && conn->connect_data == NULL)
1089 purple_connection_error_reason(gc, PURPLE_CONNECTION_ERROR_NETWORK_ERROR, _("Unable to connect"));
1090 return 0;
1093 od->default_port = port;
1095 purple_connection_update_progress(gc, _("Received authorization"), 3, OSCAR_CONNECT_STEPS);
1097 return 1;
1101 * Only used when connecting with the old-style BUCP login.
1103 static int
1104 purple_parse_auth_resp(OscarData *od, FlapConnection *conn, FlapFrame *fr, ...)
1106 PurpleConnection *gc = od->gc;
1107 PurpleAccount *account = purple_connection_get_account(gc);
1108 char *host; int port;
1109 size_t i;
1110 FlapConnection *newconn;
1111 va_list ap;
1112 struct aim_authresp_info *info;
1114 port = purple_account_get_int(account, "port", od->default_port);
1116 va_start(ap, fr);
1117 info = va_arg(ap, struct aim_authresp_info *);
1118 va_end(ap);
1120 purple_debug_info("oscar",
1121 "inside auth_resp (Username: %s)\n", info->bn);
1123 if (info->errorcode || !info->bosip || !info->cookielen || !info->cookie) {
1124 char buf[256];
1125 switch (info->errorcode) {
1126 case 0x01:
1127 /* Unregistered username */
1128 purple_connection_error_reason(gc, PURPLE_CONNECTION_ERROR_INVALID_USERNAME, _("Username does not exist"));
1129 break;
1130 case 0x05:
1131 /* Incorrect password */
1132 if (!purple_account_get_remember_password(account))
1133 purple_account_set_password(account, NULL);
1134 purple_connection_error_reason(gc, PURPLE_CONNECTION_ERROR_AUTHENTICATION_FAILED, _("Incorrect password"));
1135 break;
1136 case 0x11:
1137 /* Suspended account */
1138 purple_connection_error_reason(gc, PURPLE_CONNECTION_ERROR_AUTHENTICATION_FAILED, _("Your account is currently suspended"));
1139 break;
1140 case 0x02:
1141 case 0x14:
1142 /* service temporarily unavailable */
1143 purple_connection_error_reason(gc, PURPLE_CONNECTION_ERROR_NETWORK_ERROR, _("The AOL Instant Messenger service is temporarily unavailable."));
1144 break;
1145 case 0x18:
1146 /* username connecting too frequently */
1147 purple_connection_error_reason(gc, PURPLE_CONNECTION_ERROR_OTHER_ERROR, _("Your username has been connecting and disconnecting too frequently. Wait ten minutes and try again. If you continue to try, you will need to wait even longer."));
1148 break;
1149 case 0x1c:
1151 /* client too old */
1152 g_snprintf(buf, sizeof(buf), _("The client version you are using is too old. Please upgrade at %s"),
1153 oscar_get_ui_info_string("website", PURPLE_WEBSITE));
1154 purple_connection_error_reason(gc, PURPLE_CONNECTION_ERROR_OTHER_ERROR, buf);
1155 break;
1157 case 0x1d:
1158 /* IP address connecting too frequently */
1159 purple_connection_error_reason(gc, PURPLE_CONNECTION_ERROR_OTHER_ERROR, _("Your IP address has been connecting and disconnecting too frequently. Wait a minute and try again. If you continue to try, you will need to wait even longer."));
1160 break;
1161 default:
1162 purple_connection_error_reason(gc, PURPLE_CONNECTION_ERROR_AUTHENTICATION_FAILED, _("Unknown reason"));
1163 break;
1165 purple_debug_info("oscar", "Login Error Code 0x%04hx\n", info->errorcode);
1166 purple_debug_info("oscar", "Error URL: %s\n", info->errorurl ? info->errorurl : "");
1167 return 1;
1170 purple_debug_misc("oscar", "Reg status: %hu\n"
1171 "Email: %s\n"
1172 "BOSIP: %s\n",
1173 info->regstatus,
1174 info->email ? info->email : "null",
1175 info->bosip ? info->bosip : "null");
1176 purple_debug_info("oscar", "Closing auth connection...\n");
1177 flap_connection_schedule_destroy(conn, OSCAR_DISCONNECT_DONE, NULL);
1179 for (i = 0; i < strlen(info->bosip); i++) {
1180 if (info->bosip[i] == ':') {
1181 port = atoi(&(info->bosip[i+1]));
1182 break;
1185 host = g_strndup(info->bosip, i);
1186 newconn = flap_connection_new(od, SNAC_FAMILY_LOCATE);
1187 newconn->cookielen = info->cookielen;
1188 newconn->cookie = g_memdup(info->cookie, info->cookielen);
1190 if (od->use_ssl)
1193 * This shouldn't be hardcoded to "bos.oscar.aol.com" except that
1194 * the server isn't sending us a name to use for comparing the
1195 * certificate common name.
1197 newconn->gsc = purple_ssl_connect_with_ssl_cn(account, host, port,
1198 ssl_connection_established_cb, ssl_connection_error_cb,
1199 "bos.oscar.aol.com", newconn);
1201 else
1203 newconn->connect_data = purple_proxy_connect(NULL, account, host, port,
1204 connection_established_cb, newconn);
1207 g_free(host);
1208 if (newconn->gsc == NULL && newconn->connect_data == NULL)
1210 purple_connection_error_reason(gc, PURPLE_CONNECTION_ERROR_NETWORK_ERROR, _("Unable to connect"));
1211 return 0;
1214 purple_connection_update_progress(gc, _("Received authorization"), 3, OSCAR_CONNECT_STEPS);
1216 return 1;
1220 * Only used when connecting with the old-style BUCP login.
1222 static void
1223 purple_parse_auth_securid_request_yes_cb(gpointer user_data, const char *msg)
1225 PurpleConnection *gc = user_data;
1226 OscarData *od = purple_connection_get_protocol_data(gc);
1228 aim_auth_securid_send(od, msg);
1232 * Only used when connecting with the old-style BUCP login.
1234 static void
1235 purple_parse_auth_securid_request_no_cb(gpointer user_data, const char *value)
1237 PurpleConnection *gc = user_data;
1239 /* Disconnect */
1240 purple_connection_error_reason(gc,
1241 PURPLE_CONNECTION_ERROR_AUTHENTICATION_FAILED,
1242 _("The SecurID key entered is invalid"));
1246 * Only used when connecting with the old-style BUCP login.
1248 static int
1249 purple_parse_auth_securid_request(OscarData *od, FlapConnection *conn, FlapFrame *fr, ...)
1251 PurpleConnection *gc = od->gc;
1252 PurpleAccount *account = purple_connection_get_account(gc);
1253 gchar *primary;
1255 purple_debug_info("oscar", "Got SecurID request\n");
1257 primary = g_strdup_printf("Enter the SecurID key for %s.", purple_account_get_username(account));
1258 purple_request_input(gc, NULL, _("Enter SecurID"), primary,
1259 _("Enter the 6 digit number from the digital display."),
1260 FALSE, FALSE, NULL,
1261 _("_OK"), G_CALLBACK(purple_parse_auth_securid_request_yes_cb),
1262 _("_Cancel"), G_CALLBACK(purple_parse_auth_securid_request_no_cb),
1263 account, NULL, NULL,
1264 gc);
1265 g_free(primary);
1267 return 1;
1271 * Only used when connecting with the old-style BUCP login.
1273 static int
1274 purple_parse_login(OscarData *od, FlapConnection *conn, FlapFrame *fr, ...)
1276 PurpleConnection *gc;
1277 PurpleAccount *account;
1278 ClientInfo aiminfo = CLIENTINFO_PURPLE_AIM;
1279 ClientInfo icqinfo = CLIENTINFO_PURPLE_ICQ;
1280 va_list ap;
1281 char *key;
1282 gboolean truncate_pass;
1284 gc = od->gc;
1285 account = purple_connection_get_account(gc);
1287 va_start(ap, fr);
1288 key = va_arg(ap, char *);
1289 truncate_pass = va_arg(ap, int);
1290 va_end(ap);
1292 aim_send_login(od, conn, purple_account_get_username(account),
1293 purple_connection_get_password(gc), truncate_pass,
1294 od->icq ? &icqinfo : &aiminfo, key,
1295 purple_account_get_bool(account, "allow_multiple_logins", OSCAR_DEFAULT_ALLOW_MULTIPLE_LOGINS));
1297 purple_connection_update_progress(gc, _("Password sent"), 2, OSCAR_CONNECT_STEPS);
1299 return 1;
1302 static int
1303 purple_handle_redirect(OscarData *od, FlapConnection *conn, FlapFrame *fr, ...)
1305 PurpleConnection *gc = od->gc;
1306 PurpleAccount *account = purple_connection_get_account(gc);
1307 char *host, *separator;
1308 int port;
1309 FlapConnection *newconn;
1310 va_list ap;
1311 struct aim_redirect_data *redir;
1313 va_start(ap, fr);
1314 redir = va_arg(ap, struct aim_redirect_data *);
1315 va_end(ap);
1317 port = od->default_port;
1318 separator = strchr(redir->ip, ':');
1319 if (separator != NULL)
1321 host = g_strndup(redir->ip, separator - redir->ip);
1322 port = atoi(separator + 1);
1324 else
1325 host = g_strdup(redir->ip);
1327 if (!redir->use_ssl) {
1328 const gchar *encryption_type = purple_account_get_string(account, "encryption", OSCAR_DEFAULT_ENCRYPTION);
1329 if (purple_strequal(encryption_type, OSCAR_OPPORTUNISTIC_ENCRYPTION)) {
1330 purple_debug_warning("oscar", "We won't use SSL for FLAP type 0x%04hx.\n", redir->group);
1331 } else if (purple_strequal(encryption_type, OSCAR_REQUIRE_ENCRYPTION)) {
1332 purple_debug_error("oscar", "FLAP server %s:%d of type 0x%04hx doesn't support encryption.", host, port, redir->group);
1333 purple_connection_error_reason(
1335 PURPLE_CONNECTION_ERROR_NO_SSL_SUPPORT,
1336 _("You required encryption in your account settings, but one of the servers doesn't support it."));
1337 return 0;
1342 * These FLAP servers advertise SSL (type "0x02"), but SSL connections to these hosts
1343 * die a painful death. iChat and Miranda, when using SSL, still do these in plaintext.
1345 if (redir->use_ssl && (redir->group == SNAC_FAMILY_ADMIN ||
1346 redir->group == SNAC_FAMILY_BART))
1348 purple_debug_info("oscar", "Ignoring broken SSL for FLAP type 0x%04hx.\n", redir->group);
1349 redir->use_ssl = 0;
1352 purple_debug_info("oscar", "Connecting to FLAP server %s:%d of type 0x%04hx\n", host, port, redir->group);
1354 newconn = flap_connection_new(od, redir->group);
1355 newconn->cookielen = redir->cookielen;
1356 newconn->cookie = g_memdup(redir->cookie, redir->cookielen);
1357 if (newconn->type == SNAC_FAMILY_CHAT)
1359 struct chat_connection *cc;
1360 cc = g_new0(struct chat_connection, 1);
1361 cc->conn = newconn;
1362 cc->gc = gc;
1363 cc->name = g_strdup(redir->chat.room);
1364 cc->exchange = redir->chat.exchange;
1365 cc->instance = redir->chat.instance;
1366 cc->show = extract_name(redir->chat.room);
1367 newconn->new_conn_data = cc;
1368 purple_debug_info("oscar", "Connecting to chat room %s exchange %hu\n", cc->name, cc->exchange);
1372 if (redir->use_ssl)
1374 newconn->gsc = purple_ssl_connect_with_ssl_cn(account, host, port,
1375 ssl_connection_established_cb, ssl_connection_error_cb,
1376 redir->ssl_cert_cn, newconn);
1378 else
1380 newconn->connect_data = purple_proxy_connect(NULL, account, host, port,
1381 connection_established_cb, newconn);
1384 if (newconn->gsc == NULL && newconn->connect_data == NULL)
1386 flap_connection_schedule_destroy(newconn,
1387 OSCAR_DISCONNECT_COULD_NOT_CONNECT,
1388 _("Unable to initialize connection"));
1389 purple_debug_error("oscar", "Unable to connect to FLAP server "
1390 "of type 0x%04hx\n", redir->group);
1392 g_free(host);
1394 return 1;
1398 static int purple_parse_oncoming(OscarData *od, FlapConnection *conn, FlapFrame *fr, ...)
1400 PurpleConnection *gc;
1401 PurpleAccount *account;
1402 PurpleBuddy *buddy = NULL;
1403 PurpleStatus *previous_status = NULL;
1404 struct buddyinfo *bi;
1405 time_t time_idle = 0, signon = 0;
1406 int type = 0;
1407 gboolean buddy_is_away = FALSE;
1408 const char *status_id;
1409 va_list ap;
1410 aim_userinfo_t *info;
1411 char *message;
1412 char *itmsurl = NULL;
1414 gc = od->gc;
1415 account = purple_connection_get_account(gc);
1417 va_start(ap, fr);
1418 info = va_arg(ap, aim_userinfo_t *);
1419 va_end(ap);
1421 g_return_val_if_fail(info != NULL, 1);
1422 g_return_val_if_fail(info->bn != NULL, 1);
1424 buddy = purple_find_buddy(account, info->bn);
1425 if (buddy) {
1426 previous_status = purple_presence_get_active_status(purple_buddy_get_presence(buddy));
1430 * If this is an AIM buddy and their name has formatting, set their
1431 * server alias.
1433 if (!oscar_util_valid_name_icq(info->bn)) {
1434 gboolean bn_has_formatting = FALSE;
1435 char *c;
1436 for (c = info->bn; *c != '\0'; c++) {
1437 if (!islower(*c)) {
1438 bn_has_formatting = TRUE;
1439 break;
1442 serv_got_alias(gc, info->bn,
1443 bn_has_formatting ? info->bn : NULL);
1446 if (info->present & AIM_USERINFO_PRESENT_FLAGS) {
1447 if (info->flags & AIM_FLAG_AWAY)
1448 buddy_is_away = TRUE;
1450 if (info->present & AIM_USERINFO_PRESENT_ICQEXTSTATUS) {
1451 type = info->icqinfo.status;
1452 if (!(info->icqinfo.status & AIM_ICQ_STATE_CHAT) &&
1453 (info->icqinfo.status != AIM_ICQ_STATE_NORMAL)) {
1454 buddy_is_away = TRUE;
1458 if (oscar_util_valid_name_icq(info->bn)) {
1459 if (type & AIM_ICQ_STATE_CHAT)
1460 status_id = OSCAR_STATUS_ID_FREE4CHAT;
1461 else if (type & AIM_ICQ_STATE_DND)
1462 status_id = OSCAR_STATUS_ID_DND;
1463 else if (type & AIM_ICQ_STATE_OUT)
1464 status_id = OSCAR_STATUS_ID_NA;
1465 else if (type & AIM_ICQ_STATE_BUSY)
1466 status_id = OSCAR_STATUS_ID_OCCUPIED;
1467 else if (type & AIM_ICQ_STATE_AWAY)
1468 status_id = OSCAR_STATUS_ID_AWAY;
1469 else if (type & AIM_ICQ_STATE_INVISIBLE)
1470 status_id = OSCAR_STATUS_ID_INVISIBLE;
1471 else if (type & AIM_ICQ_STATE_EVIL)
1472 status_id = OSCAR_STATUS_ID_EVIL;
1473 else if (type & AIM_ICQ_STATE_DEPRESSION)
1474 status_id = OSCAR_STATUS_ID_DEPRESSION;
1475 else if (type & AIM_ICQ_STATE_ATHOME)
1476 status_id = OSCAR_STATUS_ID_ATHOME;
1477 else if (type & AIM_ICQ_STATE_ATWORK)
1478 status_id = OSCAR_STATUS_ID_ATWORK;
1479 else if (type & AIM_ICQ_STATE_LUNCH)
1480 status_id = OSCAR_STATUS_ID_LUNCH;
1481 else
1482 status_id = OSCAR_STATUS_ID_AVAILABLE;
1483 } else {
1484 if (type & AIM_ICQ_STATE_INVISIBLE)
1485 status_id = OSCAR_STATUS_ID_INVISIBLE;
1486 else if (buddy_is_away)
1487 status_id = OSCAR_STATUS_ID_AWAY;
1488 else
1489 status_id = OSCAR_STATUS_ID_AVAILABLE;
1492 if (info->flags & AIM_FLAG_WIRELESS) {
1493 purple_prpl_got_user_status(account, info->bn, OSCAR_STATUS_ID_MOBILE, NULL);
1494 } else {
1495 purple_prpl_got_user_status_deactive(account, info->bn, OSCAR_STATUS_ID_MOBILE);
1498 message = (info->status && info->status_len > 0)
1499 ? oscar_encoding_to_utf8(info->status_encoding, info->status, info->status_len)
1500 : NULL;
1502 if (purple_strequal(status_id, OSCAR_STATUS_ID_AVAILABLE)) {
1503 /* TODO: If itmsurl is NULL, does that mean the URL has been
1504 cleared? Or does it mean the URL should remain unchanged? */
1505 if (info->itmsurl != NULL) {
1506 itmsurl = (info->itmsurl_len > 0) ? oscar_encoding_to_utf8(info->itmsurl_encoding, info->itmsurl, info->itmsurl_len) : NULL;
1507 } else if (previous_status != NULL && purple_status_is_available(previous_status)) {
1508 itmsurl = g_strdup(purple_status_get_attr_string(previous_status, "itmsurl"));
1510 purple_debug_info("oscar", "Activating status '%s' for buddy %s, message = '%s', itmsurl = '%s'\n", status_id, info->bn, message ? message : "(null)", itmsurl ? itmsurl : "(null)");
1511 purple_prpl_got_user_status(account, info->bn, status_id, "message", message, "itmsurl", itmsurl, NULL);
1512 } else {
1513 purple_debug_info("oscar", "Activating status '%s' for buddy %s, message = '%s'\n", status_id, info->bn, message ? message : "(null)");
1514 purple_prpl_got_user_status(account, info->bn, status_id, "message", message, NULL);
1517 g_free(message);
1518 g_free(itmsurl);
1520 /* Login time stuff */
1521 if (info->present & AIM_USERINFO_PRESENT_ONLINESINCE)
1522 signon = info->onlinesince;
1523 else if (info->present & AIM_USERINFO_PRESENT_SESSIONLEN)
1524 signon = time(NULL) - info->sessionlen;
1525 purple_prpl_got_user_login_time(account, info->bn, signon);
1527 /* Idle time stuff */
1528 /* info->idletime is the number of minutes that this user has been idle */
1529 if (info->present & AIM_USERINFO_PRESENT_IDLE)
1530 time_idle = time(NULL) - info->idletime * 60;
1532 if (time_idle > 0)
1533 purple_prpl_got_user_idle(account, info->bn, TRUE, time_idle);
1534 else
1535 purple_prpl_got_user_idle(account, info->bn, FALSE, 0);
1537 /* Server stored icon stuff */
1538 bi = g_hash_table_lookup(od->buddyinfo, purple_normalize(account, info->bn));
1539 if (!bi) {
1540 bi = g_new0(struct buddyinfo, 1);
1541 g_hash_table_insert(od->buddyinfo, g_strdup(purple_normalize(account, info->bn)), bi);
1543 bi->typingnot = FALSE;
1544 bi->ico_informed = FALSE;
1545 bi->ipaddr = info->icqinfo.ipaddr;
1547 if (info->iconcsumlen) {
1548 const char *saved_b16 = NULL;
1549 char *b16 = NULL;
1550 PurpleBuddy *b = NULL;
1552 b16 = purple_base16_encode(info->iconcsum, info->iconcsumlen);
1553 b = purple_find_buddy(account, info->bn);
1554 if (b != NULL)
1555 saved_b16 = purple_buddy_icons_get_checksum_for_user(b);
1557 if (!b16 || !saved_b16 || !purple_strequal(b16, saved_b16)) {
1558 /* Invalidate the old icon for this user */
1559 purple_buddy_icons_set_for_user(account, info->bn, NULL, 0, NULL);
1561 /* Fetch the new icon (if we're not already doing so) */
1562 if (g_slist_find_custom(od->requesticon, info->bn,
1563 (GCompareFunc)oscar_util_name_compare) == NULL)
1565 od->requesticon = g_slist_prepend(od->requesticon,
1566 g_strdup(purple_normalize(account, info->bn)));
1567 purple_icons_fetch(gc);
1570 g_free(b16);
1573 return 1;
1576 static int purple_parse_offgoing(OscarData *od, FlapConnection *conn, FlapFrame *fr, ...) {
1577 PurpleConnection *gc = od->gc;
1578 PurpleAccount *account = purple_connection_get_account(gc);
1579 va_list ap;
1580 aim_userinfo_t *info;
1582 va_start(ap, fr);
1583 info = va_arg(ap, aim_userinfo_t *);
1584 va_end(ap);
1586 purple_prpl_got_user_status(account, info->bn, OSCAR_STATUS_ID_OFFLINE, NULL);
1587 purple_prpl_got_user_status_deactive(account, info->bn, OSCAR_STATUS_ID_MOBILE);
1588 g_hash_table_remove(od->buddyinfo, purple_normalize(gc->account, info->bn));
1590 return 1;
1593 static int incomingim_chan1(OscarData *od, FlapConnection *conn, aim_userinfo_t *userinfo, struct aim_incomingim_ch1_args *args) {
1594 PurpleConnection *gc = od->gc;
1595 PurpleAccount *account = purple_connection_get_account(gc);
1596 PurpleMessageFlags flags = 0;
1597 struct buddyinfo *bi;
1598 PurpleStoredImage *img;
1599 gchar *tmp;
1600 const char *start, *end;
1601 GData *attribs;
1603 purple_debug_misc("oscar", "Received IM from %s\n", userinfo->bn);
1605 bi = g_hash_table_lookup(od->buddyinfo, purple_normalize(account, userinfo->bn));
1606 if (!bi) {
1607 bi = g_new0(struct buddyinfo, 1);
1608 g_hash_table_insert(od->buddyinfo, g_strdup(purple_normalize(account, userinfo->bn)), bi);
1611 if (args->icbmflags & AIM_IMFLAGS_AWAY)
1612 flags |= PURPLE_MESSAGE_AUTO_RESP;
1614 if (args->icbmflags & AIM_IMFLAGS_TYPINGNOT)
1615 bi->typingnot = TRUE;
1616 else
1617 bi->typingnot = FALSE;
1619 if ((args->icbmflags & AIM_IMFLAGS_HASICON) && (args->iconlen) && (args->iconsum) && (args->iconstamp)) {
1620 purple_debug_misc("oscar", "%s has an icon\n", userinfo->bn);
1621 if ((args->iconlen != bi->ico_len) || (args->iconsum != bi->ico_csum) || (args->iconstamp != bi->ico_time)) {
1622 bi->ico_need = TRUE;
1623 bi->ico_len = args->iconlen;
1624 bi->ico_csum = args->iconsum;
1625 bi->ico_time = args->iconstamp;
1629 img = purple_buddy_icons_find_account_icon(account);
1630 if ((img != NULL) &&
1631 (args->icbmflags & AIM_IMFLAGS_BUDDYREQ) && !bi->ico_sent && bi->ico_informed) {
1632 gconstpointer data = purple_imgstore_get_data(img);
1633 size_t len = purple_imgstore_get_size(img);
1634 purple_debug_info("oscar",
1635 "Sending buddy icon to %s (%" G_GSIZE_FORMAT " bytes)\n",
1636 userinfo->bn, len);
1637 aim_im_sendch2_icon(od, userinfo->bn, data, len,
1638 purple_buddy_icons_get_account_icon_timestamp(account),
1639 aimutil_iconsum(data, len));
1641 purple_imgstore_unref(img);
1643 tmp = g_strdup(args->msg);
1646 * Convert iChat color tags to normal font tags.
1648 if (purple_markup_find_tag("body", tmp, &start, &end, &attribs))
1650 int len;
1651 char *tmp2, *body;
1652 const char *ichattextcolor, *ichatballooncolor;
1653 const char *slash_body_start, *slash_body_end = NULL; /* </body> */
1654 GData *unused;
1657 * Find the ending </body> so we can strip off the outer <html/>
1658 * and <body/>
1660 if (purple_markup_find_tag("/body", end + 1, &slash_body_start, &slash_body_end, &unused))
1662 body = g_strndup(start, slash_body_end - start + 1);
1663 g_datalist_clear(&unused);
1665 else
1667 purple_debug_warning("oscar", "Broken message contains <body> but not </body>!\n");
1668 /* Take everything after <body> */
1669 body = g_strdup(start);
1672 ichattextcolor = g_datalist_get_data(&attribs, "ichattextcolor");
1673 if (ichattextcolor != NULL)
1675 tmp2 = g_strdup_printf("<font color=\"%s\">%s</font>", ichattextcolor, body);
1676 g_free(body);
1677 body = tmp2;
1680 ichatballooncolor = g_datalist_get_data(&attribs, "ichatballooncolor");
1681 if (ichatballooncolor != NULL)
1683 tmp2 = g_strdup_printf("<font back=\"%s\">%s</font>", ichatballooncolor, body);
1684 g_free(body);
1685 body = tmp2;
1688 g_datalist_clear(&attribs);
1690 len = start - tmp;
1691 tmp2 = g_strdup_printf("%.*s%s%s", len, tmp, body, slash_body_end ? slash_body_end + 1: "</body>");
1692 g_free(tmp);
1693 g_free(body);
1695 tmp = tmp2;
1699 * Are there <html/> surrounding tags? If so, strip them out, too.
1701 if (purple_markup_find_tag("html", tmp, &start, &end, &attribs))
1703 gchar *tmp2;
1704 int len;
1706 g_datalist_clear(&attribs);
1708 len = start - tmp;
1709 tmp2 = g_strdup_printf("%.*s%s", len, tmp, end + 1);
1710 g_free(tmp);
1711 tmp = tmp2;
1714 if (purple_markup_find_tag("/html", tmp, &start, &end, &attribs))
1716 gchar *tmp2;
1717 int len;
1719 g_datalist_clear(&attribs);
1721 len = start - tmp;
1722 tmp2 = g_strdup_printf("%.*s%s", len, tmp, end + 1);
1723 g_free(tmp);
1724 tmp = tmp2;
1727 serv_got_im(gc, userinfo->bn, tmp, flags, (args->icbmflags & AIM_IMFLAGS_OFFLINE) ? args->timestamp : time(NULL));
1728 g_free(tmp);
1730 return 1;
1733 static int
1734 incomingim_chan2(OscarData *od, FlapConnection *conn, aim_userinfo_t *userinfo, IcbmArgsCh2 *args)
1736 PurpleConnection *gc;
1737 PurpleAccount *account;
1738 PurpleMessageFlags flags = 0;
1739 char *message = NULL;
1741 g_return_val_if_fail(od != NULL, 0);
1742 g_return_val_if_fail(od->gc != NULL, 0);
1744 gc = od->gc;
1745 account = purple_connection_get_account(gc);
1746 od = purple_connection_get_protocol_data(gc);
1748 if (args == NULL)
1749 return 0;
1751 purple_debug_misc("oscar", "Incoming rendezvous message of type %"
1752 G_GUINT64_FORMAT ", user %s, status %hu\n",
1753 args->type, userinfo->bn, args->status);
1755 if (args->msg != NULL) {
1756 message = oscar_encoding_to_utf8(args->encoding, args->msg, args->msglen);
1759 if (args->type & OSCAR_CAPABILITY_CHAT)
1761 char *utf8name, *tmp;
1762 GHashTable *components;
1764 if (!args->info.chat.roominfo.name || !args->info.chat.roominfo.exchange) {
1765 g_free(message);
1766 return 1;
1768 utf8name = oscar_encoding_to_utf8(args->encoding, args->info.chat.roominfo.name, args->info.chat.roominfo.namelen);
1770 tmp = extract_name(utf8name);
1771 if (tmp != NULL)
1773 g_free(utf8name);
1774 utf8name = tmp;
1777 components = g_hash_table_new_full(g_str_hash, g_str_equal, g_free,
1778 g_free);
1779 g_hash_table_replace(components, g_strdup("room"), utf8name);
1780 g_hash_table_replace(components, g_strdup("exchange"),
1781 g_strdup_printf("%d", args->info.chat.roominfo.exchange));
1782 serv_got_chat_invite(gc,
1783 utf8name,
1784 userinfo->bn,
1785 message,
1786 components);
1789 else if ((args->type & OSCAR_CAPABILITY_SENDFILE) || (args->type & OSCAR_CAPABILITY_DIRECTIM))
1791 if (args->status == AIM_RENDEZVOUS_PROPOSE)
1793 peer_connection_got_proposition(od, userinfo->bn, message, args);
1795 else if (args->status == AIM_RENDEZVOUS_CANCEL)
1797 /* The other user cancelled a peer request */
1798 PeerConnection *conn;
1800 conn = peer_connection_find_by_cookie(od, userinfo->bn, args->cookie);
1802 * If conn is NULL it means we haven't tried to create
1803 * a connection with that user. They may be trying to
1804 * do something malicious.
1806 if (conn != NULL)
1808 peer_connection_destroy(conn, OSCAR_DISCONNECT_REMOTE_CLOSED, NULL);
1811 else if (args->status == AIM_RENDEZVOUS_CONNECTED)
1814 * Remote user has accepted our peer request. If we
1815 * wanted to we could look up the PeerConnection using
1816 * args->cookie, but we don't need to do anything here.
1821 else if (args->type & OSCAR_CAPABILITY_GETFILE)
1825 else if (args->type & OSCAR_CAPABILITY_TALK)
1829 else if (args->type & OSCAR_CAPABILITY_BUDDYICON)
1831 purple_buddy_icons_set_for_user(account, userinfo->bn,
1832 g_memdup(args->info.icon.icon, args->info.icon.length),
1833 args->info.icon.length,
1834 NULL);
1837 else if (args->type & OSCAR_CAPABILITY_ICQSERVERRELAY)
1839 purple_debug_info("oscar", "Got an ICQ Server Relay message of "
1840 "type %d\n", args->info.rtfmsg.msgtype);
1842 if (args->info.rtfmsg.msgtype == 1) {
1843 if (args->info.rtfmsg.msg != NULL) {
1844 char *rtfmsg;
1845 const char *encoding = args->encoding;
1846 size_t len = strlen(args->info.rtfmsg.msg);
1847 char *tmp, *tmp2;
1849 if (encoding == NULL && !g_utf8_validate(args->info.rtfmsg.msg, len, NULL)) {
1850 /* Yet another wonderful Miranda-related hack. If their user disables the "Send Unicode messages" setting,
1851 * Miranda sends us ch2 messages in whatever Windows codepage is set as default on their user's system (instead of UTF-8).
1852 * Of course, they don't bother to specify that codepage. Let's just fallback to the encoding OUR users can
1853 * specify in account options as a last resort.
1855 encoding = purple_account_get_string(account, "encoding", OSCAR_DEFAULT_CUSTOM_ENCODING);
1856 purple_debug_info("oscar", "Miranda, is that you? Using '%s' as encoding\n", encoding);
1859 rtfmsg = oscar_encoding_to_utf8(encoding, args->info.rtfmsg.msg, len);
1861 /* Channel 2 messages are supposed to be plain-text (never mind the name "rtfmsg", even
1862 * the official client doesn't parse them as RTF). Therefore, we should escape them before
1863 * showing to the user. */
1864 tmp = g_markup_escape_text(rtfmsg, -1);
1865 g_free(rtfmsg);
1866 tmp2 = purple_strreplace(tmp, "\r\n", "<br>");
1867 g_free(tmp);
1869 serv_got_im(gc, userinfo->bn, tmp2, flags, time(NULL));
1870 aim_im_send_icq_confirmation(od, userinfo->bn, args->cookie);
1871 g_free(tmp2);
1873 } else if (args->info.rtfmsg.msgtype == 26) {
1874 purple_debug_info("oscar", "Sending X-Status Reply\n");
1875 icq_relay_xstatus(od, userinfo->bn, args->cookie);
1878 else
1880 purple_debug_error("oscar", "Unknown request class %"
1881 G_GUINT64_FORMAT "\n", args->type);
1884 g_free(message);
1886 return 1;
1889 /* When someone sends you buddies */
1890 static void
1891 purple_icq_buddyadd(struct name_data *data)
1893 PurpleConnection *gc = data->gc;
1895 purple_blist_request_add_buddy(purple_connection_get_account(gc), data->name, NULL, data->nick);
1897 oscar_free_name_data(data);
1900 static int
1901 incomingim_chan4(OscarData *od, FlapConnection *conn, aim_userinfo_t *userinfo, struct aim_incomingim_ch4_args *args, time_t t)
1903 PurpleConnection *gc = od->gc;
1904 PurpleAccount *account = purple_connection_get_account(gc);
1905 gchar **msg1, **msg2;
1906 int i, numtoks;
1908 if (!args->type || !args->msg || !args->uin)
1909 return 1;
1911 purple_debug_info("oscar",
1912 "Received a channel 4 message of type 0x%02hx.\n",
1913 (guint16)args->type);
1916 * Split up the message at the delimeter character, then convert each
1917 * string to UTF-8. Unless, of course, this is a type 1 message. If
1918 * this is a type 1 message, then the delimiter 0xfe could be a valid
1919 * character in whatever encoding the message was sent in. Type 1
1920 * messages are always made up of only one part, so we can easily account
1921 * for this suck-ass part of the protocol by splitting the string into at
1922 * most 1 baby string.
1924 msg1 = g_strsplit(args->msg, "\376", (args->type == 0x01 ? 1 : 0));
1925 for (numtoks=0; msg1[numtoks]; numtoks++);
1926 msg2 = (gchar **)g_malloc((numtoks+1)*sizeof(gchar *));
1927 for (i=0; msg1[i]; i++) {
1928 gchar *uin = g_strdup_printf("%u", args->uin);
1930 purple_str_strip_char(msg1[i], '\r');
1931 /* TODO: Should use an encoding other than ASCII? */
1932 msg2[i] = oscar_decode_im(account, uin, AIM_CHARSET_ASCII, msg1[i], strlen(msg1[i]));
1933 g_free(uin);
1935 msg2[i] = NULL;
1937 switch (args->type) {
1938 case 0x01: { /* MacICQ message or basic offline message */
1939 if (i >= 1) {
1940 gchar *uin = g_strdup_printf("%u", args->uin);
1941 gchar *tmp;
1943 /* If the message came from an ICQ user then escape any HTML */
1944 tmp = g_markup_escape_text(msg2[0], -1);
1946 if (t) { /* This is an offline message */
1947 /* The timestamp is UTC-ish, so we need to get the offset */
1948 #ifdef HAVE_TM_GMTOFF
1949 time_t now;
1950 struct tm *tm;
1951 now = time(NULL);
1952 tm = localtime(&now);
1953 t += tm->tm_gmtoff;
1954 #else
1955 # ifdef HAVE_TIMEZONE
1956 tzset();
1957 t -= timezone;
1958 # endif
1959 #endif
1960 serv_got_im(gc, uin, tmp, 0, t);
1961 } else { /* This is a message from MacICQ/Miranda */
1962 serv_got_im(gc, uin, tmp, 0, time(NULL));
1964 g_free(uin);
1965 g_free(tmp);
1967 } break;
1969 case 0x04: { /* Someone sent you a URL */
1970 if (i >= 2) {
1971 if (msg2[1] != NULL) {
1972 gchar *uin = g_strdup_printf("%u", args->uin);
1973 gchar *message = g_strdup_printf("<A HREF=\"%s\">%s</A>",
1974 msg2[1],
1975 (msg2[0] && msg2[0][0]) ? msg2[0] : msg2[1]);
1976 serv_got_im(gc, uin, message, 0, time(NULL));
1977 g_free(uin);
1978 g_free(message);
1981 } break;
1983 case 0x06: { /* Someone requested authorization */
1984 if (i >= 6) {
1985 gchar *bn = g_strdup_printf("%u", args->uin);
1986 gchar *reason = NULL;
1988 if (msg2[5] != NULL)
1989 reason = oscar_decode_im(account, bn, AIM_CHARSET_LATIN_1, msg2[5], strlen(msg2[5]));
1991 purple_debug_info("oscar",
1992 "Received an authorization request from UIN %u\n",
1993 args->uin);
1994 aim_icq_getalias(od, bn, TRUE, reason);
1995 g_free(bn);
1996 g_free(reason);
1998 } break;
2000 case 0x07: { /* Someone has denied you authorization */
2001 if (i >= 1) {
2002 gchar *dialog_msg = g_strdup_printf(_("The user %u has denied your request to add them to your buddy list for the following reason:\n%s"), args->uin, msg2[0] ? msg2[0] : _("No reason given."));
2003 purple_notify_info(gc, NULL, _("ICQ authorization denied."),
2004 dialog_msg);
2005 g_free(dialog_msg);
2007 } break;
2009 case 0x08: { /* Someone has granted you authorization */
2010 gchar *dialog_msg = g_strdup_printf(_("The user %u has granted your request to add them to your buddy list."), args->uin);
2011 purple_notify_info(gc, NULL, "ICQ authorization accepted.",
2012 dialog_msg);
2013 g_free(dialog_msg);
2014 } break;
2016 case 0x09: { /* Message from the Godly ICQ server itself, I think */
2017 if (i >= 5) {
2018 gchar *dialog_msg = g_strdup_printf(_("You have received a special message\n\nFrom: %s [%s]\n%s"), msg2[0], msg2[3], msg2[5]);
2019 purple_notify_info(gc, NULL, "ICQ Server Message", dialog_msg);
2020 g_free(dialog_msg);
2022 } break;
2024 case 0x0d: { /* Someone has sent you a pager message from http://www.icq.com/your_uin */
2025 if (i >= 6) {
2026 gchar *dialog_msg = g_strdup_printf(_("You have received an ICQ page\n\nFrom: %s [%s]\n%s"), msg2[0], msg2[3], msg2[5]);
2027 purple_notify_info(gc, NULL, "ICQ Page", dialog_msg);
2028 g_free(dialog_msg);
2030 } break;
2032 case 0x0e: { /* Someone has emailed you at your_uin@pager.icq.com */
2033 if (i >= 6) {
2034 gchar *dialog_msg = g_strdup_printf(_("You have received an ICQ email from %s [%s]\n\nMessage is:\n%s"), msg2[0], msg2[3], msg2[5]);
2035 purple_notify_info(gc, NULL, "ICQ Email", dialog_msg);
2036 g_free(dialog_msg);
2038 } break;
2040 case 0x12: {
2041 /* Ack for authorizing/denying someone. Or possibly an ack for sending any system notice */
2042 /* Someone added you to their buddy list? */
2043 } break;
2045 case 0x13: { /* Someone has sent you some ICQ buddies */
2046 guint i, num;
2047 gchar **text;
2048 text = g_strsplit(args->msg, "\376", 0);
2049 if (text) {
2050 /* Read the number of contacts that we were sent */
2051 errno = 0;
2052 num = text[0] ? strtoul(text[0], NULL, 10) : 0;
2054 if (num > 0 && errno == 0) {
2055 for (i=0; i<num; i++) {
2056 struct name_data *data;
2057 gchar *message;
2059 if (!text[i*2 + 1] || !text[i*2 + 2]) {
2060 /* We're missing the contact name or nickname. Bail out. */
2061 gchar *tmp = g_strescape(args->msg, NULL);
2062 purple_debug_error("oscar", "Unknown syntax parsing "
2063 "ICQ buddies. args->msg=%s\n", tmp);
2064 g_free(tmp);
2065 break;
2068 message = g_strdup_printf(_("ICQ user %u has sent you a buddy: %s (%s)"), args->uin, text[i*2+2], text[i*2+1]);
2070 data = g_new(struct name_data, 1);
2071 data->gc = gc;
2072 data->name = g_strdup(text[i*2+1]);
2073 data->nick = g_strdup(text[i*2+2]);
2075 purple_request_action(gc, NULL, message,
2076 _("Do you want to add this buddy "
2077 "to your buddy list?"),
2078 PURPLE_DEFAULT_ACTION_NONE,
2079 purple_connection_get_account(gc), data->name, NULL,
2080 data, 2,
2081 _("_Add"), G_CALLBACK(purple_icq_buddyadd),
2082 _("_Decline"), G_CALLBACK(oscar_free_name_data));
2083 g_free(message);
2085 } else {
2086 gchar *tmp = g_strescape(args->msg, NULL);
2087 purple_debug_error("oscar", "Unknown syntax parsing "
2088 "ICQ buddies. args->msg=%s\n", tmp);
2089 g_free(tmp);
2091 g_strfreev(text);
2093 } break;
2095 case 0x1a: { /* Handle SMS or someone has sent you a greeting card or requested buddies? */
2096 ByteStream qbs;
2097 guint16 smstype;
2098 guint32 taglen, smslen;
2099 char *tagstr = NULL, *smsmsg = NULL;
2100 xmlnode *xmlroot = NULL, *xmltmp = NULL;
2101 gchar *uin = NULL, *message = NULL;
2103 /* From libicq2000-0.3.2/src/ICQ.cpp */
2104 byte_stream_init(&qbs, (guint8 *)args->msg, args->msglen);
2105 byte_stream_advance(&qbs, 21);
2106 /* expected: 01 00 00 20 00 0e 28 f6 00 11 e7 d3 11 bc f3 00 04 ac 96 9d c2 | 00 00 | 06 00 00 00 | 49 43 51 53 43 53 ...*/
2107 /* unexpected: 00 00 26 00 81 1a 18 bc 0e 6c 18 47 a5 91 6f 18 dc c7 6f 1a | 00 00 | 0d 00 00 00 | 49 43 51 57 65 62 4d 65 73 73 61 67 65 ... */
2108 smstype = byte_stream_getle16(&qbs);
2109 if (smstype != 0)
2110 break;
2111 taglen = byte_stream_getle32(&qbs);
2112 if (taglen > 2000) {
2113 /* Avoid trying to allocate large amounts of memory, in
2114 case we get something unexpected. */
2115 break;
2117 tagstr = byte_stream_getstr(&qbs, taglen);
2118 if (tagstr == NULL)
2119 break;
2120 byte_stream_advance(&qbs, 3);
2121 byte_stream_advance(&qbs, 4);
2122 smslen = byte_stream_getle32(&qbs);
2123 if (smslen > 2000) {
2124 /* Avoid trying to allocate large amounts of memory, in
2125 case we get something unexpected. */
2126 g_free(tagstr);
2127 break;
2129 smsmsg = byte_stream_getstr(&qbs, smslen);
2131 /* Check if this is an SMS being sent from server */
2132 if ((smstype == 0) && (purple_strequal(tagstr, "ICQSMS")) && (smsmsg != NULL))
2134 xmlroot = xmlnode_from_str(smsmsg, -1);
2135 if (xmlroot != NULL)
2137 xmltmp = xmlnode_get_child(xmlroot, "sender");
2138 if (xmltmp != NULL)
2139 uin = xmlnode_get_data(xmltmp);
2141 xmltmp = xmlnode_get_child(xmlroot, "text");
2142 if (xmltmp != NULL)
2143 message = xmlnode_get_data(xmltmp);
2145 if ((uin != NULL) && (message != NULL))
2146 serv_got_im(gc, uin, message, 0, time(NULL));
2148 g_free(uin);
2149 g_free(message);
2150 xmlnode_free(xmlroot);
2153 g_free(tagstr);
2154 g_free(smsmsg);
2155 } break;
2157 default: {
2158 purple_debug_info("oscar",
2159 "Received a channel 4 message of unknown type "
2160 "(type 0x%02hhx).\n", args->type);
2161 } break;
2164 g_strfreev(msg1);
2165 g_strfreev(msg2);
2167 return 1;
2170 static int purple_parse_incoming_im(OscarData *od, FlapConnection *conn, FlapFrame *fr, ...) {
2171 guint16 channel;
2172 int ret = 0;
2173 aim_userinfo_t *userinfo;
2174 va_list ap;
2176 va_start(ap, fr);
2177 channel = (guint16)va_arg(ap, unsigned int);
2178 userinfo = va_arg(ap, aim_userinfo_t *);
2180 switch (channel) {
2181 case 1: { /* standard message */
2182 struct aim_incomingim_ch1_args *args;
2183 args = va_arg(ap, struct aim_incomingim_ch1_args *);
2184 ret = incomingim_chan1(od, conn, userinfo, args);
2185 } break;
2187 case 2: { /* rendezvous */
2188 IcbmArgsCh2 *args;
2189 args = va_arg(ap, IcbmArgsCh2 *);
2190 ret = incomingim_chan2(od, conn, userinfo, args);
2191 } break;
2193 case 4: { /* ICQ */
2194 struct aim_incomingim_ch4_args *args;
2195 args = va_arg(ap, struct aim_incomingim_ch4_args *);
2196 ret = incomingim_chan4(od, conn, userinfo, args, 0);
2197 } break;
2199 default: {
2200 purple_debug_warning("oscar",
2201 "ICBM received on unsupported channel (channel "
2202 "0x%04hx).", channel);
2203 } break;
2206 va_end(ap);
2208 return ret;
2211 static int purple_parse_misses(OscarData *od, FlapConnection *conn, FlapFrame *fr, ...) {
2212 PurpleConnection *gc = od->gc;
2213 PurpleAccount *account = purple_connection_get_account(gc);
2214 char *buf;
2215 va_list ap;
2216 guint16 nummissed, reason;
2217 aim_userinfo_t *userinfo;
2219 va_start(ap, fr);
2220 va_arg(ap, unsigned int); /* guint16 chan */
2221 userinfo = va_arg(ap, aim_userinfo_t *);
2222 nummissed = (guint16)va_arg(ap, unsigned int);
2223 reason = (guint16)va_arg(ap, unsigned int);
2224 va_end(ap);
2226 switch(reason) {
2227 case 0: /* Invalid (0) */
2228 buf = g_strdup_printf(
2229 dngettext(PACKAGE,
2230 "You missed %hu message from %s because it was invalid.",
2231 "You missed %hu messages from %s because they were invalid.",
2232 nummissed),
2233 nummissed,
2234 userinfo->bn);
2235 break;
2236 case 1: /* Message too large */
2237 buf = g_strdup_printf(
2238 dngettext(PACKAGE,
2239 "You missed %hu message from %s because it was too large.",
2240 "You missed %hu messages from %s because they were too large.",
2241 nummissed),
2242 nummissed,
2243 userinfo->bn);
2244 break;
2245 case 2: /* Rate exceeded */
2246 buf = g_strdup_printf(
2247 dngettext(PACKAGE,
2248 "You missed %hu message from %s because the rate limit has been exceeded.",
2249 "You missed %hu messages from %s because the rate limit has been exceeded.",
2250 nummissed),
2251 nummissed,
2252 userinfo->bn);
2253 break;
2254 case 3: /* Evil Sender */
2255 buf = g_strdup_printf(
2256 dngettext(PACKAGE,
2257 "You missed %hu message from %s because his/her warning level is too high.",
2258 "You missed %hu messages from %s because his/her warning level is too high.",
2259 nummissed),
2260 nummissed,
2261 userinfo->bn);
2262 break;
2263 case 4: /* Evil Receiver */
2264 buf = g_strdup_printf(
2265 dngettext(PACKAGE,
2266 "You missed %hu message from %s because your warning level is too high.",
2267 "You missed %hu messages from %s because your warning level is too high.",
2268 nummissed),
2269 nummissed,
2270 userinfo->bn);
2271 break;
2272 default:
2273 buf = g_strdup_printf(
2274 dngettext(PACKAGE,
2275 "You missed %hu message from %s for an unknown reason.",
2276 "You missed %hu messages from %s for an unknown reason.",
2277 nummissed),
2278 nummissed,
2279 userinfo->bn);
2280 break;
2283 if (!purple_conv_present_error(userinfo->bn, account, buf))
2284 purple_notify_error(od->gc, NULL, buf, NULL);
2285 g_free(buf);
2287 return 1;
2290 static int
2291 purple_parse_clientauto_ch2(OscarData *od, const char *who, guint16 reason, const guchar *cookie)
2293 if (reason == 0x0003)
2295 /* Rendezvous was refused. */
2296 PeerConnection *conn;
2298 conn = peer_connection_find_by_cookie(od, who, cookie);
2300 if (conn == NULL)
2302 purple_debug_info("oscar", "Received a rendezvous cancel message "
2303 "for a nonexistant connection from %s.\n", who);
2305 else
2307 peer_connection_destroy(conn, OSCAR_DISCONNECT_REMOTE_REFUSED, NULL);
2310 else
2312 purple_debug_warning("oscar", "Received an unknown rendezvous "
2313 "message from %s. Type 0x%04hx\n", who, reason);
2316 return 0;
2319 static int purple_parse_clientauto_ch4(OscarData *od, char *who, guint16 reason, guint32 state, char *msg) {
2320 PurpleConnection *gc = od->gc;
2322 switch(reason) {
2323 case 0x0003: { /* Reply from an ICQ status message request */
2324 char *statusmsg, **splitmsg;
2325 PurpleNotifyUserInfo *user_info;
2327 /* Split at (carriage return/newline)'s, then rejoin later with BRs between. */
2328 statusmsg = oscar_icqstatus(state);
2329 splitmsg = g_strsplit(msg, "\r\n", 0);
2331 user_info = purple_notify_user_info_new();
2333 purple_notify_user_info_add_pair(user_info, _("UIN"), who);
2334 purple_notify_user_info_add_pair(user_info, _("Status"), statusmsg);
2335 purple_notify_user_info_add_section_break(user_info);
2336 purple_notify_user_info_add_pair(user_info, NULL, g_strjoinv("<BR>", splitmsg));
2338 g_free(statusmsg);
2339 g_strfreev(splitmsg);
2341 purple_notify_userinfo(gc, who, user_info, NULL, NULL);
2342 purple_notify_user_info_destroy(user_info);
2344 } break;
2346 case 0x0006: { /* Reply from an ICQ status message request */
2347 char *statusmsg, **splitmsg;
2348 PurpleNotifyUserInfo *user_info;
2350 /* Split at (carriage return/newline)'s, then rejoin later with BRs between. */
2351 statusmsg = oscar_icqstatus(state);
2352 splitmsg = g_strsplit(msg, "\r\n", 0);
2354 user_info = purple_notify_user_info_new();
2356 purple_notify_user_info_add_pair(user_info, _("UIN"), who);
2357 purple_notify_user_info_add_pair(user_info, _("Status"), statusmsg);
2358 purple_notify_user_info_add_section_break(user_info);
2359 purple_notify_user_info_add_pair(user_info, NULL, g_strjoinv("<BR>", splitmsg));
2361 g_free(statusmsg);
2362 g_strfreev(splitmsg);
2364 purple_notify_userinfo(gc, who, user_info, NULL, NULL);
2365 purple_notify_user_info_destroy(user_info);
2367 } break;
2369 default: {
2370 purple_debug_warning("oscar",
2371 "Received an unknown client auto-response from %s. "
2372 "Type 0x%04hx\n", who, reason);
2373 } break;
2374 } /* end of switch */
2376 return 0;
2379 static int purple_parse_clientauto(OscarData *od, FlapConnection *conn, FlapFrame *fr, ...) {
2380 va_list ap;
2381 guint16 chan, reason;
2382 char *who;
2383 int ret = 1;
2385 va_start(ap, fr);
2386 chan = (guint16)va_arg(ap, unsigned int);
2387 who = va_arg(ap, char *);
2388 reason = (guint16)va_arg(ap, unsigned int);
2390 if (chan == 0x0002) { /* File transfer declined */
2391 guchar *cookie = va_arg(ap, guchar *);
2392 ret = purple_parse_clientauto_ch2(od, who, reason, cookie);
2393 } else if (chan == 0x0004) { /* ICQ message */
2394 guint32 state = 0;
2395 char *msg = NULL;
2396 if (reason == 0x0003) {
2397 state = va_arg(ap, guint32);
2398 msg = va_arg(ap, char *);
2400 ret = purple_parse_clientauto_ch4(od, who, reason, state, msg);
2403 va_end(ap);
2405 return ret;
2408 static int purple_parse_genericerr(OscarData *od, FlapConnection *conn, FlapFrame *fr, ...) {
2409 va_list ap;
2410 guint16 reason;
2412 va_start(ap, fr);
2413 reason = (guint16) va_arg(ap, unsigned int);
2414 va_end(ap);
2416 purple_debug_error("oscar", "snac threw error (reason 0x%04hx: %s)\n",
2417 reason, oscar_get_msgerr_reason(reason));
2418 return 1;
2421 static int purple_parse_mtn(OscarData *od, FlapConnection *conn, FlapFrame *fr, ...) {
2422 PurpleConnection *gc = od->gc;
2423 va_list ap;
2424 guint16 channel, event;
2425 char *bn;
2427 va_start(ap, fr);
2428 channel = (guint16) va_arg(ap, unsigned int);
2429 bn = va_arg(ap, char *);
2430 event = (guint16) va_arg(ap, unsigned int);
2431 va_end(ap);
2433 switch (event) {
2434 case 0x0000: { /* Text has been cleared */
2435 serv_got_typing_stopped(gc, bn);
2436 } break;
2438 case 0x0001: { /* Paused typing */
2439 serv_got_typing(gc, bn, 0, PURPLE_TYPED);
2440 } break;
2442 case 0x0002: { /* Typing */
2443 serv_got_typing(gc, bn, 0, PURPLE_TYPING);
2444 } break;
2446 case 0x000f: { /* Closed IM window */
2447 serv_got_typing_stopped(gc, bn);
2448 } break;
2450 default: {
2451 purple_debug_info("oscar", "Received unknown typing "
2452 "notification message from %s. Channel is 0x%04x "
2453 "and event is 0x%04hx.\n", bn, channel, event);
2454 } break;
2457 return 1;
2460 static int purple_parse_motd(OscarData *od, FlapConnection *conn, FlapFrame *fr, ...)
2462 char *msg;
2463 guint16 id;
2464 va_list ap;
2466 va_start(ap, fr);
2467 id = (guint16) va_arg(ap, unsigned int);
2468 msg = va_arg(ap, char *);
2469 va_end(ap);
2471 purple_debug_misc("oscar",
2472 "MOTD: %s (%hu)\n", msg ? msg : "Unknown", id);
2473 if (id < 4)
2474 purple_notify_warning(od->gc, NULL,
2475 _("Your AIM connection may be lost."), NULL);
2477 return 1;
2480 static int purple_chatnav_info(OscarData *od, FlapConnection *conn, FlapFrame *fr, ...) {
2481 va_list ap;
2482 guint16 type;
2484 va_start(ap, fr);
2485 type = (guint16) va_arg(ap, unsigned int);
2487 switch(type) {
2488 case 0x0002: {
2489 GString *msg = g_string_new("");
2490 guint8 maxrooms;
2491 struct aim_chat_exchangeinfo *exchanges;
2492 int exchangecount, i;
2494 maxrooms = (guint8) va_arg(ap, unsigned int);
2495 exchangecount = va_arg(ap, int);
2496 exchanges = va_arg(ap, struct aim_chat_exchangeinfo *);
2498 g_string_append_printf(msg, "chat info: Max Concurrent Rooms: %hhd, Exchange List (%d total): ", maxrooms, exchangecount);
2499 for (i = 0; i < exchangecount; i++) {
2500 g_string_append_printf(msg, "%hu", exchanges[i].number);
2501 if (exchanges[i].name) {
2502 g_string_append_printf(msg, " %s", exchanges[i].name);
2504 g_string_append(msg, ", ");
2506 purple_debug_misc("oscar", "%s\n", msg->str);
2507 g_string_free(msg, TRUE);
2509 while (od->create_rooms) {
2510 struct create_room *cr = od->create_rooms->data;
2511 purple_debug_info("oscar",
2512 "creating room %s\n", cr->name);
2513 aim_chatnav_createroom(od, conn, cr->name, cr->exchange);
2514 g_free(cr->name);
2515 od->create_rooms = g_slist_remove(od->create_rooms, cr);
2516 g_free(cr);
2519 break;
2520 case 0x0008: {
2521 char *fqcn, *name, *ck;
2522 guint16 instance, flags, maxmsglen, maxoccupancy, unknown, exchange;
2523 guint8 createperms;
2524 guint32 createtime;
2526 fqcn = va_arg(ap, char *);
2527 instance = (guint16)va_arg(ap, unsigned int);
2528 exchange = (guint16)va_arg(ap, unsigned int);
2529 flags = (guint16)va_arg(ap, unsigned int);
2530 createtime = va_arg(ap, guint32);
2531 maxmsglen = (guint16)va_arg(ap, unsigned int);
2532 maxoccupancy = (guint16)va_arg(ap, unsigned int);
2533 createperms = (guint8)va_arg(ap, unsigned int);
2534 unknown = (guint16)va_arg(ap, unsigned int);
2535 name = va_arg(ap, char *);
2536 ck = va_arg(ap, char *);
2538 purple_debug_misc("oscar",
2539 "created room: %s %hu %hu %hu %u %hu %hu %hhu %hu %s %s\n",
2540 fqcn ? fqcn : "(null)", exchange, instance, flags, createtime,
2541 maxmsglen, maxoccupancy, createperms, unknown,
2542 name ? name : "(null)", ck);
2543 aim_chat_join(od, exchange, ck, instance);
2545 break;
2546 default:
2547 purple_debug_warning("oscar",
2548 "chatnav info: unknown type (%04hx)\n", type);
2549 break;
2552 va_end(ap);
2554 return 1;
2557 static int purple_conv_chat_join(OscarData *od, FlapConnection *conn, FlapFrame *fr, ...) {
2558 va_list ap;
2559 int count, i;
2560 aim_userinfo_t *info;
2561 PurpleConnection *gc = od->gc;
2563 struct chat_connection *c = NULL;
2565 va_start(ap, fr);
2566 count = va_arg(ap, int);
2567 info = va_arg(ap, aim_userinfo_t *);
2568 va_end(ap);
2570 c = find_oscar_chat_by_conn(gc, conn);
2571 if (!c)
2572 return 1;
2574 for (i = 0; i < count; i++)
2575 purple_conv_chat_add_user(PURPLE_CONV_CHAT(c->conv), info[i].bn, NULL, PURPLE_CBFLAGS_NONE, TRUE);
2577 return 1;
2580 static int purple_conv_chat_leave(OscarData *od, FlapConnection *conn, FlapFrame *fr, ...) {
2581 va_list ap;
2582 int count, i;
2583 aim_userinfo_t *info;
2584 PurpleConnection *gc = od->gc;
2586 struct chat_connection *c = NULL;
2588 va_start(ap, fr);
2589 count = va_arg(ap, int);
2590 info = va_arg(ap, aim_userinfo_t *);
2591 va_end(ap);
2593 c = find_oscar_chat_by_conn(gc, conn);
2594 if (!c)
2595 return 1;
2597 for (i = 0; i < count; i++)
2598 purple_conv_chat_remove_user(PURPLE_CONV_CHAT(c->conv), info[i].bn, NULL);
2600 return 1;
2603 static int purple_conv_chat_info_update(OscarData *od, FlapConnection *conn, FlapFrame *fr, ...) {
2604 va_list ap;
2605 guint16 maxmsglen, maxvisiblemsglen;
2606 PurpleConnection *gc = od->gc;
2607 struct chat_connection *ccon = find_oscar_chat_by_conn(gc, conn);
2609 if (!ccon)
2610 return 1;
2612 va_start(ap, fr);
2613 maxmsglen = (guint16)va_arg(ap, unsigned int);
2614 maxvisiblemsglen = (guint16)va_arg(ap, unsigned int);
2615 va_end(ap);
2617 purple_debug_misc("oscar",
2618 "inside chat_info_update (maxmsglen = %hu, maxvislen = %hu)\n",
2619 maxmsglen, maxvisiblemsglen);
2621 ccon->maxlen = maxmsglen;
2622 ccon->maxvis = maxvisiblemsglen;
2624 return 1;
2627 static int purple_conv_chat_incoming_msg(OscarData *od, FlapConnection *conn, FlapFrame *fr, ...) {
2628 PurpleConnection *gc = od->gc;
2629 struct chat_connection *ccon = find_oscar_chat_by_conn(gc, conn);
2630 gchar *utf8;
2631 va_list ap;
2632 aim_userinfo_t *info;
2633 int len;
2634 char *msg;
2635 char *charset;
2637 if (!ccon)
2638 return 1;
2640 va_start(ap, fr);
2641 info = va_arg(ap, aim_userinfo_t *);
2642 len = va_arg(ap, int);
2643 msg = va_arg(ap, char *);
2644 charset = va_arg(ap, char *);
2645 va_end(ap);
2647 utf8 = oscar_encoding_to_utf8(charset, msg, len);
2648 serv_got_chat_in(gc, ccon->id, info->bn, 0, utf8, time(NULL));
2649 g_free(utf8);
2651 return 1;
2654 static int purple_email_parseupdate(OscarData *od, FlapConnection *conn, FlapFrame *fr, ...) {
2655 va_list ap;
2656 PurpleConnection *gc;
2657 PurpleAccount *account;
2658 struct aim_emailinfo *emailinfo;
2659 int havenewmail;
2660 char *alertitle, *alerturl;
2662 gc = od->gc;
2663 account = purple_connection_get_account(gc);
2665 va_start(ap, fr);
2666 emailinfo = va_arg(ap, struct aim_emailinfo *);
2667 havenewmail = va_arg(ap, int);
2668 alertitle = va_arg(ap, char *);
2669 alerturl = va_arg(ap, char *);
2670 va_end(ap);
2672 if (account != NULL && emailinfo != NULL && purple_account_get_check_mail(account) &&
2673 emailinfo->unread && havenewmail) {
2674 gchar *to = g_strdup_printf("%s%s%s",
2675 purple_account_get_username(account),
2676 emailinfo->domain ? "@" : "",
2677 emailinfo->domain ? emailinfo->domain : "");
2678 const char *tos[2] = { to };
2679 const char *urls[2] = { emailinfo->url };
2680 purple_notify_emails(gc, emailinfo->nummsgs, FALSE, NULL, NULL,
2681 tos, urls, NULL, NULL);
2682 g_free(to);
2685 if (alertitle)
2686 purple_debug_misc("oscar", "Got an alert '%s' %s\n", alertitle, alerturl ? alerturl : "");
2688 return 1;
2691 static int purple_icon_parseicon(OscarData *od, FlapConnection *conn, FlapFrame *fr, ...) {
2692 PurpleConnection *gc = od->gc;
2693 va_list ap;
2694 char *bn;
2695 guint8 *iconcsum, *icon;
2696 guint16 iconcsumlen, iconlen;
2698 va_start(ap, fr);
2699 bn = va_arg(ap, char *);
2700 va_arg(ap, int); /* iconsumtype */
2701 iconcsum = va_arg(ap, guint8 *);
2702 iconcsumlen = va_arg(ap, int);
2703 icon = va_arg(ap, guint8 *);
2704 iconlen = va_arg(ap, int);
2705 va_end(ap);
2708 * Some AIM clients will send a blank GIF image with iconlen 90 when
2709 * no icon is set. Ignore these.
2711 if ((iconlen > 0) && (iconlen != 90)) {
2712 char *b16 = purple_base16_encode(iconcsum, iconcsumlen);
2713 purple_buddy_icons_set_for_user(purple_connection_get_account(gc),
2714 bn, g_memdup(icon, iconlen), iconlen, b16);
2715 g_free(b16);
2718 return 1;
2721 static void
2722 purple_icons_fetch(PurpleConnection *gc)
2724 OscarData *od = purple_connection_get_protocol_data(gc);
2725 aim_userinfo_t *userinfo;
2726 FlapConnection *conn;
2728 conn = flap_connection_getbytype(od, SNAC_FAMILY_BART);
2729 if (!conn) {
2730 if (!od->iconconnecting) {
2731 aim_srv_requestnew(od, SNAC_FAMILY_BART);
2732 od->iconconnecting = TRUE;
2734 return;
2737 if (od->set_icon) {
2738 PurpleAccount *account = purple_connection_get_account(gc);
2739 PurpleStoredImage *img = purple_buddy_icons_find_account_icon(account);
2740 if (img == NULL) {
2741 aim_ssi_delicon(od);
2742 } else {
2743 purple_debug_info("oscar",
2744 "Uploading icon to icon server\n");
2745 aim_bart_upload(od, purple_imgstore_get_data(img),
2746 purple_imgstore_get_size(img));
2747 purple_imgstore_unref(img);
2749 od->set_icon = FALSE;
2752 while (od->requesticon != NULL)
2754 userinfo = aim_locate_finduserinfo(od, (char *)od->requesticon->data);
2755 if ((userinfo != NULL) && (userinfo->iconcsumlen > 0))
2756 aim_bart_request(od, od->requesticon->data, userinfo->iconcsumtype, userinfo->iconcsum, userinfo->iconcsumlen);
2758 g_free(od->requesticon->data);
2759 od->requesticon = g_slist_delete_link(od->requesticon, od->requesticon);
2762 purple_debug_misc("oscar", "no more icons to request\n");
2765 static int purple_selfinfo(OscarData *od, FlapConnection *conn, FlapFrame *fr, ...) {
2766 va_list ap;
2767 aim_userinfo_t *info;
2769 va_start(ap, fr);
2770 info = va_arg(ap, aim_userinfo_t *);
2771 va_end(ap);
2773 purple_connection_set_display_name(od->gc, info->bn);
2775 return 1;
2778 static int purple_connerr(OscarData *od, FlapConnection *conn, FlapFrame *fr, ...) {
2779 PurpleConnection *gc = od->gc;
2780 va_list ap;
2781 guint16 code;
2782 char *msg;
2784 va_start(ap, fr);
2785 code = (guint16)va_arg(ap, int);
2786 msg = va_arg(ap, char *);
2787 va_end(ap);
2789 purple_debug_info("oscar", "Disconnected. Code is 0x%04x and msg is %s\n",
2790 code, (msg != NULL ? msg : ""));
2792 g_return_val_if_fail(conn != NULL, 1);
2794 if (conn->type == SNAC_FAMILY_CHAT) {
2795 struct chat_connection *cc;
2796 PurpleConversation *conv = NULL;
2798 cc = find_oscar_chat_by_conn(gc, conn);
2799 if (cc != NULL)
2801 conv = purple_find_chat(gc, cc->id);
2803 if (conv != NULL)
2806 * TOOD: Have flap_connection_destroy_cb() send us the
2807 * error message stored in 'tmp', which should be
2808 * human-friendly, and print that to the chat room.
2810 gchar *buf;
2811 buf = g_strdup_printf(_("You have been disconnected from chat "
2812 "room %s."), cc->name);
2813 purple_conversation_write(conv, NULL, buf, PURPLE_MESSAGE_ERROR, time(NULL));
2814 g_free(buf);
2816 oscar_chat_kill(gc, cc);
2820 return 1;
2823 static int purple_parse_locaterights(OscarData *od, FlapConnection *conn, FlapFrame *fr, ...)
2825 PurpleConnection *gc = od->gc;
2826 PurpleAccount *account = purple_connection_get_account(gc);
2827 va_list ap;
2828 guint16 maxsiglen;
2830 va_start(ap, fr);
2831 maxsiglen = (guint16) va_arg(ap, int);
2832 va_end(ap);
2834 purple_debug_misc("oscar",
2835 "locate rights: max sig len = %d\n", maxsiglen);
2837 od->rights.maxsiglen = od->rights.maxawaymsglen = (guint)maxsiglen;
2839 aim_locate_setcaps(od, purple_caps);
2840 oscar_set_info_and_status(account, TRUE, account->user_info, TRUE,
2841 purple_account_get_active_status(account));
2843 return 1;
2846 static int purple_parse_buddyrights(OscarData *od, FlapConnection *conn, FlapFrame *fr, ...) {
2847 va_list ap;
2848 guint16 maxbuddies, maxwatchers;
2850 va_start(ap, fr);
2851 maxbuddies = (guint16) va_arg(ap, unsigned int);
2852 maxwatchers = (guint16) va_arg(ap, unsigned int);
2853 va_end(ap);
2855 purple_debug_misc("oscar",
2856 "buddy list rights: Max buddies = %hu / Max watchers = %hu\n", maxbuddies, maxwatchers);
2858 od->rights.maxbuddies = (guint)maxbuddies;
2859 od->rights.maxwatchers = (guint)maxwatchers;
2861 return 1;
2864 static void oscar_format_username(PurpleConnection *gc, const char *new_display_name)
2866 OscarData *od;
2867 const char *old_display_name, *username;
2868 char *tmp, *at_sign;
2870 old_display_name = purple_connection_get_display_name(gc);
2871 if (old_display_name && strchr(old_display_name, '@')) {
2872 purple_debug_info("oscar", "Cowardly refusing to attempt to format "
2873 "screen name because the current formatting according to "
2874 "the server (%s) appears to be an email address\n",
2875 old_display_name);
2876 return;
2879 username = purple_account_get_username(purple_connection_get_account(gc));
2880 if (oscar_util_name_compare(username, new_display_name)) {
2881 purple_notify_error(gc, NULL, _("The new formatting is invalid."),
2882 _("Username formatting can change only capitalization and whitespace."));
2883 return;
2886 tmp = g_strdup(new_display_name);
2889 * If our local username is an email address then strip off the domain.
2890 * This allows formatting to work if the user entered their username as
2891 * 'something@aim.com' or possibly other AOL-owned domains.
2893 at_sign = strchr(tmp, '@');
2894 if (at_sign)
2895 at_sign[0] = '\0';
2897 od = purple_connection_get_protocol_data(gc);
2898 if (!flap_connection_getbytype(od, SNAC_FAMILY_ADMIN)) {
2899 /* We don't have a connection to an "admin" server. Make one. */
2900 od->setnick = TRUE;
2901 g_free(od->newformatting);
2902 od->newformatting = tmp;
2903 aim_srv_requestnew(od, SNAC_FAMILY_ADMIN);
2904 } else {
2905 aim_admin_setnick(od, flap_connection_getbytype(od, SNAC_FAMILY_ADMIN), tmp);
2906 g_free(tmp);
2910 static int purple_bosrights(OscarData *od, FlapConnection *conn, FlapFrame *fr, ...) {
2911 PurpleConnection *gc;
2912 PurpleAccount *account;
2913 PurpleStatus *status;
2914 gboolean is_available;
2915 PurplePresence *presence;
2916 const char *username, *message, *itmsurl;
2917 char *tmp;
2918 va_list ap;
2919 guint16 maxpermits, maxdenies;
2921 gc = od->gc;
2922 od = purple_connection_get_protocol_data(gc);
2923 account = purple_connection_get_account(gc);
2925 va_start(ap, fr);
2926 maxpermits = (guint16) va_arg(ap, unsigned int);
2927 maxdenies = (guint16) va_arg(ap, unsigned int);
2928 va_end(ap);
2930 purple_debug_misc("oscar",
2931 "BOS rights: Max permit = %hu / Max deny = %hu\n", maxpermits, maxdenies);
2933 od->rights.maxpermits = (guint)maxpermits;
2934 od->rights.maxdenies = (guint)maxdenies;
2936 purple_debug_info("oscar", "buddy list loaded\n");
2938 if (purple_account_get_user_info(account) != NULL)
2939 serv_set_info(gc, purple_account_get_user_info(account));
2941 username = purple_account_get_username(account);
2942 if (!od->icq && !purple_strequal(username, purple_connection_get_display_name(gc))) {
2944 * Format the username for AIM accounts if it's different
2945 * than what's currently set.
2947 oscar_format_username(gc, username);
2950 /* Set our available message based on the current status */
2951 status = purple_account_get_active_status(account);
2952 is_available = purple_status_is_available(status);
2953 if (is_available)
2954 message = purple_status_get_attr_string(status, "message");
2955 else
2956 message = NULL;
2957 tmp = purple_markup_strip_html(message);
2958 itmsurl = purple_status_get_attr_string(status, "itmsurl");
2959 aim_srv_setextrainfo(od, FALSE, 0, is_available, tmp, itmsurl);
2960 aim_srv_set_dc_info(od);
2961 g_free(tmp);
2963 presence = purple_status_get_presence(status);
2964 aim_srv_setidle(od, !purple_presence_is_idle(presence) ? 0 : time(NULL) - purple_presence_get_idle_time(presence));
2966 if (od->icq) {
2967 oscar_set_extended_status(gc);
2968 aim_icq_setsecurity(od,
2969 purple_account_get_bool(account, "authorization", OSCAR_DEFAULT_AUTHORIZATION),
2970 purple_account_get_bool(account, "web_aware", OSCAR_DEFAULT_WEB_AWARE));
2973 aim_srv_requestnew(od, SNAC_FAMILY_ALERT);
2974 aim_srv_requestnew(od, SNAC_FAMILY_CHATNAV);
2976 od->bos.have_rights = TRUE;
2979 * If we've already received our feedbag data then we're not waiting on
2980 * anything else, so send the server clientready.
2982 * Normally we get bos rights before we get our feedbag data, so this
2983 * rarely (never?) happens. And I'm not sure it actually matters if we
2984 * wait for bos rights before calling clientready. But it seems safer
2985 * to do it this way.
2987 if (od->ssi.received_data) {
2988 aim_srv_clientready(od, conn);
2990 /* Request offline messages for AIM and ICQ */
2991 aim_im_reqofflinemsgs(od);
2993 purple_connection_set_state(gc, PURPLE_CONNECTED);
2996 return 1;
2999 static int purple_popup(OscarData *od, FlapConnection *conn, FlapFrame *fr, ...)
3001 PurpleConnection *gc = od->gc;
3002 gchar *text;
3003 va_list ap;
3004 char *msg, *url;
3006 va_start(ap, fr);
3007 msg = va_arg(ap, char *);
3008 url = va_arg(ap, char *);
3009 va_arg(ap, int); /* guint16 wid */
3010 va_arg(ap, int); /* guint16 hei */
3011 va_arg(ap, int); /* guint16 delay */
3012 va_end(ap);
3014 text = g_strdup_printf("%s<br><a href=\"%s\">%s</a>", msg, url, url);
3015 purple_notify_formatted(gc, NULL, _("Pop-Up Message"), NULL, text, NULL, NULL);
3016 g_free(text);
3018 return 1;
3021 static void oscar_searchresults_add_buddy_cb(PurpleConnection *gc, GList *row, void *user_data)
3023 purple_blist_request_add_buddy(purple_connection_get_account(gc),
3024 g_list_nth_data(row, 0), NULL, NULL);
3027 static int purple_parse_searchreply(OscarData *od, FlapConnection *conn, FlapFrame *fr, ...)
3029 PurpleConnection *gc = od->gc;
3030 PurpleNotifySearchResults *results;
3031 PurpleNotifySearchColumn *column;
3032 gchar *secondary;
3033 int i, num;
3034 va_list ap;
3035 char *email, *usernames;
3037 va_start(ap, fr);
3038 email = va_arg(ap, char *);
3039 num = va_arg(ap, int);
3040 usernames = va_arg(ap, char *);
3041 va_end(ap);
3043 results = purple_notify_searchresults_new();
3045 if (results == NULL) {
3046 purple_debug_error("oscar", "purple_parse_searchreply: "
3047 "Unable to display the search results.\n");
3048 purple_notify_error(gc, NULL,
3049 _("Unable to display the search results."),
3050 NULL);
3051 return 1;
3054 secondary = g_strdup_printf(
3055 dngettext(PACKAGE, "The following username is associated with %s",
3056 "The following usernames are associated with %s",
3057 num),
3058 email);
3060 column = purple_notify_searchresults_column_new(_("Username"));
3061 purple_notify_searchresults_column_add(results, column);
3063 for (i = 0; i < num; i++) {
3064 GList *row;
3065 row = g_list_append(NULL, g_strdup(&usernames[i * (MAXSNLEN + 1)]));
3066 purple_notify_searchresults_row_add(results, row);
3068 purple_notify_searchresults_button_add(results, PURPLE_NOTIFY_BUTTON_ADD,
3069 oscar_searchresults_add_buddy_cb);
3070 purple_notify_searchresults(gc, NULL, NULL, secondary, results, NULL, NULL);
3072 g_free(secondary);
3074 return 1;
3077 static int purple_parse_searcherror(OscarData *od, FlapConnection *conn, FlapFrame *fr, ...) {
3078 va_list ap;
3079 char *email;
3080 char *buf;
3082 va_start(ap, fr);
3083 email = va_arg(ap, char *);
3084 va_end(ap);
3086 buf = g_strdup_printf(_("No results found for email address %s"), email);
3087 purple_notify_error(od->gc, NULL, buf, NULL);
3088 g_free(buf);
3090 return 1;
3093 static int purple_account_confirm(OscarData *od, FlapConnection *conn, FlapFrame *fr, ...) {
3094 PurpleConnection *gc = od->gc;
3095 guint16 status;
3096 va_list ap;
3097 char msg[256];
3099 va_start(ap, fr);
3100 status = (guint16) va_arg(ap, unsigned int); /* status code of confirmation request */
3101 va_end(ap);
3103 purple_debug_info("oscar",
3104 "account confirmation returned status 0x%04x (%s)\n", status,
3105 status ? "unknown" : "email sent");
3106 if (!status) {
3107 g_snprintf(msg, sizeof(msg), _("You should receive an email asking to confirm %s."),
3108 purple_account_get_username(purple_connection_get_account(gc)));
3109 purple_notify_info(gc, NULL, _("Account Confirmation Requested"), msg);
3112 return 1;
3115 static int purple_info_change(OscarData *od, FlapConnection *conn, FlapFrame *fr, ...) {
3116 PurpleConnection *gc = od->gc;
3117 va_list ap;
3118 guint16 perms, err;
3119 char *url, *bn, *email;
3120 int change;
3122 va_start(ap, fr);
3123 change = va_arg(ap, int);
3124 perms = (guint16) va_arg(ap, unsigned int);
3125 err = (guint16) va_arg(ap, unsigned int);
3126 url = va_arg(ap, char *);
3127 bn = va_arg(ap, char *);
3128 email = va_arg(ap, char *);
3129 va_end(ap);
3131 purple_debug_misc("oscar",
3132 "account info: because of %s, perms=0x%04x, err=0x%04x, url=%s, bn=%s, email=%s\n",
3133 change ? "change" : "request", perms, err,
3134 (url != NULL) ? url : "(null)",
3135 (bn != NULL) ? bn : "(null)",
3136 (email != NULL) ? email : "(null)");
3138 if ((err > 0) && (url != NULL)) {
3139 char *dialog_msg;
3141 if (err == 0x0001)
3142 dialog_msg = g_strdup_printf(_("Error 0x%04x: Unable to format username because the requested name differs from the original."), err);
3143 else if (err == 0x0006)
3144 dialog_msg = g_strdup_printf(_("Error 0x%04x: Unable to format username because it is invalid."), err);
3145 else if (err == 0x00b)
3146 dialog_msg = g_strdup_printf(_("Error 0x%04x: Unable to format username because the requested name is too long."), err);
3147 else if (err == 0x001d)
3148 dialog_msg = g_strdup_printf(_("Error 0x%04x: Unable to change email address because there is already a request pending for this username."), err);
3149 else if (err == 0x0021)
3150 dialog_msg = g_strdup_printf(_("Error 0x%04x: Unable to change email address because the given address has too many usernames associated with it."), err);
3151 else if (err == 0x0023)
3152 dialog_msg = g_strdup_printf(_("Error 0x%04x: Unable to change email address because the given address is invalid."), err);
3153 else
3154 dialog_msg = g_strdup_printf(_("Error 0x%04x: Unknown error."), err);
3155 purple_notify_error(gc, NULL,
3156 _("Error Changing Account Info"), dialog_msg);
3157 g_free(dialog_msg);
3158 return 1;
3161 if (email != NULL) {
3162 char *dialog_msg = g_strdup_printf(_("The email address for %s is %s"),
3163 purple_account_get_username(purple_connection_get_account(gc)), email);
3164 purple_notify_info(gc, NULL, _("Account Info"), dialog_msg);
3165 g_free(dialog_msg);
3168 return 1;
3171 void
3172 oscar_keepalive(PurpleConnection *gc)
3174 OscarData *od;
3175 GSList *l;
3177 od = purple_connection_get_protocol_data(gc);
3178 for (l = od->oscar_connections; l; l = l->next) {
3179 flap_connection_send_keepalive(od, l->data);
3183 unsigned int
3184 oscar_send_typing(PurpleConnection *gc, const char *name, PurpleTypingState state)
3186 OscarData *od;
3187 PeerConnection *conn;
3189 od = purple_connection_get_protocol_data(gc);
3190 conn = peer_connection_find_by_type(od, name, OSCAR_CAPABILITY_DIRECTIM);
3192 if ((conn != NULL) && (conn->ready))
3194 peer_odc_send_typing(conn, state);
3196 else {
3197 /* Don't send if this turkey is in our deny list */
3198 GSList *list;
3199 for (list=gc->account->deny; (list && oscar_util_name_compare(name, list->data)); list=list->next);
3200 if (!list) {
3201 struct buddyinfo *bi = g_hash_table_lookup(od->buddyinfo, purple_normalize(gc->account, name));
3202 if (bi && bi->typingnot) {
3203 if (state == PURPLE_TYPING)
3204 aim_im_sendmtn(od, 0x0001, name, 0x0002);
3205 else if (state == PURPLE_TYPED)
3206 aim_im_sendmtn(od, 0x0001, name, 0x0001);
3207 else
3208 aim_im_sendmtn(od, 0x0001, name, 0x0000);
3212 return 0;
3215 /* TODO: Move this into odc.c! */
3216 static void
3217 purple_odc_send_im(PeerConnection *conn, const char *message, PurpleMessageFlags imflags)
3219 GString *msg;
3220 GString *data;
3221 gchar *tmp;
3222 gsize tmplen;
3223 guint16 charset;
3224 GData *attribs;
3225 const char *start, *end, *last;
3226 int oscar_id = 0;
3228 msg = g_string_new("<HTML><BODY>");
3229 data = g_string_new("<BINARY>");
3230 last = message;
3232 /* for each valid IMG tag... */
3233 while (last && *last && purple_markup_find_tag("img", last, &start, &end, &attribs))
3235 PurpleStoredImage *image = NULL;
3236 const char *id;
3238 if (start - last) {
3239 g_string_append_len(msg, last, start - last);
3242 id = g_datalist_get_data(&attribs, "id");
3244 /* ... if it refers to a valid purple image ... */
3245 if (id && (image = purple_imgstore_find_by_id(atoi(id)))) {
3246 /* ... append the message from start to the tag ... */
3247 unsigned long size = purple_imgstore_get_size(image);
3248 const char *filename = purple_imgstore_get_filename(image);
3249 gconstpointer imgdata = purple_imgstore_get_data(image);
3251 oscar_id++;
3253 /* ... insert a new img tag with the oscar id ... */
3254 if (filename)
3255 g_string_append_printf(msg,
3256 "<IMG SRC=\"%s\" ID=\"%d\" DATASIZE=\"%lu\">",
3257 filename, oscar_id, size);
3258 else
3259 g_string_append_printf(msg,
3260 "<IMG ID=\"%d\" DATASIZE=\"%lu\">",
3261 oscar_id, size);
3263 /* ... and append the data to the binary section ... */
3264 g_string_append_printf(data, "<DATA ID=\"%d\" SIZE=\"%lu\">",
3265 oscar_id, size);
3266 g_string_append_len(data, imgdata, size);
3267 g_string_append(data, "</DATA>");
3269 /* If the tag is invalid, skip it, thus no else here */
3271 g_datalist_clear(&attribs);
3273 /* continue from the end of the tag */
3274 last = end + 1;
3277 /* append any remaining message data */
3278 if (last && *last)
3279 g_string_append(msg, last);
3281 g_string_append(msg, "</BODY></HTML>");
3283 /* Convert the message to a good encoding */
3284 tmp = oscar_encode_im(msg->str, &tmplen, &charset, NULL);
3285 g_string_free(msg, TRUE);
3286 msg = g_string_new_len(tmp, tmplen);
3287 g_free(tmp);
3289 /* Append any binary data that we may have */
3290 if (oscar_id) {
3291 msg = g_string_append_len(msg, data->str, data->len);
3292 msg = g_string_append(msg, "</BINARY>");
3294 g_string_free(data, TRUE);
3296 purple_debug_info("oscar", "sending direct IM %s using charset %i", msg->str, charset);
3298 peer_odc_send_im(conn, msg->str, msg->len, charset,
3299 imflags & PURPLE_MESSAGE_AUTO_RESP);
3300 g_string_free(msg, TRUE);
3304 oscar_send_im(PurpleConnection *gc, const char *name, const char *message, PurpleMessageFlags imflags)
3306 OscarData *od;
3307 PurpleAccount *account;
3308 PeerConnection *conn;
3309 int ret;
3310 char *tmp1, *tmp2;
3311 gboolean is_sms, is_html;
3313 od = purple_connection_get_protocol_data(gc);
3314 account = purple_connection_get_account(gc);
3315 ret = 0;
3317 is_sms = oscar_util_valid_name_sms(name);
3319 if (od->icq && is_sms) {
3321 * We're sending to a phone number and this is ICQ,
3322 * so send the message as an SMS using aim_icq_sendsms()
3324 int ret;
3325 purple_debug_info("oscar", "Sending SMS to %s.\n", name);
3326 ret = aim_icq_sendsms(od, name, message, purple_account_get_username(account));
3327 return (ret >= 0 ? 1 : ret);
3330 if (imflags & PURPLE_MESSAGE_AUTO_RESP)
3331 tmp1 = oscar_util_format_string(message, name);
3332 else
3333 tmp1 = g_strdup(message);
3335 conn = peer_connection_find_by_type(od, name, OSCAR_CAPABILITY_DIRECTIM);
3336 if ((conn != NULL) && (conn->ready))
3338 /* If we're directly connected, send a direct IM */
3339 purple_debug_info("oscar", "Sending direct IM with flags %i\n", imflags);
3340 purple_odc_send_im(conn, tmp1, imflags);
3341 } else {
3342 struct buddyinfo *bi;
3343 struct aim_sendimext_args args;
3344 PurpleConversation *conv;
3345 PurpleStoredImage *img;
3346 PurpleBuddy *buddy;
3348 conv = purple_find_conversation_with_account(PURPLE_CONV_TYPE_IM, name, account);
3350 if (strstr(tmp1, "<IMG "))
3351 purple_conversation_write(conv, "",
3352 _("Your IM Image was not sent. "
3353 "You must be Direct Connected to send IM Images."),
3354 PURPLE_MESSAGE_ERROR, time(NULL));
3356 buddy = purple_find_buddy(account, name);
3358 bi = g_hash_table_lookup(od->buddyinfo, purple_normalize(account, name));
3359 if (!bi) {
3360 bi = g_new0(struct buddyinfo, 1);
3361 g_hash_table_insert(od->buddyinfo, g_strdup(purple_normalize(account, name)), bi);
3364 args.flags = 0;
3366 if (!is_sms && (!buddy || !PURPLE_BUDDY_IS_ONLINE(buddy)))
3367 args.flags |= AIM_IMFLAGS_OFFLINE;
3369 if (od->icq) {
3370 args.features = features_icq;
3371 args.featureslen = sizeof(features_icq);
3372 } else {
3373 args.features = features_aim;
3374 args.featureslen = sizeof(features_aim);
3376 if (imflags & PURPLE_MESSAGE_AUTO_RESP)
3377 args.flags |= AIM_IMFLAGS_AWAY;
3380 if (bi->ico_need) {
3381 purple_debug_info("oscar",
3382 "Sending buddy icon request with message\n");
3383 args.flags |= AIM_IMFLAGS_BUDDYREQ;
3384 bi->ico_need = FALSE;
3387 img = purple_buddy_icons_find_account_icon(account);
3388 if (img) {
3389 gconstpointer data = purple_imgstore_get_data(img);
3390 args.iconlen = purple_imgstore_get_size(img);
3391 args.iconsum = aimutil_iconsum(data, args.iconlen);
3392 args.iconstamp = purple_buddy_icons_get_account_icon_timestamp(account);
3394 if ((args.iconlen != bi->ico_me_len) || (args.iconsum != bi->ico_me_csum) || (args.iconstamp != bi->ico_me_time)) {
3395 bi->ico_informed = FALSE;
3396 bi->ico_sent = FALSE;
3400 * TODO:
3401 * For some reason sending our icon to people only works
3402 * when we're the ones who initiated the conversation. If
3403 * the other person sends the first IM then they never get
3404 * the icon. We should fix that.
3406 if (!bi->ico_informed) {
3407 purple_debug_info("oscar",
3408 "Claiming to have a buddy icon\n");
3409 args.flags |= AIM_IMFLAGS_HASICON;
3410 bi->ico_me_len = args.iconlen;
3411 bi->ico_me_csum = args.iconsum;
3412 bi->ico_me_time = args.iconstamp;
3413 bi->ico_informed = TRUE;
3416 purple_imgstore_unref(img);
3419 args.destbn = name;
3421 if (oscar_util_valid_name_sms(name)) {
3422 /* Messaging an SMS (mobile) user--strip HTML */
3423 tmp2 = purple_markup_strip_html(tmp1);
3424 is_html = FALSE;
3425 } else {
3426 /* ICQ 6 wants its HTML wrapped in these tags. Oblige it. */
3427 tmp2 = g_strdup_printf("<HTML><BODY>%s</BODY></HTML>", tmp1);
3428 is_html = TRUE;
3430 g_free(tmp1);
3431 tmp1 = tmp2;
3433 args.msg = oscar_encode_im(tmp1, &args.msglen, &args.charset, NULL);
3434 if (is_html && (args.msglen > MAXMSGLEN)) {
3435 /* If the length was too long, try stripping the HTML and then running it back through
3436 * purple_strdup_withhtml() and the encoding process. The result may be shorter. */
3437 g_free((char *)args.msg);
3439 tmp2 = purple_markup_strip_html(tmp1);
3440 g_free(tmp1);
3442 /* re-escape the entities */
3443 tmp1 = g_markup_escape_text(tmp2, -1);
3444 g_free(tmp2);
3446 tmp2 = purple_strdup_withhtml(tmp1);
3447 g_free(tmp1);
3448 tmp1 = tmp2;
3450 args.msg = oscar_encode_im(tmp1, &args.msglen, &args.charset, NULL);
3451 purple_debug_info("oscar", "Sending %s as %s because the original was too long.\n",
3452 message, (char *)args.msg);
3455 purple_debug_info("oscar", "Sending IM, charset=0x%04hx, length=%" G_GSIZE_FORMAT "\n", args.charset, args.msglen);
3456 ret = aim_im_sendch1_ext(od, &args);
3457 g_free((char *)args.msg);
3460 g_free(tmp1);
3462 if (ret >= 0)
3463 return 1;
3465 return ret;
3469 * As of 26 June 2006, ICQ users can request AIM info from
3470 * everyone, and can request ICQ info from ICQ users, and
3471 * AIM users can only request AIM info.
3473 void oscar_get_info(PurpleConnection *gc, const char *name) {
3474 OscarData *od = purple_connection_get_protocol_data(gc);
3476 if (od->icq && oscar_util_valid_name_icq(name))
3477 aim_icq_getallinfo(od, name);
3478 else
3479 aim_locate_getinfoshort(od, name, 0x00000003);
3482 void oscar_set_idle(PurpleConnection *gc, int time) {
3483 OscarData *od = purple_connection_get_protocol_data(gc);
3484 aim_srv_setidle(od, time);
3487 void
3488 oscar_set_info(PurpleConnection *gc, const char *rawinfo)
3490 PurpleAccount *account;
3491 PurpleStatus *status;
3493 account = purple_connection_get_account(gc);
3494 status = purple_account_get_active_status(account);
3495 oscar_set_info_and_status(account, TRUE, rawinfo, FALSE, status);
3498 static guint32
3499 oscar_get_extended_status(PurpleConnection *gc)
3501 PurpleAccount *account;
3502 PurpleStatus *status;
3503 const gchar *status_id;
3504 guint32 data = 0x00000000;
3506 account = purple_connection_get_account(gc);
3507 status = purple_account_get_active_status(account);
3508 status_id = purple_status_get_id(status);
3510 data |= AIM_ICQ_STATE_HIDEIP;
3511 if (purple_account_get_bool(account, "web_aware", OSCAR_DEFAULT_WEB_AWARE))
3512 data |= AIM_ICQ_STATE_WEBAWARE;
3514 if (purple_strequal(status_id, OSCAR_STATUS_ID_AVAILABLE))
3515 data |= AIM_ICQ_STATE_NORMAL;
3516 else if (purple_strequal(status_id, OSCAR_STATUS_ID_AWAY))
3517 data |= AIM_ICQ_STATE_AWAY;
3518 else if (purple_strequal(status_id, OSCAR_STATUS_ID_DND))
3519 data |= AIM_ICQ_STATE_AWAY | AIM_ICQ_STATE_DND | AIM_ICQ_STATE_BUSY;
3520 else if (purple_strequal(status_id, OSCAR_STATUS_ID_NA))
3521 data |= AIM_ICQ_STATE_OUT | AIM_ICQ_STATE_AWAY;
3522 else if (purple_strequal(status_id, OSCAR_STATUS_ID_OCCUPIED))
3523 data |= AIM_ICQ_STATE_AWAY | AIM_ICQ_STATE_BUSY;
3524 else if (purple_strequal(status_id, OSCAR_STATUS_ID_FREE4CHAT))
3525 data |= AIM_ICQ_STATE_CHAT;
3526 else if (purple_strequal(status_id, OSCAR_STATUS_ID_INVISIBLE))
3527 data |= AIM_ICQ_STATE_INVISIBLE;
3528 else if (purple_strequal(status_id, OSCAR_STATUS_ID_EVIL))
3529 data |= AIM_ICQ_STATE_EVIL;
3530 else if (purple_strequal(status_id, OSCAR_STATUS_ID_DEPRESSION))
3531 data |= AIM_ICQ_STATE_DEPRESSION;
3532 else if (purple_strequal(status_id, OSCAR_STATUS_ID_ATWORK))
3533 data |= AIM_ICQ_STATE_ATWORK;
3534 else if (purple_strequal(status_id, OSCAR_STATUS_ID_ATHOME))
3535 data |= AIM_ICQ_STATE_ATHOME;
3536 else if (purple_strequal(status_id, OSCAR_STATUS_ID_LUNCH))
3537 data |= AIM_ICQ_STATE_LUNCH;
3538 else if (purple_strequal(status_id, OSCAR_STATUS_ID_CUSTOM))
3539 data |= AIM_ICQ_STATE_OUT | AIM_ICQ_STATE_AWAY;
3541 return data;
3544 static void
3545 oscar_set_extended_status(PurpleConnection *gc)
3547 aim_srv_setextrainfo(purple_connection_get_protocol_data(gc), TRUE, oscar_get_extended_status(gc), FALSE, NULL, NULL);
3550 static void
3551 oscar_set_info_and_status(PurpleAccount *account, gboolean setinfo, const char *rawinfo,
3552 gboolean setstatus, PurpleStatus *status)
3554 PurpleConnection *gc = purple_account_get_connection(account);
3555 OscarData *od = purple_connection_get_protocol_data(gc);
3556 PurpleStatusType *status_type;
3557 PurpleStatusPrimitive primitive;
3559 char *info_encoding = NULL;
3560 char *info = NULL;
3561 gsize infolen = 0;
3563 char *away_encoding = NULL;
3564 char *away = NULL;
3565 gsize awaylen = 0;
3567 char *status_text = NULL;
3568 const char *itmsurl = NULL;
3570 status_type = purple_status_get_type(status);
3571 primitive = purple_status_type_get_primitive(status_type);
3573 if (!setinfo)
3575 /* Do nothing! */
3577 else if (od->rights.maxsiglen == 0)
3579 purple_notify_warning(gc, NULL, _("Unable to set AIM profile."),
3580 _("You have probably requested to set your "
3581 "profile before the login procedure completed. "
3582 "Your profile remains unset; try setting it "
3583 "again when you are fully connected."));
3585 else if (rawinfo != NULL)
3587 char *htmlinfo = purple_strdup_withhtml(rawinfo);
3588 info = oscar_encode_im(htmlinfo, &infolen, NULL, &info_encoding);
3589 g_free(htmlinfo);
3591 if (infolen > od->rights.maxsiglen)
3593 gchar *errstr;
3594 errstr = g_strdup_printf(dngettext(PACKAGE, "The maximum profile length of %d byte "
3595 "has been exceeded. It has been truncated for you.",
3596 "The maximum profile length of %d bytes "
3597 "has been exceeded. It has been truncated for you.",
3598 od->rights.maxsiglen), od->rights.maxsiglen);
3599 purple_notify_warning(gc, NULL, _("Profile too long."), errstr);
3600 g_free(errstr);
3604 if (setstatus)
3606 const char *status_html;
3608 status_html = purple_status_get_attr_string(status, "message");
3610 if (status_html == NULL || primitive == PURPLE_STATUS_AVAILABLE || primitive == PURPLE_STATUS_INVISIBLE)
3612 /* This is needed for us to un-set any previous away message. */
3613 away = g_strdup("");
3615 else
3617 gchar *linkified;
3619 /* We do this for icq too so that they work for old third party clients */
3620 linkified = purple_markup_linkify(status_html);
3621 away = oscar_encode_im(linkified, &awaylen, NULL, &away_encoding);
3622 g_free(linkified);
3624 if (awaylen > od->rights.maxawaymsglen)
3626 gchar *errstr;
3628 errstr = g_strdup_printf(dngettext(PACKAGE, "The maximum away message length of %d byte "
3629 "has been exceeded. It has been truncated for you.",
3630 "The maximum away message length of %d bytes "
3631 "has been exceeded. It has been truncated for you.",
3632 od->rights.maxawaymsglen), od->rights.maxawaymsglen);
3633 purple_notify_warning(gc, NULL, _("Away message too long."), errstr);
3634 g_free(errstr);
3639 aim_locate_setprofile(od,
3640 info_encoding, info, MIN(infolen, od->rights.maxsiglen),
3641 away_encoding, away, MIN(awaylen, od->rights.maxawaymsglen));
3642 g_free(info);
3643 g_free(away);
3645 if (setstatus)
3647 const char *status_html;
3649 status_html = purple_status_get_attr_string(status, "message");
3650 if (status_html != NULL)
3652 status_text = purple_markup_strip_html(status_html);
3653 /* If the status_text is longer than 251 characters then truncate it */
3654 if (strlen(status_text) > MAXAVAILMSGLEN)
3656 char *tmp = g_utf8_find_prev_char(status_text, &status_text[MAXAVAILMSGLEN - 2]);
3657 strcpy(tmp, "...");
3661 itmsurl = purple_status_get_attr_string(status, "itmsurl");
3663 aim_srv_setextrainfo(od, TRUE, oscar_get_extended_status(gc), TRUE, status_text, itmsurl);
3664 g_free(status_text);
3668 static void
3669 oscar_set_icq_permdeny(PurpleAccount *account)
3671 PurpleConnection *gc = purple_account_get_connection(account);
3672 OscarData *od = purple_connection_get_protocol_data(gc);
3673 gboolean invisible = purple_account_is_status_active(account, OSCAR_STATUS_ID_INVISIBLE);
3676 * For ICQ the permit/deny setting controls who can see you
3677 * online. Mimicking the official client's behavior, we use PURPLE_PRIVACY_ALLOW_USERS
3678 * when our status is "invisible" and PURPLE_PRIVACY_DENY_USERS otherwise.
3679 * In the former case, we are visible only to buddies on our "permanently visible" list.
3680 * In the latter, we are invisible only to buddies on our "permanently invisible" list.
3682 aim_ssi_setpermdeny(od, invisible ? PURPLE_PRIVACY_ALLOW_USERS : PURPLE_PRIVACY_DENY_USERS);
3685 void
3686 oscar_set_status(PurpleAccount *account, PurpleStatus *status)
3688 PurpleConnection *pc;
3689 OscarData *od;
3691 purple_debug_info("oscar", "Set status to %s\n", purple_status_get_name(status));
3693 /* Either setting a new status active or setting a status inactive.
3694 * (Only possible for independent status (i.e. X-Status moods.) */
3695 if (!purple_status_is_active(status) && !purple_status_is_independent(status))
3696 return;
3698 if (!purple_account_is_connected(account))
3699 return;
3701 pc = purple_account_get_connection(account);
3702 od = purple_connection_get_protocol_data(pc);
3704 /* There's no need to do the stuff below for mood updates. */
3705 if (purple_status_type_get_primitive(purple_status_get_type(status)) == PURPLE_STATUS_MOOD) {
3706 aim_locate_setcaps(od, purple_caps);
3707 return;
3710 if (od->icq) {
3711 /* Set visibility */
3712 oscar_set_icq_permdeny(account);
3715 /* Set the AIM-style away message for both AIM and ICQ accounts */
3716 oscar_set_info_and_status(account, FALSE, NULL, TRUE, status);
3719 void
3720 oscar_add_buddy(PurpleConnection *gc, PurpleBuddy *buddy, PurpleGroup *group, const char *msg)
3722 OscarData *od;
3723 PurpleAccount *account;
3724 const char *bname, *gname;
3726 od = purple_connection_get_protocol_data(gc);
3727 account = purple_connection_get_account(gc);
3728 bname = purple_buddy_get_name(buddy);
3729 gname = purple_group_get_name(group);
3731 if (!oscar_util_valid_name(bname)) {
3732 gchar *buf;
3733 buf = g_strdup_printf(_("Unable to add the buddy %s because the username is invalid. Usernames must be a valid email address, or start with a letter and contain only letters, numbers and spaces, or contain only numbers."), bname);
3734 if (!purple_conv_present_error(bname, account, buf))
3735 purple_notify_error(gc, NULL, _("Unable to Add"), buf);
3736 g_free(buf);
3738 /* Remove from local list */
3739 purple_blist_remove_buddy(buddy);
3741 return;
3744 if (od->ssi.received_data) {
3745 if (!aim_ssi_itemlist_finditem(od->ssi.local, gname, bname, AIM_SSI_TYPE_BUDDY)) {
3746 purple_debug_info("oscar",
3747 "ssi: adding buddy %s to group %s\n", bname, gname);
3748 aim_ssi_addbuddy(od, bname, gname, NULL, purple_buddy_get_alias_only(buddy), NULL, NULL, 0);
3750 /* Mobile users should always be online */
3751 if (bname[0] == '+') {
3752 purple_prpl_got_user_status(account, bname,
3753 OSCAR_STATUS_ID_AVAILABLE, NULL);
3754 purple_prpl_got_user_status(account, bname,
3755 OSCAR_STATUS_ID_MOBILE, NULL);
3757 } else if (aim_ssi_waitingforauth(od->ssi.local,
3758 aim_ssi_itemlist_findparentname(od->ssi.local, bname),
3759 bname)) {
3760 /* Not authorized -- Re-request authorization */
3761 oscar_auth_sendrequest(gc, bname, msg);
3765 /* XXX - Should this be done from AIM accounts, as well? */
3766 if (od->icq)
3767 aim_icq_getalias(od, bname, FALSE, NULL);
3770 void oscar_remove_buddy(PurpleConnection *gc, PurpleBuddy *buddy, PurpleGroup *group) {
3771 OscarData *od = purple_connection_get_protocol_data(gc);
3773 if (od->ssi.received_data) {
3774 const char *gname = purple_group_get_name(group);
3775 const char *bname = purple_buddy_get_name(buddy);
3776 purple_debug_info("oscar",
3777 "ssi: deleting buddy %s from group %s\n", bname, gname);
3778 aim_ssi_delbuddy(od, bname, gname);
3782 void oscar_move_buddy(PurpleConnection *gc, const char *name, const char *old_group, const char *new_group) {
3783 OscarData *od = purple_connection_get_protocol_data(gc);
3785 if (od->ssi.received_data && !purple_strequal(old_group, new_group)) {
3786 purple_debug_info("oscar",
3787 "ssi: moving buddy %s from group %s to group %s\n", name, old_group, new_group);
3788 aim_ssi_movebuddy(od, old_group, new_group, name);
3792 void oscar_alias_buddy(PurpleConnection *gc, const char *name, const char *alias) {
3793 OscarData *od = purple_connection_get_protocol_data(gc);
3795 if (od->ssi.received_data) {
3796 char *gname = aim_ssi_itemlist_findparentname(od->ssi.local, name);
3797 if (gname) {
3798 purple_debug_info("oscar",
3799 "ssi: changing the alias for buddy %s to %s\n", name, alias ? alias : "(none)");
3800 aim_ssi_aliasbuddy(od, gname, name, alias);
3806 * FYI, the OSCAR SSI code removes empty groups automatically.
3808 void oscar_rename_group(PurpleConnection *gc, const char *old_name, PurpleGroup *group, GList *moved_buddies) {
3809 OscarData *od = purple_connection_get_protocol_data(gc);
3811 if (od->ssi.received_data) {
3812 const char *gname = purple_group_get_name(group);
3813 if (aim_ssi_itemlist_finditem(od->ssi.local, gname, NULL, AIM_SSI_TYPE_GROUP)) {
3814 GList *cur, *groups = NULL;
3815 PurpleAccount *account = purple_connection_get_account(gc);
3817 /* Make a list of what the groups each buddy is in */
3818 for (cur = moved_buddies; cur != NULL; cur = cur->next) {
3819 PurpleBlistNode *node = cur->data;
3820 /* node is PurpleBuddy, parent is a PurpleContact.
3821 * We must go two levels up to get the Group */
3822 groups = g_list_append(groups,
3823 purple_buddy_get_group((PurpleBuddy*)node));
3826 purple_account_remove_buddies(account, moved_buddies, groups);
3827 purple_account_add_buddies(account, moved_buddies);
3828 g_list_free(groups);
3829 purple_debug_info("oscar",
3830 "ssi: moved all buddies from group %s to %s\n", old_name, gname);
3831 } else {
3832 aim_ssi_rename_group(od, old_name, gname);
3833 purple_debug_info("oscar",
3834 "ssi: renamed group %s to %s\n", old_name, gname);
3839 void oscar_remove_group(PurpleConnection *gc, PurpleGroup *group)
3841 aim_ssi_delgroup(purple_connection_get_protocol_data(gc), purple_group_get_name(group));
3844 static gboolean purple_ssi_rerequestdata(gpointer data) {
3845 OscarData *od = data;
3847 aim_ssi_reqdata(od);
3849 return TRUE;
3852 static int purple_ssi_parseerr(OscarData *od, FlapConnection *conn, FlapFrame *fr, ...) {
3853 PurpleConnection *gc = od->gc;
3854 va_list ap;
3855 guint16 reason;
3857 va_start(ap, fr);
3858 reason = (guint16)va_arg(ap, unsigned int);
3859 va_end(ap);
3861 purple_debug_error("oscar", "ssi: SNAC error %hu\n", reason);
3863 if (reason == 0x0005) {
3864 if (od->getblisttimer > 0)
3865 purple_timeout_remove(od->getblisttimer);
3866 else
3867 /* We only show this error the first time it happens */
3868 purple_notify_error(gc, NULL,
3869 _("Unable to Retrieve Buddy List"),
3870 _("The AIM servers were temporarily unable to send "
3871 "your buddy list. Your buddy list is not lost, and "
3872 "will probably become available in a few minutes."));
3873 od->getblisttimer = purple_timeout_add_seconds(30, purple_ssi_rerequestdata, od);
3874 return 1;
3877 return 1;
3880 static int purple_ssi_parserights(OscarData *od, FlapConnection *conn, FlapFrame *fr, ...) {
3881 int i;
3882 va_list ap;
3883 int numtypes;
3884 guint16 *maxitems;
3885 GString *msg;
3887 va_start(ap, fr);
3888 numtypes = va_arg(ap, int);
3889 maxitems = va_arg(ap, guint16 *);
3890 va_end(ap);
3892 msg = g_string_new("ssi rights:");
3893 for (i=0; i<numtypes; i++)
3894 g_string_append_printf(msg, " max type 0x%04x=%hd,", i, maxitems[i]);
3895 g_string_append(msg, "\n");
3896 purple_debug_misc("oscar", "%s", msg->str);
3897 g_string_free(msg, TRUE);
3899 if (numtypes >= 0)
3900 od->rights.maxbuddies = maxitems[0];
3901 if (numtypes >= 1)
3902 od->rights.maxgroups = maxitems[1];
3903 if (numtypes >= 2)
3904 od->rights.maxpermits = maxitems[2];
3905 if (numtypes >= 3)
3906 od->rights.maxdenies = maxitems[3];
3908 return 1;
3911 static int purple_ssi_parselist(OscarData *od, FlapConnection *conn, FlapFrame *fr, ...)
3913 PurpleConnection *gc;
3914 PurpleAccount *account;
3915 PurpleGroup *g;
3916 PurpleBuddy *b;
3917 GSList *cur, *next, *buddies;
3918 struct aim_ssi_item *curitem;
3919 guint32 tmp;
3920 PurpleStoredImage *img;
3921 va_list ap;
3922 guint16 deny_entry_type = aim_ssi_getdenyentrytype(od);
3924 gc = od->gc;
3925 od = purple_connection_get_protocol_data(gc);
3926 account = purple_connection_get_account(gc);
3928 va_start(ap, fr);
3929 va_arg(ap, int); /* guint16 fmtver */
3930 va_arg(ap, int); /* guint16 numitems */
3931 va_arg(ap, guint32); /* timestamp */
3932 va_end(ap);
3934 /* Don't attempt to re-request our buddy list later */
3935 if (od->getblisttimer != 0) {
3936 purple_timeout_remove(od->getblisttimer);
3937 od->getblisttimer = 0;
3940 purple_debug_info("oscar", "ssi: syncing local list and server list\n");
3942 /* Clean the buddy list */
3943 aim_ssi_cleanlist(od);
3945 /*** Begin code for pruning buddies from local list if they're not in server list ***/
3947 /* Buddies */
3948 cur = NULL;
3949 for (buddies = purple_find_buddies(account, NULL);
3950 buddies;
3951 buddies = g_slist_delete_link(buddies, buddies))
3953 PurpleGroup *g;
3954 const char *gname;
3955 const char *bname;
3957 b = buddies->data;
3958 g = purple_buddy_get_group(b);
3959 gname = purple_group_get_name(g);
3960 bname = purple_buddy_get_name(b);
3962 if (aim_ssi_itemlist_exists(od->ssi.local, bname)) {
3963 /* If the buddy is an ICQ user then load his nickname */
3964 const char *servernick = purple_blist_node_get_string((PurpleBlistNode*)b, "servernick");
3965 char *alias;
3966 const char *balias;
3967 if (servernick)
3968 serv_got_alias(gc, bname, servernick);
3970 /* Store local alias on server */
3971 alias = aim_ssi_getalias(od->ssi.local, gname, bname);
3972 balias = purple_buddy_get_local_buddy_alias(b);
3973 if (!alias && balias && *balias)
3974 aim_ssi_aliasbuddy(od, gname, bname, balias);
3975 g_free(alias);
3976 } else {
3977 purple_debug_info("oscar",
3978 "ssi: removing buddy %s from local list\n", bname);
3979 /* Queue the buddy for removal from the local list */
3980 cur = g_slist_prepend(cur, b);
3983 while (cur != NULL) {
3984 purple_blist_remove_buddy(cur->data);
3985 cur = g_slist_delete_link(cur, cur);
3988 /* Permit list (ICQ doesn't have one) */
3989 if (!od->icq) {
3990 next = account->permit;
3991 while (next != NULL) {
3992 cur = next;
3993 next = next->next;
3994 if (!aim_ssi_itemlist_finditem(od->ssi.local, NULL, cur->data, AIM_SSI_TYPE_PERMIT)) {
3995 purple_debug_info("oscar",
3996 "ssi: removing permit %s from local list\n", (const char *)cur->data);
3997 purple_privacy_permit_remove(account, cur->data, TRUE);
4002 /* Deny list */
4003 next = account->deny;
4004 while (next != NULL) {
4005 cur = next;
4006 next = next->next;
4007 if (!aim_ssi_itemlist_finditem(od->ssi.local, NULL, cur->data, deny_entry_type)) {
4008 purple_debug_info("oscar",
4009 "ssi: removing deny %s from local list\n", (const char *)cur->data);
4010 purple_privacy_deny_remove(account, cur->data, TRUE);
4014 /* Presence settings (idle time visibility) */
4015 tmp = aim_ssi_getpresence(od->ssi.local);
4016 if (tmp != 0xFFFFFFFF) {
4017 const char *idle_reporting_pref;
4018 gboolean report_idle;
4020 idle_reporting_pref = purple_prefs_get_string("/purple/away/idle_reporting");
4021 report_idle = !purple_strequal(idle_reporting_pref, "none");
4023 if (report_idle)
4024 aim_ssi_setpresence(od, tmp | AIM_SSI_PRESENCE_FLAG_SHOWIDLE);
4025 else
4026 aim_ssi_setpresence(od, tmp & ~AIM_SSI_PRESENCE_FLAG_SHOWIDLE);
4029 /*** End code for pruning buddies from local list ***/
4031 /*** Begin code for adding from server list to local list ***/
4033 for (curitem=od->ssi.local; curitem; curitem=curitem->next) {
4034 if (curitem->name && !g_utf8_validate(curitem->name, -1, NULL)) {
4035 /* Got node with invalid UTF-8 in the name. Skip it. */
4036 purple_debug_warning("oscar", "ssi: server list contains item of "
4037 "type 0x%04hx with a non-utf8 name\n", curitem->type);
4038 continue;
4041 switch (curitem->type) {
4042 case AIM_SSI_TYPE_BUDDY: { /* Buddy */
4043 if (curitem->name) {
4044 struct aim_ssi_item *groupitem;
4045 char *gname, *gname_utf8, *alias, *alias_utf8;
4047 groupitem = aim_ssi_itemlist_find(od->ssi.local, curitem->gid, 0x0000);
4048 gname = groupitem ? groupitem->name : NULL;
4049 gname_utf8 = oscar_utf8_try_convert(account, od, gname);
4051 g = purple_find_group(gname_utf8 ? gname_utf8 : _("Orphans"));
4052 if (g == NULL) {
4053 g = purple_group_new(gname_utf8 ? gname_utf8 : _("Orphans"));
4054 purple_blist_add_group(g, NULL);
4057 alias = aim_ssi_getalias(od->ssi.local, gname, curitem->name);
4058 alias_utf8 = oscar_utf8_try_convert(account, od, alias);
4060 b = purple_find_buddy_in_group(account, curitem->name, g);
4061 if (b) {
4062 /* Get server stored alias */
4063 purple_blist_alias_buddy(b, alias_utf8);
4064 } else {
4065 b = purple_buddy_new(account, curitem->name, alias_utf8);
4067 purple_debug_info("oscar",
4068 "ssi: adding buddy %s to group %s to local list\n", curitem->name, gname);
4069 purple_blist_add_buddy(b, NULL, g, NULL);
4072 /* Mobile users should always be online */
4073 if (curitem->name[0] == '+') {
4074 purple_prpl_got_user_status(account,
4075 purple_buddy_get_name(b),
4076 OSCAR_STATUS_ID_AVAILABLE, NULL);
4077 purple_prpl_got_user_status(account,
4078 purple_buddy_get_name(b),
4079 OSCAR_STATUS_ID_MOBILE, NULL);
4082 g_free(gname_utf8);
4083 g_free(alias);
4084 g_free(alias_utf8);
4086 } break;
4088 case AIM_SSI_TYPE_GROUP: { /* Group */
4089 if (curitem->name != NULL && purple_find_group(curitem->name) == NULL) {
4090 g = purple_group_new(curitem->name);
4091 purple_blist_add_group(g, NULL);
4093 } break;
4095 case AIM_SSI_TYPE_PERMIT: { /* Permit buddy (unless we're on ICQ) */
4096 if (!od->icq && curitem->name) {
4097 for (cur = account->permit; (cur && oscar_util_name_compare(curitem->name, cur->data)); cur = cur->next);
4098 if (!cur) {
4099 purple_debug_info("oscar",
4100 "ssi: adding permit buddy %s to local list\n", curitem->name);
4101 purple_privacy_permit_add(account, curitem->name, TRUE);
4104 } break;
4106 case AIM_SSI_TYPE_ICQDENY:
4107 case AIM_SSI_TYPE_DENY: { /* Deny buddy */
4108 if (curitem->type == deny_entry_type && curitem->name) {
4109 for (cur = account->deny; (cur && oscar_util_name_compare(curitem->name, cur->data)); cur = cur->next);
4110 if (!cur) {
4111 purple_debug_info("oscar",
4112 "ssi: adding deny buddy %s to local list\n", curitem->name);
4113 purple_privacy_deny_add(account, curitem->name, TRUE);
4116 } break;
4118 case AIM_SSI_TYPE_PDINFO: { /* Permit/deny setting */
4120 * We don't inherit the permit/deny setting from the server
4121 * for ICQ because, for ICQ, this setting controls who can
4122 * see your online status when you are invisible. Thus it is
4123 * a part of your status and not really related to blocking.
4125 if (!od->icq && curitem->data) {
4126 guint8 perm_deny = aim_ssi_getpermdeny(od->ssi.local);
4127 if (perm_deny != 0 && perm_deny != account->perm_deny)
4129 purple_debug_info("oscar",
4130 "ssi: changing permdeny from %d to %hhu\n", account->perm_deny, perm_deny);
4131 account->perm_deny = perm_deny;
4134 } break;
4136 case AIM_SSI_TYPE_PRESENCEPREFS: { /* Presence setting */
4137 /* We don't want to change Purple's setting because it applies to all accounts */
4138 } break;
4139 } /* End of switch on curitem->type */
4140 } /* End of for loop */
4142 /*** End code for adding from server list to local list ***/
4144 if (od->icq) {
4145 oscar_set_icq_permdeny(account);
4146 } else {
4147 oscar_set_aim_permdeny(gc);
4150 /* Activate SSI */
4151 /* Sending the enable causes other people to be able to see you, and you to see them */
4152 /* Make sure your privacy setting/invisibility is set how you want it before this! */
4153 purple_debug_info("oscar",
4154 "ssi: activating server-stored buddy list\n");
4155 aim_ssi_enable(od);
4158 * Make sure our server-stored icon is updated correctly in
4159 * the event that the local user set a new icon while this
4160 * account was offline.
4162 img = purple_buddy_icons_find_account_icon(account);
4163 oscar_set_icon(gc, img);
4164 purple_imgstore_unref(img);
4167 * If we've already received our bos rights then we're not waiting on
4168 * anything else, so send the server clientready.
4170 if (od->bos.have_rights) {
4171 aim_srv_clientready(od, conn);
4173 /* Request offline messages for AIM and ICQ */
4174 aim_im_reqofflinemsgs(od);
4176 purple_connection_set_state(gc, PURPLE_CONNECTED);
4179 return 1;
4182 static int purple_ssi_parseack(OscarData *od, FlapConnection *conn, FlapFrame *fr, ...) {
4183 PurpleConnection *gc = od->gc;
4184 va_list ap;
4185 struct aim_ssi_tmp *retval;
4187 va_start(ap, fr);
4188 retval = va_arg(ap, struct aim_ssi_tmp *);
4189 va_end(ap);
4191 while (retval) {
4192 purple_debug_misc("oscar",
4193 "ssi: status is 0x%04hx for a 0x%04hx action with name %s\n", retval->ack, retval->action, retval->item ? (retval->item->name ? retval->item->name : "no name") : "no item");
4195 if (retval->ack != 0xffff)
4196 switch (retval->ack) {
4197 case 0x0000: { /* added successfully */
4198 } break;
4200 case 0x000c: { /* you are over the limit, the cheat is to the limit, come on fhqwhgads */
4201 gchar *buf;
4202 buf = g_strdup_printf(_("Unable to add the buddy %s because you have too many buddies in your buddy list. Please remove one and try again."), (retval->name ? retval->name : _("(no name)")));
4203 if ((retval->name != NULL) && !purple_conv_present_error(retval->name, purple_connection_get_account(gc), buf))
4204 purple_notify_error(gc, NULL, _("Unable to Add"), buf);
4205 g_free(buf);
4206 } break;
4208 case 0x000e: { /* buddy requires authorization */
4209 if ((retval->action == SNAC_SUBTYPE_FEEDBAG_ADD) && (retval->name))
4210 oscar_auth_sendrequest(gc, retval->name, NULL);
4211 } break;
4213 default: { /* La la la */
4214 gchar *buf;
4215 purple_debug_error("oscar", "ssi: Action 0x%04hx was unsuccessful with error 0x%04hx\n", retval->action, retval->ack);
4216 buf = g_strdup_printf(_("Unable to add the buddy %s for an unknown reason."),
4217 (retval->name ? retval->name : _("(no name)")));
4218 if ((retval->name != NULL) && !purple_conv_present_error(retval->name, purple_connection_get_account(gc), buf))
4219 purple_notify_error(gc, NULL, _("Unable to Add"), buf);
4220 g_free(buf);
4221 } break;
4224 retval = retval->next;
4227 return 1;
4230 static int
4231 purple_ssi_parseaddmod(OscarData *od, FlapConnection *conn, FlapFrame *fr, ...)
4233 PurpleConnection *gc;
4234 PurpleAccount *account;
4235 char *gname, *gname_utf8, *alias, *alias_utf8;
4236 PurpleBuddy *b;
4237 PurpleGroup *g;
4238 struct aim_ssi_item *ssi_item;
4239 va_list ap;
4240 guint16 snac_subtype, type;
4241 const char *name;
4243 gc = od->gc;
4244 account = purple_connection_get_account(gc);
4246 va_start(ap, fr);
4247 snac_subtype = (guint16)va_arg(ap, int);
4248 type = (guint16)va_arg(ap, int);
4249 name = va_arg(ap, char *);
4250 va_end(ap);
4252 if ((type != 0x0000) || (name == NULL))
4253 return 1;
4255 gname = aim_ssi_itemlist_findparentname(od->ssi.local, name);
4256 gname_utf8 = gname ? oscar_utf8_try_convert(account, od, gname) : NULL;
4258 alias = aim_ssi_getalias(od->ssi.local, gname, name);
4259 alias_utf8 = oscar_utf8_try_convert(account, od, alias);
4260 g_free(alias);
4262 b = purple_find_buddy(account, name);
4263 if (b) {
4265 * You're logged in somewhere else and you aliased one
4266 * of your buddies, so update our local buddy list with
4267 * the person's new alias.
4269 purple_blist_alias_buddy(b, alias_utf8);
4270 } else if (snac_subtype == 0x0008) {
4272 * You're logged in somewhere else and you added a buddy to
4273 * your server list, so add them to your local buddy list.
4275 b = purple_buddy_new(account, name, alias_utf8);
4277 if (!(g = purple_find_group(gname_utf8 ? gname_utf8 : _("Orphans")))) {
4278 g = purple_group_new(gname_utf8 ? gname_utf8 : _("Orphans"));
4279 purple_blist_add_group(g, NULL);
4282 purple_debug_info("oscar",
4283 "ssi: adding buddy %s to group %s to local list\n", name, gname_utf8 ? gname_utf8 : _("Orphans"));
4284 purple_blist_add_buddy(b, NULL, g, NULL);
4286 /* Mobile users should always be online */
4287 if (name[0] == '+') {
4288 purple_prpl_got_user_status(account,
4289 name, OSCAR_STATUS_ID_AVAILABLE, NULL);
4290 purple_prpl_got_user_status(account,
4291 name, OSCAR_STATUS_ID_MOBILE, NULL);
4296 ssi_item = aim_ssi_itemlist_finditem(od->ssi.local,
4297 gname, name, AIM_SSI_TYPE_BUDDY);
4298 if (ssi_item == NULL)
4300 purple_debug_error("oscar", "purple_ssi_parseaddmod: "
4301 "Could not find ssi item for oncoming buddy %s, "
4302 "group %s\n", name, gname);
4305 g_free(gname_utf8);
4306 g_free(alias_utf8);
4308 return 1;
4311 static int purple_ssi_authgiven(OscarData *od, FlapConnection *conn, FlapFrame *fr, ...) {
4312 PurpleConnection *gc = od->gc;
4313 va_list ap;
4314 char *bn;
4315 gchar *dialog_msg, *nombre;
4316 struct name_data *data;
4317 PurpleBuddy *buddy;
4319 va_start(ap, fr);
4320 bn = va_arg(ap, char *);
4321 va_arg(ap, char *); /* msg */
4322 va_end(ap);
4324 purple_debug_info("oscar",
4325 "ssi: %s has given you permission to add him to your buddy list\n", bn);
4327 buddy = purple_find_buddy(purple_connection_get_account(gc), bn);
4328 if (buddy && (purple_buddy_get_alias_only(buddy)))
4329 nombre = g_strdup_printf("%s (%s)", bn, purple_buddy_get_alias_only(buddy));
4330 else
4331 nombre = g_strdup(bn);
4333 dialog_msg = g_strdup_printf(_("The user %s has given you permission to add him or her to your buddy list. Do you want to add this user?"), nombre);
4334 g_free(nombre);
4336 data = g_new(struct name_data, 1);
4337 data->gc = gc;
4338 data->name = g_strdup(bn);
4339 data->nick = (buddy ? g_strdup(purple_buddy_get_alias_only(buddy)) : NULL);
4341 purple_request_yes_no(gc, NULL, _("Authorization Given"), dialog_msg,
4342 PURPLE_DEFAULT_ACTION_NONE,
4343 purple_connection_get_account(gc), bn, NULL,
4344 data,
4345 G_CALLBACK(purple_icq_buddyadd),
4346 G_CALLBACK(oscar_free_name_data));
4347 g_free(dialog_msg);
4349 return 1;
4352 static int purple_ssi_authrequest(OscarData *od, FlapConnection *conn, FlapFrame *fr, ...)
4354 va_list ap;
4355 const char *bn;
4356 char *msg;
4358 va_start(ap, fr);
4359 bn = va_arg(ap, const char *);
4360 msg = va_arg(ap, char *);
4361 va_end(ap);
4363 purple_debug_info("oscar",
4364 "ssi: received authorization request from %s\n", bn);
4366 if (!msg) {
4367 purple_debug_warning("oscar", "Received auth request from %s with "
4368 "empty message\n", bn);
4369 } else if (!g_utf8_validate(msg, -1, NULL)) {
4370 purple_debug_warning("oscar", "Received auth request from %s with "
4371 "invalid UTF-8 message\n", bn);
4372 msg = NULL;
4375 aim_icq_getalias(od, bn, TRUE, msg);
4376 return 1;
4379 static int purple_ssi_authreply(OscarData *od, FlapConnection *conn, FlapFrame *fr, ...) {
4380 PurpleConnection *gc = od->gc;
4381 va_list ap;
4382 char *bn, *msg;
4383 gchar *dialog_msg, *nombre;
4384 guint8 reply;
4385 PurpleBuddy *buddy;
4387 va_start(ap, fr);
4388 bn = va_arg(ap, char *);
4389 reply = (guint8)va_arg(ap, int);
4390 msg = va_arg(ap, char *);
4391 va_end(ap);
4393 purple_debug_info("oscar",
4394 "ssi: received authorization reply from %s. Reply is 0x%04hhx\n", bn, reply);
4396 buddy = purple_find_buddy(purple_connection_get_account(gc), bn);
4397 if (buddy && (purple_buddy_get_alias_only(buddy)))
4398 nombre = g_strdup_printf("%s (%s)", bn, purple_buddy_get_alias_only(buddy));
4399 else
4400 nombre = g_strdup(bn);
4402 if (reply) {
4403 /* Granted */
4404 dialog_msg = g_strdup_printf(_("The user %s has granted your request to add them to your buddy list."), nombre);
4405 purple_notify_info(gc, NULL, _("Authorization Granted"), dialog_msg);
4406 } else {
4407 /* Denied */
4408 dialog_msg = g_strdup_printf(_("The user %s has denied your request to add them to your buddy list for the following reason:\n%s"), nombre, msg ? msg : _("No reason given."));
4409 purple_notify_info(gc, NULL, _("Authorization Denied"), dialog_msg);
4411 g_free(dialog_msg);
4412 g_free(nombre);
4414 return 1;
4417 static int purple_ssi_gotadded(OscarData *od, FlapConnection *conn, FlapFrame *fr, ...) {
4418 PurpleConnection *gc = od->gc;
4419 PurpleAccount *account = purple_connection_get_account(gc);
4420 va_list ap;
4421 char *bn;
4422 PurpleBuddy *buddy;
4424 va_start(ap, fr);
4425 bn = va_arg(ap, char *);
4426 va_end(ap);
4428 buddy = purple_find_buddy(account, bn);
4429 purple_debug_info("oscar", "ssi: %s added you to their buddy list\n", bn);
4430 purple_account_notify_added(account, bn, NULL,
4431 (buddy ? purple_buddy_get_alias_only(buddy) : NULL), NULL);
4433 return 1;
4436 GList *oscar_chat_info(PurpleConnection *gc) {
4437 GList *m = NULL;
4438 struct proto_chat_entry *pce;
4440 pce = g_new0(struct proto_chat_entry, 1);
4441 pce->label = _("_Room:");
4442 pce->identifier = "room";
4443 pce->required = TRUE;
4444 m = g_list_append(m, pce);
4446 pce = g_new0(struct proto_chat_entry, 1);
4447 pce->label = _("_Exchange:");
4448 pce->identifier = "exchange";
4449 pce->required = TRUE;
4450 pce->is_int = TRUE;
4451 pce->min = 4;
4452 pce->max = 20;
4453 m = g_list_append(m, pce);
4455 return m;
4458 GHashTable *oscar_chat_info_defaults(PurpleConnection *gc, const char *chat_name)
4460 GHashTable *defaults;
4462 defaults = g_hash_table_new_full(g_str_hash, g_str_equal, NULL, g_free);
4464 if (chat_name != NULL)
4465 g_hash_table_insert(defaults, "room", g_strdup(chat_name));
4466 g_hash_table_insert(defaults, "exchange", g_strdup("4"));
4468 return defaults;
4471 char *
4472 oscar_get_chat_name(GHashTable *data)
4474 return g_strdup(g_hash_table_lookup(data, "room"));
4477 void
4478 oscar_join_chat(PurpleConnection *gc, GHashTable *data)
4480 OscarData *od = purple_connection_get_protocol_data(gc);
4481 FlapConnection *conn;
4482 char *name, *exchange;
4483 int exchange_int;
4485 name = g_hash_table_lookup(data, "room");
4486 exchange = g_hash_table_lookup(data, "exchange");
4488 g_return_if_fail(name != NULL && *name != '\0');
4489 g_return_if_fail(exchange != NULL);
4491 errno = 0;
4492 exchange_int = strtol(exchange, NULL, 10);
4493 g_return_if_fail(errno == 0);
4495 purple_debug_info("oscar", "Attempting to join chat room %s.\n", name);
4497 if ((conn = flap_connection_getbytype(od, SNAC_FAMILY_CHATNAV)))
4499 purple_debug_info("oscar", "chatnav exists, creating room\n");
4500 aim_chatnav_createroom(od, conn, name, exchange_int);
4501 } else {
4502 /* this gets tricky */
4503 struct create_room *cr = g_new0(struct create_room, 1);
4504 purple_debug_info("oscar", "chatnav does not exist, opening chatnav\n");
4505 cr->exchange = exchange_int;
4506 cr->name = g_strdup(name);
4507 od->create_rooms = g_slist_prepend(od->create_rooms, cr);
4508 aim_srv_requestnew(od, SNAC_FAMILY_CHATNAV);
4512 void
4513 oscar_chat_invite(PurpleConnection *gc, int id, const char *message, const char *name)
4515 OscarData *od = purple_connection_get_protocol_data(gc);
4516 struct chat_connection *ccon = find_oscar_chat(gc, id);
4518 if (ccon == NULL)
4519 return;
4521 aim_im_sendch2_chatinvite(od, name, message ? message : "",
4522 ccon->exchange, ccon->name, 0x0);
4525 void
4526 oscar_chat_leave(PurpleConnection *gc, int id)
4528 PurpleConversation *conv;
4529 struct chat_connection *cc;
4531 conv = purple_find_chat(gc, id);
4533 g_return_if_fail(conv != NULL);
4535 purple_debug_info("oscar", "Leaving chat room %s\n",
4536 purple_conversation_get_name(conv));
4538 cc = find_oscar_chat(gc, purple_conv_chat_get_id(PURPLE_CONV_CHAT(conv)));
4539 flap_connection_schedule_destroy(cc->conn, OSCAR_DISCONNECT_DONE, NULL);
4540 oscar_chat_kill(gc, cc);
4543 int oscar_send_chat(PurpleConnection *gc, int id, const char *message, PurpleMessageFlags flags)
4545 OscarData *od = purple_connection_get_protocol_data(gc);
4546 PurpleConversation *conv = NULL;
4547 struct chat_connection *c = NULL;
4548 char *buf, *buf2, *buf3;
4549 guint16 charset;
4550 char *charsetstr;
4551 gsize len;
4553 if (!(conv = purple_find_chat(gc, id)))
4554 return -EINVAL;
4556 if (!(c = find_oscar_chat_by_conv(gc, conv)))
4557 return -EINVAL;
4559 buf = purple_strdup_withhtml(message);
4561 if (strstr(buf, "<IMG "))
4562 purple_conversation_write(conv, "",
4563 _("Your IM Image was not sent. "
4564 "You cannot send IM Images in AIM chats."),
4565 PURPLE_MESSAGE_ERROR, time(NULL));
4567 buf2 = oscar_encode_im(buf, &len, &charset, &charsetstr);
4569 * Evan S. suggested that maxvis really does mean "number of
4570 * visible characters" and not "number of bytes"
4572 if ((len > c->maxlen) || (len > c->maxvis)) {
4573 /* If the length was too long, try stripping the HTML and then running it back through
4574 * purple_strdup_withhtml() and the encoding process. The result may be shorter. */
4575 g_free(buf2);
4577 buf3 = purple_markup_strip_html(buf);
4578 g_free(buf);
4580 buf = purple_strdup_withhtml(buf3);
4581 g_free(buf3);
4583 buf2 = oscar_encode_im(buf, &len, &charset, &charsetstr);
4585 if ((len > c->maxlen) || (len > c->maxvis)) {
4586 purple_debug_warning("oscar",
4587 "Could not send %s because (%" G_GSIZE_FORMAT " > maxlen %i) or (%" G_GSIZE_FORMAT " > maxvis %i)\n",
4588 buf2, len, c->maxlen, len, c->maxvis);
4589 g_free(buf);
4590 g_free(buf2);
4591 return -E2BIG;
4594 purple_debug_info("oscar", "Sending %s as %s because the original was too long.\n",
4595 message, buf2);
4598 aim_chat_send_im(od, c->conn, 0, buf2, len, charsetstr, "en");
4599 g_free(buf2);
4600 g_free(buf);
4602 return 0;
4605 PurpleMood* oscar_get_purple_moods(PurpleAccount *account)
4607 return icq_get_purple_moods(account);
4610 const char *oscar_list_icon_icq(PurpleAccount *a, PurpleBuddy *b)
4612 const char *name = b ? purple_buddy_get_name(b) : NULL;
4613 if (name && !oscar_util_valid_name_sms(name) && oscar_util_valid_name_icq(name))
4614 return "icq";
4616 return "icq";
4619 const char *oscar_list_icon_aim(PurpleAccount *a, PurpleBuddy *b)
4621 const char *name = b ? purple_buddy_get_name(b) : NULL;
4622 if (name && !oscar_util_valid_name_sms(name) && oscar_util_valid_name_icq(name))
4623 return "icq";
4625 return "aim";
4628 const char *oscar_list_emblem(PurpleBuddy *b)
4630 PurpleConnection *gc = NULL;
4631 OscarData *od = NULL;
4632 PurpleAccount *account = NULL;
4633 PurplePresence *presence;
4634 aim_userinfo_t *userinfo = NULL;
4635 const char *name;
4637 account = purple_buddy_get_account(b);
4638 name = purple_buddy_get_name(b);
4639 if (account != NULL)
4640 gc = purple_account_get_connection(account);
4641 if (gc != NULL)
4642 od = purple_connection_get_protocol_data(gc);
4643 if (od != NULL)
4644 userinfo = aim_locate_finduserinfo(od, name);
4646 presence = purple_buddy_get_presence(b);
4648 if (purple_presence_is_online(presence) == FALSE) {
4649 char *gname;
4650 if ((name) && (od) && (od->ssi.received_data) &&
4651 (gname = aim_ssi_itemlist_findparentname(od->ssi.local, name)) &&
4652 (aim_ssi_waitingforauth(od->ssi.local, gname, name))) {
4653 return "not-authorized";
4657 if (userinfo != NULL ) {
4658 if (userinfo->flags & AIM_FLAG_ADMINISTRATOR)
4659 return "admin";
4660 if (userinfo->flags & AIM_FLAG_ACTIVEBUDDY)
4661 return "bot";
4662 if (userinfo->capabilities & OSCAR_CAPABILITY_SECUREIM)
4663 return "secure";
4664 if (userinfo->icqinfo.status & AIM_ICQ_STATE_BIRTHDAY)
4665 return "birthday";
4667 /* Make the mood icon override anything below this. */
4668 if (purple_presence_is_status_primitive_active(presence, PURPLE_STATUS_MOOD))
4669 return NULL;
4671 if (userinfo->capabilities & OSCAR_CAPABILITY_HIPTOP)
4672 return "hiptop";
4674 return NULL;
4677 void oscar_tooltip_text(PurpleBuddy *b, PurpleNotifyUserInfo *user_info, gboolean full)
4679 PurpleConnection *gc;
4680 PurpleAccount *account;
4681 OscarData *od;
4682 aim_userinfo_t *userinfo;
4684 if (!PURPLE_BUDDY_IS_ONLINE(b))
4685 return;
4687 account = purple_buddy_get_account(b);
4688 gc = purple_account_get_connection(account);
4689 od = purple_connection_get_protocol_data(gc);
4690 userinfo = aim_locate_finduserinfo(od, purple_buddy_get_name(b));
4692 oscar_user_info_append_status(gc, user_info, b, userinfo, /* use_html_status */ FALSE);
4694 if (full)
4695 oscar_user_info_append_extra_info(gc, user_info, b, userinfo);
4698 char *oscar_status_text(PurpleBuddy *b)
4700 PurpleConnection *gc;
4701 PurpleAccount *account;
4702 OscarData *od;
4703 const PurplePresence *presence;
4704 const PurpleStatus *status;
4705 const char *message;
4706 gchar *ret = NULL;
4708 gc = purple_account_get_connection(purple_buddy_get_account(b));
4709 account = purple_connection_get_account(gc);
4710 od = purple_connection_get_protocol_data(gc);
4711 presence = purple_buddy_get_presence(b);
4712 status = purple_presence_get_active_status(presence);
4714 if ((od != NULL) && !purple_presence_is_online(presence))
4716 const char *name = purple_buddy_get_name(b);
4717 char *gname = aim_ssi_itemlist_findparentname(od->ssi.local, name);
4718 if (aim_ssi_waitingforauth(od->ssi.local, gname, name))
4719 ret = g_strdup(_("Not Authorized"));
4720 else
4721 ret = g_strdup(_("Offline"));
4723 else
4725 message = purple_status_get_attr_string(status, "message");
4726 if (message != NULL)
4728 gchar *tmp = oscar_util_format_string(message, purple_account_get_username(account));
4729 ret = purple_markup_escape_text(tmp, -1);
4730 g_free(tmp);
4732 else if (purple_status_is_available(status))
4734 /* Don't show "Available" as status message in case buddy doesn't have a status message */
4736 else
4738 ret = g_strdup(purple_status_get_name(status));
4742 return ret;
4745 void oscar_set_aim_permdeny(PurpleConnection *gc) {
4746 PurpleAccount *account = purple_connection_get_account(gc);
4747 OscarData *od = purple_connection_get_protocol_data(gc);
4750 * Conveniently there is a one-to-one mapping between the
4751 * values of libpurple's PurplePrivacyType and the values used
4752 * by the oscar protocol.
4754 aim_ssi_setpermdeny(od, account->perm_deny);
4757 void oscar_add_permit(PurpleConnection *gc, const char *who) {
4758 OscarData *od = purple_connection_get_protocol_data(gc);
4759 purple_debug_info("oscar", "ssi: About to add a permit\n");
4760 aim_ssi_add_to_private_list(od, who, AIM_SSI_TYPE_PERMIT);
4763 void oscar_add_deny(PurpleConnection *gc, const char *who) {
4764 OscarData *od = purple_connection_get_protocol_data(gc);
4765 purple_debug_info("oscar", "ssi: About to add a deny\n");
4766 aim_ssi_add_to_private_list(od, who, aim_ssi_getdenyentrytype(od));
4769 void oscar_rem_permit(PurpleConnection *gc, const char *who) {
4770 OscarData *od = purple_connection_get_protocol_data(gc);
4771 purple_debug_info("oscar", "ssi: About to delete a permit\n");
4772 aim_ssi_del_from_private_list(od, who, AIM_SSI_TYPE_PERMIT);
4775 void oscar_rem_deny(PurpleConnection *gc, const char *who) {
4776 OscarData *od = purple_connection_get_protocol_data(gc);
4777 purple_debug_info("oscar", "ssi: About to delete a deny\n");
4778 aim_ssi_del_from_private_list(od, who, aim_ssi_getdenyentrytype(od));
4781 GList *
4782 oscar_status_types(PurpleAccount *account)
4784 gboolean is_icq;
4785 GList *status_types = NULL;
4786 PurpleStatusType *type;
4788 g_return_val_if_fail(account != NULL, NULL);
4790 /* Used to flag some statuses as "user settable" or not */
4791 is_icq = oscar_util_valid_name_icq(purple_account_get_username(account));
4793 /* Common status types */
4794 /* Really the available message should only be settable for AIM accounts */
4795 type = purple_status_type_new_with_attrs(PURPLE_STATUS_AVAILABLE,
4796 OSCAR_STATUS_ID_AVAILABLE,
4797 NULL, TRUE, TRUE, FALSE,
4798 "message", _("Message"),
4799 purple_value_new(PURPLE_TYPE_STRING),
4800 "itmsurl", _("iTunes Music Store Link"),
4801 purple_value_new(PURPLE_TYPE_STRING), NULL);
4802 status_types = g_list_prepend(status_types, type);
4804 type = purple_status_type_new_with_attrs(PURPLE_STATUS_AVAILABLE,
4805 OSCAR_STATUS_ID_FREE4CHAT,
4806 _("Free For Chat"), TRUE, is_icq, FALSE,
4807 "message", _("Message"),
4808 purple_value_new(PURPLE_TYPE_STRING), NULL);
4810 status_types = g_list_prepend(status_types, type);
4812 type = purple_status_type_new_with_attrs(PURPLE_STATUS_AVAILABLE,
4813 OSCAR_STATUS_ID_EVIL,
4814 _("Evil"), TRUE, is_icq, FALSE,
4815 "message", _("Message"),
4816 purple_value_new(PURPLE_TYPE_STRING), NULL);
4817 status_types = g_list_prepend(status_types, type);
4820 type = purple_status_type_new_with_attrs(PURPLE_STATUS_AVAILABLE,
4821 OSCAR_STATUS_ID_DEPRESSION,
4822 _("Depression"), TRUE, is_icq, FALSE,
4823 "message", _("Message"),
4824 purple_value_new(PURPLE_TYPE_STRING), NULL);
4825 status_types = g_list_prepend(status_types, type);
4828 type = purple_status_type_new_with_attrs(PURPLE_STATUS_AVAILABLE,
4829 OSCAR_STATUS_ID_ATHOME,
4830 _("At home"), TRUE, is_icq, FALSE,
4831 "message", _("Message"),
4832 purple_value_new(PURPLE_TYPE_STRING), NULL);
4833 status_types = g_list_prepend(status_types, type);
4836 type = purple_status_type_new_with_attrs(PURPLE_STATUS_AVAILABLE,
4837 OSCAR_STATUS_ID_ATWORK,
4838 _("At work"), TRUE, is_icq, FALSE,
4839 "message", _("Message"),
4840 purple_value_new(PURPLE_TYPE_STRING), NULL);
4842 status_types = g_list_prepend(status_types, type);
4845 type = purple_status_type_new_with_attrs(PURPLE_STATUS_AVAILABLE,
4846 OSCAR_STATUS_ID_LUNCH,
4847 _("Lunch"), TRUE, is_icq, FALSE,
4848 "message", _("Message"),
4849 purple_value_new(PURPLE_TYPE_STRING), NULL);
4851 status_types = g_list_prepend(status_types, type);
4853 type = purple_status_type_new_with_attrs(PURPLE_STATUS_AWAY,
4854 OSCAR_STATUS_ID_AWAY,
4855 NULL, TRUE, TRUE, FALSE,
4856 "message", _("Message"),
4857 purple_value_new(PURPLE_TYPE_STRING), NULL);
4858 status_types = g_list_prepend(status_types, type);
4860 type = purple_status_type_new_with_attrs(PURPLE_STATUS_INVISIBLE,
4861 OSCAR_STATUS_ID_INVISIBLE,
4862 NULL, TRUE, TRUE, FALSE,
4863 "message", _("Message"),
4864 purple_value_new(PURPLE_TYPE_STRING), NULL);
4866 status_types = g_list_prepend(status_types, type);
4868 type = purple_status_type_new_full(PURPLE_STATUS_MOBILE, OSCAR_STATUS_ID_MOBILE, NULL, FALSE, FALSE, TRUE);
4869 status_types = g_list_prepend(status_types, type);
4871 /* ICQ-specific status types */
4872 type = purple_status_type_new_with_attrs(PURPLE_STATUS_UNAVAILABLE,
4873 OSCAR_STATUS_ID_OCCUPIED,
4874 _("Occupied"), TRUE, is_icq, FALSE,
4875 "message", _("Message"),
4876 purple_value_new(PURPLE_TYPE_STRING), NULL);
4877 status_types = g_list_prepend(status_types, type);
4879 type = purple_status_type_new_with_attrs(PURPLE_STATUS_UNAVAILABLE,
4880 OSCAR_STATUS_ID_DND,
4881 _("Do Not Disturb"), TRUE, is_icq, FALSE,
4882 "message", _("Message"),
4883 purple_value_new(PURPLE_TYPE_STRING), NULL);
4884 status_types = g_list_prepend(status_types, type);
4886 type = purple_status_type_new_with_attrs(PURPLE_STATUS_EXTENDED_AWAY,
4887 OSCAR_STATUS_ID_NA,
4888 _("Not Available"), TRUE, is_icq, FALSE,
4889 "message", _("Message"),
4890 purple_value_new(PURPLE_TYPE_STRING), NULL);
4891 status_types = g_list_prepend(status_types, type);
4893 type = purple_status_type_new_full(PURPLE_STATUS_OFFLINE,
4894 OSCAR_STATUS_ID_OFFLINE,
4895 NULL, TRUE, TRUE, FALSE);
4896 status_types = g_list_prepend(status_types, type);
4898 type = purple_status_type_new_with_attrs(PURPLE_STATUS_MOOD,
4899 "mood", NULL, TRUE, is_icq, TRUE,
4900 PURPLE_MOOD_NAME, _("Mood Name"), purple_value_new(PURPLE_TYPE_STRING),
4901 PURPLE_MOOD_COMMENT, _("Mood Comment"), purple_value_new(PURPLE_TYPE_STRING),
4902 NULL);
4903 status_types = g_list_prepend(status_types, type);
4905 return g_list_reverse(status_types);
4908 static void oscar_ssi_editcomment(struct name_data *data, const char *text) {
4909 PurpleConnection *gc;
4910 PurpleAccount *account;
4911 OscarData *od;
4912 PurpleBuddy *b;
4913 PurpleGroup *g;
4915 gc = data->gc;
4916 od = purple_connection_get_protocol_data(gc);
4917 account = purple_connection_get_account(gc);
4919 b = purple_find_buddy(account, data->name);
4920 if (b == NULL) {
4921 oscar_free_name_data(data);
4922 return;
4925 g = purple_buddy_get_group(b);
4926 if (g == NULL) {
4927 oscar_free_name_data(data);
4928 return;
4931 aim_ssi_editcomment(od, purple_group_get_name(g), data->name, text);
4932 oscar_free_name_data(data);
4935 static void oscar_buddycb_edit_comment(PurpleBlistNode *node, gpointer ignore) {
4937 PurpleBuddy *buddy;
4938 PurpleConnection *gc;
4939 OscarData *od;
4940 struct name_data *data;
4941 PurpleGroup *g;
4942 char *comment;
4943 gchar *comment_utf8;
4944 gchar *title;
4945 PurpleAccount *account;
4946 const char *name;
4948 g_return_if_fail(PURPLE_BLIST_NODE_IS_BUDDY(node));
4950 buddy = (PurpleBuddy *) node;
4951 name = purple_buddy_get_name(buddy);
4952 account = purple_buddy_get_account(buddy);
4953 gc = purple_account_get_connection(account);
4954 od = purple_connection_get_protocol_data(gc);
4956 if (!(g = purple_buddy_get_group(buddy)))
4957 return;
4959 data = g_new(struct name_data, 1);
4961 comment = aim_ssi_getcomment(od->ssi.local, purple_group_get_name(g), name);
4962 comment_utf8 = comment ? oscar_utf8_try_convert(account, od, comment) : NULL;
4964 data->gc = gc;
4965 data->name = g_strdup(name);
4966 data->nick = g_strdup(purple_buddy_get_alias_only(buddy));
4968 title = g_strdup_printf(_("Buddy Comment for %s"), data->name);
4969 purple_request_input(gc, title, _("Buddy Comment:"), NULL,
4970 comment_utf8, TRUE, FALSE, NULL,
4971 _("_OK"), G_CALLBACK(oscar_ssi_editcomment),
4972 _("_Cancel"), G_CALLBACK(oscar_free_name_data),
4973 account, data->name, NULL,
4974 data);
4975 g_free(title);
4977 g_free(comment);
4978 g_free(comment_utf8);
4981 static void
4982 oscar_ask_directim_yes_cb(struct oscar_ask_directim_data *data)
4984 peer_connection_propose(data->od, OSCAR_CAPABILITY_DIRECTIM, data->who);
4985 g_free(data->who);
4986 g_free(data);
4989 static void
4990 oscar_ask_directim_no_cb(struct oscar_ask_directim_data *data)
4992 g_free(data->who);
4993 g_free(data);
4996 /* This is called from right-click menu on a buddy node. */
4997 static void
4998 oscar_ask_directim(gpointer object, gpointer ignored)
5000 PurpleBlistNode *node;
5001 PurpleBuddy *buddy;
5002 PurpleConnection *gc;
5003 gchar *buf;
5004 struct oscar_ask_directim_data *data;
5005 PurpleAccount *account;
5007 node = object;
5009 g_return_if_fail(PURPLE_BLIST_NODE_IS_BUDDY(node));
5011 buddy = (PurpleBuddy *)node;
5012 account = purple_buddy_get_account(buddy);
5013 gc = purple_account_get_connection(account);
5015 data = g_new0(struct oscar_ask_directim_data, 1);
5016 data->who = g_strdup(purple_buddy_get_name(buddy));
5017 data->od = purple_connection_get_protocol_data(gc);
5018 buf = g_strdup_printf(_("You have selected to open a Direct IM connection with %s."),
5019 data->who);
5021 purple_request_action(gc, NULL, buf,
5022 _("Because this reveals your IP address, it "
5023 "may be considered a security risk. Do you "
5024 "wish to continue?"),
5025 0, /* Default action is "connect" */
5026 account, data->who, NULL,
5027 data, 2,
5028 _("C_onnect"), G_CALLBACK(oscar_ask_directim_yes_cb),
5029 _("_Cancel"), G_CALLBACK(oscar_ask_directim_no_cb));
5030 g_free(buf);
5033 static void
5034 oscar_close_directim(gpointer object, gpointer ignored)
5036 PurpleBlistNode *node;
5037 PurpleBuddy *buddy;
5038 PurpleAccount *account;
5039 PurpleConnection *gc;
5040 PurpleConversation *conv;
5041 OscarData *od;
5042 PeerConnection *conn;
5043 const char *name;
5045 node = object;
5047 g_return_if_fail(PURPLE_BLIST_NODE_IS_BUDDY(node));
5049 buddy = (PurpleBuddy*)node;
5050 name = purple_buddy_get_name(buddy);
5051 account = purple_buddy_get_account(buddy);
5052 gc = purple_account_get_connection(account);
5053 od = gc->proto_data;
5054 conn = peer_connection_find_by_type(od, name, OSCAR_CAPABILITY_DIRECTIM);
5056 if (conn != NULL)
5058 if (!conn->ready)
5059 aim_im_sendch2_cancel(conn);
5061 peer_connection_destroy(conn, OSCAR_DISCONNECT_LOCAL_CLOSED, NULL);
5063 /* OSCAR_DISCONNECT_LOCAL_CLOSED doesn't write anything to the convo
5064 * window. Let the user know that we cancelled the Direct IM. */
5065 conv = purple_conversation_new(PURPLE_CONV_TYPE_IM, account, name);
5066 purple_conversation_write(conv, NULL, _("You closed the connection."),
5067 PURPLE_MESSAGE_SYSTEM, time(NULL));
5071 static void oscar_get_icqxstatusmsg(PurpleBlistNode *node, gpointer ignore)
5073 PurpleBuddy *buddy;
5074 PurpleConnection *gc;
5075 OscarData *od;
5076 PurpleAccount *account;
5077 const char *bname;
5079 g_return_if_fail(PURPLE_BLIST_NODE_IS_BUDDY(node));
5081 buddy = (PurpleBuddy *)node;
5082 bname = purple_buddy_get_name(buddy);
5084 account = purple_buddy_get_account(buddy);
5085 gc = purple_account_get_connection(account);
5086 od = purple_connection_get_protocol_data(gc);
5088 purple_debug_info("oscar", "Manual X-Status Get From %s to %s:\n", bname, purple_account_get_username(account));
5090 icq_im_xstatus_request(od, bname);
5093 static void
5094 oscar_get_aim_info_cb(PurpleBlistNode *node, gpointer ignore)
5096 PurpleBuddy *buddy;
5097 PurpleConnection *gc;
5099 g_return_if_fail(PURPLE_BLIST_NODE_IS_BUDDY(node));
5101 buddy = (PurpleBuddy *)node;
5102 gc = purple_account_get_connection(purple_buddy_get_account(buddy));
5104 aim_locate_getinfoshort(purple_connection_get_protocol_data(gc),
5105 purple_buddy_get_name(buddy), 0x00000003);
5108 static GList *
5109 oscar_buddy_menu(PurpleBuddy *buddy) {
5110 PurpleConnection *gc;
5111 OscarData *od;
5112 GList *menu;
5113 PurpleMenuAction *act;
5114 aim_userinfo_t *userinfo;
5115 PurpleAccount *account;
5116 const char *bname = purple_buddy_get_name(buddy);
5118 account = purple_buddy_get_account(buddy);
5119 gc = purple_account_get_connection(account);
5120 od = purple_connection_get_protocol_data(gc);
5121 userinfo = aim_locate_finduserinfo(od, bname);
5122 menu = NULL;
5124 if (od->icq && oscar_util_valid_name_icq(bname))
5126 act = purple_menu_action_new(_("Get AIM Info"),
5127 PURPLE_CALLBACK(oscar_get_aim_info_cb),
5128 NULL, NULL);
5129 menu = g_list_prepend(menu, act);
5132 if (purple_buddy_get_group(buddy) != NULL)
5134 /* We only do this if the user is in our buddy list */
5135 act = purple_menu_action_new(_("Edit Buddy Comment"),
5136 PURPLE_CALLBACK(oscar_buddycb_edit_comment),
5137 NULL, NULL);
5138 menu = g_list_prepend(menu, act);
5141 if (od->icq)
5143 act = purple_menu_action_new(_("Get X-Status Msg"),
5144 PURPLE_CALLBACK(oscar_get_icqxstatusmsg),
5145 NULL, NULL);
5146 menu = g_list_prepend(menu, act);
5147 menu = g_list_prepend(menu, create_visibility_menu_item(od, bname));
5150 if (userinfo &&
5151 oscar_util_name_compare(purple_account_get_username(account), bname) &&
5152 PURPLE_BUDDY_IS_ONLINE(buddy))
5154 PeerConnection *conn;
5155 conn = peer_connection_find_by_type(od, bname, OSCAR_CAPABILITY_DIRECTIM);
5157 if (userinfo->capabilities & OSCAR_CAPABILITY_DIRECTIM)
5159 if (conn)
5161 act = purple_menu_action_new(_("End Direct IM Session"),
5162 PURPLE_CALLBACK(oscar_close_directim),
5163 NULL, NULL);
5165 else
5167 act = purple_menu_action_new(_("Direct IM"),
5168 PURPLE_CALLBACK(oscar_ask_directim),
5169 NULL, NULL);
5171 menu = g_list_prepend(menu, act);
5175 if (od->ssi.received_data && purple_buddy_get_group(buddy) != NULL)
5178 * We only do this if the user is in our buddy list and we're
5179 * waiting for authorization.
5181 char *gname;
5182 gname = aim_ssi_itemlist_findparentname(od->ssi.local, bname);
5183 if (gname && aim_ssi_waitingforauth(od->ssi.local, gname, bname))
5185 act = purple_menu_action_new(_("Re-request Authorization"),
5186 PURPLE_CALLBACK(oscar_auth_sendrequest_menu),
5187 NULL, NULL);
5188 menu = g_list_prepend(menu, act);
5192 menu = g_list_reverse(menu);
5194 return menu;
5198 GList *oscar_blist_node_menu(PurpleBlistNode *node) {
5199 if(PURPLE_BLIST_NODE_IS_BUDDY(node)) {
5200 return oscar_buddy_menu((PurpleBuddy *) node);
5201 } else {
5202 return NULL;
5206 static void
5207 oscar_icq_privacy_opts(PurpleConnection *gc, PurpleRequestFields *fields)
5209 OscarData *od = purple_connection_get_protocol_data(gc);
5210 PurpleAccount *account = purple_connection_get_account(gc);
5211 PurpleRequestField *f;
5212 gboolean auth, web_aware;
5214 f = purple_request_fields_get_field(fields, "authorization");
5215 auth = purple_request_field_bool_get_value(f);
5217 f = purple_request_fields_get_field(fields, "web_aware");
5218 web_aware = purple_request_field_bool_get_value(f);
5220 purple_account_set_bool(account, "authorization", auth);
5221 purple_account_set_bool(account, "web_aware", web_aware);
5223 oscar_set_extended_status(gc);
5224 aim_icq_setsecurity(od, auth, web_aware);
5227 static void
5228 oscar_show_icq_privacy_opts(PurplePluginAction *action)
5230 PurpleConnection *gc = (PurpleConnection *) action->context;
5231 PurpleAccount *account = purple_connection_get_account(gc);
5232 PurpleRequestFields *fields;
5233 PurpleRequestFieldGroup *g;
5234 PurpleRequestField *f;
5235 gboolean auth, web_aware;
5237 auth = purple_account_get_bool(account, "authorization", OSCAR_DEFAULT_AUTHORIZATION);
5238 web_aware = purple_account_get_bool(account, "web_aware", OSCAR_DEFAULT_WEB_AWARE);
5240 fields = purple_request_fields_new();
5242 g = purple_request_field_group_new(NULL);
5244 f = purple_request_field_bool_new("authorization", _("Require authorization"), auth);
5245 purple_request_field_group_add_field(g, f);
5247 f = purple_request_field_bool_new("web_aware", _("Web aware (enabling this will cause you to receive SPAM!)"), web_aware);
5248 purple_request_field_group_add_field(g, f);
5250 purple_request_fields_add_group(fields, g);
5252 purple_request_fields(gc, _("ICQ Privacy Options"), _("ICQ Privacy Options"),
5253 NULL, fields,
5254 _("OK"), G_CALLBACK(oscar_icq_privacy_opts),
5255 _("Cancel"), NULL,
5256 purple_connection_get_account(gc), NULL, NULL,
5257 gc);
5260 static void oscar_confirm_account(PurplePluginAction *action)
5262 PurpleConnection *gc;
5263 OscarData *od;
5264 FlapConnection *conn;
5266 gc = (PurpleConnection *)action->context;
5267 od = purple_connection_get_protocol_data(gc);
5269 conn = flap_connection_getbytype(od, SNAC_FAMILY_ADMIN);
5270 if (conn != NULL) {
5271 aim_admin_reqconfirm(od, conn);
5272 } else {
5273 od->conf = TRUE;
5274 aim_srv_requestnew(od, SNAC_FAMILY_ADMIN);
5278 static void oscar_show_email(PurplePluginAction *action)
5280 PurpleConnection *gc = (PurpleConnection *) action->context;
5281 OscarData *od = purple_connection_get_protocol_data(gc);
5282 FlapConnection *conn = flap_connection_getbytype(od, SNAC_FAMILY_ADMIN);
5284 if (conn) {
5285 aim_admin_getinfo(od, conn, 0x11);
5286 } else {
5287 od->reqemail = TRUE;
5288 aim_srv_requestnew(od, SNAC_FAMILY_ADMIN);
5292 static void oscar_change_email(PurpleConnection *gc, const char *email)
5294 OscarData *od = purple_connection_get_protocol_data(gc);
5295 FlapConnection *conn = flap_connection_getbytype(od, SNAC_FAMILY_ADMIN);
5297 if (conn) {
5298 aim_admin_setemail(od, conn, email);
5299 } else {
5300 od->setemail = TRUE;
5301 od->email = g_strdup(email);
5302 aim_srv_requestnew(od, SNAC_FAMILY_ADMIN);
5306 static void oscar_show_change_email(PurplePluginAction *action)
5308 PurpleConnection *gc = (PurpleConnection *) action->context;
5309 purple_request_input(gc, NULL, _("Change Address To:"), NULL, NULL,
5310 FALSE, FALSE, NULL,
5311 _("_OK"), G_CALLBACK(oscar_change_email),
5312 _("_Cancel"), NULL,
5313 purple_connection_get_account(gc), NULL, NULL,
5314 gc);
5317 static void oscar_show_awaitingauth(PurplePluginAction *action)
5319 PurpleConnection *gc = (PurpleConnection *) action->context;
5320 OscarData *od = purple_connection_get_protocol_data(gc);
5321 PurpleAccount *account = purple_connection_get_account(gc);
5322 GSList *buddies, *filtered_buddies, *cur;
5323 gchar *text;
5325 buddies = purple_find_buddies(account, NULL);
5326 filtered_buddies = NULL;
5327 for (cur = buddies; cur != NULL; cur = cur->next) {
5328 PurpleBuddy *buddy;
5329 const gchar *bname, *gname;
5331 buddy = cur->data;
5332 bname = purple_buddy_get_name(buddy);
5333 gname = purple_group_get_name(purple_buddy_get_group(buddy));
5334 if (aim_ssi_waitingforauth(od->ssi.local, gname, bname)) {
5335 filtered_buddies = g_slist_prepend(filtered_buddies, buddy);
5339 g_slist_free(buddies);
5341 filtered_buddies = g_slist_reverse(filtered_buddies);
5342 text = oscar_format_buddies(filtered_buddies, _("you are not waiting for authorization"));
5343 g_slist_free(filtered_buddies);
5345 purple_notify_formatted(gc, NULL, _("You are awaiting authorization from "
5346 "the following buddies"), _("You can re-request "
5347 "authorization from these buddies by "
5348 "right-clicking on them and selecting "
5349 "\"Re-request Authorization.\""), text, NULL, NULL);
5350 g_free(text);
5353 static void search_by_email_cb(PurpleConnection *gc, const char *email)
5355 OscarData *od = purple_connection_get_protocol_data(gc);
5357 aim_search_address(od, email);
5360 static void oscar_show_find_email(PurplePluginAction *action)
5362 PurpleConnection *gc = (PurpleConnection *) action->context;
5363 purple_request_input(gc, _("Find Buddy by Email"),
5364 _("Search for a buddy by email address"),
5365 _("Type the email address of the buddy you are "
5366 "searching for."),
5367 NULL, FALSE, FALSE, NULL,
5368 _("_Search"), G_CALLBACK(search_by_email_cb),
5369 _("_Cancel"), NULL,
5370 purple_connection_get_account(gc), NULL, NULL,
5371 gc);
5374 static void oscar_show_set_info(PurplePluginAction *action)
5376 PurpleConnection *gc = (PurpleConnection *) action->context;
5377 purple_account_request_change_user_info(purple_connection_get_account(gc));
5380 static void oscar_show_set_info_icqurl(PurplePluginAction *action)
5382 PurpleConnection *gc = (PurpleConnection *) action->context;
5383 purple_notify_uri(gc, "http://www.icq.com/whitepages/user_details.php");
5386 static void oscar_change_pass(PurplePluginAction *action)
5388 PurpleConnection *gc = (PurpleConnection *) action->context;
5389 purple_account_request_change_password(purple_connection_get_account(gc));
5393 * Only used when connecting with the old-style BUCP login.
5395 static void oscar_show_chpassurl(PurplePluginAction *action)
5397 PurpleConnection *gc = (PurpleConnection *) action->context;
5398 OscarData *od = purple_connection_get_protocol_data(gc);
5399 gchar *substituted = purple_strreplace(od->authinfo->chpassurl, "%s", purple_account_get_username(purple_connection_get_account(gc)));
5400 purple_notify_uri(gc, substituted);
5401 g_free(substituted);
5404 static void oscar_show_imforwardingurl(PurplePluginAction *action)
5406 PurpleConnection *gc = (PurpleConnection *) action->context;
5407 purple_notify_uri(gc, "http://mymobile.aol.com/dbreg/register?action=imf&clientID=1");
5410 void oscar_set_icon(PurpleConnection *gc, PurpleStoredImage *img)
5412 OscarData *od = purple_connection_get_protocol_data(gc);
5414 if (img == NULL) {
5415 aim_ssi_delicon(od);
5416 } else {
5417 PurpleCipherContext *context;
5418 guchar md5[16];
5419 gconstpointer data = purple_imgstore_get_data(img);
5420 size_t len = purple_imgstore_get_size(img);
5422 context = purple_cipher_context_new_by_name("md5", NULL);
5423 purple_cipher_context_append(context, data, len);
5424 purple_cipher_context_digest(context, 16, md5, NULL);
5425 purple_cipher_context_destroy(context);
5427 aim_ssi_seticon(od, md5, 16);
5432 * Called by the Purple core to determine whether or not we're
5433 * allowed to send a file to this user.
5435 gboolean
5436 oscar_can_receive_file(PurpleConnection *gc, const char *who)
5438 OscarData *od;
5439 PurpleAccount *account;
5441 od = purple_connection_get_protocol_data(gc);
5442 account = purple_connection_get_account(gc);
5444 if (od != NULL)
5446 aim_userinfo_t *userinfo;
5447 userinfo = aim_locate_finduserinfo(od, who);
5450 * Don't allowing sending a file to a user that does not support
5451 * file transfer, and don't allow sending to ourselves.
5453 if (((userinfo == NULL) ||
5454 (userinfo->capabilities & OSCAR_CAPABILITY_SENDFILE)) &&
5455 oscar_util_name_compare(who, purple_account_get_username(account)))
5457 return TRUE;
5461 return FALSE;
5464 PurpleXfer *
5465 oscar_new_xfer(PurpleConnection *gc, const char *who)
5467 PurpleXfer *xfer;
5468 OscarData *od;
5469 PurpleAccount *account;
5470 PeerConnection *conn;
5472 od = purple_connection_get_protocol_data(gc);
5473 account = purple_connection_get_account(gc);
5475 xfer = purple_xfer_new(account, PURPLE_XFER_SEND, who);
5476 if (xfer)
5478 purple_xfer_ref(xfer);
5479 purple_xfer_set_init_fnc(xfer, peer_oft_sendcb_init);
5480 purple_xfer_set_cancel_send_fnc(xfer, peer_oft_cb_generic_cancel);
5481 purple_xfer_set_request_denied_fnc(xfer, peer_oft_cb_generic_cancel);
5482 purple_xfer_set_ack_fnc(xfer, peer_oft_sendcb_ack);
5484 conn = peer_connection_new(od, OSCAR_CAPABILITY_SENDFILE, who);
5485 conn->flags |= PEER_CONNECTION_FLAG_INITIATED_BY_ME;
5486 conn->flags |= PEER_CONNECTION_FLAG_APPROVED;
5487 aim_icbm_makecookie(conn->cookie);
5488 conn->xfer = xfer;
5489 xfer->data = conn;
5492 return xfer;
5496 * Called by the Purple core when the user indicates that a
5497 * file is to be sent to a special someone.
5499 void
5500 oscar_send_file(PurpleConnection *gc, const char *who, const char *file)
5502 PurpleXfer *xfer;
5504 xfer = oscar_new_xfer(gc, who);
5506 if (file != NULL)
5507 purple_xfer_request_accepted(xfer, file);
5508 else
5509 purple_xfer_request(xfer);
5512 GList *
5513 oscar_actions(PurplePlugin *plugin, gpointer context)
5515 PurpleConnection *gc = (PurpleConnection *) context;
5516 OscarData *od = purple_connection_get_protocol_data(gc);
5517 GList *menu = NULL;
5518 PurplePluginAction *act;
5520 act = purple_plugin_action_new(_("Set User Info..."),
5521 oscar_show_set_info);
5522 menu = g_list_prepend(menu, act);
5524 if (od->icq)
5526 act = purple_plugin_action_new(_("Set User Info (web)..."),
5527 oscar_show_set_info_icqurl);
5528 menu = g_list_prepend(menu, act);
5531 act = purple_plugin_action_new(_("Change Password..."),
5532 oscar_change_pass);
5533 menu = g_list_prepend(menu, act);
5535 if (od->authinfo != NULL && od->authinfo->chpassurl != NULL)
5537 /* This only happens when connecting with the old-style BUCP login */
5538 act = purple_plugin_action_new(_("Change Password (web)"),
5539 oscar_show_chpassurl);
5540 menu = g_list_prepend(menu, act);
5543 if (!od->icq)
5545 act = purple_plugin_action_new(_("Configure IM Forwarding (web)"),
5546 oscar_show_imforwardingurl);
5547 menu = g_list_prepend(menu, act);
5550 menu = g_list_prepend(menu, NULL);
5552 if (od->icq)
5554 /* ICQ actions */
5555 act = purple_plugin_action_new(_("Set Privacy Options..."),
5556 oscar_show_icq_privacy_opts);
5557 menu = g_list_prepend(menu, act);
5559 act = purple_plugin_action_new(_("Show Visible List"), oscar_show_visible_list);
5560 menu = g_list_prepend(menu, act);
5562 act = purple_plugin_action_new(_("Show Invisible List"), oscar_show_invisible_list);
5563 menu = g_list_prepend(menu, act);
5565 else
5567 /* AIM actions */
5568 act = purple_plugin_action_new(_("Confirm Account"),
5569 oscar_confirm_account);
5570 menu = g_list_prepend(menu, act);
5572 act = purple_plugin_action_new(_("Display Currently Registered Email Address"),
5573 oscar_show_email);
5574 menu = g_list_prepend(menu, act);
5576 act = purple_plugin_action_new(_("Change Currently Registered Email Address..."),
5577 oscar_show_change_email);
5578 menu = g_list_prepend(menu, act);
5581 menu = g_list_prepend(menu, NULL);
5583 act = purple_plugin_action_new(_("Show Buddies Awaiting Authorization"),
5584 oscar_show_awaitingauth);
5585 menu = g_list_prepend(menu, act);
5587 menu = g_list_prepend(menu, NULL);
5589 act = purple_plugin_action_new(_("Search for Buddy by Email Address..."),
5590 oscar_show_find_email);
5591 menu = g_list_prepend(menu, act);
5593 menu = g_list_reverse(menu);
5595 return menu;
5598 void oscar_change_passwd(PurpleConnection *gc, const char *old, const char *new)
5600 OscarData *od = purple_connection_get_protocol_data(gc);
5602 if (od->icq) {
5603 aim_icq_changepasswd(od, new);
5604 } else {
5605 FlapConnection *conn;
5606 conn = flap_connection_getbytype(od, SNAC_FAMILY_ADMIN);
5607 if (conn) {
5608 aim_admin_changepasswd(od, conn, new, old);
5609 } else {
5610 od->chpass = TRUE;
5611 od->oldp = g_strdup(old);
5612 od->newp = g_strdup(new);
5613 aim_srv_requestnew(od, SNAC_FAMILY_ADMIN);
5618 void
5619 oscar_convo_closed(PurpleConnection *gc, const char *who)
5621 OscarData *od;
5622 PeerConnection *conn;
5624 od = purple_connection_get_protocol_data(gc);
5625 conn = peer_connection_find_by_type(od, who, OSCAR_CAPABILITY_DIRECTIM);
5627 if (conn != NULL)
5629 if (!conn->ready)
5630 aim_im_sendch2_cancel(conn);
5632 peer_connection_destroy(conn, OSCAR_DISCONNECT_LOCAL_CLOSED, NULL);
5636 const char *
5637 oscar_normalize(const PurpleAccount *account, const char *str)
5639 static char buf[BUF_LEN];
5640 char *tmp1, *tmp2;
5641 int i, j;
5643 g_return_val_if_fail(str != NULL, NULL);
5645 /* copy str to buf and skip all blanks */
5646 i = 0;
5647 for (j = 0; str[j]; j++) {
5648 if (str[j] != ' ') {
5649 buf[i++] = str[j];
5650 if (i >= BUF_LEN - 1)
5651 break;
5654 buf[i] = '\0';
5656 tmp1 = g_utf8_strdown(buf, -1);
5657 tmp2 = g_utf8_normalize(tmp1, -1, G_NORMALIZE_DEFAULT);
5658 if (strlen(tmp2) > sizeof(buf) - 1) {
5659 purple_debug_error("oscar", "normalized string exceeds buffer length!\n");
5661 g_strlcpy(buf, tmp2, sizeof(buf));
5662 g_free(tmp2);
5663 g_free(tmp1);
5665 return buf;
5668 gboolean
5669 oscar_offline_message(const PurpleBuddy *buddy)
5671 return TRUE;
5674 /* TODO: Find somewhere to put this instead of including it in a bunch of places.
5675 * Maybe just change purple_accounts_find() to return anything for the prpl if there is no acct_id.
5677 static PurpleAccount *find_acct(const char *prpl, const char *acct_id)
5679 PurpleAccount *acct = NULL;
5681 /* If we have a specific acct, use it */
5682 if (acct_id) {
5683 acct = purple_accounts_find(acct_id, prpl);
5684 if (acct && !purple_account_is_connected(acct))
5685 acct = NULL;
5686 } else { /* Otherwise find an active account for the protocol */
5687 GList *l = purple_accounts_get_all();
5688 while (l) {
5689 if (purple_strequal(prpl, purple_account_get_protocol_id(l->data))
5690 && purple_account_is_connected(l->data)) {
5691 acct = l->data;
5692 break;
5694 l = l->next;
5698 return acct;
5702 static gboolean oscar_uri_handler(const char *proto, const char *cmd, GHashTable *params)
5704 char *acct_id = g_hash_table_lookup(params, "account");
5705 char prpl[11];
5706 PurpleAccount *acct;
5708 if (g_ascii_strcasecmp(proto, "aim") && g_ascii_strcasecmp(proto, "icq"))
5709 return FALSE;
5711 g_snprintf(prpl, sizeof(prpl), "prpl-%s", proto);
5713 acct = find_acct(prpl, acct_id);
5715 if (!acct)
5716 return FALSE;
5718 /* aim:GoIM?screenname=SCREENNAME&message=MESSAGE */
5719 if (!g_ascii_strcasecmp(cmd, "GoIM")) {
5720 char *bname = g_hash_table_lookup(params, "screenname");
5721 if (bname) {
5722 char *message = g_hash_table_lookup(params, "message");
5724 PurpleConversation *conv = purple_find_conversation_with_account(
5725 PURPLE_CONV_TYPE_IM, bname, acct);
5726 if (conv == NULL)
5727 conv = purple_conversation_new(PURPLE_CONV_TYPE_IM, acct, bname);
5728 purple_conversation_present(conv);
5730 if (message) {
5731 /* Spaces are encoded as '+' */
5732 g_strdelimit(message, "+", ' ');
5733 purple_conv_send_confirm(conv, message);
5736 /*else
5737 **If pidgindialogs_im() was in the core, we could use it here.
5738 * It is all purple_request_* based, but I'm not sure it really belongs in the core
5739 pidgindialogs_im();*/
5741 return TRUE;
5743 /* aim:GoChat?roomname=CHATROOMNAME&exchange=4 */
5744 else if (!g_ascii_strcasecmp(cmd, "GoChat")) {
5745 char *rname = g_hash_table_lookup(params, "roomname");
5746 if (rname) {
5747 /* This is somewhat hacky, but the params aren't useful after this command */
5748 g_hash_table_insert(params, g_strdup("exchange"), g_strdup("4"));
5749 g_hash_table_insert(params, g_strdup("room"), g_strdup(rname));
5750 serv_join_chat(purple_account_get_connection(acct), params);
5752 /*else
5753 ** Same as above (except that this would have to be re-written using purple_request_*)
5754 pidgin_blist_joinchat_show(); */
5756 return TRUE;
5758 /* aim:AddBuddy?screenname=SCREENNAME&groupname=GROUPNAME*/
5759 else if (!g_ascii_strcasecmp(cmd, "AddBuddy")) {
5760 char *bname = g_hash_table_lookup(params, "screenname");
5761 char *gname = g_hash_table_lookup(params, "groupname");
5762 purple_blist_request_add_buddy(acct, bname, gname, NULL);
5763 return TRUE;
5766 return FALSE;
5769 void oscar_init(PurplePlugin *plugin, gboolean is_icq)
5771 PurplePluginProtocolInfo *prpl_info = PURPLE_PLUGIN_PROTOCOL_INFO(plugin);
5772 PurpleAccountOption *option;
5773 static gboolean init = FALSE;
5774 static const gchar *encryption_keys[] = {
5775 N_("Use encryption if available"),
5776 N_("Require encryption"),
5777 N_("Don't use encryption"),
5778 NULL
5780 static const gchar *encryption_values[] = {
5781 OSCAR_OPPORTUNISTIC_ENCRYPTION,
5782 OSCAR_REQUIRE_ENCRYPTION,
5783 OSCAR_NO_ENCRYPTION,
5784 NULL
5786 static const gchar *aim_login_keys[] = {
5787 N_("clientLogin"),
5788 N_("Kerberos"),
5789 N_("MD5-based"),
5790 NULL
5792 static const gchar *aim_login_values[] = {
5793 OSCAR_CLIENT_LOGIN,
5794 OSCAR_KERBEROS_LOGIN,
5795 OSCAR_MD5_LOGIN,
5796 NULL
5798 static const gchar *icq_login_keys[] = {
5799 N_("clientLogin"),
5800 N_("MD5-based"),
5801 NULL
5803 static const gchar *icq_login_values[] = {
5804 OSCAR_CLIENT_LOGIN,
5805 OSCAR_MD5_LOGIN,
5806 NULL
5808 const gchar **login_keys;
5809 const gchar **login_values;
5810 GList *encryption_options = NULL;
5811 GList *login_options = NULL;
5812 int i;
5814 option = purple_account_option_string_new(_("Server"), "server", get_login_server(is_icq, TRUE));
5815 prpl_info->protocol_options = g_list_append(prpl_info->protocol_options, option);
5817 option = purple_account_option_int_new(_("Port"), "port", OSCAR_DEFAULT_LOGIN_PORT);
5818 prpl_info->protocol_options = g_list_append(prpl_info->protocol_options, option);
5820 for (i = 0; encryption_keys[i]; i++) {
5821 PurpleKeyValuePair *kvp = g_new0(PurpleKeyValuePair, 1);
5822 kvp->key = g_strdup(_(encryption_keys[i]));
5823 kvp->value = g_strdup(encryption_values[i]);
5824 encryption_options = g_list_append(encryption_options, kvp);
5826 option = purple_account_option_list_new(_("Connection security"), "encryption", encryption_options);
5827 prpl_info->protocol_options = g_list_append(prpl_info->protocol_options, option);
5829 if (is_icq) {
5830 login_keys = icq_login_keys;
5831 login_values = icq_login_values;
5832 } else {
5833 login_keys = aim_login_keys;
5834 login_values = aim_login_values;
5836 for (i = 0; login_keys[i]; i++) {
5837 PurpleKeyValuePair *kvp = g_new0(PurpleKeyValuePair, 1);
5838 kvp->key = g_strdup(_(login_keys[i]));
5839 kvp->value = g_strdup(login_values[i]);
5840 login_options = g_list_append(login_options, kvp);
5842 option = purple_account_option_list_new(_("Authentication method"), "login_type", login_options);
5843 prpl_info->protocol_options = g_list_append(prpl_info->protocol_options, option);
5845 option = purple_account_option_bool_new(
5846 _("Always use AIM/ICQ proxy server for\nfile transfers and direct IM (slower,\nbut does not reveal your IP address)"), "always_use_rv_proxy",
5847 OSCAR_DEFAULT_ALWAYS_USE_RV_PROXY);
5848 prpl_info->protocol_options = g_list_append(prpl_info->protocol_options, option);
5850 if (purple_strequal(purple_plugin_get_id(plugin), "prpl-aim")) {
5851 option = purple_account_option_bool_new(_("Allow multiple simultaneous logins"), "allow_multiple_logins",
5852 OSCAR_DEFAULT_ALLOW_MULTIPLE_LOGINS);
5853 prpl_info->protocol_options = g_list_append(prpl_info->protocol_options, option);
5856 if (init)
5857 return;
5858 init = TRUE;
5860 /* Preferences */
5861 purple_prefs_add_none("/plugins/prpl/oscar");
5862 purple_prefs_add_bool("/plugins/prpl/oscar/recent_buddies", FALSE);
5864 purple_prefs_remove("/plugins/prpl/oscar/show_idle");
5865 purple_prefs_remove("/plugins/prpl/oscar/always_use_rv_proxy");
5867 /* protocol handler */
5868 /* TODO: figure out a good instance to use here */
5869 purple_signal_connect(purple_get_core(), "uri-handler", &init,
5870 PURPLE_CALLBACK(oscar_uri_handler), NULL);