File:  [Coherent Logic Development] / ChivanetAimPidgin / oscarprpl / src / c / oscar.c
Revision 1.1.1.1 (vendor branch): download - view: text, annotated - select for diffs
Mon Jan 27 19:48:25 2025 UTC (6 months ago) by snw
Branches: MAIN, CoherentLogicDevelopment
CVS tags: test-tag, start, HEAD
Pidgin AIM Plugin for ChivaNet

    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>