Annotation of ChivanetAimPidgin/oscarprpl/src/c/oscar.c, revision 1.1.1.1
1.1 snw 1: /*
2: * purple
3: *
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
11: *
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>
14: *
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.
19: *
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.
24: *
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
28: *
29: */
30:
31: #include "internal.h"
32:
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"
51: #include <ctype.h>
52: #include "oscarcommon.h"
53: #include "oscar.h"
54: #include "peer.h"
55:
56: #define AIMHASHDATA "http://pidgin.im/aim_data.php3"
57:
58: #define OSCAR_CONNECT_STEPS 6
59:
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;
73:
74: static guint8 features_aim[] = {0x01, 0x01, 0x01, 0x02};
75: static guint8 features_icq[] = {0x01};
76:
77: struct create_room {
78: char *name;
79: int exchange;
80: };
81:
82: struct oscar_ask_directim_data
83: {
84: OscarData *od;
85: char *who;
86: };
87:
88: /* All the libfaim->purple callback functions */
89:
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 *, ...);
94:
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 *, ...);
131:
132: static void purple_icons_fetch(PurpleConnection *gc);
133:
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);
138:
139: void oscar_free_name_data(struct name_data *data) {
140: g_free(data->name);
141: g_free(data->nick);
142: g_free(data);
143: }
144:
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;
151: }
152: #endif
153:
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"));
182: }
183:
184: static char *extract_name(const char *name) {
185: char *tmp, *x;
186: int i, j;
187:
188: if (!name)
189: return NULL;
190:
191: x = strchr(name, '-');
192: if (!x)
193: return NULL;
194:
195: x = strchr(x + 1, '-');
196: if (!x)
197: return NULL;
198:
199: tmp = g_strdup(++x);
200:
201: for (i = 0, j = 0; x[i]; i++) {
202: char hex[3];
203: if (x[i] != '%') {
204: tmp[j++] = x[i];
205: continue;
206: }
207: strncpy(hex, x + ++i, 2);
208: hex[2] = 0;
209: i++;
210: tmp[j++] = strtol(hex, NULL, 16);
211: }
212:
213: tmp[j] = 0;
214: return tmp;
215: }
216:
217: static struct chat_connection *
218: find_oscar_chat(PurpleConnection *gc, int id)
219: {
220: OscarData *od = purple_connection_get_protocol_data(gc);
221: GSList *cur;
222: struct chat_connection *cc;
223:
224: for (cur = od->oscar_chats; cur != NULL; cur = cur->next)
225: {
226: cc = (struct chat_connection *)cur->data;
227: if (cc->id == id)
228: return cc;
229: }
230:
231: return NULL;
232: }
233:
234: static struct chat_connection *
235: find_oscar_chat_by_conn(PurpleConnection *gc, FlapConnection *conn)
236: {
237: OscarData *od = purple_connection_get_protocol_data(gc);
238: GSList *cur;
239: struct chat_connection *cc;
240:
241: for (cur = od->oscar_chats; cur != NULL; cur = cur->next)
242: {
243: cc = (struct chat_connection *)cur->data;
244: if (cc->conn == conn)
245: return cc;
246: }
247:
248: return NULL;
249: }
250:
251: static struct chat_connection *
252: find_oscar_chat_by_conv(PurpleConnection *gc, PurpleConversation *conv)
253: {
254: OscarData *od = purple_connection_get_protocol_data(gc);
255: GSList *cur;
256: struct chat_connection *cc;
257:
258: for (cur = od->oscar_chats; cur != NULL; cur = cur->next)
259: {
260: cc = (struct chat_connection *)cur->data;
261: if (cc->conv == conv)
262: return cc;
263: }
264:
265: return NULL;
266: }
267:
268: void
269: oscar_chat_destroy(struct chat_connection *cc)
270: {
271: g_free(cc->name);
272: g_free(cc->show);
273: g_free(cc);
274: }
275:
276: static void
277: oscar_chat_kill(PurpleConnection *gc, struct chat_connection *cc)
278: {
279: OscarData *od = purple_connection_get_protocol_data(gc);
280:
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)));
283:
284: /* Destroy the chat_connection */
285: od->oscar_chats = g_slist_remove(od->oscar_chats, cc);
286: oscar_chat_destroy(cc);
287: }
288:
289: /**
290: * This is called from the callback functions for establishing
291: * a TCP connection with an oscar host if an error occurred.
292: */
293: static void
294: connection_common_error_cb(FlapConnection *conn, const gchar *error_message)
295: {
296: OscarData *od;
297: PurpleConnection *gc;
298:
299: od = conn->od;
300: gc = od->gc;
301:
302: purple_debug_error("oscar", "unable to connect to FLAP "
303: "server of type 0x%04hx\n", conn->type);
304:
305: if (conn->type == SNAC_FAMILY_AUTH)
306: {
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);
313: }
314: else if (conn->type == SNAC_FAMILY_LOCATE)
315: {
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);
321: }
322: else
323: {
324: /* Maybe we should call this for BOS connections, too? */
325: flap_connection_schedule_destroy(conn,
326: OSCAR_DISCONNECT_COULD_NOT_CONNECT, error_message);
327: }
328: }
329:
330: /**
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.
334: */
335: static void
336: connection_common_established_cb(FlapConnection *conn)
337: {
338: OscarData *od;
339: PurpleConnection *gc;
340: PurpleAccount *account;
341:
342: od = conn->od;
343: gc = od->gc;
344: account = purple_connection_get_account(gc);
345:
346: purple_debug_info("oscar", "connected to FLAP server of type 0x%04hx\n",
347: conn->type);
348:
349: if (conn->cookie == NULL)
350: flap_connection_send_version(od, conn);
351: else
352: {
353: const gchar *login_type = purple_account_get_string(account, "login_type", OSCAR_DEFAULT_LOGIN);
354:
355: if (!purple_strequal(login_type, OSCAR_MD5_LOGIN))
356: {
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);
366: }
367:
368:
369: g_free(conn->cookie);
370: conn->cookie = NULL;
371: }
372:
373: if (conn->type == SNAC_FAMILY_AUTH)
374: {
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);
379: }
380: else if (conn->type == SNAC_FAMILY_LOCATE)
381: {
382: purple_connection_update_progress(gc, _("Connection established, cookie sent"), 4, OSCAR_CONNECT_STEPS);
383: }
384: else if (conn->type == SNAC_FAMILY_CHAT)
385: {
386: od->oscar_chats = g_slist_prepend(od->oscar_chats, conn->new_conn_data);
387: conn->new_conn_data = NULL;
388: }
389: }
390:
391: static void
392: connection_established_cb(gpointer data, gint source, const gchar *error_message)
393: {
394: FlapConnection *conn;
395:
396: conn = data;
397:
398: conn->connect_data = NULL;
399: conn->fd = source;
400:
401: if (source < 0)
402: {
403: connection_common_error_cb(conn, error_message);
404: return;
405: }
406:
407: conn->watcher_incoming = purple_input_add(conn->fd,
408: PURPLE_INPUT_READ, flap_connection_recv_cb, conn);
409: connection_common_established_cb(conn);
410: }
411:
412: static void
413: ssl_connection_established_cb(gpointer data, PurpleSslConnection *gsc,
414: PurpleInputCondition cond)
415: {
416: FlapConnection *conn;
417:
418: conn = data;
419:
420: purple_ssl_input_add(gsc, flap_connection_recv_cb_ssl, conn);
421: connection_common_established_cb(conn);
422: }
423:
424: static void
425: ssl_connection_error_cb(PurpleSslConnection *gsc, PurpleSslErrorType error,
426: gpointer data)
427: {
428: FlapConnection *conn;
429:
430: conn = data;
431:
432: if (conn->watcher_outgoing)
433: {
434: purple_input_remove(conn->watcher_outgoing);
435: conn->watcher_outgoing = 0;
436: }
437:
438: /* sslconn frees the connection on error */
439: conn->gsc = NULL;
440:
441: connection_common_error_cb(conn, purple_ssl_strerror(error));
442: }
443:
444: static void
445: flap_connection_established_bos(OscarData *od, FlapConnection *conn)
446: {
447: PurpleConnection *gc = od->gc;
448:
449: aim_srv_reqpersonalinfo(od, conn);
450:
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);
457:
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 */
462:
463: purple_connection_update_progress(gc, _("Finalizing connection"), 5, OSCAR_CONNECT_STEPS);
464: }
465:
466: static void
467: flap_connection_established_admin(OscarData *od, FlapConnection *conn)
468: {
469: aim_srv_clientready(od, conn);
470: purple_debug_info("oscar", "connected to admin\n");
471:
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;
480: }
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;
487: }
488: if (od->conf) {
489: purple_debug_info("oscar", "confirming account\n");
490: aim_admin_reqconfirm(od, conn);
491: od->conf = FALSE;
492: }
493: if (od->reqemail) {
494: purple_debug_info("oscar", "requesting email address\n");
495: aim_admin_getinfo(od, conn, 0x0011);
496: od->reqemail = FALSE;
497: }
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;
504: }
505: }
506:
507: static void
508: flap_connection_established_chat(OscarData *od, FlapConnection *conn)
509: {
510: PurpleConnection *gc = od->gc;
511: struct chat_connection *chatcon;
512: static int id = 1;
513:
514: aim_srv_clientready(od, conn);
515:
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);
520: }
521: }
522:
523: static void
524: flap_connection_established_chatnav(OscarData *od, FlapConnection *conn)
525: {
526: aim_srv_clientready(od, conn);
527: aim_chatnav_reqrights(od, conn);
528: }
529:
530: static void
531: flap_connection_established_alert(OscarData *od, FlapConnection *conn)
532: {
533: aim_email_sendcookies(od);
534: aim_email_activate(od);
535: aim_srv_clientready(od, conn);
536: }
537:
538: static void
539: flap_connection_established_bart(OscarData *od, FlapConnection *conn)
540: {
541: PurpleConnection *gc = od->gc;
542:
543: aim_srv_clientready(od, conn);
544:
545: od->iconconnecting = FALSE;
546:
547: purple_icons_fetch(gc);
548: }
549:
550: static int
551: flap_connection_established(OscarData *od, FlapConnection *conn, FlapFrame *fr, ...)
552: {
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);
568:
569: return 1;
570: }
571:
572: static void
573: idle_reporting_pref_cb(const char *name, PurplePrefType type,
574: gconstpointer value, gpointer data)
575: {
576: PurpleConnection *gc;
577: OscarData *od;
578: gboolean report_idle;
579: guint32 presence;
580:
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);
585:
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);
590: }
591:
592: /**
593: * Should probably make a "Use recent buddies group" account preference
594: * so that this option is surfaced to the user.
595: */
596: static void
597: recent_buddies_pref_cb(const char *name, PurplePrefType type,
598: gconstpointer value, gpointer data)
599: {
600: PurpleConnection *gc;
601: OscarData *od;
602: guint32 presence;
603:
604: gc = data;
605: od = purple_connection_get_protocol_data(gc);
606: presence = aim_ssi_getpresence(od->ssi.local);
607:
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);
612: }
613:
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,
619: };
620:
621: static const gchar *
622: get_login_server(gboolean is_icq, gboolean use_ssl)
623: {
624: return login_servers[(is_icq ? 2 : 0) + (use_ssl ? 1 : 0)];
625: }
626:
627: static gint
628: compare_handlers(gconstpointer a, gconstpointer b)
629: {
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;
638: }
639: return subtype1 - subtype2;
640: }
641:
642: #if !GLIB_CHECK_VERSION(2,14,0)
643: static void hash_table_get_list_of_keys(gpointer key, gpointer value, gpointer user_data)
644: {
645: GList **handlers = (GList **)user_data;
646:
647: *handlers = g_list_prepend(*handlers, key);
648: }
649: #endif /* GLIB < 2.14.0 */
650:
651: void
652: oscar_login(PurpleAccount *account)
653: {
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("");
662:
663: gc = purple_account_get_connection(account);
664: od = oscar_data_new();
665: od->gc = gc;
666: purple_connection_set_protocol_data(gc, od);
667:
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);
670:
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);
676:
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);
681:
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);
719:
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);
731: }
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);
736:
737: purple_debug_misc("oscar", "oscar_login: gc = %p\n", gc);
738:
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;
745: }
746:
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;
752: }
753:
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;
758:
759: od->default_port = purple_account_get_int(account, "port", OSCAR_DEFAULT_LOGIN_PORT);
760:
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(
765: gc,
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;
769: }
770: od->use_ssl = purple_ssl_is_supported() && !purple_strequal(encryption_type, OSCAR_NO_ENCRYPTION);
771:
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);
775:
776: /*
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.
781: *
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.
785: */
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;
791:
792: if (!od->use_ssl) {
793: purple_connection_error_reason(
794: gc,
795: PURPLE_CONNECTION_ERROR_NO_SSL_SUPPORT,
796: _("You required Kerberos authentication but encryption is disabled in your account settings."));
797: return;
798: }
799: server = purple_account_get_string(account, "server", AIM_DEFAULT_KDC_SERVER);
800: /*
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.
806: */
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);
814: }
815: send_kerberos_login(od, purple_account_get_username(account));
816: } else {
817: FlapConnection *newconn;
818: const char *server;
819:
820: newconn = flap_connection_new(od, SNAC_FAMILY_AUTH);
821:
822: if (od->use_ssl) {
823: server = purple_account_get_string(account, "server", get_login_server(od->icq, TRUE));
824:
825: /*
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.
831: */
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);
840: }
841:
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));
847:
848: /*
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.
852: */
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);
860: }
861:
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);
865: }
866:
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;
871: }
872: }
873:
874: purple_connection_update_progress(gc, _("Connecting"), 0, OSCAR_CONNECT_STEPS);
875: }
876:
877: void
878: oscar_close(PurpleConnection *gc)
879: {
880: OscarData *od;
881:
882: od = purple_connection_get_protocol_data(gc);
883:
884: while (od->oscar_chats)
885: {
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);
889: }
890: while (od->create_rooms)
891: {
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);
896: }
897: oscar_data_destroy(od);
898: purple_connection_set_protocol_data(gc, NULL);
899:
900: purple_prefs_disconnect_by_handle(gc);
901:
902: purple_debug_info("oscar", "Signed off.\n");
903: }
904:
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;
914: };
915:
916: static void damn_you(gpointer data, gint source, PurpleInputCondition c)
917: {
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;
924:
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';
933: }
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;
946: }
947: if (read(pos->fd, m, 16) != 16)
948: {
949: purple_debug_warning("oscar", "Could not read full AIM login hash "
950: "from " AIMHASHDATA "--that's bad.\n");
951: }
952: m[16] = '\0';
953:
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);
960:
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);
965: }
966:
967: static void
968: straight_to_hell(gpointer data, gint source, const gchar *error_message)
969: {
970: struct pieceofcrap *pos = data;
971: gchar *buf;
972: gssize result;
973:
974: pos->fd = source;
975:
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;
987: }
988:
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;
1005: }
1006:
1007: /* size of icbmui.ocm, the largest module in AIM 3.5 */
1008: #define AIM_MAX_FILE_SIZE 98304
1009:
1010: static int purple_memrequest(OscarData *od, FlapConnection *conn, FlapFrame *fr, ...)
1011: {
1012: va_list ap;
1013: struct pieceofcrap *pos;
1014: guint32 offset, len;
1015: char *modname;
1016:
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);
1022:
1023: purple_debug_misc("oscar", "offset: %u, len: %u, file: %s\n",
1024: offset, len, (modname ? modname : "aim.exe"));
1025:
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;
1031: }
1032:
1033: pos = g_new0(struct pieceofcrap, 1);
1034: pos->gc = od->gc;
1035: pos->conn = conn;
1036:
1037: pos->offset = offset;
1038: pos->len = len;
1039: pos->modname = g_strdup(modname);
1040:
1041: if (purple_proxy_connect(pos->gc, pos->gc->account, "pidgin.im", 80,
1042: straight_to_hell, pos) == NULL)
1043: {
1044: char buf[256];
1045: g_free(pos->modname);
1046: g_free(pos);
1047:
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);
1054: }
1055:
1056: return 1;
1057: }
1058:
1059: int oscar_connect_to_bos(PurpleConnection *gc, OscarData *od, const char *host, guint16 port, guint8 *cookie, guint16 cookielen, const char *tls_certname)
1060: {
1061: PurpleAccount *account;
1062: FlapConnection *conn;
1063:
1064: account = purple_connection_get_account(gc);
1065:
1066: conn = flap_connection_new(od, SNAC_FAMILY_LOCATE);
1067: conn->cookielen = cookielen;
1068: conn->cookie = g_memdup(cookie, cookielen);
1069:
1070: /*
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.
1073: */
1074: if (tls_certname)
1075: {
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);
1079: }
1080: else
1081: {
1082: conn->connect_data = purple_proxy_connect(NULL,
1083: account, host, port,
1084: connection_established_cb, conn);
1085: }
1086:
1087: if (conn->gsc == NULL && conn->connect_data == NULL)
1088: {
1089: purple_connection_error_reason(gc, PURPLE_CONNECTION_ERROR_NETWORK_ERROR, _("Unable to connect"));
1090: return 0;
1091: }
1092:
1093: od->default_port = port;
1094:
1095: purple_connection_update_progress(gc, _("Received authorization"), 3, OSCAR_CONNECT_STEPS);
1096:
1097: return 1;
1098: }
1099:
1100: /**
1101: * Only used when connecting with the old-style BUCP login.
1102: */
1103: static int
1104: purple_parse_auth_resp(OscarData *od, FlapConnection *conn, FlapFrame *fr, ...)
1105: {
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;
1113:
1114: port = purple_account_get_int(account, "port", od->default_port);
1115:
1116: va_start(ap, fr);
1117: info = va_arg(ap, struct aim_authresp_info *);
1118: va_end(ap);
1119:
1120: purple_debug_info("oscar",
1121: "inside auth_resp (Username: %s)\n", info->bn);
1122:
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:
1150: {
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;
1156: }
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;
1164: }
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;
1168: }
1169:
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);
1178:
1179: for (i = 0; i < strlen(info->bosip); i++) {
1180: if (info->bosip[i] == ':') {
1181: port = atoi(&(info->bosip[i+1]));
1182: break;
1183: }
1184: }
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);
1189:
1190: if (od->use_ssl)
1191: {
1192: /*
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.
1196: */
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);
1200: }
1201: else
1202: {
1203: newconn->connect_data = purple_proxy_connect(NULL, account, host, port,
1204: connection_established_cb, newconn);
1205: }
1206:
1207: g_free(host);
1208: if (newconn->gsc == NULL && newconn->connect_data == NULL)
1209: {
1210: purple_connection_error_reason(gc, PURPLE_CONNECTION_ERROR_NETWORK_ERROR, _("Unable to connect"));
1211: return 0;
1212: }
1213:
1214: purple_connection_update_progress(gc, _("Received authorization"), 3, OSCAR_CONNECT_STEPS);
1215:
1216: return 1;
1217: }
1218:
1219: /**
1220: * Only used when connecting with the old-style BUCP login.
1221: */
1222: static void
1223: purple_parse_auth_securid_request_yes_cb(gpointer user_data, const char *msg)
1224: {
1225: PurpleConnection *gc = user_data;
1226: OscarData *od = purple_connection_get_protocol_data(gc);
1227:
1228: aim_auth_securid_send(od, msg);
1229: }
1230:
1231: /**
1232: * Only used when connecting with the old-style BUCP login.
1233: */
1234: static void
1235: purple_parse_auth_securid_request_no_cb(gpointer user_data, const char *value)
1236: {
1237: PurpleConnection *gc = user_data;
1238:
1239: /* Disconnect */
1240: purple_connection_error_reason(gc,
1241: PURPLE_CONNECTION_ERROR_AUTHENTICATION_FAILED,
1242: _("The SecurID key entered is invalid"));
1243: }
1244:
1245: /**
1246: * Only used when connecting with the old-style BUCP login.
1247: */
1248: static int
1249: purple_parse_auth_securid_request(OscarData *od, FlapConnection *conn, FlapFrame *fr, ...)
1250: {
1251: PurpleConnection *gc = od->gc;
1252: PurpleAccount *account = purple_connection_get_account(gc);
1253: gchar *primary;
1254:
1255: purple_debug_info("oscar", "Got SecurID request\n");
1256:
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);
1266:
1267: return 1;
1268: }
1269:
1270: /**
1271: * Only used when connecting with the old-style BUCP login.
1272: */
1273: static int
1274: purple_parse_login(OscarData *od, FlapConnection *conn, FlapFrame *fr, ...)
1275: {
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;
1283:
1284: gc = od->gc;
1285: account = purple_connection_get_account(gc);
1286:
1287: va_start(ap, fr);
1288: key = va_arg(ap, char *);
1289: truncate_pass = va_arg(ap, int);
1290: va_end(ap);
1291:
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));
1296:
1297: purple_connection_update_progress(gc, _("Password sent"), 2, OSCAR_CONNECT_STEPS);
1298:
1299: return 1;
1300: }
1301:
1302: static int
1303: purple_handle_redirect(OscarData *od, FlapConnection *conn, FlapFrame *fr, ...)
1304: {
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;
1312:
1313: va_start(ap, fr);
1314: redir = va_arg(ap, struct aim_redirect_data *);
1315: va_end(ap);
1316:
1317: port = od->default_port;
1318: separator = strchr(redir->ip, ':');
1319: if (separator != NULL)
1320: {
1321: host = g_strndup(redir->ip, separator - redir->ip);
1322: port = atoi(separator + 1);
1323: }
1324: else
1325: host = g_strdup(redir->ip);
1326:
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(
1334: gc,
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;
1338: }
1339: }
1340:
1341: /*
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.
1344: */
1345: if (redir->use_ssl && (redir->group == SNAC_FAMILY_ADMIN ||
1346: redir->group == SNAC_FAMILY_BART))
1347: {
1348: purple_debug_info("oscar", "Ignoring broken SSL for FLAP type 0x%04hx.\n", redir->group);
1349: redir->use_ssl = 0;
1350: }
1351:
1352: purple_debug_info("oscar", "Connecting to FLAP server %s:%d of type 0x%04hx\n", host, port, redir->group);
1353:
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)
1358: {
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);
1369: }
1370:
1371:
1372: if (redir->use_ssl)
1373: {
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);
1377: }
1378: else
1379: {
1380: newconn->connect_data = purple_proxy_connect(NULL, account, host, port,
1381: connection_established_cb, newconn);
1382: }
1383:
1384: if (newconn->gsc == NULL && newconn->connect_data == NULL)
1385: {
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);
1391: }
1392: g_free(host);
1393:
1394: return 1;
1395: }
1396:
1397:
1398: static int purple_parse_oncoming(OscarData *od, FlapConnection *conn, FlapFrame *fr, ...)
1399: {
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;
1413:
1414: gc = od->gc;
1415: account = purple_connection_get_account(gc);
1416:
1417: va_start(ap, fr);
1418: info = va_arg(ap, aim_userinfo_t *);
1419: va_end(ap);
1420:
1421: g_return_val_if_fail(info != NULL, 1);
1422: g_return_val_if_fail(info->bn != NULL, 1);
1423:
1424: buddy = purple_find_buddy(account, info->bn);
1425: if (buddy) {
1426: previous_status = purple_presence_get_active_status(purple_buddy_get_presence(buddy));
1427: }
1428:
1429: /*
1430: * If this is an AIM buddy and their name has formatting, set their
1431: * server alias.
1432: */
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;
1440: }
1441: }
1442: serv_got_alias(gc, info->bn,
1443: bn_has_formatting ? info->bn : NULL);
1444: }
1445:
1446: if (info->present & AIM_USERINFO_PRESENT_FLAGS) {
1447: if (info->flags & AIM_FLAG_AWAY)
1448: buddy_is_away = TRUE;
1449: }
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;
1455: }
1456: }
1457:
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;
1490: }
1491:
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);
1496: }
1497:
1498: message = (info->status && info->status_len > 0)
1499: ? oscar_encoding_to_utf8(info->status_encoding, info->status, info->status_len)
1500: : NULL;
1501:
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"));
1509: }
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);
1515: }
1516:
1517: g_free(message);
1518: g_free(itmsurl);
1519:
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);
1526:
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;
1531:
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);
1536:
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);
1542: }
1543: bi->typingnot = FALSE;
1544: bi->ico_informed = FALSE;
1545: bi->ipaddr = info->icqinfo.ipaddr;
1546:
1547: if (info->iconcsumlen) {
1548: const char *saved_b16 = NULL;
1549: char *b16 = NULL;
1550: PurpleBuddy *b = NULL;
1551:
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);
1556:
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);
1560:
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)
1564: {
1565: od->requesticon = g_slist_prepend(od->requesticon,
1566: g_strdup(purple_normalize(account, info->bn)));
1567: purple_icons_fetch(gc);
1568: }
1569: }
1570: g_free(b16);
1571: }
1572:
1573: return 1;
1574: }
1575:
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;
1581:
1582: va_start(ap, fr);
1583: info = va_arg(ap, aim_userinfo_t *);
1584: va_end(ap);
1585:
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));
1589:
1590: return 1;
1591: }
1592:
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;
1602:
1603: purple_debug_misc("oscar", "Received IM from %s\n", userinfo->bn);
1604:
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);
1609: }
1610:
1611: if (args->icbmflags & AIM_IMFLAGS_AWAY)
1612: flags |= PURPLE_MESSAGE_AUTO_RESP;
1613:
1614: if (args->icbmflags & AIM_IMFLAGS_TYPINGNOT)
1615: bi->typingnot = TRUE;
1616: else
1617: bi->typingnot = FALSE;
1618:
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;
1626: }
1627: }
1628:
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));
1640: }
1641: purple_imgstore_unref(img);
1642:
1643: tmp = g_strdup(args->msg);
1644:
1645: /*
1646: * Convert iChat color tags to normal font tags.
1647: */
1648: if (purple_markup_find_tag("body", tmp, &start, &end, &attribs))
1649: {
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;
1655:
1656: /*
1657: * Find the ending </body> so we can strip off the outer <html/>
1658: * and <body/>
1659: */
1660: if (purple_markup_find_tag("/body", end + 1, &slash_body_start, &slash_body_end, &unused))
1661: {
1662: body = g_strndup(start, slash_body_end - start + 1);
1663: g_datalist_clear(&unused);
1664: }
1665: else
1666: {
1667: purple_debug_warning("oscar", "Broken message contains <body> but not </body>!\n");
1668: /* Take everything after <body> */
1669: body = g_strdup(start);
1670: }
1671:
1672: ichattextcolor = g_datalist_get_data(&attribs, "ichattextcolor");
1673: if (ichattextcolor != NULL)
1674: {
1675: tmp2 = g_strdup_printf("<font color=\"%s\">%s</font>", ichattextcolor, body);
1676: g_free(body);
1677: body = tmp2;
1678: }
1679:
1680: ichatballooncolor = g_datalist_get_data(&attribs, "ichatballooncolor");
1681: if (ichatballooncolor != NULL)
1682: {
1683: tmp2 = g_strdup_printf("<font back=\"%s\">%s</font>", ichatballooncolor, body);
1684: g_free(body);
1685: body = tmp2;
1686: }
1687:
1688: g_datalist_clear(&attribs);
1689:
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);
1694:
1695: tmp = tmp2;
1696: }
1697:
1698: /*
1699: * Are there <html/> surrounding tags? If so, strip them out, too.
1700: */
1701: if (purple_markup_find_tag("html", tmp, &start, &end, &attribs))
1702: {
1703: gchar *tmp2;
1704: int len;
1705:
1706: g_datalist_clear(&attribs);
1707:
1708: len = start - tmp;
1709: tmp2 = g_strdup_printf("%.*s%s", len, tmp, end + 1);
1710: g_free(tmp);
1711: tmp = tmp2;
1712: }
1713:
1714: if (purple_markup_find_tag("/html", tmp, &start, &end, &attribs))
1715: {
1716: gchar *tmp2;
1717: int len;
1718:
1719: g_datalist_clear(&attribs);
1720:
1721: len = start - tmp;
1722: tmp2 = g_strdup_printf("%.*s%s", len, tmp, end + 1);
1723: g_free(tmp);
1724: tmp = tmp2;
1725: }
1726:
1727: serv_got_im(gc, userinfo->bn, tmp, flags, (args->icbmflags & AIM_IMFLAGS_OFFLINE) ? args->timestamp : time(NULL));
1728: g_free(tmp);
1729:
1730: return 1;
1731: }
1732:
1733: static int
1734: incomingim_chan2(OscarData *od, FlapConnection *conn, aim_userinfo_t *userinfo, IcbmArgsCh2 *args)
1735: {
1736: PurpleConnection *gc;
1737: PurpleAccount *account;
1738: PurpleMessageFlags flags = 0;
1739: char *message = NULL;
1740:
1741: g_return_val_if_fail(od != NULL, 0);
1742: g_return_val_if_fail(od->gc != NULL, 0);
1743:
1744: gc = od->gc;
1745: account = purple_connection_get_account(gc);
1746: od = purple_connection_get_protocol_data(gc);
1747:
1748: if (args == NULL)
1749: return 0;
1750:
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);
1754:
1755: if (args->msg != NULL) {
1756: message = oscar_encoding_to_utf8(args->encoding, args->msg, args->msglen);
1757: }
1758:
1759: if (args->type & OSCAR_CAPABILITY_CHAT)
1760: {
1761: char *utf8name, *tmp;
1762: GHashTable *components;
1763:
1764: if (!args->info.chat.roominfo.name || !args->info.chat.roominfo.exchange) {
1765: g_free(message);
1766: return 1;
1767: }
1768: utf8name = oscar_encoding_to_utf8(args->encoding, args->info.chat.roominfo.name, args->info.chat.roominfo.namelen);
1769:
1770: tmp = extract_name(utf8name);
1771: if (tmp != NULL)
1772: {
1773: g_free(utf8name);
1774: utf8name = tmp;
1775: }
1776:
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);
1787: }
1788:
1789: else if ((args->type & OSCAR_CAPABILITY_SENDFILE) || (args->type & OSCAR_CAPABILITY_DIRECTIM))
1790: {
1791: if (args->status == AIM_RENDEZVOUS_PROPOSE)
1792: {
1793: peer_connection_got_proposition(od, userinfo->bn, message, args);
1794: }
1795: else if (args->status == AIM_RENDEZVOUS_CANCEL)
1796: {
1797: /* The other user cancelled a peer request */
1798: PeerConnection *conn;
1799:
1800: conn = peer_connection_find_by_cookie(od, userinfo->bn, args->cookie);
1801: /*
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.
1805: */
1806: if (conn != NULL)
1807: {
1808: peer_connection_destroy(conn, OSCAR_DISCONNECT_REMOTE_CLOSED, NULL);
1809: }
1810: }
1811: else if (args->status == AIM_RENDEZVOUS_CONNECTED)
1812: {
1813: /*
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.
1817: */
1818: }
1819: }
1820:
1821: else if (args->type & OSCAR_CAPABILITY_GETFILE)
1822: {
1823: }
1824:
1825: else if (args->type & OSCAR_CAPABILITY_TALK)
1826: {
1827: }
1828:
1829: else if (args->type & OSCAR_CAPABILITY_BUDDYICON)
1830: {
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);
1835: }
1836:
1837: else if (args->type & OSCAR_CAPABILITY_ICQSERVERRELAY)
1838: {
1839: purple_debug_info("oscar", "Got an ICQ Server Relay message of "
1840: "type %d\n", args->info.rtfmsg.msgtype);
1841:
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;
1848:
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.
1854: */
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);
1857: }
1858:
1859: rtfmsg = oscar_encoding_to_utf8(encoding, args->info.rtfmsg.msg, len);
1860:
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);
1868:
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);
1872: }
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);
1876: }
1877: }
1878: else
1879: {
1880: purple_debug_error("oscar", "Unknown request class %"
1881: G_GUINT64_FORMAT "\n", args->type);
1882: }
1883:
1884: g_free(message);
1885:
1886: return 1;
1887: }
1888:
1889: /* When someone sends you buddies */
1890: static void
1891: purple_icq_buddyadd(struct name_data *data)
1892: {
1893: PurpleConnection *gc = data->gc;
1894:
1895: purple_blist_request_add_buddy(purple_connection_get_account(gc), data->name, NULL, data->nick);
1896:
1897: oscar_free_name_data(data);
1898: }
1899:
1900: static int
1901: incomingim_chan4(OscarData *od, FlapConnection *conn, aim_userinfo_t *userinfo, struct aim_incomingim_ch4_args *args, time_t t)
1902: {
1903: PurpleConnection *gc = od->gc;
1904: PurpleAccount *account = purple_connection_get_account(gc);
1905: gchar **msg1, **msg2;
1906: int i, numtoks;
1907:
1908: if (!args->type || !args->msg || !args->uin)
1909: return 1;
1910:
1911: purple_debug_info("oscar",
1912: "Received a channel 4 message of type 0x%02hx.\n",
1913: (guint16)args->type);
1914:
1915: /*
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.
1923: */
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);
1929:
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);
1934: }
1935: msg2[i] = NULL;
1936:
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;
1942:
1943: /* If the message came from an ICQ user then escape any HTML */
1944: tmp = g_markup_escape_text(msg2[0], -1);
1945:
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));
1963: }
1964: g_free(uin);
1965: g_free(tmp);
1966: }
1967: } break;
1968:
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);
1979: }
1980: }
1981: } break;
1982:
1983: case 0x06: { /* Someone requested authorization */
1984: if (i >= 6) {
1985: gchar *bn = g_strdup_printf("%u", args->uin);
1986: gchar *reason = NULL;
1987:
1988: if (msg2[5] != NULL)
1989: reason = oscar_decode_im(account, bn, AIM_CHARSET_LATIN_1, msg2[5], strlen(msg2[5]));
1990:
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);
1997: }
1998: } break;
1999:
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);
2006: }
2007: } break;
2008:
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;
2015:
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);
2021: }
2022: } break;
2023:
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);
2029: }
2030: } break;
2031:
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);
2037: }
2038: } break;
2039:
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;
2044:
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;
2053:
2054: if (num > 0 && errno == 0) {
2055: for (i=0; i<num; i++) {
2056: struct name_data *data;
2057: gchar *message;
2058:
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;
2066: }
2067:
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]);
2069:
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]);
2074:
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);
2084: }
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);
2090: }
2091: g_strfreev(text);
2092: }
2093: } break;
2094:
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;
2102:
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;
2116: }
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;
2128: }
2129: smsmsg = byte_stream_getstr(&qbs, smslen);
2130:
2131: /* Check if this is an SMS being sent from server */
2132: if ((smstype == 0) && (purple_strequal(tagstr, "ICQSMS")) && (smsmsg != NULL))
2133: {
2134: xmlroot = xmlnode_from_str(smsmsg, -1);
2135: if (xmlroot != NULL)
2136: {
2137: xmltmp = xmlnode_get_child(xmlroot, "sender");
2138: if (xmltmp != NULL)
2139: uin = xmlnode_get_data(xmltmp);
2140:
2141: xmltmp = xmlnode_get_child(xmlroot, "text");
2142: if (xmltmp != NULL)
2143: message = xmlnode_get_data(xmltmp);
2144:
2145: if ((uin != NULL) && (message != NULL))
2146: serv_got_im(gc, uin, message, 0, time(NULL));
2147:
2148: g_free(uin);
2149: g_free(message);
2150: xmlnode_free(xmlroot);
2151: }
2152: }
2153: g_free(tagstr);
2154: g_free(smsmsg);
2155: } break;
2156:
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;
2162: }
2163:
2164: g_strfreev(msg1);
2165: g_strfreev(msg2);
2166:
2167: return 1;
2168: }
2169:
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;
2175:
2176: va_start(ap, fr);
2177: channel = (guint16)va_arg(ap, unsigned int);
2178: userinfo = va_arg(ap, aim_userinfo_t *);
2179:
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;
2186:
2187: case 2: { /* rendezvous */
2188: IcbmArgsCh2 *args;
2189: args = va_arg(ap, IcbmArgsCh2 *);
2190: ret = incomingim_chan2(od, conn, userinfo, args);
2191: } break;
2192:
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;
2198:
2199: default: {
2200: purple_debug_warning("oscar",
2201: "ICBM received on unsupported channel (channel "
2202: "0x%04hx).", channel);
2203: } break;
2204: }
2205:
2206: va_end(ap);
2207:
2208: return ret;
2209: }
2210:
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;
2218:
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);
2225:
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;
2281: }
2282:
2283: if (!purple_conv_present_error(userinfo->bn, account, buf))
2284: purple_notify_error(od->gc, NULL, buf, NULL);
2285: g_free(buf);
2286:
2287: return 1;
2288: }
2289:
2290: static int
2291: purple_parse_clientauto_ch2(OscarData *od, const char *who, guint16 reason, const guchar *cookie)
2292: {
2293: if (reason == 0x0003)
2294: {
2295: /* Rendezvous was refused. */
2296: PeerConnection *conn;
2297:
2298: conn = peer_connection_find_by_cookie(od, who, cookie);
2299:
2300: if (conn == NULL)
2301: {
2302: purple_debug_info("oscar", "Received a rendezvous cancel message "
2303: "for a nonexistant connection from %s.\n", who);
2304: }
2305: else
2306: {
2307: peer_connection_destroy(conn, OSCAR_DISCONNECT_REMOTE_REFUSED, NULL);
2308: }
2309: }
2310: else
2311: {
2312: purple_debug_warning("oscar", "Received an unknown rendezvous "
2313: "message from %s. Type 0x%04hx\n", who, reason);
2314: }
2315:
2316: return 0;
2317: }
2318:
2319: static int purple_parse_clientauto_ch4(OscarData *od, char *who, guint16 reason, guint32 state, char *msg) {
2320: PurpleConnection *gc = od->gc;
2321:
2322: switch(reason) {
2323: case 0x0003: { /* Reply from an ICQ status message request */
2324: char *statusmsg, **splitmsg;
2325: PurpleNotifyUserInfo *user_info;
2326:
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);
2330:
2331: user_info = purple_notify_user_info_new();
2332:
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));
2337:
2338: g_free(statusmsg);
2339: g_strfreev(splitmsg);
2340:
2341: purple_notify_userinfo(gc, who, user_info, NULL, NULL);
2342: purple_notify_user_info_destroy(user_info);
2343:
2344: } break;
2345:
2346: case 0x0006: { /* Reply from an ICQ status message request */
2347: char *statusmsg, **splitmsg;
2348: PurpleNotifyUserInfo *user_info;
2349:
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);
2353:
2354: user_info = purple_notify_user_info_new();
2355:
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));
2360:
2361: g_free(statusmsg);
2362: g_strfreev(splitmsg);
2363:
2364: purple_notify_userinfo(gc, who, user_info, NULL, NULL);
2365: purple_notify_user_info_destroy(user_info);
2366:
2367: } break;
2368:
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 */
2375:
2376: return 0;
2377: }
2378:
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;
2384:
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);
2389:
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 *);
2399: }
2400: ret = purple_parse_clientauto_ch4(od, who, reason, state, msg);
2401: }
2402:
2403: va_end(ap);
2404:
2405: return ret;
2406: }
2407:
2408: static int purple_parse_genericerr(OscarData *od, FlapConnection *conn, FlapFrame *fr, ...) {
2409: va_list ap;
2410: guint16 reason;
2411:
2412: va_start(ap, fr);
2413: reason = (guint16) va_arg(ap, unsigned int);
2414: va_end(ap);
2415:
2416: purple_debug_error("oscar", "snac threw error (reason 0x%04hx: %s)\n",
2417: reason, oscar_get_msgerr_reason(reason));
2418: return 1;
2419: }
2420:
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;
2426:
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);
2432:
2433: switch (event) {
2434: case 0x0000: { /* Text has been cleared */
2435: serv_got_typing_stopped(gc, bn);
2436: } break;
2437:
2438: case 0x0001: { /* Paused typing */
2439: serv_got_typing(gc, bn, 0, PURPLE_TYPED);
2440: } break;
2441:
2442: case 0x0002: { /* Typing */
2443: serv_got_typing(gc, bn, 0, PURPLE_TYPING);
2444: } break;
2445:
2446: case 0x000f: { /* Closed IM window */
2447: serv_got_typing_stopped(gc, bn);
2448: } break;
2449:
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;
2455: }
2456:
2457: return 1;
2458: }
2459:
2460: static int purple_parse_motd(OscarData *od, FlapConnection *conn, FlapFrame *fr, ...)
2461: {
2462: char *msg;
2463: guint16 id;
2464: va_list ap;
2465:
2466: va_start(ap, fr);
2467: id = (guint16) va_arg(ap, unsigned int);
2468: msg = va_arg(ap, char *);
2469: va_end(ap);
2470:
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);
2476:
2477: return 1;
2478: }
2479:
2480: static int purple_chatnav_info(OscarData *od, FlapConnection *conn, FlapFrame *fr, ...) {
2481: va_list ap;
2482: guint16 type;
2483:
2484: va_start(ap, fr);
2485: type = (guint16) va_arg(ap, unsigned int);
2486:
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;
2493:
2494: maxrooms = (guint8) va_arg(ap, unsigned int);
2495: exchangecount = va_arg(ap, int);
2496: exchanges = va_arg(ap, struct aim_chat_exchangeinfo *);
2497:
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);
2503: }
2504: g_string_append(msg, ", ");
2505: }
2506: purple_debug_misc("oscar", "%s\n", msg->str);
2507: g_string_free(msg, TRUE);
2508:
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);
2517: }
2518: }
2519: break;
2520: case 0x0008: {
2521: char *fqcn, *name, *ck;
2522: guint16 instance, flags, maxmsglen, maxoccupancy, unknown, exchange;
2523: guint8 createperms;
2524: guint32 createtime;
2525:
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 *);
2537:
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);
2544: }
2545: break;
2546: default:
2547: purple_debug_warning("oscar",
2548: "chatnav info: unknown type (%04hx)\n", type);
2549: break;
2550: }
2551:
2552: va_end(ap);
2553:
2554: return 1;
2555: }
2556:
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;
2562:
2563: struct chat_connection *c = NULL;
2564:
2565: va_start(ap, fr);
2566: count = va_arg(ap, int);
2567: info = va_arg(ap, aim_userinfo_t *);
2568: va_end(ap);
2569:
2570: c = find_oscar_chat_by_conn(gc, conn);
2571: if (!c)
2572: return 1;
2573:
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);
2576:
2577: return 1;
2578: }
2579:
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;
2585:
2586: struct chat_connection *c = NULL;
2587:
2588: va_start(ap, fr);
2589: count = va_arg(ap, int);
2590: info = va_arg(ap, aim_userinfo_t *);
2591: va_end(ap);
2592:
2593: c = find_oscar_chat_by_conn(gc, conn);
2594: if (!c)
2595: return 1;
2596:
2597: for (i = 0; i < count; i++)
2598: purple_conv_chat_remove_user(PURPLE_CONV_CHAT(c->conv), info[i].bn, NULL);
2599:
2600: return 1;
2601: }
2602:
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);
2608:
2609: if (!ccon)
2610: return 1;
2611:
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);
2616:
2617: purple_debug_misc("oscar",
2618: "inside chat_info_update (maxmsglen = %hu, maxvislen = %hu)\n",
2619: maxmsglen, maxvisiblemsglen);
2620:
2621: ccon->maxlen = maxmsglen;
2622: ccon->maxvis = maxvisiblemsglen;
2623:
2624: return 1;
2625: }
2626:
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;
2636:
2637: if (!ccon)
2638: return 1;
2639:
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);
2646:
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);
2650:
2651: return 1;
2652: }
2653:
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;
2661:
2662: gc = od->gc;
2663: account = purple_connection_get_account(gc);
2664:
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);
2671:
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);
2683: }
2684:
2685: if (alertitle)
2686: purple_debug_misc("oscar", "Got an alert '%s' %s\n", alertitle, alerturl ? alerturl : "");
2687:
2688: return 1;
2689: }
2690:
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;
2697:
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);
2706:
2707: /*
2708: * Some AIM clients will send a blank GIF image with iconlen 90 when
2709: * no icon is set. Ignore these.
2710: */
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);
2716: }
2717:
2718: return 1;
2719: }
2720:
2721: static void
2722: purple_icons_fetch(PurpleConnection *gc)
2723: {
2724: OscarData *od = purple_connection_get_protocol_data(gc);
2725: aim_userinfo_t *userinfo;
2726: FlapConnection *conn;
2727:
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;
2733: }
2734: return;
2735: }
2736:
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);
2748: }
2749: od->set_icon = FALSE;
2750: }
2751:
2752: while (od->requesticon != NULL)
2753: {
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);
2757:
2758: g_free(od->requesticon->data);
2759: od->requesticon = g_slist_delete_link(od->requesticon, od->requesticon);
2760: }
2761:
2762: purple_debug_misc("oscar", "no more icons to request\n");
2763: }
2764:
2765: static int purple_selfinfo(OscarData *od, FlapConnection *conn, FlapFrame *fr, ...) {
2766: va_list ap;
2767: aim_userinfo_t *info;
2768:
2769: va_start(ap, fr);
2770: info = va_arg(ap, aim_userinfo_t *);
2771: va_end(ap);
2772:
2773: purple_connection_set_display_name(od->gc, info->bn);
2774:
2775: return 1;
2776: }
2777:
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;
2783:
2784: va_start(ap, fr);
2785: code = (guint16)va_arg(ap, int);
2786: msg = va_arg(ap, char *);
2787: va_end(ap);
2788:
2789: purple_debug_info("oscar", "Disconnected. Code is 0x%04x and msg is %s\n",
2790: code, (msg != NULL ? msg : ""));
2791:
2792: g_return_val_if_fail(conn != NULL, 1);
2793:
2794: if (conn->type == SNAC_FAMILY_CHAT) {
2795: struct chat_connection *cc;
2796: PurpleConversation *conv = NULL;
2797:
2798: cc = find_oscar_chat_by_conn(gc, conn);
2799: if (cc != NULL)
2800: {
2801: conv = purple_find_chat(gc, cc->id);
2802:
2803: if (conv != NULL)
2804: {
2805: /*
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.
2809: */
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);
2815: }
2816: oscar_chat_kill(gc, cc);
2817: }
2818: }
2819:
2820: return 1;
2821: }
2822:
2823: static int purple_parse_locaterights(OscarData *od, FlapConnection *conn, FlapFrame *fr, ...)
2824: {
2825: PurpleConnection *gc = od->gc;
2826: PurpleAccount *account = purple_connection_get_account(gc);
2827: va_list ap;
2828: guint16 maxsiglen;
2829:
2830: va_start(ap, fr);
2831: maxsiglen = (guint16) va_arg(ap, int);
2832: va_end(ap);
2833:
2834: purple_debug_misc("oscar",
2835: "locate rights: max sig len = %d\n", maxsiglen);
2836:
2837: od->rights.maxsiglen = od->rights.maxawaymsglen = (guint)maxsiglen;
2838:
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));
2842:
2843: return 1;
2844: }
2845:
2846: static int purple_parse_buddyrights(OscarData *od, FlapConnection *conn, FlapFrame *fr, ...) {
2847: va_list ap;
2848: guint16 maxbuddies, maxwatchers;
2849:
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);
2854:
2855: purple_debug_misc("oscar",
2856: "buddy list rights: Max buddies = %hu / Max watchers = %hu\n", maxbuddies, maxwatchers);
2857:
2858: od->rights.maxbuddies = (guint)maxbuddies;
2859: od->rights.maxwatchers = (guint)maxwatchers;
2860:
2861: return 1;
2862: }
2863:
2864: static void oscar_format_username(PurpleConnection *gc, const char *new_display_name)
2865: {
2866: OscarData *od;
2867: const char *old_display_name, *username;
2868: char *tmp, *at_sign;
2869:
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;
2877: }
2878:
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;
2884: }
2885:
2886: tmp = g_strdup(new_display_name);
2887:
2888: /*
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.
2892: */
2893: at_sign = strchr(tmp, '@');
2894: if (at_sign)
2895: at_sign[0] = '\0';
2896:
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);
2907: }
2908: }
2909:
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;
2920:
2921: gc = od->gc;
2922: od = purple_connection_get_protocol_data(gc);
2923: account = purple_connection_get_account(gc);
2924:
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);
2929:
2930: purple_debug_misc("oscar",
2931: "BOS rights: Max permit = %hu / Max deny = %hu\n", maxpermits, maxdenies);
2932:
2933: od->rights.maxpermits = (guint)maxpermits;
2934: od->rights.maxdenies = (guint)maxdenies;
2935:
2936: purple_debug_info("oscar", "buddy list loaded\n");
2937:
2938: if (purple_account_get_user_info(account) != NULL)
2939: serv_set_info(gc, purple_account_get_user_info(account));
2940:
2941: username = purple_account_get_username(account);
2942: if (!od->icq && !purple_strequal(username, purple_connection_get_display_name(gc))) {
2943: /*
2944: * Format the username for AIM accounts if it's different
2945: * than what's currently set.
2946: */
2947: oscar_format_username(gc, username);
2948: }
2949:
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);
2962:
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));
2965:
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));
2971: }
2972:
2973: aim_srv_requestnew(od, SNAC_FAMILY_ALERT);
2974: aim_srv_requestnew(od, SNAC_FAMILY_CHATNAV);
2975:
2976: od->bos.have_rights = TRUE;
2977:
2978: /*
2979: * If we've already received our feedbag data then we're not waiting on
2980: * anything else, so send the server clientready.
2981: *
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.
2986: */
2987: if (od->ssi.received_data) {
2988: aim_srv_clientready(od, conn);
2989:
2990: /* Request offline messages for AIM and ICQ */
2991: aim_im_reqofflinemsgs(od);
2992:
2993: purple_connection_set_state(gc, PURPLE_CONNECTED);
2994: }
2995:
2996: return 1;
2997: }
2998:
2999: static int purple_popup(OscarData *od, FlapConnection *conn, FlapFrame *fr, ...)
3000: {
3001: PurpleConnection *gc = od->gc;
3002: gchar *text;
3003: va_list ap;
3004: char *msg, *url;
3005:
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);
3013:
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);
3017:
3018: return 1;
3019: }
3020:
3021: static void oscar_searchresults_add_buddy_cb(PurpleConnection *gc, GList *row, void *user_data)
3022: {
3023: purple_blist_request_add_buddy(purple_connection_get_account(gc),
3024: g_list_nth_data(row, 0), NULL, NULL);
3025: }
3026:
3027: static int purple_parse_searchreply(OscarData *od, FlapConnection *conn, FlapFrame *fr, ...)
3028: {
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;
3036:
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);
3042:
3043: results = purple_notify_searchresults_new();
3044:
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;
3052: }
3053:
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);
3059:
3060: column = purple_notify_searchresults_column_new(_("Username"));
3061: purple_notify_searchresults_column_add(results, column);
3062:
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);
3067: }
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);
3071:
3072: g_free(secondary);
3073:
3074: return 1;
3075: }
3076:
3077: static int purple_parse_searcherror(OscarData *od, FlapConnection *conn, FlapFrame *fr, ...) {
3078: va_list ap;
3079: char *email;
3080: char *buf;
3081:
3082: va_start(ap, fr);
3083: email = va_arg(ap, char *);
3084: va_end(ap);
3085:
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);
3089:
3090: return 1;
3091: }
3092:
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];
3098:
3099: va_start(ap, fr);
3100: status = (guint16) va_arg(ap, unsigned int); /* status code of confirmation request */
3101: va_end(ap);
3102:
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);
3110: }
3111:
3112: return 1;
3113: }
3114:
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;
3121:
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);
3130:
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)");
3137:
3138: if ((err > 0) && (url != NULL)) {
3139: char *dialog_msg;
3140:
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;
3159: }
3160:
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);
3166: }
3167:
3168: return 1;
3169: }
3170:
3171: void
3172: oscar_keepalive(PurpleConnection *gc)
3173: {
3174: OscarData *od;
3175: GSList *l;
3176:
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);
3180: }
3181: }
3182:
3183: unsigned int
3184: oscar_send_typing(PurpleConnection *gc, const char *name, PurpleTypingState state)
3185: {
3186: OscarData *od;
3187: PeerConnection *conn;
3188:
3189: od = purple_connection_get_protocol_data(gc);
3190: conn = peer_connection_find_by_type(od, name, OSCAR_CAPABILITY_DIRECTIM);
3191:
3192: if ((conn != NULL) && (conn->ready))
3193: {
3194: peer_odc_send_typing(conn, state);
3195: }
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);
3209: }
3210: }
3211: }
3212: return 0;
3213: }
3214:
3215: /* TODO: Move this into odc.c! */
3216: static void
3217: purple_odc_send_im(PeerConnection *conn, const char *message, PurpleMessageFlags imflags)
3218: {
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;
3227:
3228: msg = g_string_new("<HTML><BODY>");
3229: data = g_string_new("<BINARY>");
3230: last = message;
3231:
3232: /* for each valid IMG tag... */
3233: while (last && *last && purple_markup_find_tag("img", last, &start, &end, &attribs))
3234: {
3235: PurpleStoredImage *image = NULL;
3236: const char *id;
3237:
3238: if (start - last) {
3239: g_string_append_len(msg, last, start - last);
3240: }
3241:
3242: id = g_datalist_get_data(&attribs, "id");
3243:
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);
3250:
3251: oscar_id++;
3252:
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);
3262:
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>");
3268: }
3269: /* If the tag is invalid, skip it, thus no else here */
3270:
3271: g_datalist_clear(&attribs);
3272:
3273: /* continue from the end of the tag */
3274: last = end + 1;
3275: }
3276:
3277: /* append any remaining message data */
3278: if (last && *last)
3279: g_string_append(msg, last);
3280:
3281: g_string_append(msg, "</BODY></HTML>");
3282:
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);
3288:
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>");
3293: }
3294: g_string_free(data, TRUE);
3295:
3296: purple_debug_info("oscar", "sending direct IM %s using charset %i", msg->str, charset);
3297:
3298: peer_odc_send_im(conn, msg->str, msg->len, charset,
3299: imflags & PURPLE_MESSAGE_AUTO_RESP);
3300: g_string_free(msg, TRUE);
3301: }
3302:
3303: int
3304: oscar_send_im(PurpleConnection *gc, const char *name, const char *message, PurpleMessageFlags imflags)
3305: {
3306: OscarData *od;
3307: PurpleAccount *account;
3308: PeerConnection *conn;
3309: int ret;
3310: char *tmp1, *tmp2;
3311: gboolean is_sms, is_html;
3312:
3313: od = purple_connection_get_protocol_data(gc);
3314: account = purple_connection_get_account(gc);
3315: ret = 0;
3316:
3317: is_sms = oscar_util_valid_name_sms(name);
3318:
3319: if (od->icq && is_sms) {
3320: /*
3321: * We're sending to a phone number and this is ICQ,
3322: * so send the message as an SMS using aim_icq_sendsms()
3323: */
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);
3328: }
3329:
3330: if (imflags & PURPLE_MESSAGE_AUTO_RESP)
3331: tmp1 = oscar_util_format_string(message, name);
3332: else
3333: tmp1 = g_strdup(message);
3334:
3335: conn = peer_connection_find_by_type(od, name, OSCAR_CAPABILITY_DIRECTIM);
3336: if ((conn != NULL) && (conn->ready))
3337: {
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;
3347:
3348: conv = purple_find_conversation_with_account(PURPLE_CONV_TYPE_IM, name, account);
3349:
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));
3355:
3356: buddy = purple_find_buddy(account, name);
3357:
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);
3362: }
3363:
3364: args.flags = 0;
3365:
3366: if (!is_sms && (!buddy || !PURPLE_BUDDY_IS_ONLINE(buddy)))
3367: args.flags |= AIM_IMFLAGS_OFFLINE;
3368:
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);
3375:
3376: if (imflags & PURPLE_MESSAGE_AUTO_RESP)
3377: args.flags |= AIM_IMFLAGS_AWAY;
3378: }
3379:
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;
3385: }
3386:
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);
3393:
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;
3397: }
3398:
3399: /*
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.
3405: */
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;
3414: }
3415:
3416: purple_imgstore_unref(img);
3417: }
3418:
3419: args.destbn = name;
3420:
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;
3429: }
3430: g_free(tmp1);
3431: tmp1 = tmp2;
3432:
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);
3438:
3439: tmp2 = purple_markup_strip_html(tmp1);
3440: g_free(tmp1);
3441:
3442: /* re-escape the entities */
3443: tmp1 = g_markup_escape_text(tmp2, -1);
3444: g_free(tmp2);
3445:
3446: tmp2 = purple_strdup_withhtml(tmp1);
3447: g_free(tmp1);
3448: tmp1 = tmp2;
3449:
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);
3453: }
3454:
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);
3458: }
3459:
3460: g_free(tmp1);
3461:
3462: if (ret >= 0)
3463: return 1;
3464:
3465: return ret;
3466: }
3467:
3468: /*
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.
3472: */
3473: void oscar_get_info(PurpleConnection *gc, const char *name) {
3474: OscarData *od = purple_connection_get_protocol_data(gc);
3475:
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);
3480: }
3481:
3482: void oscar_set_idle(PurpleConnection *gc, int time) {
3483: OscarData *od = purple_connection_get_protocol_data(gc);
3484: aim_srv_setidle(od, time);
3485: }
3486:
3487: void
3488: oscar_set_info(PurpleConnection *gc, const char *rawinfo)
3489: {
3490: PurpleAccount *account;
3491: PurpleStatus *status;
3492:
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);
3496: }
3497:
3498: static guint32
3499: oscar_get_extended_status(PurpleConnection *gc)
3500: {
3501: PurpleAccount *account;
3502: PurpleStatus *status;
3503: const gchar *status_id;
3504: guint32 data = 0x00000000;
3505:
3506: account = purple_connection_get_account(gc);
3507: status = purple_account_get_active_status(account);
3508: status_id = purple_status_get_id(status);
3509:
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;
3513:
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;
3540:
3541: return data;
3542: }
3543:
3544: static void
3545: oscar_set_extended_status(PurpleConnection *gc)
3546: {
3547: aim_srv_setextrainfo(purple_connection_get_protocol_data(gc), TRUE, oscar_get_extended_status(gc), FALSE, NULL, NULL);
3548: }
3549:
3550: static void
3551: oscar_set_info_and_status(PurpleAccount *account, gboolean setinfo, const char *rawinfo,
3552: gboolean setstatus, PurpleStatus *status)
3553: {
3554: PurpleConnection *gc = purple_account_get_connection(account);
3555: OscarData *od = purple_connection_get_protocol_data(gc);
3556: PurpleStatusType *status_type;
3557: PurpleStatusPrimitive primitive;
3558:
3559: char *info_encoding = NULL;
3560: char *info = NULL;
3561: gsize infolen = 0;
3562:
3563: char *away_encoding = NULL;
3564: char *away = NULL;
3565: gsize awaylen = 0;
3566:
3567: char *status_text = NULL;
3568: const char *itmsurl = NULL;
3569:
3570: status_type = purple_status_get_type(status);
3571: primitive = purple_status_type_get_primitive(status_type);
3572:
3573: if (!setinfo)
3574: {
3575: /* Do nothing! */
3576: }
3577: else if (od->rights.maxsiglen == 0)
3578: {
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."));
3584: }
3585: else if (rawinfo != NULL)
3586: {
3587: char *htmlinfo = purple_strdup_withhtml(rawinfo);
3588: info = oscar_encode_im(htmlinfo, &infolen, NULL, &info_encoding);
3589: g_free(htmlinfo);
3590:
3591: if (infolen > od->rights.maxsiglen)
3592: {
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);
3601: }
3602: }
3603:
3604: if (setstatus)
3605: {
3606: const char *status_html;
3607:
3608: status_html = purple_status_get_attr_string(status, "message");
3609:
3610: if (status_html == NULL || primitive == PURPLE_STATUS_AVAILABLE || primitive == PURPLE_STATUS_INVISIBLE)
3611: {
3612: /* This is needed for us to un-set any previous away message. */
3613: away = g_strdup("");
3614: }
3615: else
3616: {
3617: gchar *linkified;
3618:
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);
3623:
3624: if (awaylen > od->rights.maxawaymsglen)
3625: {
3626: gchar *errstr;
3627:
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);
3635: }
3636: }
3637: }
3638:
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);
3644:
3645: if (setstatus)
3646: {
3647: const char *status_html;
3648:
3649: status_html = purple_status_get_attr_string(status, "message");
3650: if (status_html != NULL)
3651: {
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)
3655: {
3656: char *tmp = g_utf8_find_prev_char(status_text, &status_text[MAXAVAILMSGLEN - 2]);
3657: strcpy(tmp, "...");
3658: }
3659: }
3660:
3661: itmsurl = purple_status_get_attr_string(status, "itmsurl");
3662:
3663: aim_srv_setextrainfo(od, TRUE, oscar_get_extended_status(gc), TRUE, status_text, itmsurl);
3664: g_free(status_text);
3665: }
3666: }
3667:
3668: static void
3669: oscar_set_icq_permdeny(PurpleAccount *account)
3670: {
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);
3674:
3675: /*
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.
3681: */
3682: aim_ssi_setpermdeny(od, invisible ? PURPLE_PRIVACY_ALLOW_USERS : PURPLE_PRIVACY_DENY_USERS);
3683: }
3684:
3685: void
3686: oscar_set_status(PurpleAccount *account, PurpleStatus *status)
3687: {
3688: PurpleConnection *pc;
3689: OscarData *od;
3690:
3691: purple_debug_info("oscar", "Set status to %s\n", purple_status_get_name(status));
3692:
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;
3697:
3698: if (!purple_account_is_connected(account))
3699: return;
3700:
3701: pc = purple_account_get_connection(account);
3702: od = purple_connection_get_protocol_data(pc);
3703:
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;
3708: }
3709:
3710: if (od->icq) {
3711: /* Set visibility */
3712: oscar_set_icq_permdeny(account);
3713: }
3714:
3715: /* Set the AIM-style away message for both AIM and ICQ accounts */
3716: oscar_set_info_and_status(account, FALSE, NULL, TRUE, status);
3717: }
3718:
3719: void
3720: oscar_add_buddy(PurpleConnection *gc, PurpleBuddy *buddy, PurpleGroup *group, const char *msg)
3721: {
3722: OscarData *od;
3723: PurpleAccount *account;
3724: const char *bname, *gname;
3725:
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);
3730:
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);
3737:
3738: /* Remove from local list */
3739: purple_blist_remove_buddy(buddy);
3740:
3741: return;
3742: }
3743:
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);
3749:
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);
3756: }
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);
3762: }
3763: }
3764:
3765: /* XXX - Should this be done from AIM accounts, as well? */
3766: if (od->icq)
3767: aim_icq_getalias(od, bname, FALSE, NULL);
3768: }
3769:
3770: void oscar_remove_buddy(PurpleConnection *gc, PurpleBuddy *buddy, PurpleGroup *group) {
3771: OscarData *od = purple_connection_get_protocol_data(gc);
3772:
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);
3779: }
3780: }
3781:
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);
3784:
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);
3789: }
3790: }
3791:
3792: void oscar_alias_buddy(PurpleConnection *gc, const char *name, const char *alias) {
3793: OscarData *od = purple_connection_get_protocol_data(gc);
3794:
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);
3801: }
3802: }
3803: }
3804:
3805: /*
3806: * FYI, the OSCAR SSI code removes empty groups automatically.
3807: */
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);
3810:
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);
3816:
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));
3824: }
3825:
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);
3835: }
3836: }
3837: }
3838:
3839: void oscar_remove_group(PurpleConnection *gc, PurpleGroup *group)
3840: {
3841: aim_ssi_delgroup(purple_connection_get_protocol_data(gc), purple_group_get_name(group));
3842: }
3843:
3844: static gboolean purple_ssi_rerequestdata(gpointer data) {
3845: OscarData *od = data;
3846:
3847: aim_ssi_reqdata(od);
3848:
3849: return TRUE;
3850: }
3851:
3852: static int purple_ssi_parseerr(OscarData *od, FlapConnection *conn, FlapFrame *fr, ...) {
3853: PurpleConnection *gc = od->gc;
3854: va_list ap;
3855: guint16 reason;
3856:
3857: va_start(ap, fr);
3858: reason = (guint16)va_arg(ap, unsigned int);
3859: va_end(ap);
3860:
3861: purple_debug_error("oscar", "ssi: SNAC error %hu\n", reason);
3862:
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;
3875: }
3876:
3877: return 1;
3878: }
3879:
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;
3886:
3887: va_start(ap, fr);
3888: numtypes = va_arg(ap, int);
3889: maxitems = va_arg(ap, guint16 *);
3890: va_end(ap);
3891:
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);
3898:
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];
3907:
3908: return 1;
3909: }
3910:
3911: static int purple_ssi_parselist(OscarData *od, FlapConnection *conn, FlapFrame *fr, ...)
3912: {
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);
3923:
3924: gc = od->gc;
3925: od = purple_connection_get_protocol_data(gc);
3926: account = purple_connection_get_account(gc);
3927:
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);
3933:
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;
3938: }
3939:
3940: purple_debug_info("oscar", "ssi: syncing local list and server list\n");
3941:
3942: /* Clean the buddy list */
3943: aim_ssi_cleanlist(od);
3944:
3945: /*** Begin code for pruning buddies from local list if they're not in server list ***/
3946:
3947: /* Buddies */
3948: cur = NULL;
3949: for (buddies = purple_find_buddies(account, NULL);
3950: buddies;
3951: buddies = g_slist_delete_link(buddies, buddies))
3952: {
3953: PurpleGroup *g;
3954: const char *gname;
3955: const char *bname;
3956:
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);
3961:
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);
3969:
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);
3981: }
3982: }
3983: while (cur != NULL) {
3984: purple_blist_remove_buddy(cur->data);
3985: cur = g_slist_delete_link(cur, cur);
3986: }
3987:
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);
3998: }
3999: }
4000: }
4001:
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);
4011: }
4012: }
4013:
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;
4019:
4020: idle_reporting_pref = purple_prefs_get_string("/purple/away/idle_reporting");
4021: report_idle = !purple_strequal(idle_reporting_pref, "none");
4022:
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);
4027: }
4028:
4029: /*** End code for pruning buddies from local list ***/
4030:
4031: /*** Begin code for adding from server list to local list ***/
4032:
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;
4039: }
4040:
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;
4046:
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);
4050:
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);
4055: }
4056:
4057: alias = aim_ssi_getalias(od->ssi.local, gname, curitem->name);
4058: alias_utf8 = oscar_utf8_try_convert(account, od, alias);
4059:
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);
4066:
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);
4070: }
4071:
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);
4080: }
4081:
4082: g_free(gname_utf8);
4083: g_free(alias);
4084: g_free(alias_utf8);
4085: }
4086: } break;
4087:
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);
4092: }
4093: } break;
4094:
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);
4102: }
4103: }
4104: } break;
4105:
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);
4114: }
4115: }
4116: } break;
4117:
4118: case AIM_SSI_TYPE_PDINFO: { /* Permit/deny setting */
4119: /*
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.
4124: */
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)
4128: {
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;
4132: }
4133: }
4134: } break;
4135:
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 */
4141:
4142: /*** End code for adding from server list to local list ***/
4143:
4144: if (od->icq) {
4145: oscar_set_icq_permdeny(account);
4146: } else {
4147: oscar_set_aim_permdeny(gc);
4148: }
4149:
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);
4156:
4157: /*
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.
4161: */
4162: img = purple_buddy_icons_find_account_icon(account);
4163: oscar_set_icon(gc, img);
4164: purple_imgstore_unref(img);
4165:
4166: /*
4167: * If we've already received our bos rights then we're not waiting on
4168: * anything else, so send the server clientready.
4169: */
4170: if (od->bos.have_rights) {
4171: aim_srv_clientready(od, conn);
4172:
4173: /* Request offline messages for AIM and ICQ */
4174: aim_im_reqofflinemsgs(od);
4175:
4176: purple_connection_set_state(gc, PURPLE_CONNECTED);
4177: }
4178:
4179: return 1;
4180: }
4181:
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;
4186:
4187: va_start(ap, fr);
4188: retval = va_arg(ap, struct aim_ssi_tmp *);
4189: va_end(ap);
4190:
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");
4194:
4195: if (retval->ack != 0xffff)
4196: switch (retval->ack) {
4197: case 0x0000: { /* added successfully */
4198: } break;
4199:
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;
4207:
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;
4212:
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;
4222: }
4223:
4224: retval = retval->next;
4225: }
4226:
4227: return 1;
4228: }
4229:
4230: static int
4231: purple_ssi_parseaddmod(OscarData *od, FlapConnection *conn, FlapFrame *fr, ...)
4232: {
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;
4242:
4243: gc = od->gc;
4244: account = purple_connection_get_account(gc);
4245:
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);
4251:
4252: if ((type != 0x0000) || (name == NULL))
4253: return 1;
4254:
4255: gname = aim_ssi_itemlist_findparentname(od->ssi.local, name);
4256: gname_utf8 = gname ? oscar_utf8_try_convert(account, od, gname) : NULL;
4257:
4258: alias = aim_ssi_getalias(od->ssi.local, gname, name);
4259: alias_utf8 = oscar_utf8_try_convert(account, od, alias);
4260: g_free(alias);
4261:
4262: b = purple_find_buddy(account, name);
4263: if (b) {
4264: /*
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.
4268: */
4269: purple_blist_alias_buddy(b, alias_utf8);
4270: } else if (snac_subtype == 0x0008) {
4271: /*
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.
4274: */
4275: b = purple_buddy_new(account, name, alias_utf8);
4276:
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);
4280: }
4281:
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);
4285:
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);
4292: }
4293:
4294: }
4295:
4296: ssi_item = aim_ssi_itemlist_finditem(od->ssi.local,
4297: gname, name, AIM_SSI_TYPE_BUDDY);
4298: if (ssi_item == NULL)
4299: {
4300: purple_debug_error("oscar", "purple_ssi_parseaddmod: "
4301: "Could not find ssi item for oncoming buddy %s, "
4302: "group %s\n", name, gname);
4303: }
4304:
4305: g_free(gname_utf8);
4306: g_free(alias_utf8);
4307:
4308: return 1;
4309: }
4310:
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;
4318:
4319: va_start(ap, fr);
4320: bn = va_arg(ap, char *);
4321: va_arg(ap, char *); /* msg */
4322: va_end(ap);
4323:
4324: purple_debug_info("oscar",
4325: "ssi: %s has given you permission to add him to your buddy list\n", bn);
4326:
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);
4332:
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);
4335:
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);
4340:
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);
4348:
4349: return 1;
4350: }
4351:
4352: static int purple_ssi_authrequest(OscarData *od, FlapConnection *conn, FlapFrame *fr, ...)
4353: {
4354: va_list ap;
4355: const char *bn;
4356: char *msg;
4357:
4358: va_start(ap, fr);
4359: bn = va_arg(ap, const char *);
4360: msg = va_arg(ap, char *);
4361: va_end(ap);
4362:
4363: purple_debug_info("oscar",
4364: "ssi: received authorization request from %s\n", bn);
4365:
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;
4373: }
4374:
4375: aim_icq_getalias(od, bn, TRUE, msg);
4376: return 1;
4377: }
4378:
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;
4386:
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);
4392:
4393: purple_debug_info("oscar",
4394: "ssi: received authorization reply from %s. Reply is 0x%04hhx\n", bn, reply);
4395:
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);
4401:
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);
4410: }
4411: g_free(dialog_msg);
4412: g_free(nombre);
4413:
4414: return 1;
4415: }
4416:
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;
4423:
4424: va_start(ap, fr);
4425: bn = va_arg(ap, char *);
4426: va_end(ap);
4427:
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);
4432:
4433: return 1;
4434: }
4435:
4436: GList *oscar_chat_info(PurpleConnection *gc) {
4437: GList *m = NULL;
4438: struct proto_chat_entry *pce;
4439:
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);
4445:
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);
4454:
4455: return m;
4456: }
4457:
4458: GHashTable *oscar_chat_info_defaults(PurpleConnection *gc, const char *chat_name)
4459: {
4460: GHashTable *defaults;
4461:
4462: defaults = g_hash_table_new_full(g_str_hash, g_str_equal, NULL, g_free);
4463:
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"));
4467:
4468: return defaults;
4469: }
4470:
4471: char *
4472: oscar_get_chat_name(GHashTable *data)
4473: {
4474: return g_strdup(g_hash_table_lookup(data, "room"));
4475: }
4476:
4477: void
4478: oscar_join_chat(PurpleConnection *gc, GHashTable *data)
4479: {
4480: OscarData *od = purple_connection_get_protocol_data(gc);
4481: FlapConnection *conn;
4482: char *name, *exchange;
4483: int exchange_int;
4484:
4485: name = g_hash_table_lookup(data, "room");
4486: exchange = g_hash_table_lookup(data, "exchange");
4487:
4488: g_return_if_fail(name != NULL && *name != '\0');
4489: g_return_if_fail(exchange != NULL);
4490:
4491: errno = 0;
4492: exchange_int = strtol(exchange, NULL, 10);
4493: g_return_if_fail(errno == 0);
4494:
4495: purple_debug_info("oscar", "Attempting to join chat room %s.\n", name);
4496:
4497: if ((conn = flap_connection_getbytype(od, SNAC_FAMILY_CHATNAV)))
4498: {
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);
4509: }
4510: }
4511:
4512: void
4513: oscar_chat_invite(PurpleConnection *gc, int id, const char *message, const char *name)
4514: {
4515: OscarData *od = purple_connection_get_protocol_data(gc);
4516: struct chat_connection *ccon = find_oscar_chat(gc, id);
4517:
4518: if (ccon == NULL)
4519: return;
4520:
4521: aim_im_sendch2_chatinvite(od, name, message ? message : "",
4522: ccon->exchange, ccon->name, 0x0);
4523: }
4524:
4525: void
4526: oscar_chat_leave(PurpleConnection *gc, int id)
4527: {
4528: PurpleConversation *conv;
4529: struct chat_connection *cc;
4530:
4531: conv = purple_find_chat(gc, id);
4532:
4533: g_return_if_fail(conv != NULL);
4534:
4535: purple_debug_info("oscar", "Leaving chat room %s\n",
4536: purple_conversation_get_name(conv));
4537:
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);
4541: }
4542:
4543: int oscar_send_chat(PurpleConnection *gc, int id, const char *message, PurpleMessageFlags flags)
4544: {
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;
4552:
4553: if (!(conv = purple_find_chat(gc, id)))
4554: return -EINVAL;
4555:
4556: if (!(c = find_oscar_chat_by_conv(gc, conv)))
4557: return -EINVAL;
4558:
4559: buf = purple_strdup_withhtml(message);
4560:
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));
4566:
4567: buf2 = oscar_encode_im(buf, &len, &charset, &charsetstr);
4568: /*
4569: * Evan S. suggested that maxvis really does mean "number of
4570: * visible characters" and not "number of bytes"
4571: */
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);
4576:
4577: buf3 = purple_markup_strip_html(buf);
4578: g_free(buf);
4579:
4580: buf = purple_strdup_withhtml(buf3);
4581: g_free(buf3);
4582:
4583: buf2 = oscar_encode_im(buf, &len, &charset, &charsetstr);
4584:
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;
4592: }
4593:
4594: purple_debug_info("oscar", "Sending %s as %s because the original was too long.\n",
4595: message, buf2);
4596: }
4597:
4598: aim_chat_send_im(od, c->conn, 0, buf2, len, charsetstr, "en");
4599: g_free(buf2);
4600: g_free(buf);
4601:
4602: return 0;
4603: }
4604:
4605: PurpleMood* oscar_get_purple_moods(PurpleAccount *account)
4606: {
4607: return icq_get_purple_moods(account);
4608: }
4609:
4610: const char *oscar_list_icon_icq(PurpleAccount *a, PurpleBuddy *b)
4611: {
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";
4615:
4616: return "icq";
4617: }
4618:
4619: const char *oscar_list_icon_aim(PurpleAccount *a, PurpleBuddy *b)
4620: {
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";
4624:
4625: return "aim";
4626: }
4627:
4628: const char *oscar_list_emblem(PurpleBuddy *b)
4629: {
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;
4636:
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);
4645:
4646: presence = purple_buddy_get_presence(b);
4647:
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";
4654: }
4655: }
4656:
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";
4666:
4667: /* Make the mood icon override anything below this. */
4668: if (purple_presence_is_status_primitive_active(presence, PURPLE_STATUS_MOOD))
4669: return NULL;
4670:
4671: if (userinfo->capabilities & OSCAR_CAPABILITY_HIPTOP)
4672: return "hiptop";
4673: }
4674: return NULL;
4675: }
4676:
4677: void oscar_tooltip_text(PurpleBuddy *b, PurpleNotifyUserInfo *user_info, gboolean full)
4678: {
4679: PurpleConnection *gc;
4680: PurpleAccount *account;
4681: OscarData *od;
4682: aim_userinfo_t *userinfo;
4683:
4684: if (!PURPLE_BUDDY_IS_ONLINE(b))
4685: return;
4686:
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));
4691:
4692: oscar_user_info_append_status(gc, user_info, b, userinfo, /* use_html_status */ FALSE);
4693:
4694: if (full)
4695: oscar_user_info_append_extra_info(gc, user_info, b, userinfo);
4696: }
4697:
4698: char *oscar_status_text(PurpleBuddy *b)
4699: {
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;
4707:
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);
4713:
4714: if ((od != NULL) && !purple_presence_is_online(presence))
4715: {
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"));
4722: }
4723: else
4724: {
4725: message = purple_status_get_attr_string(status, "message");
4726: if (message != NULL)
4727: {
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);
4731: }
4732: else if (purple_status_is_available(status))
4733: {
4734: /* Don't show "Available" as status message in case buddy doesn't have a status message */
4735: }
4736: else
4737: {
4738: ret = g_strdup(purple_status_get_name(status));
4739: }
4740: }
4741:
4742: return ret;
4743: }
4744:
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);
4748:
4749: /*
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.
4753: */
4754: aim_ssi_setpermdeny(od, account->perm_deny);
4755: }
4756:
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);
4761: }
4762:
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));
4767: }
4768:
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);
4773: }
4774:
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));
4779: }
4780:
4781: GList *
4782: oscar_status_types(PurpleAccount *account)
4783: {
4784: gboolean is_icq;
4785: GList *status_types = NULL;
4786: PurpleStatusType *type;
4787:
4788: g_return_val_if_fail(account != NULL, NULL);
4789:
4790: /* Used to flag some statuses as "user settable" or not */
4791: is_icq = oscar_util_valid_name_icq(purple_account_get_username(account));
4792:
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);
4803:
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);
4809:
4810: status_types = g_list_prepend(status_types, type);
4811:
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);
4818:
4819:
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);
4826:
4827:
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);
4834:
4835:
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);
4841:
4842: status_types = g_list_prepend(status_types, type);
4843:
4844:
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);
4850:
4851: status_types = g_list_prepend(status_types, type);
4852:
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);
4859:
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);
4865:
4866: status_types = g_list_prepend(status_types, type);
4867:
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);
4870:
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);
4878:
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);
4885:
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);
4892:
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);
4897:
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);
4904:
4905: return g_list_reverse(status_types);
4906: }
4907:
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;
4914:
4915: gc = data->gc;
4916: od = purple_connection_get_protocol_data(gc);
4917: account = purple_connection_get_account(gc);
4918:
4919: b = purple_find_buddy(account, data->name);
4920: if (b == NULL) {
4921: oscar_free_name_data(data);
4922: return;
4923: }
4924:
4925: g = purple_buddy_get_group(b);
4926: if (g == NULL) {
4927: oscar_free_name_data(data);
4928: return;
4929: }
4930:
4931: aim_ssi_editcomment(od, purple_group_get_name(g), data->name, text);
4932: oscar_free_name_data(data);
4933: }
4934:
4935: static void oscar_buddycb_edit_comment(PurpleBlistNode *node, gpointer ignore) {
4936:
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;
4947:
4948: g_return_if_fail(PURPLE_BLIST_NODE_IS_BUDDY(node));
4949:
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);
4955:
4956: if (!(g = purple_buddy_get_group(buddy)))
4957: return;
4958:
4959: data = g_new(struct name_data, 1);
4960:
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;
4963:
4964: data->gc = gc;
4965: data->name = g_strdup(name);
4966: data->nick = g_strdup(purple_buddy_get_alias_only(buddy));
4967:
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);
4976:
4977: g_free(comment);
4978: g_free(comment_utf8);
4979: }
4980:
4981: static void
4982: oscar_ask_directim_yes_cb(struct oscar_ask_directim_data *data)
4983: {
4984: peer_connection_propose(data->od, OSCAR_CAPABILITY_DIRECTIM, data->who);
4985: g_free(data->who);
4986: g_free(data);
4987: }
4988:
4989: static void
4990: oscar_ask_directim_no_cb(struct oscar_ask_directim_data *data)
4991: {
4992: g_free(data->who);
4993: g_free(data);
4994: }
4995:
4996: /* This is called from right-click menu on a buddy node. */
4997: static void
4998: oscar_ask_directim(gpointer object, gpointer ignored)
4999: {
5000: PurpleBlistNode *node;
5001: PurpleBuddy *buddy;
5002: PurpleConnection *gc;
5003: gchar *buf;
5004: struct oscar_ask_directim_data *data;
5005: PurpleAccount *account;
5006:
5007: node = object;
5008:
5009: g_return_if_fail(PURPLE_BLIST_NODE_IS_BUDDY(node));
5010:
5011: buddy = (PurpleBuddy *)node;
5012: account = purple_buddy_get_account(buddy);
5013: gc = purple_account_get_connection(account);
5014:
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);
5020:
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);
5031: }
5032:
5033: static void
5034: oscar_close_directim(gpointer object, gpointer ignored)
5035: {
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;
5044:
5045: node = object;
5046:
5047: g_return_if_fail(PURPLE_BLIST_NODE_IS_BUDDY(node));
5048:
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);
5055:
5056: if (conn != NULL)
5057: {
5058: if (!conn->ready)
5059: aim_im_sendch2_cancel(conn);
5060:
5061: peer_connection_destroy(conn, OSCAR_DISCONNECT_LOCAL_CLOSED, NULL);
5062:
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));
5068: }
5069: }
5070:
5071: static void oscar_get_icqxstatusmsg(PurpleBlistNode *node, gpointer ignore)
5072: {
5073: PurpleBuddy *buddy;
5074: PurpleConnection *gc;
5075: OscarData *od;
5076: PurpleAccount *account;
5077: const char *bname;
5078:
5079: g_return_if_fail(PURPLE_BLIST_NODE_IS_BUDDY(node));
5080:
5081: buddy = (PurpleBuddy *)node;
5082: bname = purple_buddy_get_name(buddy);
5083:
5084: account = purple_buddy_get_account(buddy);
5085: gc = purple_account_get_connection(account);
5086: od = purple_connection_get_protocol_data(gc);
5087:
5088: purple_debug_info("oscar", "Manual X-Status Get From %s to %s:\n", bname, purple_account_get_username(account));
5089:
5090: icq_im_xstatus_request(od, bname);
5091: }
5092:
5093: static void
5094: oscar_get_aim_info_cb(PurpleBlistNode *node, gpointer ignore)
5095: {
5096: PurpleBuddy *buddy;
5097: PurpleConnection *gc;
5098:
5099: g_return_if_fail(PURPLE_BLIST_NODE_IS_BUDDY(node));
5100:
5101: buddy = (PurpleBuddy *)node;
5102: gc = purple_account_get_connection(purple_buddy_get_account(buddy));
5103:
5104: aim_locate_getinfoshort(purple_connection_get_protocol_data(gc),
5105: purple_buddy_get_name(buddy), 0x00000003);
5106: }
5107:
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);
5117:
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;
5123:
5124: if (od->icq && oscar_util_valid_name_icq(bname))
5125: {
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);
5130: }
5131:
5132: if (purple_buddy_get_group(buddy) != NULL)
5133: {
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);
5139: }
5140:
5141: if (od->icq)
5142: {
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));
5148: }
5149:
5150: if (userinfo &&
5151: oscar_util_name_compare(purple_account_get_username(account), bname) &&
5152: PURPLE_BUDDY_IS_ONLINE(buddy))
5153: {
5154: PeerConnection *conn;
5155: conn = peer_connection_find_by_type(od, bname, OSCAR_CAPABILITY_DIRECTIM);
5156:
5157: if (userinfo->capabilities & OSCAR_CAPABILITY_DIRECTIM)
5158: {
5159: if (conn)
5160: {
5161: act = purple_menu_action_new(_("End Direct IM Session"),
5162: PURPLE_CALLBACK(oscar_close_directim),
5163: NULL, NULL);
5164: }
5165: else
5166: {
5167: act = purple_menu_action_new(_("Direct IM"),
5168: PURPLE_CALLBACK(oscar_ask_directim),
5169: NULL, NULL);
5170: }
5171: menu = g_list_prepend(menu, act);
5172: }
5173: }
5174:
5175: if (od->ssi.received_data && purple_buddy_get_group(buddy) != NULL)
5176: {
5177: /*
5178: * We only do this if the user is in our buddy list and we're
5179: * waiting for authorization.
5180: */
5181: char *gname;
5182: gname = aim_ssi_itemlist_findparentname(od->ssi.local, bname);
5183: if (gname && aim_ssi_waitingforauth(od->ssi.local, gname, bname))
5184: {
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);
5189: }
5190: }
5191:
5192: menu = g_list_reverse(menu);
5193:
5194: return menu;
5195: }
5196:
5197:
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;
5203: }
5204: }
5205:
5206: static void
5207: oscar_icq_privacy_opts(PurpleConnection *gc, PurpleRequestFields *fields)
5208: {
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;
5213:
5214: f = purple_request_fields_get_field(fields, "authorization");
5215: auth = purple_request_field_bool_get_value(f);
5216:
5217: f = purple_request_fields_get_field(fields, "web_aware");
5218: web_aware = purple_request_field_bool_get_value(f);
5219:
5220: purple_account_set_bool(account, "authorization", auth);
5221: purple_account_set_bool(account, "web_aware", web_aware);
5222:
5223: oscar_set_extended_status(gc);
5224: aim_icq_setsecurity(od, auth, web_aware);
5225: }
5226:
5227: static void
5228: oscar_show_icq_privacy_opts(PurplePluginAction *action)
5229: {
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;
5236:
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);
5239:
5240: fields = purple_request_fields_new();
5241:
5242: g = purple_request_field_group_new(NULL);
5243:
5244: f = purple_request_field_bool_new("authorization", _("Require authorization"), auth);
5245: purple_request_field_group_add_field(g, f);
5246:
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);
5249:
5250: purple_request_fields_add_group(fields, g);
5251:
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);
5258: }
5259:
5260: static void oscar_confirm_account(PurplePluginAction *action)
5261: {
5262: PurpleConnection *gc;
5263: OscarData *od;
5264: FlapConnection *conn;
5265:
5266: gc = (PurpleConnection *)action->context;
5267: od = purple_connection_get_protocol_data(gc);
5268:
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);
5275: }
5276: }
5277:
5278: static void oscar_show_email(PurplePluginAction *action)
5279: {
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);
5283:
5284: if (conn) {
5285: aim_admin_getinfo(od, conn, 0x11);
5286: } else {
5287: od->reqemail = TRUE;
5288: aim_srv_requestnew(od, SNAC_FAMILY_ADMIN);
5289: }
5290: }
5291:
5292: static void oscar_change_email(PurpleConnection *gc, const char *email)
5293: {
5294: OscarData *od = purple_connection_get_protocol_data(gc);
5295: FlapConnection *conn = flap_connection_getbytype(od, SNAC_FAMILY_ADMIN);
5296:
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);
5303: }
5304: }
5305:
5306: static void oscar_show_change_email(PurplePluginAction *action)
5307: {
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);
5315: }
5316:
5317: static void oscar_show_awaitingauth(PurplePluginAction *action)
5318: {
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;
5324:
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;
5330:
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);
5336: }
5337: }
5338:
5339: g_slist_free(buddies);
5340:
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);
5344:
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);
5351: }
5352:
5353: static void search_by_email_cb(PurpleConnection *gc, const char *email)
5354: {
5355: OscarData *od = purple_connection_get_protocol_data(gc);
5356:
5357: aim_search_address(od, email);
5358: }
5359:
5360: static void oscar_show_find_email(PurplePluginAction *action)
5361: {
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);
5372: }
5373:
5374: static void oscar_show_set_info(PurplePluginAction *action)
5375: {
5376: PurpleConnection *gc = (PurpleConnection *) action->context;
5377: purple_account_request_change_user_info(purple_connection_get_account(gc));
5378: }
5379:
5380: static void oscar_show_set_info_icqurl(PurplePluginAction *action)
5381: {
5382: PurpleConnection *gc = (PurpleConnection *) action->context;
5383: purple_notify_uri(gc, "http://www.icq.com/whitepages/user_details.php");
5384: }
5385:
5386: static void oscar_change_pass(PurplePluginAction *action)
5387: {
5388: PurpleConnection *gc = (PurpleConnection *) action->context;
5389: purple_account_request_change_password(purple_connection_get_account(gc));
5390: }
5391:
5392: /**
5393: * Only used when connecting with the old-style BUCP login.
5394: */
5395: static void oscar_show_chpassurl(PurplePluginAction *action)
5396: {
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);
5402: }
5403:
5404: static void oscar_show_imforwardingurl(PurplePluginAction *action)
5405: {
5406: PurpleConnection *gc = (PurpleConnection *) action->context;
5407: purple_notify_uri(gc, "http://mymobile.aol.com/dbreg/register?action=imf&clientID=1");
5408: }
5409:
5410: void oscar_set_icon(PurpleConnection *gc, PurpleStoredImage *img)
5411: {
5412: OscarData *od = purple_connection_get_protocol_data(gc);
5413:
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);
5421:
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);
5426:
5427: aim_ssi_seticon(od, md5, 16);
5428: }
5429: }
5430:
5431: /**
5432: * Called by the Purple core to determine whether or not we're
5433: * allowed to send a file to this user.
5434: */
5435: gboolean
5436: oscar_can_receive_file(PurpleConnection *gc, const char *who)
5437: {
5438: OscarData *od;
5439: PurpleAccount *account;
5440:
5441: od = purple_connection_get_protocol_data(gc);
5442: account = purple_connection_get_account(gc);
5443:
5444: if (od != NULL)
5445: {
5446: aim_userinfo_t *userinfo;
5447: userinfo = aim_locate_finduserinfo(od, who);
5448:
5449: /*
5450: * Don't allowing sending a file to a user that does not support
5451: * file transfer, and don't allow sending to ourselves.
5452: */
5453: if (((userinfo == NULL) ||
5454: (userinfo->capabilities & OSCAR_CAPABILITY_SENDFILE)) &&
5455: oscar_util_name_compare(who, purple_account_get_username(account)))
5456: {
5457: return TRUE;
5458: }
5459: }
5460:
5461: return FALSE;
5462: }
5463:
5464: PurpleXfer *
5465: oscar_new_xfer(PurpleConnection *gc, const char *who)
5466: {
5467: PurpleXfer *xfer;
5468: OscarData *od;
5469: PurpleAccount *account;
5470: PeerConnection *conn;
5471:
5472: od = purple_connection_get_protocol_data(gc);
5473: account = purple_connection_get_account(gc);
5474:
5475: xfer = purple_xfer_new(account, PURPLE_XFER_SEND, who);
5476: if (xfer)
5477: {
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);
5483:
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;
5490: }
5491:
5492: return xfer;
5493: }
5494:
5495: /*
5496: * Called by the Purple core when the user indicates that a
5497: * file is to be sent to a special someone.
5498: */
5499: void
5500: oscar_send_file(PurpleConnection *gc, const char *who, const char *file)
5501: {
5502: PurpleXfer *xfer;
5503:
5504: xfer = oscar_new_xfer(gc, who);
5505:
5506: if (file != NULL)
5507: purple_xfer_request_accepted(xfer, file);
5508: else
5509: purple_xfer_request(xfer);
5510: }
5511:
5512: GList *
5513: oscar_actions(PurplePlugin *plugin, gpointer context)
5514: {
5515: PurpleConnection *gc = (PurpleConnection *) context;
5516: OscarData *od = purple_connection_get_protocol_data(gc);
5517: GList *menu = NULL;
5518: PurplePluginAction *act;
5519:
5520: act = purple_plugin_action_new(_("Set User Info..."),
5521: oscar_show_set_info);
5522: menu = g_list_prepend(menu, act);
5523:
5524: if (od->icq)
5525: {
5526: act = purple_plugin_action_new(_("Set User Info (web)..."),
5527: oscar_show_set_info_icqurl);
5528: menu = g_list_prepend(menu, act);
5529: }
5530:
5531: act = purple_plugin_action_new(_("Change Password..."),
5532: oscar_change_pass);
5533: menu = g_list_prepend(menu, act);
5534:
5535: if (od->authinfo != NULL && od->authinfo->chpassurl != NULL)
5536: {
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);
5541: }
5542:
5543: if (!od->icq)
5544: {
5545: act = purple_plugin_action_new(_("Configure IM Forwarding (web)"),
5546: oscar_show_imforwardingurl);
5547: menu = g_list_prepend(menu, act);
5548: }
5549:
5550: menu = g_list_prepend(menu, NULL);
5551:
5552: if (od->icq)
5553: {
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);
5558:
5559: act = purple_plugin_action_new(_("Show Visible List"), oscar_show_visible_list);
5560: menu = g_list_prepend(menu, act);
5561:
5562: act = purple_plugin_action_new(_("Show Invisible List"), oscar_show_invisible_list);
5563: menu = g_list_prepend(menu, act);
5564: }
5565: else
5566: {
5567: /* AIM actions */
5568: act = purple_plugin_action_new(_("Confirm Account"),
5569: oscar_confirm_account);
5570: menu = g_list_prepend(menu, act);
5571:
5572: act = purple_plugin_action_new(_("Display Currently Registered Email Address"),
5573: oscar_show_email);
5574: menu = g_list_prepend(menu, act);
5575:
5576: act = purple_plugin_action_new(_("Change Currently Registered Email Address..."),
5577: oscar_show_change_email);
5578: menu = g_list_prepend(menu, act);
5579: }
5580:
5581: menu = g_list_prepend(menu, NULL);
5582:
5583: act = purple_plugin_action_new(_("Show Buddies Awaiting Authorization"),
5584: oscar_show_awaitingauth);
5585: menu = g_list_prepend(menu, act);
5586:
5587: menu = g_list_prepend(menu, NULL);
5588:
5589: act = purple_plugin_action_new(_("Search for Buddy by Email Address..."),
5590: oscar_show_find_email);
5591: menu = g_list_prepend(menu, act);
5592:
5593: menu = g_list_reverse(menu);
5594:
5595: return menu;
5596: }
5597:
5598: void oscar_change_passwd(PurpleConnection *gc, const char *old, const char *new)
5599: {
5600: OscarData *od = purple_connection_get_protocol_data(gc);
5601:
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);
5614: }
5615: }
5616: }
5617:
5618: void
5619: oscar_convo_closed(PurpleConnection *gc, const char *who)
5620: {
5621: OscarData *od;
5622: PeerConnection *conn;
5623:
5624: od = purple_connection_get_protocol_data(gc);
5625: conn = peer_connection_find_by_type(od, who, OSCAR_CAPABILITY_DIRECTIM);
5626:
5627: if (conn != NULL)
5628: {
5629: if (!conn->ready)
5630: aim_im_sendch2_cancel(conn);
5631:
5632: peer_connection_destroy(conn, OSCAR_DISCONNECT_LOCAL_CLOSED, NULL);
5633: }
5634: }
5635:
5636: const char *
5637: oscar_normalize(const PurpleAccount *account, const char *str)
5638: {
5639: static char buf[BUF_LEN];
5640: char *tmp1, *tmp2;
5641: int i, j;
5642:
5643: g_return_val_if_fail(str != NULL, NULL);
5644:
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;
5652: }
5653: }
5654: buf[i] = '\0';
5655:
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");
5660: }
5661: g_strlcpy(buf, tmp2, sizeof(buf));
5662: g_free(tmp2);
5663: g_free(tmp1);
5664:
5665: return buf;
5666: }
5667:
5668: gboolean
5669: oscar_offline_message(const PurpleBuddy *buddy)
5670: {
5671: return TRUE;
5672: }
5673:
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.
5676: */
5677: static PurpleAccount *find_acct(const char *prpl, const char *acct_id)
5678: {
5679: PurpleAccount *acct = NULL;
5680:
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;
5693: }
5694: l = l->next;
5695: }
5696: }
5697:
5698: return acct;
5699: }
5700:
5701:
5702: static gboolean oscar_uri_handler(const char *proto, const char *cmd, GHashTable *params)
5703: {
5704: char *acct_id = g_hash_table_lookup(params, "account");
5705: char prpl[11];
5706: PurpleAccount *acct;
5707:
5708: if (g_ascii_strcasecmp(proto, "aim") && g_ascii_strcasecmp(proto, "icq"))
5709: return FALSE;
5710:
5711: g_snprintf(prpl, sizeof(prpl), "prpl-%s", proto);
5712:
5713: acct = find_acct(prpl, acct_id);
5714:
5715: if (!acct)
5716: return FALSE;
5717:
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");
5723:
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);
5729:
5730: if (message) {
5731: /* Spaces are encoded as '+' */
5732: g_strdelimit(message, "+", ' ');
5733: purple_conv_send_confirm(conv, message);
5734: }
5735: }
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();*/
5740:
5741: return TRUE;
5742: }
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);
5751: }
5752: /*else
5753: ** Same as above (except that this would have to be re-written using purple_request_*)
5754: pidgin_blist_joinchat_show(); */
5755:
5756: return TRUE;
5757: }
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;
5764: }
5765:
5766: return FALSE;
5767: }
5768:
5769: void oscar_init(PurplePlugin *plugin, gboolean is_icq)
5770: {
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
5779: };
5780: static const gchar *encryption_values[] = {
5781: OSCAR_OPPORTUNISTIC_ENCRYPTION,
5782: OSCAR_REQUIRE_ENCRYPTION,
5783: OSCAR_NO_ENCRYPTION,
5784: NULL
5785: };
5786: static const gchar *aim_login_keys[] = {
5787: N_("clientLogin"),
5788: N_("Kerberos"),
5789: N_("MD5-based"),
5790: NULL
5791: };
5792: static const gchar *aim_login_values[] = {
5793: OSCAR_CLIENT_LOGIN,
5794: OSCAR_KERBEROS_LOGIN,
5795: OSCAR_MD5_LOGIN,
5796: NULL
5797: };
5798: static const gchar *icq_login_keys[] = {
5799: N_("clientLogin"),
5800: N_("MD5-based"),
5801: NULL
5802: };
5803: static const gchar *icq_login_values[] = {
5804: OSCAR_CLIENT_LOGIN,
5805: OSCAR_MD5_LOGIN,
5806: NULL
5807: };
5808: const gchar **login_keys;
5809: const gchar **login_values;
5810: GList *encryption_options = NULL;
5811: GList *login_options = NULL;
5812: int i;
5813:
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);
5816:
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);
5819:
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);
5825: }
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);
5828:
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;
5835: }
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);
5841: }
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);
5844:
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);
5849:
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);
5854: }
5855:
5856: if (init)
5857: return;
5858: init = TRUE;
5859:
5860: /* Preferences */
5861: purple_prefs_add_none("/plugins/prpl/oscar");
5862: purple_prefs_add_bool("/plugins/prpl/oscar/recent_buddies", FALSE);
5863:
5864: purple_prefs_remove("/plugins/prpl/oscar/show_idle");
5865: purple_prefs_remove("/plugins/prpl/oscar/always_use_rv_proxy");
5866:
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);
5871: }
5872:
FreeBSD-CVSweb <freebsd-cvsweb@FreeBSD.org>