Mercurial > pidgin
view src/protocols/oscar/oscar.c @ 13711:967ef719cb62
[gaim-migrate @ 16115]
Resolve some unused value CIDs. This changeset just removes dead code.
jabber
chat.c
CID 180
CID 181
msn
httpconn.c
CID 179
msn.c
CID 185
CID 186
notification.c
CID 189
slp.c
CID 183
CID 184
switchboard.c
CID 178
sync.c
CID 187
CID 188
user.c
CID 182
novell
novell.c
CID 190
sametime
sametime.c
CID 214
CID 215
yahoo
yahoo.c
CID 200
CID 201
CID 202
yahoo_doodle.c
CID 168
CID 169
yahoo_filexfer.c
CID 170
CID 171
CID 172
committer: Tailor Script <tailor@pidgin.im>
author | Richard Laager <rlaager@wiktel.com> |
---|---|
date | Mon, 01 May 2006 20:37:37 +0000 |
parents | 2f7b356812bc |
children | f65e8d30ae53 |
line wrap: on
line source
/* * gaim * * Some code copyright (C) 1998-1999, Mark Spencer <markster@marko.net> * Some code copyright (C) 1999-2001, Eric Warmenhoven * Some code copyright (C) 2001-2003, Sean Egan * Some code copyright (C) 2001-2005, Mark Doliner <thekingant@users.sourceforge.net> * Some code copyright (C) 2005, Jonathan Clark <ardentlygnarly@users.sourceforge.net> * * Most libfaim code copyright (C) 1998-2001 Adam Fritzler <afritz@auk.cx> * Some libfaim code copyright (C) 2001-2004 Mark Doliner <thekingant@users.sourceforge.net> * * This program is free software; you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation; either version 2 of the License, or * (at your option) any later version. * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. * * You should have received a copy of the GNU General Public License * along with this program; if not, write to the Free Software * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA * */ #include "internal.h" #include "account.h" #include "accountopt.h" #include "buddyicon.h" #include "cipher.h" #include "conversation.h" #include "core.h" #include "debug.h" #include "imgstore.h" #include "network.h" #include "notify.h" #include "privacy.h" #include "prpl.h" #include "proxy.h" #include "request.h" #include "util.h" #include "version.h" #include "oscar.h" #include "peer.h" #define OSCAR_STATUS_ID_INVISIBLE "invisible" #define OSCAR_STATUS_ID_OFFLINE "offline" #define OSCAR_STATUS_ID_AVAILABLE "available" #define OSCAR_STATUS_ID_AWAY "away" #define OSCAR_STATUS_ID_DND "dnd" #define OSCAR_STATUS_ID_NA "na" #define OSCAR_STATUS_ID_OCCUPIED "occupied" #define OSCAR_STATUS_ID_FREE4CHAT "free4chat" #define OSCAR_STATUS_ID_CUSTOM "custom" #define AIMHASHDATA "http://gaim.sourceforge.net/aim_data.php3" #define OSCAR_CONNECT_STEPS 6 #define OSCAR_DEFAULT_LOGIN_SERVER "login.oscar.aol.com" #define OSCAR_DEFAULT_LOGIN_PORT 5190 #define OSCAR_DEFAULT_CUSTOM_ENCODING "ISO-8859-1" #define OSCAR_DEFAULT_AUTHORIZATION TRUE #define OSCAR_DEFAULT_HIDE_IP TRUE #define OSCAR_DEFAULT_WEB_AWARE FALSE #define OSCAR_DEFAULT_ALWAYS_USE_RV_PROXY FALSE static int caps_aim = OSCAR_CAPABILITY_CHAT | OSCAR_CAPABILITY_BUDDYICON | OSCAR_CAPABILITY_DIRECTIM | OSCAR_CAPABILITY_SENDFILE | OSCAR_CAPABILITY_INTEROPERATE | OSCAR_CAPABILITY_ICHAT; static int caps_icq = OSCAR_CAPABILITY_BUDDYICON | OSCAR_CAPABILITY_DIRECTIM | OSCAR_CAPABILITY_SENDFILE | OSCAR_CAPABILITY_ICQUTF8 | OSCAR_CAPABILITY_INTEROPERATE | OSCAR_CAPABILITY_ICHAT; static guint8 features_aim[] = {0x01, 0x01, 0x01, 0x02}; static guint8 features_icq[] = {0x01, 0x06}; static guint8 features_icq_offline[] = {0x01}; static guint8 ck[] = {0x00, 0x00, 0x00, 0x00, 0x00, 0x00}; typedef struct _NewFlapConnectionData NewFlapConnectionData; struct _NewFlapConnectionData { GaimConnection *gc; FlapConnection *conn; guint16 cookielen; guint8 *cookie; gpointer data; }; struct create_room { char *name; int exchange; }; struct chat_connection { char *name; char *show; /* AOL did something funny to us */ guint16 exchange; guint16 instance; FlapConnection *conn; int id; GaimConnection *gc; /* i hate this. */ GaimConversation *conv; /* bah. */ int maxlen; int maxvis; }; struct oscar_ask_directim_data { OscarData *od; char *who; }; /* * Various PRPL-specific buddy info that we want to keep track of * Some other info is maintained by locate.c, and I'd like to move * the rest of this to libfaim, mostly im.c * * TODO: More of this should use the status API. */ struct buddyinfo { gboolean typingnot; guint32 ipaddr; unsigned long ico_me_len; unsigned long ico_me_csum; time_t ico_me_time; gboolean ico_informed; unsigned long ico_len; unsigned long ico_csum; time_t ico_time; gboolean ico_need; gboolean ico_sent; }; struct name_data { GaimConnection *gc; gchar *name; gchar *nick; }; static char *msgerrreason[] = { N_("Invalid error"), N_("Invalid SNAC"), N_("Rate to host"), N_("Rate to client"), N_("Not logged in"), N_("Service unavailable"), N_("Service not defined"), N_("Obsolete SNAC"), N_("Not supported by host"), N_("Not supported by client"), N_("Refused by client"), N_("Reply too big"), N_("Responses lost"), N_("Request denied"), N_("Busted SNAC payload"), N_("Insufficient rights"), N_("In local permit/deny"), N_("Too evil (sender)"), N_("Too evil (receiver)"), N_("User temporarily unavailable"), N_("No match"), N_("List overflow"), N_("Request ambiguous"), N_("Queue full"), N_("Not while on AOL") }; static int msgerrreasonlen = 25; /* All the libfaim->gaim callback functions */ static int gaim_parse_auth_resp (OscarData *, FlapConnection *, FlapFrame *, ...); static int gaim_parse_login (OscarData *, FlapConnection *, FlapFrame *, ...); static int gaim_parse_auth_securid_request(OscarData *, FlapConnection *, FlapFrame *, ...); static int gaim_handle_redirect (OscarData *, FlapConnection *, FlapFrame *, ...); static int gaim_info_change (OscarData *, FlapConnection *, FlapFrame *, ...); static int gaim_account_confirm (OscarData *, FlapConnection *, FlapFrame *, ...); static int gaim_parse_oncoming (OscarData *, FlapConnection *, FlapFrame *, ...); static int gaim_parse_offgoing (OscarData *, FlapConnection *, FlapFrame *, ...); static int gaim_parse_incoming_im(OscarData *, FlapConnection *, FlapFrame *, ...); static int gaim_parse_misses (OscarData *, FlapConnection *, FlapFrame *, ...); static int gaim_parse_clientauto (OscarData *, FlapConnection *, FlapFrame *, ...); static int gaim_parse_userinfo (OscarData *, FlapConnection *, FlapFrame *, ...); static int gaim_got_infoblock (OscarData *, FlapConnection *, FlapFrame *, ...); static int gaim_parse_motd (OscarData *, FlapConnection *, FlapFrame *, ...); static int gaim_chatnav_info (OscarData *, FlapConnection *, FlapFrame *, ...); static int gaim_conv_chat_join (OscarData *, FlapConnection *, FlapFrame *, ...); static int gaim_conv_chat_leave (OscarData *, FlapConnection *, FlapFrame *, ...); static int gaim_conv_chat_info_update (OscarData *, FlapConnection *, FlapFrame *, ...); static int gaim_conv_chat_incoming_msg(OscarData *, FlapConnection *, FlapFrame *, ...); static int gaim_email_parseupdate(OscarData *, FlapConnection *, FlapFrame *, ...); static int gaim_icon_error (OscarData *, FlapConnection *, FlapFrame *, ...); static int gaim_icon_parseicon (OscarData *, FlapConnection *, FlapFrame *, ...); static int oscar_icon_req (OscarData *, FlapConnection *, FlapFrame *, ...); static int gaim_parse_msgack (OscarData *, FlapConnection *, FlapFrame *, ...); static int gaim_parse_ratechange (OscarData *, FlapConnection *, FlapFrame *, ...); static int gaim_parse_evilnotify (OscarData *, FlapConnection *, FlapFrame *, ...); static int gaim_parse_searcherror(OscarData *, FlapConnection *, FlapFrame *, ...); static int gaim_parse_searchreply(OscarData *, FlapConnection *, FlapFrame *, ...); static int gaim_bosrights (OscarData *, FlapConnection *, FlapFrame *, ...); static int gaim_connerr (OscarData *, FlapConnection *, FlapFrame *, ...); static int gaim_parse_msgerr (OscarData *, FlapConnection *, FlapFrame *, ...); static int gaim_parse_mtn (OscarData *, FlapConnection *, FlapFrame *, ...); static int gaim_parse_locaterights(OscarData *, FlapConnection *, FlapFrame *, ...); static int gaim_parse_buddyrights(OscarData *, FlapConnection *, FlapFrame *, ...); static int gaim_parse_locerr (OscarData *, FlapConnection *, FlapFrame *, ...); static int gaim_icbm_param_info (OscarData *, FlapConnection *, FlapFrame *, ...); static int gaim_parse_genericerr (OscarData *, FlapConnection *, FlapFrame *, ...); static int gaim_memrequest (OscarData *, FlapConnection *, FlapFrame *, ...); static int gaim_selfinfo (OscarData *, FlapConnection *, FlapFrame *, ...); static int gaim_offlinemsg (OscarData *, FlapConnection *, FlapFrame *, ...); static int gaim_offlinemsgdone (OscarData *, FlapConnection *, FlapFrame *, ...); static int gaim_icqalias (OscarData *, FlapConnection *, FlapFrame *, ...); static int gaim_icqinfo (OscarData *, FlapConnection *, FlapFrame *, ...); static int gaim_popup (OscarData *, FlapConnection *, FlapFrame *, ...); static int gaim_ssi_parseerr (OscarData *, FlapConnection *, FlapFrame *, ...); static int gaim_ssi_parserights (OscarData *, FlapConnection *, FlapFrame *, ...); static int gaim_ssi_parselist (OscarData *, FlapConnection *, FlapFrame *, ...); static int gaim_ssi_parseack (OscarData *, FlapConnection *, FlapFrame *, ...); static int gaim_ssi_parseadd (OscarData *, FlapConnection *, FlapFrame *, ...); static int gaim_ssi_authgiven (OscarData *, FlapConnection *, FlapFrame *, ...); static int gaim_ssi_authrequest (OscarData *, FlapConnection *, FlapFrame *, ...); static int gaim_ssi_authreply (OscarData *, FlapConnection *, FlapFrame *, ...); static int gaim_ssi_gotadded (OscarData *, FlapConnection *, FlapFrame *, ...); static gboolean gaim_icon_timerfunc(gpointer data); static void recent_buddies_cb(const char *name, GaimPrefType type, gconstpointer value, gpointer data); static void oscar_set_info(GaimConnection *gc, const char *info); static void oscar_set_info_and_status(GaimAccount *account, gboolean setinfo, const char *rawinfo, gboolean setstatus, GaimStatus *status); static void oscar_set_extendedstatus(GaimConnection *gc); static gboolean gaim_ssi_rerequestdata(gpointer data); static void oscar_free_name_data(struct name_data *data) { g_free(data->name); g_free(data->nick); g_free(data); } /** * Determine how we can send this message. Per the warnings elsewhere * in this file, these little checks determine the simplest encoding * we can use for a given message send using it. */ static guint32 oscar_charset_check(const char *utf8) { int i = 0; int charset = AIM_CHARSET_ASCII; /* * Can we get away with using our custom encoding? */ while (utf8[i]) { if ((unsigned char)utf8[i] > 0x7f) { /* not ASCII! */ charset = AIM_CHARSET_CUSTOM; break; } i++; } /* * Must we send this message as UNICODE (in the UCS-2BE encoding)? */ while (utf8[i]) { /* ISO-8859-1 is 0x00-0xbf in the first byte * followed by 0xc0-0xc3 in the second */ if ((unsigned char)utf8[i] < 0x80) { i++; continue; } else if (((unsigned char)utf8[i] & 0xfc) == 0xc0 && ((unsigned char)utf8[i + 1] & 0xc0) == 0x80) { i += 2; continue; } charset = AIM_CHARSET_UNICODE; break; } return charset; } /** * Take a string of the form charset="bleh" where bleh is * one of us-ascii, utf-8, iso-8859-1, or unicode-2-0, and * return a newly allocated string containing bleh. */ static gchar * oscar_encoding_extract(const char *encoding) { gchar *ret = NULL; char *begin, *end; g_return_val_if_fail(encoding != NULL, NULL); /* Make sure encoding begins with charset= */ if (strncmp(encoding, "text/aolrtf; charset=", 21) && strncmp(encoding, "text/x-aolrtf; charset=", 23)) { return NULL; } begin = strchr(encoding, '"'); end = strrchr(encoding, '"'); if ((begin == NULL) || (end == NULL) || (begin >= end)) return NULL; ret = g_strndup(begin+1, (end-1) - begin); return ret; } gchar * oscar_encoding_to_utf8(const char *encoding, const char *text, int textlen) { gchar *utf8 = NULL; if ((encoding == NULL) || encoding[0] == '\0') { gaim_debug_info("oscar", "Empty encoding, assuming UTF-8\n"); } else if (!strcasecmp(encoding, "iso-8859-1")) { utf8 = g_convert(text, textlen, "UTF-8", "iso-8859-1", NULL, NULL, NULL); } else if (!strcasecmp(encoding, "ISO-8859-1-Windows-3.1-Latin-1")) { utf8 = g_convert(text, textlen, "UTF-8", "Windows-1252", NULL, NULL, NULL); } else if (!strcasecmp(encoding, "unicode-2-0")) { utf8 = g_convert(text, textlen, "UTF-8", "UCS-2BE", NULL, NULL, NULL); } else if (strcasecmp(encoding, "us-ascii") && strcmp(encoding, "utf-8")) { gaim_debug_warning("oscar", "Unrecognized character encoding \"%s\", " "attempting to convert to UTF-8 anyway\n", encoding); utf8 = g_convert(text, textlen, "UTF-8", encoding, NULL, NULL, NULL); } /* * If utf8 is still NULL then either the encoding is us-ascii/utf-8 or * we have been unable to convert the text to utf-8 from the encoding * that was specified. So we check if the text is valid utf-8 then * just copy it. */ if (utf8 == NULL) { if (textlen != 0 && *text != '\0' && !g_utf8_validate(text, textlen, NULL)) utf8 = g_strdup(_("(There was an error receiving this message. The buddy you are speaking with is probably using a different encoding than expected. If you know what encoding he is using, you can specify it in the advanced account options for your AIM/ICQ account.)")); else utf8 = g_strndup(text, textlen); } return utf8; } static gchar * oscar_utf8_try_convert(GaimAccount *account, const gchar *msg) { const char *charset = NULL; char *ret = NULL; if(aim_sn_is_icq(gaim_account_get_username(account))) charset = gaim_account_get_string(account, "encoding", NULL); if(charset && *charset) ret = g_convert(msg, -1, "UTF-8", charset, NULL, NULL, NULL); if(!ret) ret = gaim_utf8_try_convert(msg); return ret; } static gchar * gaim_plugin_oscar_convert_to_utf8(const gchar *data, gsize datalen, const char *charsetstr, gboolean fallback) { gchar *ret = NULL; GError *err = NULL; if ((charsetstr == NULL) || (*charsetstr == '\0')) return NULL; if (strcasecmp("UTF-8", charsetstr)) { if (fallback) ret = g_convert_with_fallback(data, datalen, "UTF-8", charsetstr, "?", NULL, NULL, &err); else ret = g_convert(data, datalen, "UTF-8", charsetstr, NULL, NULL, &err); if (err != NULL) { gaim_debug_warning("oscar", "Conversation from %s failed: %s.\n", charsetstr, err->message); g_error_free(err); } } else { if (g_utf8_validate(data, datalen, NULL)) ret = g_strndup(data, datalen); else gaim_debug_warning("oscar", "String is not valid UTF-8.\n"); } return ret; } /** * This attemps to decode an incoming IM into a UTF8 string. * * We try decoding using two different character sets. The charset * specified in the IM determines the order in which we attempt to * decode. We do this because there are lots of broken ICQ clients * that don't correctly send non-ASCII messages. And if Gaim isn't * able to deal with that crap, then people complain like banshees. * charsetstr1 is always set to what the correct encoding should be. */ gchar * gaim_plugin_oscar_decode_im_part(GaimAccount *account, const char *sourcesn, guint16 charset, guint16 charsubset, const gchar *data, gsize datalen) { gchar *ret = NULL; const gchar *charsetstr1, *charsetstr2; gaim_debug_info("oscar", "Parsing IM part, charset=0x%04hx, charsubset=0x%04hx, datalen=%hd\n", charset, charsubset, datalen); if ((datalen == 0) || (data == NULL)) return NULL; if (charset == AIM_CHARSET_UNICODE) { charsetstr1 = "UCS-2BE"; charsetstr2 = "UTF-8"; } else if (charset == AIM_CHARSET_CUSTOM) { if ((sourcesn != NULL) && isdigit(sourcesn[0])) charsetstr1 = gaim_account_get_string(account, "encoding", OSCAR_DEFAULT_CUSTOM_ENCODING); else charsetstr1 = "ISO-8859-1"; charsetstr2 = "UTF-8"; } else if (charset == AIM_CHARSET_ASCII) { /* Should just be "ASCII" */ charsetstr1 = "ASCII"; charsetstr2 = gaim_account_get_string(account, "encoding", OSCAR_DEFAULT_CUSTOM_ENCODING); } else if (charset == 0x000d) { /* Mobile AIM client on a Nokia 3100 and an LG VX6000 */ charsetstr1 = "ISO-8859-1"; charsetstr2 = gaim_account_get_string(account, "encoding", OSCAR_DEFAULT_CUSTOM_ENCODING); } else { /* Unknown, hope for valid UTF-8... */ charsetstr1 = "UTF-8"; charsetstr2 = gaim_account_get_string(account, "encoding", OSCAR_DEFAULT_CUSTOM_ENCODING); } ret = gaim_plugin_oscar_convert_to_utf8(data, datalen, charsetstr1, FALSE); if (ret == NULL) ret = gaim_plugin_oscar_convert_to_utf8(data, datalen, charsetstr2, TRUE); if (ret == NULL) ret = g_strdup(_("(There was an error receiving this message. The buddy you are speaking to most likely has a buggy client.)")); return ret; } /** * Figure out what encoding to use when sending a given outgoing message. */ static void gaim_plugin_oscar_convert_to_best_encoding(GaimConnection *gc, const char *destsn, const gchar *from, gchar **msg, int *msglen_int, guint16 *charset, guint16 *charsubset) { OscarData *od = gc->proto_data; GaimAccount *account = gaim_connection_get_account(gc); GError *err = NULL; aim_userinfo_t *userinfo = NULL; const gchar *charsetstr; gsize msglen; /* Attempt to send as ASCII */ if (oscar_charset_check(from) == AIM_CHARSET_ASCII) { *msg = g_convert(from, strlen(from), "ASCII", "UTF-8", NULL, &msglen, NULL); *charset = AIM_CHARSET_ASCII; *charsubset = 0x0000; *msglen_int = msglen; return; } /* * If we're sending to an ICQ user, and they are in our * buddy list, and they are advertising the Unicode * capability, and they are online, then attempt to send * as UCS-2BE. */ if ((destsn != NULL) && aim_sn_is_icq(destsn)) userinfo = aim_locate_finduserinfo(od, destsn); if ((userinfo != NULL) && (userinfo->capabilities & OSCAR_CAPABILITY_ICQUTF8)) { GaimBuddy *b; b = gaim_find_buddy(account, destsn); if ((b != NULL) && (GAIM_BUDDY_IS_ONLINE(b))) { *msg = g_convert(from, strlen(from), "UCS-2BE", "UTF-8", NULL, &msglen, NULL); if (*msg != NULL) { *charset = AIM_CHARSET_UNICODE; *charsubset = 0x0000; *msglen_int = msglen; return; } } } /* * If this is AIM then attempt to send as ISO-8859-1. If this is * ICQ then attempt to send as the user specified character encoding. */ charsetstr = "ISO-8859-1"; if ((destsn != NULL) && aim_sn_is_icq(destsn)) charsetstr = gaim_account_get_string(account, "encoding", OSCAR_DEFAULT_CUSTOM_ENCODING); /* * XXX - We need a way to only attempt to convert if we KNOW "from" * can be converted to "charsetstr" */ *msg = g_convert(from, strlen(from), charsetstr, "UTF-8", NULL, &msglen, NULL); if (*msg != NULL) { *charset = AIM_CHARSET_CUSTOM; *charsubset = 0x0000; *msglen_int = msglen; return; } /* * Nothing else worked, so send as UCS-2BE. */ *msg = g_convert(from, strlen(from), "UCS-2BE", "UTF-8", NULL, &msglen, &err); if (*msg != NULL) { *charset = AIM_CHARSET_UNICODE; *charsubset = 0x0000; *msglen_int = msglen; return; } gaim_debug_error("oscar", "Error converting a Unicode message: %s\n", err->message); g_error_free(err); gaim_debug_error("oscar", "This should NEVER happen! Sending UTF-8 text flagged as ASCII.\n"); *msg = g_strdup(from); *msglen_int = strlen(*msg); *charset = AIM_CHARSET_ASCII; *charsubset = 0x0000; return; } /** * Looks for %n, %d, or %t in a string, and replaces them with the * specified name, date, and time, respectively. * * @param str The string that may contain the special variables. * @param name The sender name. * * @return A newly allocated string where the special variables are * expanded. This should be g_free'd by the caller. */ static gchar * gaim_str_sub_away_formatters(const char *str, const char *name) { char *c; GString *cpy; time_t t; struct tm *tme; g_return_val_if_fail(str != NULL, NULL); g_return_val_if_fail(name != NULL, NULL); /* Create an empty GString that is hopefully big enough for most messages */ cpy = g_string_sized_new(1024); t = time(NULL); tme = localtime(&t); c = (char *)str; while (*c) { switch (*c) { case '%': if (*(c + 1)) { switch (*(c + 1)) { case 'n': /* append name */ g_string_append(cpy, name); c++; break; case 'd': /* append date */ g_string_append(cpy, gaim_date_format_short(tme)); c++; break; case 't': /* append time */ g_string_append(cpy, gaim_time_format(tme)); c++; break; default: g_string_append_c(cpy, *c); } } else { g_string_append_c(cpy, *c); } break; default: g_string_append_c(cpy, *c); } c++; } return g_string_free(cpy, FALSE); } static gchar *oscar_caps_to_string(guint caps) { GString *str; const gchar *tmp; guint bit = 1; str = g_string_new(""); if (!caps) { return NULL; } else while (bit <= OSCAR_CAPABILITY_LAST) { if (bit & caps) { switch (bit) { case OSCAR_CAPABILITY_BUDDYICON: tmp = _("Buddy Icon"); break; case OSCAR_CAPABILITY_TALK: tmp = _("Voice"); break; case OSCAR_CAPABILITY_DIRECTIM: tmp = _("AIM Direct IM"); break; case OSCAR_CAPABILITY_CHAT: tmp = _("Chat"); break; case OSCAR_CAPABILITY_GETFILE: tmp = _("Get File"); break; case OSCAR_CAPABILITY_SENDFILE: tmp = _("Send File"); break; case OSCAR_CAPABILITY_GAMES: case OSCAR_CAPABILITY_GAMES2: tmp = _("Games"); break; case OSCAR_CAPABILITY_ADDINS: tmp = _("Add-Ins"); break; case OSCAR_CAPABILITY_SENDBUDDYLIST: tmp = _("Send Buddy List"); break; case OSCAR_CAPABILITY_ICQ_DIRECT: tmp = _("ICQ Direct Connect"); break; case OSCAR_CAPABILITY_APINFO: tmp = _("AP User"); break; case OSCAR_CAPABILITY_ICQRTF: tmp = _("ICQ RTF"); break; case OSCAR_CAPABILITY_EMPTY: tmp = _("Nihilist"); break; case OSCAR_CAPABILITY_ICQSERVERRELAY: tmp = _("ICQ Server Relay"); break; case OSCAR_CAPABILITY_ICQUTF8OLD: tmp = _("Old ICQ UTF8"); break; case OSCAR_CAPABILITY_TRILLIANCRYPT: tmp = _("Trillian Encryption"); break; case OSCAR_CAPABILITY_ICQUTF8: tmp = _("ICQ UTF8"); break; case OSCAR_CAPABILITY_HIPTOP: tmp = _("Hiptop"); break; case OSCAR_CAPABILITY_SECUREIM: tmp = _("Security Enabled"); break; case OSCAR_CAPABILITY_VIDEO: tmp = _("Video Chat"); break; /* Not actually sure about this one... WinAIM doesn't show anything */ case OSCAR_CAPABILITY_ICHATAV: tmp = _("iChat AV"); break; case OSCAR_CAPABILITY_LIVEVIDEO: tmp = _("Live Video"); break; case OSCAR_CAPABILITY_CAMERA: tmp = _("Camera"); break; default: tmp = NULL; break; } if (tmp) g_string_append_printf(str, "%s%s", (*(str->str) == '\0' ? "" : ", "), tmp); } bit <<= 1; } return g_string_free(str, FALSE); } static char *oscar_icqstatus(int state) { /* Make a cute little string that shows the status of the dude or dudet */ if (state & AIM_ICQ_STATE_CHAT) return g_strdup_printf(_("Free For Chat")); else if (state & AIM_ICQ_STATE_DND) return g_strdup_printf(_("Do Not Disturb")); else if (state & AIM_ICQ_STATE_OUT) return g_strdup_printf(_("Not Available")); else if (state & AIM_ICQ_STATE_BUSY) return g_strdup_printf(_("Occupied")); else if (state & AIM_ICQ_STATE_AWAY) return g_strdup_printf(_("Away")); else if (state & AIM_ICQ_STATE_WEBAWARE) return g_strdup_printf(_("Web Aware")); else if (state & AIM_ICQ_STATE_INVISIBLE) return g_strdup_printf(_("Invisible")); else return g_strdup_printf(_("Online")); } static void oscar_string_append(GaimAccount *account, GString *str, const char *newline, const char *name, const char *value) { gchar *utf8; if (value && value[0] && (utf8 = oscar_utf8_try_convert(account, value))) { g_string_append_printf(str, "%s<b>%s:</b> %s", newline, name, utf8); g_free(utf8); } } static void oscar_string_append_info(GaimConnection *gc, GString *str, const char *newline, GaimBuddy *b, aim_userinfo_t *userinfo) { OscarData *od; GaimAccount *account; GaimPresence *presence = NULL; GaimStatus *status = NULL; GaimGroup *g = NULL; struct buddyinfo *bi = NULL; char *tmp; od = gc->proto_data; account = gaim_connection_get_account(gc); if ((str == NULL) || (newline == NULL) || ((b == NULL) && (userinfo == NULL))) return; if (userinfo == NULL) userinfo = aim_locate_finduserinfo(od, b->name); if (b == NULL) b = gaim_find_buddy(account, userinfo->sn); if (b != NULL) { g = gaim_buddy_get_group(b); presence = gaim_buddy_get_presence(b); status = gaim_presence_get_active_status(presence); } if (userinfo != NULL) bi = g_hash_table_lookup(od->buddyinfo, gaim_normalize(account, userinfo->sn)); if (b != NULL) { if (gaim_presence_is_online(presence)) { if (aim_sn_is_icq(b->name)) { GaimStatus *status = gaim_presence_get_active_status(presence); oscar_string_append(gc->account, str, newline, _("Status"), gaim_status_get_name(status)); } } else { tmp = aim_ssi_itemlist_findparentname(od->ssi.local, b->name); if (aim_ssi_waitingforauth(od->ssi.local, tmp, b->name)) oscar_string_append(gc->account, str, newline, _("Status"), _("Not Authorized")); else oscar_string_append(gc->account, str, newline, _("Status"), _("Offline")); } } if ((bi != NULL) && (bi->ipaddr != 0)) { tmp = g_strdup_printf("%hhu.%hhu.%hhu.%hhu", (bi->ipaddr & 0xff000000) >> 24, (bi->ipaddr & 0x00ff0000) >> 16, (bi->ipaddr & 0x0000ff00) >> 8, (bi->ipaddr & 0x000000ff)); oscar_string_append(gc->account, str, newline, _("IP Address"), tmp); g_free(tmp); } if ((userinfo != NULL) && (userinfo->warnlevel != 0)) { tmp = g_strdup_printf("%d", (int)(userinfo->warnlevel/10.0 + .5)); oscar_string_append(gc->account, str, newline, _("Warning Level"), tmp); g_free(tmp); } if ((b != NULL) && (b->name != NULL) && (g != NULL) && (g->name != NULL)) { tmp = aim_ssi_getcomment(od->ssi.local, g->name, b->name); if (tmp != NULL) { char *tmp2 = g_markup_escape_text(tmp, strlen(tmp)); g_free(tmp); oscar_string_append(gc->account, str, newline, _("Buddy Comment"), tmp2); g_free(tmp2); } } } static char *extract_name(const char *name) { char *tmp, *x; int i, j; if (!name) return NULL; x = strchr(name, '-'); if (!x) return NULL; x = strchr(++x, '-'); if (!x) return NULL; tmp = g_strdup(++x); for (i = 0, j = 0; x[i]; i++) { char hex[3]; if (x[i] != '%') { tmp[j++] = x[i]; continue; } strncpy(hex, x + ++i, 2); hex[2] = 0; i++; tmp[j++] = strtol(hex, NULL, 16); } tmp[j] = 0; return tmp; } static struct chat_connection * find_oscar_chat(GaimConnection *gc, int id) { OscarData *od = (OscarData *)gc->proto_data; GSList *cur; struct chat_connection *cc; for (cur = od->oscar_chats; cur != NULL; cur = cur->next) { cc = (struct chat_connection *)cur->data; if (cc->id == id) return cc; } return NULL; } static struct chat_connection * find_oscar_chat_by_conn(GaimConnection *gc, FlapConnection *conn) { OscarData *od = (OscarData *)gc->proto_data; GSList *cur; struct chat_connection *cc; for (cur = od->oscar_chats; cur != NULL; cur = cur->next) { cc = (struct chat_connection *)cur->data; if (cc->conn == conn) return cc; } return NULL; } static struct chat_connection * find_oscar_chat_by_conv(GaimConnection *gc, GaimConversation *conv) { OscarData *od = (OscarData *)gc->proto_data; GSList *cur; struct chat_connection *cc; for (cur = od->oscar_chats; cur != NULL; cur = cur->next) { cc = (struct chat_connection *)cur->data; if (cc->conv == conv) return cc; } return NULL; } static void oscar_chat_destroy(struct chat_connection *cc) { g_free(cc->name); g_free(cc->show); g_free(cc); } static void oscar_chat_kill(GaimConnection *gc, struct chat_connection *cc) { OscarData *od = (OscarData *)gc->proto_data; /* Notify the conversation window that we've left the chat */ serv_got_chat_left(gc, gaim_conv_chat_get_id(GAIM_CONV_CHAT(cc->conv))); /* Destroy the chat_connection */ od->oscar_chats = g_slist_remove(od->oscar_chats, cc); flap_connection_schedule_destroy(cc->conn, OSCAR_DISCONNECT_DONE); oscar_chat_destroy(cc); } static void destroy_new_conn_data(NewFlapConnectionData *new_conn_data) { if ((new_conn_data->data != NULL) && (new_conn_data->conn->type == SNAC_FAMILY_CHAT)) { oscar_chat_destroy(new_conn_data->data); } g_free(new_conn_data->cookie); g_free(new_conn_data); } /** * This is the callback function anytime gaim_proxy_connect() * establishes a new TCP connection with an oscar host. Depending * on the type of host, we do a few different things here. */ static void connection_established_cb(gpointer data, gint source, GaimInputCondition cond) { NewFlapConnectionData *new_conn_data; GaimConnection *gc; OscarData *od; GaimAccount *account; FlapConnection *conn; new_conn_data = data; gc = new_conn_data->gc; if (!g_list_find(gaim_connections_get_all(), gc)) { if (source >= 0) close(source); destroy_new_conn_data(new_conn_data); return; } od = gc->proto_data; account = gaim_connection_get_account(gc); conn = new_conn_data->conn; conn->fd = source; if (source < 0) { gaim_debug_error("oscar", "unable to connect FLAP server " "of type 0x%04hx\n", conn->type); if (conn->type == SNAC_FAMILY_AUTH) gaim_connection_error(gc, _("Could not connect to authentication server")); if (conn->type == SNAC_FAMILY_LOCATE) gaim_connection_error(gc, _("Could not connect to BOS server")); else /* Maybe we should call this for BOS connections, too? */ flap_connection_schedule_destroy(conn, OSCAR_DISCONNECT_COULD_NOT_CONNECT); destroy_new_conn_data(new_conn_data); return; } gaim_debug_info("oscar", "connected to FLAP server of type 0x%04hx\n", conn->type); conn->watcher_incoming = gaim_input_add(conn->fd, GAIM_INPUT_READ, flap_connection_recv_cb, conn); if (new_conn_data->cookie == NULL) { if (!aim_sn_is_icq(gaim_account_get_username(account))) /* * We don't send this when authenticating an ICQ account * because for some reason ICQ is still using the * assy/insecure authentication procedure. */ flap_connection_send_version(od, conn); } else flap_connection_send_version_with_cookie(od, conn, new_conn_data->cookielen, new_conn_data->cookie); if (conn->type == SNAC_FAMILY_AUTH) { aim_request_login(od, conn, gaim_account_get_username(account)); gaim_debug_info("oscar", "Screen name sent, waiting for response\n"); gaim_connection_update_progress(gc, _("Screen name sent"), 1, OSCAR_CONNECT_STEPS); ck[1] = 0x65; } else if (conn->type == SNAC_FAMILY_LOCATE) { gaim_connection_update_progress(gc, _("Connection established, cookie sent"), 4, OSCAR_CONNECT_STEPS); ck[4] = 0x61; } else if (conn->type == SNAC_FAMILY_CHAT) { od->oscar_chats = g_slist_append(od->oscar_chats, new_conn_data->data); new_conn_data->data = NULL; } destroy_new_conn_data(new_conn_data); } static void flap_connection_established_bos(OscarData *od, FlapConnection *conn) { GaimConnection *gc = od->gc; aim_reqpersonalinfo(od, conn); gaim_debug_info("oscar", "ssi: requesting rights and list\n"); aim_ssi_reqrights(od); aim_ssi_reqdata(od); if (od->getblisttimer > 0) gaim_timeout_remove(od->getblisttimer); od->getblisttimer = gaim_timeout_add(30000, gaim_ssi_rerequestdata, od); aim_locate_reqrights(od); aim_buddylist_reqrights(od, conn); aim_im_reqparams(od); aim_bos_reqrights(od, conn); /* TODO: Don't call this with ssi */ gaim_connection_update_progress(gc, _("Finalizing connection"), 5, OSCAR_CONNECT_STEPS); } static void flap_connection_established_admin(OscarData *od, FlapConnection *conn) { aim_clientready(od, conn); gaim_debug_info("oscar", "connected to admin\n"); if (od->chpass) { gaim_debug_info("oscar", "changing password\n"); aim_admin_changepasswd(od, conn, od->newp, od->oldp); g_free(od->oldp); od->oldp = NULL; g_free(od->newp); od->newp = NULL; od->chpass = FALSE; } if (od->setnick) { gaim_debug_info("oscar", "formatting screen name\n"); aim_admin_setnick(od, conn, od->newsn); g_free(od->newsn); od->newsn = NULL; od->setnick = FALSE; } if (od->conf) { gaim_debug_info("oscar", "confirming account\n"); aim_admin_reqconfirm(od, conn); od->conf = FALSE; } if (od->reqemail) { gaim_debug_info("oscar", "requesting e-mail address\n"); aim_admin_getinfo(od, conn, 0x0011); od->reqemail = FALSE; } if (od->setemail) { gaim_debug_info("oscar", "setting e-mail address\n"); aim_admin_setemail(od, conn, od->email); g_free(od->email); od->email = NULL; od->setemail = FALSE; } } static void flap_connection_established_chat(OscarData *od, FlapConnection *conn) { GaimConnection *gc = od->gc; struct chat_connection *chatcon; static int id = 1; aim_clientready(od, conn); chatcon = find_oscar_chat_by_conn(gc, conn); chatcon->id = id; chatcon->conv = serv_got_joined_chat(gc, id++, chatcon->show); } static void flap_connection_established_chatnav(OscarData *od, FlapConnection *conn) { aim_clientready(od, conn); aim_chatnav_reqrights(od, conn); } static void flap_connection_established_alert(OscarData *od, FlapConnection *conn) { aim_email_sendcookies(od); aim_email_activate(od); aim_clientready(od, conn); } static void flap_connection_established_bart(OscarData *od, FlapConnection *conn) { GaimConnection *gc = od->gc; aim_clientready(od, conn); od->iconconnecting = FALSE; if (od->icontimer) gaim_timeout_remove(od->icontimer); od->icontimer = gaim_timeout_add(100, gaim_icon_timerfunc, gc); } static int flap_connection_established(OscarData *od, FlapConnection *conn, FlapFrame *fr, ...) { gaim_debug_info("oscar", "FLAP connection of type 0x%04hx is " "now fully connected\n", conn->type); if (conn->type == SNAC_FAMILY_LOCATE) flap_connection_established_bos(od, conn); else if (conn->type == SNAC_FAMILY_ADMIN) flap_connection_established_admin(od, conn); else if (conn->type == SNAC_FAMILY_CHAT) flap_connection_established_chat(od, conn); else if (conn->type == SNAC_FAMILY_CHATNAV) flap_connection_established_chatnav(od, conn); else if (conn->type == SNAC_FAMILY_ALERT) flap_connection_established_alert(od, conn); else if (conn->type == SNAC_FAMILY_BART) flap_connection_established_bart(od, conn); return 1; } static void oscar_login(GaimAccount *account) { GaimConnection *gc; OscarData *od; NewFlapConnectionData *new_conn_data; gc = gaim_account_get_connection(account); od = gc->proto_data = oscar_data_new(); od->gc = gc; oscar_data_addhandler(od, AIM_CB_FAM_SPECIAL, AIM_CB_SPECIAL_CONNERR, gaim_connerr, 0); oscar_data_addhandler(od, AIM_CB_FAM_SPECIAL, AIM_CB_SPECIAL_CONNINITDONE, flap_connection_established, 0); oscar_data_addhandler(od, SNAC_FAMILY_ADMIN, 0x0003, gaim_info_change, 0); oscar_data_addhandler(od, SNAC_FAMILY_ADMIN, 0x0005, gaim_info_change, 0); oscar_data_addhandler(od, SNAC_FAMILY_ADMIN, 0x0007, gaim_account_confirm, 0); oscar_data_addhandler(od, SNAC_FAMILY_ALERT, 0x0001, gaim_parse_genericerr, 0); oscar_data_addhandler(od, SNAC_FAMILY_ALERT, SNAC_SUBTYPE_ALERT_MAILSTATUS, gaim_email_parseupdate, 0); oscar_data_addhandler(od, SNAC_FAMILY_AUTH, 0x0003, gaim_parse_auth_resp, 0); oscar_data_addhandler(od, SNAC_FAMILY_AUTH, 0x0007, gaim_parse_login, 0); oscar_data_addhandler(od, SNAC_FAMILY_AUTH, SNAC_SUBTYPE_AUTH_SECURID_REQUEST, gaim_parse_auth_securid_request, 0); oscar_data_addhandler(od, SNAC_FAMILY_BART, SNAC_SUBTYPE_BART_ERROR, gaim_icon_error, 0); oscar_data_addhandler(od, SNAC_FAMILY_BART, SNAC_SUBTYPE_BART_RESPONSE, gaim_icon_parseicon, 0); oscar_data_addhandler(od, SNAC_FAMILY_BOS, 0x0001, gaim_parse_genericerr, 0); oscar_data_addhandler(od, SNAC_FAMILY_BOS, 0x0003, gaim_bosrights, 0); oscar_data_addhandler(od, SNAC_FAMILY_BUDDY, 0x0001, gaim_parse_genericerr, 0); oscar_data_addhandler(od, SNAC_FAMILY_BUDDY, SNAC_SUBTYPE_BUDDY_RIGHTSINFO, gaim_parse_buddyrights, 0); oscar_data_addhandler(od, SNAC_FAMILY_BUDDY, SNAC_SUBTYPE_BUDDY_ONCOMING, gaim_parse_oncoming, 0); oscar_data_addhandler(od, SNAC_FAMILY_BUDDY, SNAC_SUBTYPE_BUDDY_OFFGOING, gaim_parse_offgoing, 0); oscar_data_addhandler(od, SNAC_FAMILY_CHAT, 0x0001, gaim_parse_genericerr, 0); oscar_data_addhandler(od, SNAC_FAMILY_CHAT, SNAC_SUBTYPE_CHAT_USERJOIN, gaim_conv_chat_join, 0); oscar_data_addhandler(od, SNAC_FAMILY_CHAT, SNAC_SUBTYPE_CHAT_USERLEAVE, gaim_conv_chat_leave, 0); oscar_data_addhandler(od, SNAC_FAMILY_CHAT, SNAC_SUBTYPE_CHAT_ROOMINFOUPDATE, gaim_conv_chat_info_update, 0); oscar_data_addhandler(od, SNAC_FAMILY_CHAT, SNAC_SUBTYPE_CHAT_INCOMINGMSG, gaim_conv_chat_incoming_msg, 0); oscar_data_addhandler(od, SNAC_FAMILY_CHATNAV, 0x0001, gaim_parse_genericerr, 0); oscar_data_addhandler(od, SNAC_FAMILY_CHATNAV, SNAC_SUBTYPE_CHATNAV_INFO, gaim_chatnav_info, 0); oscar_data_addhandler(od, SNAC_FAMILY_FEEDBAG, SNAC_SUBTYPE_FEEDBAG_ERROR, gaim_ssi_parseerr, 0); oscar_data_addhandler(od, SNAC_FAMILY_FEEDBAG, SNAC_SUBTYPE_FEEDBAG_RIGHTSINFO, gaim_ssi_parserights, 0); oscar_data_addhandler(od, SNAC_FAMILY_FEEDBAG, SNAC_SUBTYPE_FEEDBAG_LIST, gaim_ssi_parselist, 0); oscar_data_addhandler(od, SNAC_FAMILY_FEEDBAG, SNAC_SUBTYPE_FEEDBAG_SRVACK, gaim_ssi_parseack, 0); oscar_data_addhandler(od, SNAC_FAMILY_FEEDBAG, SNAC_SUBTYPE_FEEDBAG_ADD, gaim_ssi_parseadd, 0); oscar_data_addhandler(od, SNAC_FAMILY_FEEDBAG, SNAC_SUBTYPE_FEEDBAG_RECVAUTH, gaim_ssi_authgiven, 0); oscar_data_addhandler(od, SNAC_FAMILY_FEEDBAG, SNAC_SUBTYPE_FEEDBAG_RECVAUTHREQ, gaim_ssi_authrequest, 0); oscar_data_addhandler(od, SNAC_FAMILY_FEEDBAG, SNAC_SUBTYPE_FEEDBAG_RECVAUTHREP, gaim_ssi_authreply, 0); oscar_data_addhandler(od, SNAC_FAMILY_FEEDBAG, SNAC_SUBTYPE_FEEDBAG_ADDED, gaim_ssi_gotadded, 0); oscar_data_addhandler(od, SNAC_FAMILY_ICBM, 0x0005, gaim_icbm_param_info, 0); oscar_data_addhandler(od, SNAC_FAMILY_ICBM, SNAC_SUBTYPE_ICBM_INCOMING, gaim_parse_incoming_im, 0); oscar_data_addhandler(od, SNAC_FAMILY_ICBM, SNAC_SUBTYPE_ICBM_MISSEDCALL, gaim_parse_misses, 0); oscar_data_addhandler(od, SNAC_FAMILY_ICBM, SNAC_SUBTYPE_ICBM_CLIENTAUTORESP, gaim_parse_clientauto, 0); oscar_data_addhandler(od, SNAC_FAMILY_ICBM, SNAC_SUBTYPE_ICBM_ERROR, gaim_parse_msgerr, 0); oscar_data_addhandler(od, SNAC_FAMILY_ICBM, SNAC_SUBTYPE_ICBM_MTN, gaim_parse_mtn, 0); oscar_data_addhandler(od, SNAC_FAMILY_ICBM, SNAC_SUBTYPE_ICBM_ACK, gaim_parse_msgack, 0); oscar_data_addhandler(od, SNAC_FAMILY_ICQ, SNAC_SUBTYPE_ICQ_OFFLINEMSG, gaim_offlinemsg, 0); oscar_data_addhandler(od, SNAC_FAMILY_ICQ, SNAC_SUBTYPE_ICQ_OFFLINEMSGCOMPLETE, gaim_offlinemsgdone, 0); oscar_data_addhandler(od, SNAC_FAMILY_ICQ, SNAC_SUBTYPE_ICQ_ALIAS, gaim_icqalias, 0); oscar_data_addhandler(od, SNAC_FAMILY_ICQ, SNAC_SUBTYPE_ICQ_INFO, gaim_icqinfo, 0); oscar_data_addhandler(od, SNAC_FAMILY_LOCATE, SNAC_SUBTYPE_LOCATE_RIGHTSINFO, gaim_parse_locaterights, 0); oscar_data_addhandler(od, SNAC_FAMILY_LOCATE, SNAC_SUBTYPE_LOCATE_USERINFO, gaim_parse_userinfo, 0); oscar_data_addhandler(od, SNAC_FAMILY_LOCATE, SNAC_SUBTYPE_LOCATE_ERROR, gaim_parse_locerr, 0); oscar_data_addhandler(od, SNAC_FAMILY_LOCATE, SNAC_SUBTYPE_LOCATE_GOTINFOBLOCK, gaim_got_infoblock, 0); oscar_data_addhandler(od, SNAC_FAMILY_OSERVICE, 0x0001, gaim_parse_genericerr, 0); oscar_data_addhandler(od, SNAC_FAMILY_OSERVICE, 0x000f, gaim_selfinfo, 0); oscar_data_addhandler(od, SNAC_FAMILY_OSERVICE, 0x001f, gaim_memrequest, 0); oscar_data_addhandler(od, SNAC_FAMILY_OSERVICE, 0x0021, oscar_icon_req,0); oscar_data_addhandler(od, SNAC_FAMILY_OSERVICE, SNAC_SUBTYPE_OSERVICE_RATECHANGE, gaim_parse_ratechange, 0); oscar_data_addhandler(od, SNAC_FAMILY_OSERVICE, SNAC_SUBTYPE_OSERVICE_REDIRECT, gaim_handle_redirect, 0); oscar_data_addhandler(od, SNAC_FAMILY_OSERVICE, SNAC_SUBTYPE_OSERVICE_MOTD, gaim_parse_motd, 0); oscar_data_addhandler(od, SNAC_FAMILY_OSERVICE, SNAC_SUBTYPE_OSERVICE_EVIL, gaim_parse_evilnotify, 0); oscar_data_addhandler(od, SNAC_FAMILY_POPUP, 0x0002, gaim_popup, 0); oscar_data_addhandler(od, SNAC_FAMILY_USERLOOKUP, SNAC_SUBTYPE_USERLOOKUP_ERROR, gaim_parse_searcherror, 0); oscar_data_addhandler(od, SNAC_FAMILY_USERLOOKUP, 0x0003, gaim_parse_searchreply, 0); gaim_debug_misc("oscar", "oscar_login: gc = %p\n", gc); if (!aim_snvalid(gaim_account_get_username(account))) { gchar *buf; buf = g_strdup_printf(_("Unable to login: Could not sign on as %s because the screen name is invalid. Screen names must either start with a letter and contain only letters, numbers and spaces, or contain only numbers."), gaim_account_get_username(account)); gaim_connection_error(gc, buf); g_free(buf); } if (aim_sn_is_icq((gaim_account_get_username(account)))) { od->icq = TRUE; } else { gc->flags |= GAIM_CONNECTION_HTML; gc->flags |= GAIM_CONNECTION_AUTO_RESP; } /* Connect to core Gaim signals */ gaim_prefs_connect_callback(gc, "/plugins/prpl/oscar/recent_buddies", recent_buddies_cb, gc); new_conn_data = g_new(NewFlapConnectionData, 1); new_conn_data->gc = gc; new_conn_data->conn = flap_connection_new(od, SNAC_FAMILY_AUTH); new_conn_data->cookielen = 0; new_conn_data->cookie = NULL; new_conn_data->data = NULL; if (gaim_proxy_connect(account, gaim_account_get_string(account, "server", OSCAR_DEFAULT_LOGIN_SERVER), gaim_account_get_int(account, "port", OSCAR_DEFAULT_LOGIN_PORT), connection_established_cb, new_conn_data) < 0) { gaim_connection_error(gc, _("Couldn't connect to host")); return; } gaim_connection_update_progress(gc, _("Connecting"), 0, OSCAR_CONNECT_STEPS); ck[0] = 0x5a; } static void oscar_close(GaimConnection *gc) { OscarData *od; od = (OscarData *)gc->proto_data; while (od->oscar_chats) { struct chat_connection *cc = od->oscar_chats->data; od->oscar_chats = g_slist_remove(od->oscar_chats, cc); oscar_chat_destroy(cc); } while (od->create_rooms) { struct create_room *cr = od->create_rooms->data; g_free(cr->name); od->create_rooms = g_slist_remove(od->create_rooms, cr); g_free(cr); } oscar_data_destroy(od); gc->proto_data = NULL; gaim_prefs_disconnect_by_handle(gc); gaim_debug_info("oscar", "Signed off.\n"); } static int gaim_parse_auth_resp(OscarData *od, FlapConnection *conn, FlapFrame *fr, ...) { GaimConnection *gc = od->gc; GaimAccount *account = gc->account; char *host; int port; int i, rc; NewFlapConnectionData *new_conn_data; va_list ap; struct aim_authresp_info *info; port = gaim_account_get_int(account, "port", OSCAR_DEFAULT_LOGIN_PORT); va_start(ap, fr); info = va_arg(ap, struct aim_authresp_info *); va_end(ap); gaim_debug_info("oscar", "inside auth_resp (Screen name: %s)\n", info->sn); if (info->errorcode || !info->bosip || !info->cookielen || !info->cookie) { char buf[256]; switch (info->errorcode) { case 0x05: /* Incorrect nick/password */ gc->wants_to_die = TRUE; gaim_connection_error(gc, _("Incorrect nickname or password.")); break; case 0x11: /* Suspended account */ gc->wants_to_die = TRUE; gaim_connection_error(gc, _("Your account is currently suspended.")); break; case 0x14: /* service temporarily unavailable */ gaim_connection_error(gc, _("The AOL Instant Messenger service is temporarily unavailable.")); break; case 0x18: /* connecting too frequently */ gc->wants_to_die = TRUE; gaim_connection_error(gc, _("You have been connecting and disconnecting too frequently. Wait ten minutes and try again. If you continue to try, you will need to wait even longer.")); break; case 0x1c: /* client too old */ gc->wants_to_die = TRUE; g_snprintf(buf, sizeof(buf), _("The client version you are using is too old. Please upgrade at %s"), GAIM_WEBSITE); gaim_connection_error(gc, buf); break; default: gaim_connection_error(gc, _("Authentication failed")); break; } gaim_debug_error("oscar", "Login Error Code 0x%04hx\n", info->errorcode); gaim_debug_error("oscar", "Error URL: %s\n", info->errorurl); od->killme = TRUE; return 1; } gaim_debug_misc("oscar", "Reg status: %hu\n", info->regstatus); gaim_debug_misc("oscar", "E-mail: %s\n", (info->email != NULL) ? info->email : "null"); gaim_debug_misc("oscar", "BOSIP: %s\n", info->bosip); gaim_debug_info("oscar", "Closing auth connection...\n"); flap_connection_schedule_destroy(conn, OSCAR_DISCONNECT_DONE); for (i = 0; i < strlen(info->bosip); i++) { if (info->bosip[i] == ':') { port = atoi(&(info->bosip[i+1])); break; } } host = g_strndup(info->bosip, i); new_conn_data = g_new(NewFlapConnectionData, 1); new_conn_data->gc = gc; new_conn_data->conn = flap_connection_new(od, SNAC_FAMILY_LOCATE); new_conn_data->cookielen = info->cookielen; new_conn_data->cookie = g_memdup(info->cookie, info->cookielen); new_conn_data->data = NULL; rc = gaim_proxy_connect(gc->account, host, port, connection_established_cb, new_conn_data); g_free(host); if (rc < 0) { gaim_connection_error(gc, _("Could Not Connect")); od->killme = TRUE; return 0; } gaim_connection_update_progress(gc, _("Received authorization"), 3, OSCAR_CONNECT_STEPS); ck[3] = 0x64; return 1; } static void gaim_parse_auth_securid_request_yes_cb(gpointer user_data, const char *msg) { GaimConnection *gc = user_data; OscarData *od = gc->proto_data; aim_auth_securid_send(od, msg); } static void gaim_parse_auth_securid_request_no_cb(gpointer user_data, const char *value) { GaimConnection *gc = user_data; OscarData *od = gc->proto_data; /* Disconnect */ gc->wants_to_die = TRUE; gaim_connection_error(gc, _("The SecurID key entered is invalid.")); od->killme = TRUE; } static int gaim_parse_auth_securid_request(OscarData *od, FlapConnection *conn, FlapFrame *fr, ...) { GaimConnection *gc = od->gc; GaimAccount *account = gaim_connection_get_account(gc); gchar *primary; gaim_debug_info("oscar", "Got SecurID request\n"); primary = g_strdup_printf("Enter the SecurID key for %s.", gaim_account_get_username(account)); gaim_request_input(gc, NULL, _("Enter SecurID"), primary, _("Enter the 6 digit number from the digital display."), FALSE, FALSE, NULL, _("OK"), G_CALLBACK(gaim_parse_auth_securid_request_yes_cb), _("Cancel"), G_CALLBACK(gaim_parse_auth_securid_request_no_cb), gc); g_free(primary); return 1; } /* XXX - Should use gaim_url_fetch for the below stuff */ struct pieceofcrap { GaimConnection *gc; unsigned long offset; unsigned long len; char *modname; int fd; FlapConnection *conn; unsigned int inpa; }; static void damn_you(gpointer data, gint source, GaimInputCondition c) { struct pieceofcrap *pos = data; OscarData *od = pos->gc->proto_data; char in = '\0'; int x = 0; unsigned char m[17]; while (read(pos->fd, &in, 1) == 1) { if (in == '\n') x++; else if (in != '\r') x = 0; if (x == 2) break; in = '\0'; } if (in != '\n') { char buf[256]; g_snprintf(buf, sizeof(buf), _("You may be disconnected shortly. You may want to use TOC until " "this is fixed. Check %s for updates."), GAIM_WEBSITE); gaim_notify_warning(pos->gc, NULL, _("Gaim was unable to get a valid AIM login hash."), buf); gaim_input_remove(pos->inpa); close(pos->fd); g_free(pos); return; } if (read(pos->fd, m, 16) != 16) { gaim_debug_warning("oscar", "Could not read full AIM login hash " "from " AIMHASHDATA "--that's bad.\n"); } m[16] = '\0'; gaim_debug_misc("oscar", "Sending hash: "); for (x = 0; x < 16; x++) gaim_debug_misc(NULL, "%02hhx ", (unsigned char)m[x]); gaim_debug_misc(NULL, "\n"); gaim_input_remove(pos->inpa); close(pos->fd); aim_sendmemblock(od, pos->conn, 0, 16, m, AIM_SENDMEMBLOCK_FLAG_ISHASH); g_free(pos); } static void straight_to_hell(gpointer data, gint source, GaimInputCondition cond) { struct pieceofcrap *pos = data; gchar *buf; pos->fd = source; if (source < 0) { buf = g_strdup_printf(_("You may be disconnected shortly. You may want to use TOC until " "this is fixed. Check %s for updates."), GAIM_WEBSITE); gaim_notify_warning(pos->gc, NULL, _("Gaim was unable to get a valid AIM login hash."), buf); g_free(buf); if (pos->modname) g_free(pos->modname); g_free(pos); return; } buf = g_strdup_printf("GET " AIMHASHDATA "?offset=%ld&len=%ld&modname=%s HTTP/1.0\n\n", pos->offset, pos->len, pos->modname ? pos->modname : ""); write(pos->fd, buf, strlen(buf)); g_free(buf); if (pos->modname) g_free(pos->modname); pos->inpa = gaim_input_add(pos->fd, GAIM_INPUT_READ, damn_you, pos); return; } /* size of icbmui.ocm, the largest module in AIM 3.5 */ #define AIM_MAX_FILE_SIZE 98304 int gaim_memrequest(OscarData *od, FlapConnection *conn, FlapFrame *fr, ...) { va_list ap; struct pieceofcrap *pos; guint32 offset, len; char *modname; va_start(ap, fr); offset = va_arg(ap, guint32); len = va_arg(ap, guint32); modname = va_arg(ap, char *); va_end(ap); gaim_debug_misc("oscar", "offset: %u, len: %u, file: %s\n", offset, len, (modname ? modname : "aim.exe")); if (len == 0) { gaim_debug_misc("oscar", "len is 0, hashing NULL\n"); aim_sendmemblock(od, conn, offset, len, NULL, AIM_SENDMEMBLOCK_FLAG_ISREQUEST); return 1; } /* uncomment this when you're convinced it's right. remember, it's been wrong before. */ #if 0 if (offset > AIM_MAX_FILE_SIZE || len > AIM_MAX_FILE_SIZE) { char *buf; int i = 8; if (modname) i += strlen(modname); buf = g_malloc(i); i = 0; if (modname) { memcpy(buf, modname, strlen(modname)); i += strlen(modname); } buf[i++] = offset & 0xff; buf[i++] = (offset >> 8) & 0xff; buf[i++] = (offset >> 16) & 0xff; buf[i++] = (offset >> 24) & 0xff; buf[i++] = len & 0xff; buf[i++] = (len >> 8) & 0xff; buf[i++] = (len >> 16) & 0xff; buf[i++] = (len >> 24) & 0xff; gaim_debug_misc("oscar", "len + offset is invalid, " "hashing request\n"); aim_sendmemblock(od, command->conn, offset, i, buf, AIM_SENDMEMBLOCK_FLAG_ISREQUEST); g_free(buf); return 1; } #endif pos = g_new0(struct pieceofcrap, 1); pos->gc = od->gc; pos->conn = conn; pos->offset = offset; pos->len = len; pos->modname = modname ? g_strdup(modname) : NULL; if (gaim_proxy_connect(pos->gc->account, "gaim.sourceforge.net", 80, straight_to_hell, pos) != 0) { char buf[256]; if (pos->modname) g_free(pos->modname); g_free(pos); g_snprintf(buf, sizeof(buf), _("You may be disconnected shortly. " "Check %s for updates."), GAIM_WEBSITE); gaim_notify_warning(pos->gc, NULL, _("Gaim was unable to get a valid login hash."), buf); } return 1; } static int gaim_parse_login(OscarData *od, FlapConnection *conn, FlapFrame *fr, ...) { GaimConnection *gc; GaimAccount *account; ClientInfo info = CLIENTINFO_GAIM; va_list ap; char *key; gc = od->gc; account = gaim_connection_get_account(gc); va_start(ap, fr); key = va_arg(ap, char *); va_end(ap); aim_send_login(od, conn, gaim_account_get_username(account), gaim_connection_get_password(gc), &info, key); gaim_connection_update_progress(gc, _("Password sent"), 2, OSCAR_CONNECT_STEPS); ck[2] = 0x6c; return 1; } static int gaim_handle_redirect(OscarData *od, FlapConnection *conn, FlapFrame *fr, ...) { GaimConnection *gc = od->gc; GaimAccount *account = gaim_connection_get_account(gc); char *host, *separator; int port; NewFlapConnectionData *new_conn_data; va_list ap; struct aim_redirect_data *redir; va_start(ap, fr); redir = va_arg(ap, struct aim_redirect_data *); va_end(ap); port = gaim_account_get_int(account, "port", OSCAR_DEFAULT_LOGIN_PORT); separator = strchr(redir->ip, ':'); if (separator != NULL) { host = g_strndup(redir->ip, (int)separator - (int)redir->ip); port = atoi(separator + 1); } else host = g_strdup(redir->ip); gaim_debug_info("oscar", "Connecting to FLAP server %s:%d of type 0x%04hx\n", host, port, redir->group); new_conn_data = g_new(NewFlapConnectionData, 1); new_conn_data->gc = gc; new_conn_data->conn = flap_connection_new(od, redir->group); new_conn_data->cookielen = redir->cookielen; new_conn_data->cookie = g_memdup(redir->cookie, redir->cookielen); if (new_conn_data->conn->type == SNAC_FAMILY_CHAT) { struct chat_connection *cc; cc = g_new0(struct chat_connection, 1); cc->conn = new_conn_data->conn; cc->gc = gc; cc->name = g_strdup(redir->chat.room); cc->exchange = redir->chat.exchange; cc->instance = redir->chat.instance; cc->show = extract_name(redir->chat.room); new_conn_data->data = cc; gaim_debug_info("oscar", "Connecting to chat room %s exchange %hu\n", cc->name, cc->exchange); } else { new_conn_data->data = NULL; } if (gaim_proxy_connect(account, host, port, connection_established_cb, new_conn_data) != 0) { flap_connection_schedule_destroy(new_conn_data->conn, OSCAR_DISCONNECT_COULD_NOT_CONNECT); gaim_debug_error("oscar", "Unable to connect to FLAP server " "of type 0x%04hx\n", redir->group); destroy_new_conn_data(new_conn_data); } g_free(host); return 1; } static int gaim_parse_oncoming(OscarData *od, FlapConnection *conn, FlapFrame *fr, ...) { GaimConnection *gc; GaimAccount *account; GaimPresence *presence; struct buddyinfo *bi; time_t time_idle = 0, signon = 0; int type = 0; gboolean buddy_is_away = FALSE; const char *status_id; gboolean have_status_message = FALSE; char *message = NULL; va_list ap; aim_userinfo_t *info; gc = od->gc; account = gaim_connection_get_account(gc); presence = gaim_account_get_presence(account); va_start(ap, fr); info = va_arg(ap, aim_userinfo_t *); va_end(ap); g_return_val_if_fail(info != NULL, 1); g_return_val_if_fail(info->sn != NULL, 1); if (info->present & AIM_USERINFO_PRESENT_FLAGS) { if (info->flags & AIM_FLAG_AWAY) buddy_is_away = TRUE; } if (info->present & AIM_USERINFO_PRESENT_ICQEXTSTATUS) { type = info->icqinfo.status; if (!(info->icqinfo.status & AIM_ICQ_STATE_CHAT) && (info->icqinfo.status != AIM_ICQ_STATE_NORMAL)) { buddy_is_away = TRUE; } } if (aim_sn_is_icq(info->sn)) { if (type & AIM_ICQ_STATE_CHAT) status_id = OSCAR_STATUS_ID_FREE4CHAT; else if (type & AIM_ICQ_STATE_DND) status_id = OSCAR_STATUS_ID_DND; else if (type & AIM_ICQ_STATE_OUT) status_id = OSCAR_STATUS_ID_NA; else if (type & AIM_ICQ_STATE_BUSY) status_id = OSCAR_STATUS_ID_OCCUPIED; else if (type & AIM_ICQ_STATE_AWAY) status_id = OSCAR_STATUS_ID_AWAY; else if (type & AIM_ICQ_STATE_INVISIBLE) status_id = OSCAR_STATUS_ID_INVISIBLE; else status_id = OSCAR_STATUS_ID_AVAILABLE; } else { if (buddy_is_away) status_id = OSCAR_STATUS_ID_AWAY; else status_id = OSCAR_STATUS_ID_AVAILABLE; } /* * Handle the available message. If info->status is NULL then the user * may or may not have an available message, so don't do anything. If * info->status is set to the empty string, then the user's client DOES * support available messages and the user DOES NOT have one set. * Otherwise info->status contains the available message. */ if (info->status != NULL) { have_status_message = TRUE; if (info->status[0] != '\0') message = oscar_encoding_to_utf8(info->status_encoding, info->status, info->status_len); } if (have_status_message) { gaim_prpl_got_user_status(account, info->sn, status_id, "message", message, NULL); g_free(message); } else gaim_prpl_got_user_status(account, info->sn, status_id, NULL); /* Login time stuff */ if (info->present & AIM_USERINFO_PRESENT_ONLINESINCE) signon = info->onlinesince; else if (info->present & AIM_USERINFO_PRESENT_SESSIONLEN) signon = time(NULL) - info->sessionlen; if (!aim_sncmp(gaim_account_get_username(account), info->sn)) { gaim_connection_set_display_name(gc, info->sn); od->timeoffset = signon - gaim_presence_get_login_time(presence); } gaim_prpl_got_user_login_time(account, info->sn, signon - od->timeoffset); /* Idle time stuff */ /* info->idletime is the number of minutes that this user has been idle */ if (info->present & AIM_USERINFO_PRESENT_IDLE) time_idle = time(NULL) - info->idletime * 60; if (time_idle > 0) gaim_prpl_got_user_idle(account, info->sn, TRUE, time_idle); else gaim_prpl_got_user_idle(account, info->sn, FALSE, 0); /* Server stored icon stuff */ bi = g_hash_table_lookup(od->buddyinfo, gaim_normalize(account, info->sn)); if (!bi) { bi = g_new0(struct buddyinfo, 1); g_hash_table_insert(od->buddyinfo, g_strdup(gaim_normalize(account, info->sn)), bi); } bi->typingnot = FALSE; bi->ico_informed = FALSE; bi->ipaddr = info->icqinfo.ipaddr; if (info->iconcsumlen) { const char *filename, *saved_b16 = NULL; char *b16 = NULL, *filepath = NULL; GaimBuddy *b = NULL; b16 = gaim_base16_encode(info->iconcsum, info->iconcsumlen); b = gaim_find_buddy(account, info->sn); /* * If for some reason the checksum is valid, but cached file is not.. * we want to know. */ if (b != NULL) filename = gaim_blist_node_get_string((GaimBlistNode*)b, "buddy_icon"); else filename = NULL; if (filename != NULL) { if (g_file_test(filename, G_FILE_TEST_EXISTS)) saved_b16 = gaim_blist_node_get_string((GaimBlistNode*)b, "icon_checksum"); else { filepath = g_build_filename(gaim_buddy_icons_get_cache_dir(), filename, NULL); if (g_file_test(filepath, G_FILE_TEST_EXISTS)) saved_b16 = gaim_blist_node_get_string((GaimBlistNode*)b, "icon_checksum"); g_free(filepath); } } else saved_b16 = NULL; if (!b16 || !saved_b16 || strcmp(b16, saved_b16)) { GSList *cur = od->requesticon; while (cur && aim_sncmp((char *)cur->data, info->sn)) cur = cur->next; if (!cur) { od->requesticon = g_slist_append(od->requesticon, g_strdup(gaim_normalize(account, info->sn))); if (od->icontimer) gaim_timeout_remove(od->icontimer); od->icontimer = gaim_timeout_add(500, gaim_icon_timerfunc, gc); } } g_free(b16); } return 1; } static void gaim_check_comment(OscarData *od, const char *str) { if ((str == NULL) || strcmp(str, (const char *)ck)) aim_locate_setcaps(od, caps_aim); else aim_locate_setcaps(od, caps_aim | OSCAR_CAPABILITY_SECUREIM); } static int gaim_parse_offgoing(OscarData *od, FlapConnection *conn, FlapFrame *fr, ...) { GaimConnection *gc = od->gc; GaimAccount *account = gaim_connection_get_account(gc); va_list ap; aim_userinfo_t *info; va_start(ap, fr); info = va_arg(ap, aim_userinfo_t *); va_end(ap); gaim_prpl_got_user_status(account, info->sn, OSCAR_STATUS_ID_OFFLINE, NULL); g_hash_table_remove(od->buddyinfo, gaim_normalize(gc->account, info->sn)); return 1; } static int incomingim_chan1(OscarData *od, FlapConnection *conn, aim_userinfo_t *userinfo, struct aim_incomingim_ch1_args *args) { GaimConnection *gc = od->gc; GaimAccount *account = gaim_connection_get_account(gc); GaimMessageFlags flags = 0; struct buddyinfo *bi; char *iconfile; GString *message; gchar *tmp; aim_mpmsg_section_t *curpart; const char *start; GData *attribs; gaim_debug_misc("oscar", "Received IM from %s with %d parts\n", userinfo->sn, args->mpmsg.numparts); if (args->mpmsg.numparts == 0) return 1; bi = g_hash_table_lookup(od->buddyinfo, gaim_normalize(account, userinfo->sn)); if (!bi) { bi = g_new0(struct buddyinfo, 1); g_hash_table_insert(od->buddyinfo, g_strdup(gaim_normalize(account, userinfo->sn)), bi); } if (args->icbmflags & AIM_IMFLAGS_AWAY) flags |= GAIM_MESSAGE_AUTO_RESP; if (args->icbmflags & AIM_IMFLAGS_TYPINGNOT) bi->typingnot = TRUE; else bi->typingnot = FALSE; if ((args->icbmflags & AIM_IMFLAGS_HASICON) && (args->iconlen) && (args->iconsum) && (args->iconstamp)) { gaim_debug_misc("oscar", "%s has an icon\n", userinfo->sn); if ((args->iconlen != bi->ico_len) || (args->iconsum != bi->ico_csum) || (args->iconstamp != bi->ico_time)) { bi->ico_need = TRUE; bi->ico_len = args->iconlen; bi->ico_csum = args->iconsum; bi->ico_time = args->iconstamp; } } iconfile = gaim_buddy_icons_get_full_path(gaim_account_get_buddy_icon(account)); if ((iconfile != NULL) && (args->icbmflags & AIM_IMFLAGS_BUDDYREQ) && !bi->ico_sent && bi->ico_informed) { FILE *file; struct stat st; if (!g_stat(iconfile, &st)) { guchar *buf = g_malloc(st.st_size); file = g_fopen(iconfile, "rb"); if (file) { /* XXX - Use g_file_get_contents() */ int len = fread(buf, 1, st.st_size, file); gaim_debug_info("oscar", "Sending buddy icon to %s (%d bytes, " "%lu reported)\n", userinfo->sn, len, st.st_size); aim_im_sendch2_icon(od, userinfo->sn, buf, st.st_size, st.st_mtime, aimutil_iconsum(buf, st.st_size)); fclose(file); } else gaim_debug_error("oscar", "Can't open buddy icon file!\n"); g_free(buf); } else gaim_debug_error("oscar", "Can't stat buddy icon file!\n"); } g_free(iconfile); message = g_string_new(""); curpart = args->mpmsg.parts; while (curpart != NULL) { tmp = gaim_plugin_oscar_decode_im_part(account, userinfo->sn, curpart->charset, curpart->charsubset, curpart->data, curpart->datalen); if (tmp != NULL) { g_string_append(message, tmp); g_free(tmp); } curpart = curpart->next; } tmp = g_string_free(message, FALSE); /* * If the message is from an ICQ user and to an ICQ user then escape any HTML, * because HTML is not sent over ICQ as a means to format a message. * So any HTML we receive is intended to be displayed. Also, \r\n must be * replaced with <br> * * Note: There *may* be some clients which send messages as HTML formatted - * they need to be special-cased somehow. */ if (aim_sn_is_icq(gaim_account_get_username(account)) && aim_sn_is_icq(userinfo->sn)) { /* being recevied by ICQ from ICQ - escape HTML so it is displayed as sent */ gchar *tmp2 = g_markup_escape_text(tmp, -1); g_free(tmp); tmp = tmp2; tmp2 = gaim_strreplace(tmp, "\r\n", "<br>"); g_free(tmp); tmp = tmp2; } /* * Convert iChat color tags to normal font tags. */ if (gaim_markup_find_tag("body", tmp, &start, NULL, &attribs)) { const char *ichattextcolor, *ichatballooncolor; ichattextcolor = g_datalist_get_data(&attribs, "ichattextcolor"); if (ichattextcolor != NULL) { gchar *tmp2; tmp2 = g_strdup_printf("<font color=\"%s\">%s</font>", ichattextcolor, tmp); g_free(tmp); tmp = tmp2; } ichatballooncolor = g_datalist_get_data(&attribs, "ichatballooncolor"); if (ichatballooncolor != NULL) { gchar *tmp2; tmp2 = g_strdup_printf("<font back=\"%s\">%s</font>", ichatballooncolor, tmp); g_free(tmp); tmp = tmp2; } g_datalist_clear(&attribs); } serv_got_im(gc, userinfo->sn, tmp, flags, time(NULL)); g_free(tmp); return 1; } static int incomingim_chan2(OscarData *od, FlapConnection *conn, aim_userinfo_t *userinfo, IcbmArgsCh2 *args) { GaimConnection *gc; GaimAccount *account; char *message = NULL; g_return_val_if_fail(od != NULL, 0); g_return_val_if_fail(od->gc != NULL, 0); gc = od->gc; account = gaim_connection_get_account(gc); od = gc->proto_data; if (args == NULL) return 0; gaim_debug_misc("oscar", "Incoming rendezvous message of type %u, " "user %s, status %hu\n", args->type, userinfo->sn, args->status); if (args->msg != NULL) { if (args->encoding != NULL) { char *encoding = NULL; encoding = oscar_encoding_extract(args->encoding); message = oscar_encoding_to_utf8(encoding, args->msg, args->msglen); g_free(encoding); } else { if (g_utf8_validate(args->msg, args->msglen, NULL)) message = g_strdup(args->msg); } } if (args->type & OSCAR_CAPABILITY_CHAT) { char *name; GHashTable *components; if (!args->info.chat.roominfo.name || !args->info.chat.roominfo.exchange) { g_free(message); return 1; } components = g_hash_table_new_full(g_str_hash, g_str_equal, g_free, g_free); name = extract_name(args->info.chat.roominfo.name); g_hash_table_replace(components, g_strdup("room"), g_strdup(name ? name : args->info.chat.roominfo.name)); g_hash_table_replace(components, g_strdup("exchange"), g_strdup_printf("%d", args->info.chat.roominfo.exchange)); serv_got_chat_invite(gc, name ? name : args->info.chat.roominfo.name, userinfo->sn, message, components); if (name) g_free(name); } else if ((args->type & OSCAR_CAPABILITY_SENDFILE) || (args->type & OSCAR_CAPABILITY_DIRECTIM)) { if (args->status == AIM_RENDEZVOUS_PROPOSE) { peer_connection_got_proposition(od, userinfo->sn, message, args); } else if (args->status == AIM_RENDEZVOUS_CANCEL) { /* The other user canceled a peer request */ PeerConnection *conn; conn = peer_connection_find_by_cookie(od, userinfo->sn, args->cookie); /* * If conn is NULL it means we haven't tried to create * a connection with that user. They may be trying to * do something malicious. */ if (conn != NULL) { peer_connection_destroy(conn, OSCAR_DISCONNECT_REMOTE_CLOSED); } } else if (args->status == AIM_RENDEZVOUS_CONNECTED) { /* Remote user has accepted our peer request */ PeerConnection *conn; conn = peer_connection_find_by_cookie(od, userinfo->sn, args->cookie); /* * If conn is NULL it means we haven't tried to create * a connection with that user. They may be trying to * do something malicious. */ if (conn != NULL) { if (conn->listenerfd != -1) { /* * If they are connecting directly to us then * continue the peer negotiation by * accepting connections on our listener port. */ conn->watcher_incoming = gaim_input_add(conn->listenerfd, GAIM_INPUT_READ, peer_connection_listen_cb, conn); } } } } else if (args->type & OSCAR_CAPABILITY_GETFILE) { } else if (args->type & OSCAR_CAPABILITY_TALK) { } else if (args->type & OSCAR_CAPABILITY_BUDDYICON) { gaim_buddy_icons_set_for_user(account, userinfo->sn, args->info.icon.icon, args->info.icon.length); } else if (args->type & OSCAR_CAPABILITY_ICQSERVERRELAY) { gaim_debug_error("oscar", "Got an ICQ Server Relay message of " "type %d\n", args->info.rtfmsg.msgtype); } else { gaim_debug_error("oscar", "Unknown request class %hu\n", args->type); } g_free(message); return 1; } /* * Authorization Functions * Most of these are callbacks from dialogs. They're used by both * methods of authorization (SSI and old-school channel 4 ICBM) */ /* When you ask other people for authorization */ static void gaim_auth_request(struct name_data *data, char *msg) { GaimConnection *gc; OscarData *od; GaimBuddy *buddy; GaimGroup *group; gc = data->gc; od = gc->proto_data; buddy = gaim_find_buddy(gaim_connection_get_account(gc), data->name); if (buddy != NULL) group = gaim_buddy_get_group(buddy); else group = NULL; if (group != NULL) { gaim_debug_info("oscar", "ssi: adding buddy %s to group %s\n", buddy->name, group->name); aim_ssi_sendauthrequest(od, data->name, msg ? msg : _("Please authorize me so I can add you to my buddy list.")); if (!aim_ssi_itemlist_finditem(od->ssi.local, group->name, buddy->name, AIM_SSI_TYPE_BUDDY)) aim_ssi_addbuddy(od, buddy->name, group->name, gaim_buddy_get_alias_only(buddy), NULL, NULL, 1); } } static void gaim_auth_request_msgprompt(struct name_data *data) { gaim_request_input(data->gc, NULL, _("Authorization Request Message:"), NULL, _("Please authorize me!"), TRUE, FALSE, NULL, _("OK"), G_CALLBACK(gaim_auth_request), _("Cancel"), G_CALLBACK(oscar_free_name_data), data); } static void gaim_auth_dontrequest(struct name_data *data) { GaimConnection *gc = data->gc; if (g_list_find(gaim_connections_get_all(), gc)) { /* Remove from local list */ GaimBuddy *b = gaim_find_buddy(gaim_connection_get_account(gc), data->name); gaim_blist_remove_buddy(b); } oscar_free_name_data(data); } static void gaim_auth_sendrequest(GaimConnection *gc, char *name) { struct name_data *data = g_new0(struct name_data, 1); GaimBuddy *buddy; gchar *dialog_msg, *nombre; buddy = gaim_find_buddy(gc->account, name); if (buddy && (gaim_buddy_get_alias_only(buddy))) nombre = g_strdup_printf("%s (%s)", name, gaim_buddy_get_alias_only(buddy)); else nombre = NULL; dialog_msg = g_strdup_printf(_("The user %s requires authorization before being added to a buddy list. Do you want to send an authorization request?"), (nombre ? nombre : name)); data->gc = gc; data->name = g_strdup(name); data->nick = NULL; gaim_request_action(gc, NULL, _("Request Authorization"), dialog_msg, 0, data, 2, _("_Request Authorization"), G_CALLBACK(gaim_auth_request_msgprompt), _("Cancel"), G_CALLBACK(gaim_auth_dontrequest)); g_free(dialog_msg); g_free(nombre); } static void gaim_auth_sendrequest_menu(GaimBlistNode *node, gpointer ignored) { GaimBuddy *buddy; GaimConnection *gc; g_return_if_fail(GAIM_BLIST_NODE_IS_BUDDY(node)); buddy = (GaimBuddy *) node; gc = gaim_account_get_connection(buddy->account); gaim_auth_sendrequest(gc, buddy->name); } /* When other people ask you for authorization */ static void gaim_auth_grant(struct name_data *data) { GaimConnection *gc = data->gc; if (g_list_find(gaim_connections_get_all(), gc)) { OscarData *od = gc->proto_data; aim_ssi_sendauthreply(od, data->name, 0x01, NULL); } oscar_free_name_data(data); } /* When other people ask you for authorization */ static void gaim_auth_dontgrant(struct name_data *data, char *msg) { GaimConnection *gc = data->gc; if (g_list_find(gaim_connections_get_all(), gc)) { OscarData *od = gc->proto_data; aim_ssi_sendauthreply(od, data->name, 0x00, msg ? msg : _("No reason given.")); } } static void gaim_auth_dontgrant_msgprompt(struct name_data *data) { gaim_request_input(data->gc, NULL, _("Authorization Denied Message:"), NULL, _("No reason given."), TRUE, FALSE, NULL, _("OK"), G_CALLBACK(gaim_auth_dontgrant), _("Cancel"), G_CALLBACK(oscar_free_name_data), data); } /* When someone sends you buddies */ static void gaim_icq_buddyadd(struct name_data *data) { GaimConnection *gc = data->gc; if (g_list_find(gaim_connections_get_all(), gc)) { gaim_blist_request_add_buddy(gaim_connection_get_account(gc), data->name, NULL, data->nick); } oscar_free_name_data(data); } static int incomingim_chan4(OscarData *od, FlapConnection *conn, aim_userinfo_t *userinfo, struct aim_incomingim_ch4_args *args, time_t t) { GaimConnection *gc = od->gc; GaimAccount *account = gaim_connection_get_account(gc); gchar **msg1, **msg2; int i, numtoks; if (!args->type || !args->msg || !args->uin) return 1; gaim_debug_info("oscar", "Received a channel 4 message of type 0x%02hx.\n", args->type); /* * Split up the message at the delimeter character, then convert each * string to UTF-8. Unless, of course, this is a type 1 message. If * this is a type 1 message, then the delimiter 0xfe could be a valid * character in whatever encoding the message was sent in. Type 1 * messages are always made up of only one part, so we can easily account * for this suck-ass part of the protocol by splitting the string into at * most 1 baby string. */ msg1 = g_strsplit(args->msg, "\376", (args->type == 0x01 ? 1 : 0)); for (numtoks=0; msg1[numtoks]; numtoks++); msg2 = (gchar **)g_malloc((numtoks+1)*sizeof(gchar *)); for (i=0; msg1[i]; i++) { gaim_str_strip_char(msg1[i], '\r'); msg2[i] = gaim_plugin_oscar_decode_im_part(account, "1", AIM_CHARSET_ASCII, 0x0000, msg1[i], strlen(msg1[i])); } msg2[i] = NULL; switch (args->type) { case 0x01: { /* MacICQ message or basic offline message */ if (i >= 1) { gchar *uin = g_strdup_printf("%u", args->uin); gchar *tmp; /* If the message came from an ICQ user then escape any HTML */ tmp = g_markup_escape_text(msg2[0], -1); if (t) { /* This is an offline message */ /* The timestamp is UTC-ish, so we need to get the offset */ #ifdef HAVE_TM_GMTOFF time_t now; struct tm *tm; now = time(NULL); tm = localtime(&now); t += tm->tm_gmtoff; #else # ifdef HAVE_TIMEZONE tzset(); t -= timezone; # endif #endif serv_got_im(gc, uin, tmp, 0, t); } else { /* This is a message from MacICQ/Miranda */ serv_got_im(gc, uin, tmp, 0, time(NULL)); } g_free(uin); g_free(tmp); } } break; case 0x04: { /* Someone sent you a URL */ if (i >= 2) { if (msg2[1] != NULL) { gchar *uin = g_strdup_printf("%u", args->uin); gchar *message = g_strdup_printf("<A HREF=\"%s\">%s</A>", msg2[1], (msg2[0] && msg2[0][0]) ? msg2[0] : msg2[1]); serv_got_im(gc, uin, message, 0, time(NULL)); g_free(uin); g_free(message); } } } break; case 0x06: { /* Someone requested authorization */ if (i >= 6) { struct name_data *data = g_new(struct name_data, 1); gchar *sn = g_strdup_printf("%u", args->uin); gchar *reason; gchar *dialog_msg; if (msg2[5] != NULL) reason = gaim_plugin_oscar_decode_im_part(account, sn, AIM_CHARSET_CUSTOM, 0x0000, msg2[5], strlen(msg2[5])); else reason = g_strdup(_("No reason given.")); dialog_msg = g_strdup_printf(_("The user %u wants to add %s to their buddy list for the following reason:\n%s"), args->uin, gaim_account_get_username(gc->account), reason); g_free(reason); gaim_debug_info("oscar", "Received an authorization request from UIN %u\n", args->uin); data->gc = gc; data->name = sn; data->nick = NULL; gaim_request_action(gc, NULL, _("Authorization Request"), dialog_msg, GAIM_DEFAULT_ACTION_NONE, data, 2, _("_Authorize"), G_CALLBACK(gaim_auth_grant), _("_Deny"), G_CALLBACK(gaim_auth_dontgrant_msgprompt)); g_free(dialog_msg); } } break; case 0x07: { /* Someone has denied you authorization */ if (i >= 1) { 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.")); gaim_notify_info(gc, NULL, _("ICQ authorization denied."), dialog_msg); g_free(dialog_msg); } } break; case 0x08: { /* Someone has granted you authorization */ gchar *dialog_msg = g_strdup_printf(_("The user %u has granted your request to add them to your buddy list."), args->uin); gaim_notify_info(gc, NULL, "ICQ authorization accepted.", dialog_msg); g_free(dialog_msg); } break; case 0x09: { /* Message from the Godly ICQ server itself, I think */ if (i >= 5) { gchar *dialog_msg = g_strdup_printf(_("You have received a special message\n\nFrom: %s [%s]\n%s"), msg2[0], msg2[3], msg2[5]); gaim_notify_info(gc, NULL, "ICQ Server Message", dialog_msg); g_free(dialog_msg); } } break; case 0x0d: { /* Someone has sent you a pager message from http://www.icq.com/your_uin */ if (i >= 6) { gchar *dialog_msg = g_strdup_printf(_("You have received an ICQ page\n\nFrom: %s [%s]\n%s"), msg2[0], msg2[3], msg2[5]); gaim_notify_info(gc, NULL, "ICQ Page", dialog_msg); g_free(dialog_msg); } } break; case 0x0e: { /* Someone has emailed you at your_uin@pager.icq.com */ if (i >= 6) { gchar *dialog_msg = g_strdup_printf(_("You have received an ICQ e-mail from %s [%s]\n\nMessage is:\n%s"), msg2[0], msg2[3], msg2[5]); gaim_notify_info(gc, NULL, "ICQ E-Mail", dialog_msg); g_free(dialog_msg); } } break; case 0x12: { /* Ack for authorizing/denying someone. Or possibly an ack for sending any system notice */ /* Someone added you to their buddy list? */ } break; case 0x13: { /* Someone has sent you some ICQ buddies */ guint i, num; gchar **text; text = g_strsplit(args->msg, "\376", 0); if (text) { num = 0; for (i=0; i<strlen(text[0]); i++) num = num*10 + text[0][i]-48; for (i=0; i<num; i++) { struct name_data *data = g_new(struct name_data, 1); gchar *message = g_strdup_printf(_("ICQ user %u has sent you a buddy: %s (%s)"), args->uin, text[i*2+2], text[i*2+1]); data->gc = gc; data->name = g_strdup(text[i*2+1]); data->nick = g_strdup(text[i*2+2]); gaim_request_action(gc, NULL, message, _("Do you want to add this buddy " "to your buddy list?"), GAIM_DEFAULT_ACTION_NONE, data, 2, _("Add"), G_CALLBACK(gaim_icq_buddyadd), _("_Decline"), G_CALLBACK(oscar_free_name_data)); g_free(message); } g_strfreev(text); } } break; case 0x1a: { /* Someone has sent you a greeting card or requested buddies? */ /* This is boring and silly. */ } break; default: { gaim_debug_info("oscar", "Received a channel 4 message of unknown type " "(type 0x%02hhx).\n", args->type); } break; } g_strfreev(msg1); g_strfreev(msg2); return 1; } static int gaim_parse_incoming_im(OscarData *od, FlapConnection *conn, FlapFrame *fr, ...) { guint16 channel; int ret = 0; aim_userinfo_t *userinfo; va_list ap; va_start(ap, fr); channel = (guint16)va_arg(ap, unsigned int); userinfo = va_arg(ap, aim_userinfo_t *); switch (channel) { case 1: { /* standard message */ struct aim_incomingim_ch1_args *args; args = va_arg(ap, struct aim_incomingim_ch1_args *); ret = incomingim_chan1(od, conn, userinfo, args); } break; case 2: { /* rendezvous */ IcbmArgsCh2 *args; args = va_arg(ap, IcbmArgsCh2 *); ret = incomingim_chan2(od, conn, userinfo, args); } break; case 4: { /* ICQ */ struct aim_incomingim_ch4_args *args; args = va_arg(ap, struct aim_incomingim_ch4_args *); ret = incomingim_chan4(od, conn, userinfo, args, 0); } break; default: { gaim_debug_warning("oscar", "ICBM received on unsupported channel (channel " "0x%04hx).", channel); } break; } va_end(ap); return ret; } static int gaim_parse_misses(OscarData *od, FlapConnection *conn, FlapFrame *fr, ...) { GaimConnection *gc = od->gc; GaimAccount *account = gaim_connection_get_account(gc); char *buf; va_list ap; guint16 chan, nummissed, reason; aim_userinfo_t *userinfo; va_start(ap, fr); chan = (guint16)va_arg(ap, unsigned int); userinfo = va_arg(ap, aim_userinfo_t *); nummissed = (guint16)va_arg(ap, unsigned int); reason = (guint16)va_arg(ap, unsigned int); va_end(ap); switch(reason) { case 0: /* Invalid (0) */ buf = g_strdup_printf( ngettext( "You missed %hu message from %s because it was invalid.", "You missed %hu messages from %s because they were invalid.", nummissed), nummissed, userinfo->sn); break; case 1: /* Message too large */ buf = g_strdup_printf( ngettext( "You missed %hu message from %s because it was too large.", "You missed %hu messages from %s because they were too large.", nummissed), nummissed, userinfo->sn); break; case 2: /* Rate exceeded */ buf = g_strdup_printf( ngettext( "You missed %hu message from %s because the rate limit has been exceeded.", "You missed %hu messages from %s because the rate limit has been exceeded.", nummissed), nummissed, userinfo->sn); break; case 3: /* Evil Sender */ buf = g_strdup_printf( ngettext( "You missed %hu message from %s because he/she was too evil.", "You missed %hu messages from %s because he/she was too evil.", nummissed), nummissed, userinfo->sn); break; case 4: /* Evil Receiver */ buf = g_strdup_printf( ngettext( "You missed %hu message from %s because you are too evil.", "You missed %hu messages from %s because you are too evil.", nummissed), nummissed, userinfo->sn); break; default: buf = g_strdup_printf( ngettext( "You missed %hu message from %s for an unknown reason.", "You missed %hu messages from %s for an unknown reason.", nummissed), nummissed, userinfo->sn); break; } if (!gaim_conv_present_error(userinfo->sn, account, buf)) gaim_notify_error(od->gc, NULL, buf, NULL); g_free(buf); return 1; } static int gaim_parse_clientauto_ch2(OscarData *od, const char *who, guint16 reason, const guchar *cookie) { if (reason == 0x0003) { /* Rendezvous was refused. */ PeerConnection *conn; conn = peer_connection_find_by_cookie(od, who, cookie); if (conn == NULL) { gaim_debug_info("oscar", "Received a rendezvous cancel message " "for a nonexistant connection from %s.\n", who); } else { peer_connection_destroy(conn, OSCAR_DISCONNECT_REMOTE_REFUSED); } } else { gaim_debug_warning("oscar", "Received an unknown rendezvous " "message from %s. Type 0x%04hx\n", who, reason); } return 0; } static int gaim_parse_clientauto_ch4(OscarData *od, char *who, guint16 reason, guint32 state, char *msg) { GaimConnection *gc = od->gc; switch(reason) { case 0x0003: { /* Reply from an ICQ status message request */ char *title, *statusmsg, **splitmsg, *dialogmsg; title = g_strdup_printf(_("Info for %s"), who); /* Split at (carriage return/newline)'s, then rejoin later with BRs between. */ statusmsg = oscar_icqstatus(state); splitmsg = g_strsplit(msg, "\r\n", 0); dialogmsg = g_strdup_printf(_("<B>UIN:</B> %s<BR><B>Status:</B> %s<HR>%s"), who, statusmsg, g_strjoinv("<BR>", splitmsg)); g_free(statusmsg); g_strfreev(splitmsg); gaim_notify_userinfo(gc, who, dialogmsg, NULL, NULL); g_free(title); g_free(dialogmsg); } break; default: { gaim_debug_warning("oscar", "Received an unknown client auto-response from %s. " "Type 0x%04hx\n", who, reason); } break; } /* end of switch */ return 0; } static int gaim_parse_clientauto(OscarData *od, FlapConnection *conn, FlapFrame *fr, ...) { va_list ap; guint16 chan, reason; char *who; va_start(ap, fr); chan = (guint16)va_arg(ap, unsigned int); who = va_arg(ap, char *); reason = (guint16)va_arg(ap, unsigned int); if (chan == 0x0002) { /* File transfer declined */ guchar *cookie = va_arg(ap, guchar *); return gaim_parse_clientauto_ch2(od, who, reason, cookie); } else if (chan == 0x0004) { /* ICQ message */ guint32 state = 0; char *msg = NULL; if (reason == 0x0003) { state = va_arg(ap, guint32); msg = va_arg(ap, char *); } return gaim_parse_clientauto_ch4(od, who, reason, state, msg); } va_end(ap); return 1; } static int gaim_parse_genericerr(OscarData *od, FlapConnection *conn, FlapFrame *fr, ...) { va_list ap; guint16 reason; char *m; va_start(ap, fr); reason = (guint16) va_arg(ap, unsigned int); va_end(ap); gaim_debug_error("oscar", "snac threw error (reason 0x%04hx: %s)\n", reason, (reason < msgerrreasonlen) ? msgerrreason[reason] : "unknown"); m = g_strdup_printf(_("SNAC threw error: %s\n"), reason < msgerrreasonlen ? _(msgerrreason[reason]) : _("Unknown error")); gaim_notify_error(od->gc, NULL, m, NULL); g_free(m); return 1; } static int gaim_parse_msgerr(OscarData *od, FlapConnection *conn, FlapFrame *fr, ...) { GaimConnection *gc = od->gc; #ifdef TODOFT OscarData *od = gc->proto_data; GaimXfer *xfer; #endif va_list ap; guint16 reason; char *data, *buf; va_start(ap, fr); reason = (guint16)va_arg(ap, unsigned int); data = va_arg(ap, char *); va_end(ap); gaim_debug_error("oscar", "Message error with data %s and reason %hu\n", (data != NULL ? data : ""), reason); #ifdef TODOFT /* If this was a file transfer request, data is a cookie */ if ((xfer = oscar_find_xfer_by_cookie(od->file_transfers, data))) { gaim_xfer_cancel_remote(xfer); return 1; } #endif /* Data is assumed to be the destination sn */ buf = g_strdup_printf(_("Unable to send message: %s"), (reason < msgerrreasonlen) ? msgerrreason[reason] : _("Unknown reason.")); if (!gaim_conv_present_error(data, gaim_connection_get_account(gc), buf)) { g_free(buf); buf = g_strdup_printf(_("Unable to send message to %s:"), data ? data : "(unknown)"); gaim_notify_error(od->gc, NULL, buf, (reason < msgerrreasonlen) ? _(msgerrreason[reason]) : _("Unknown reason.")); } g_free(buf); return 1; } static int gaim_parse_mtn(OscarData *od, FlapConnection *conn, FlapFrame *fr, ...) { GaimConnection *gc = od->gc; va_list ap; guint16 type1, type2; char *sn; va_start(ap, fr); type1 = (guint16) va_arg(ap, unsigned int); sn = va_arg(ap, char *); type2 = (guint16) va_arg(ap, unsigned int); va_end(ap); switch (type2) { case 0x0000: { /* Text has been cleared */ serv_got_typing_stopped(gc, sn); } break; case 0x0001: { /* Paused typing */ serv_got_typing(gc, sn, 0, GAIM_TYPED); } break; case 0x0002: { /* Typing */ serv_got_typing(gc, sn, 0, GAIM_TYPING); } break; default: { gaim_debug_error("oscar", "Received unknown typing notification message from %s. Type1 is 0x%04x and type2 is 0x%04hx.\n", sn, type1, type2); } break; } return 1; } /* * We get this error when there was an error in the locate family. This * happens when you request info of someone who is offline. */ static int gaim_parse_locerr(OscarData *od, FlapConnection *conn, FlapFrame *fr, ...) { gchar *buf; va_list ap; guint16 reason; char *destn; va_start(ap, fr); reason = (guint16) va_arg(ap, unsigned int); destn = va_arg(ap, char *); va_end(ap); if (destn == NULL) return 1; buf = g_strdup_printf(_("User information not available: %s"), (reason < msgerrreasonlen) ? _(msgerrreason[reason]) : _("Unknown reason.")); if (!gaim_conv_present_error(destn, gaim_connection_get_account((GaimConnection*)od->gc), buf)) { g_free(buf); buf = g_strdup_printf(_("User information for %s unavailable:"), destn); gaim_notify_error(od->gc, NULL, buf, (reason < msgerrreasonlen) ? _(msgerrreason[reason]) : _("Unknown reason.")); } g_free(buf); return 1; } static int gaim_parse_userinfo(OscarData *od, FlapConnection *conn, FlapFrame *fr, ...) { GaimConnection *gc = od->gc; GaimAccount *account = gaim_connection_get_account(gc); GString *str; gchar *tmp = NULL, *info_utf8 = NULL, *away_utf8 = NULL; va_list ap; aim_userinfo_t *userinfo; va_start(ap, fr); userinfo = va_arg(ap, aim_userinfo_t *); va_end(ap); str = g_string_new(""); g_string_append_printf(str, "<b>%s:</b> %s", _("Screen Name"), userinfo->sn); g_string_append_printf(str, "\n<br><b>%s</b>: %d%%", _("Warning Level"), (int)((userinfo->warnlevel/10.0) + 0.5)); if (userinfo->present & AIM_USERINFO_PRESENT_ONLINESINCE) { time_t t = userinfo->onlinesince - od->timeoffset; oscar_string_append(gc->account, str, "\n<br>", _("Online Since"), gaim_date_format_full(localtime(&t))); } if (userinfo->present & AIM_USERINFO_PRESENT_MEMBERSINCE) { time_t t = userinfo->membersince - od->timeoffset; oscar_string_append(gc->account, str, "\n<br>", _("Member Since"), gaim_date_format_full(localtime(&t))); } if (userinfo->capabilities != 0) { tmp = oscar_caps_to_string(userinfo->capabilities); oscar_string_append(gc->account, str, "\n<br>", _("Capabilities"), tmp); g_free(tmp); } if (userinfo->present & AIM_USERINFO_PRESENT_IDLE) { tmp = gaim_str_seconds_to_string(userinfo->idletime*60); oscar_string_append(gc->account, str, "\n<br>", _("Idle"), tmp); g_free(tmp); } oscar_string_append_info(gc, str, "\n<br>", NULL, userinfo); /* Available message */ if ((userinfo->status != NULL) && !(userinfo->flags & AIM_FLAG_AWAY)) { if (userinfo->status[0] != '\0') tmp = oscar_encoding_to_utf8(userinfo->status_encoding, userinfo->status, userinfo->status_len); oscar_string_append(gc->account, str, "\n<br>", _("Available Message"), tmp); g_free(tmp); } /* Away message */ if ((userinfo->flags & AIM_FLAG_AWAY) && (userinfo->away_len > 0) && (userinfo->away != NULL) && (userinfo->away_encoding != NULL)) { tmp = oscar_encoding_extract(userinfo->away_encoding); away_utf8 = oscar_encoding_to_utf8(tmp, userinfo->away, userinfo->away_len); g_free(tmp); if (away_utf8 != NULL) { g_string_append_printf(str, "\n<hr>%s", away_utf8); g_free(away_utf8); } } /* Info */ if ((userinfo->info_len > 0) && (userinfo->info != NULL) && (userinfo->info_encoding != NULL)) { tmp = oscar_encoding_extract(userinfo->info_encoding); info_utf8 = oscar_encoding_to_utf8(tmp, userinfo->info, userinfo->info_len); g_free(tmp); if (info_utf8 != NULL) { g_string_append_printf(str, "\n<hr>%s", info_utf8); g_free(info_utf8); } } tmp = gaim_str_sub_away_formatters(str->str, gaim_account_get_username(account)); g_string_free(str, TRUE); gaim_str_strip_char(tmp, '\r'); gaim_notify_userinfo(gc, userinfo->sn, tmp, NULL, NULL); g_free(tmp); return 1; } static int gaim_got_infoblock(OscarData *od, FlapConnection *conn, FlapFrame *fr, ...) { GaimConnection *gc = od->gc; GaimBuddy *b; GaimPresence *presence; GaimStatus *status; gchar *message = NULL; va_list ap; aim_userinfo_t *userinfo; va_start(ap, fr); userinfo = va_arg(ap, aim_userinfo_t *); va_end(ap); b = gaim_find_buddy(gaim_connection_get_account(gc), userinfo->sn); if (b == NULL) return 1; presence = gaim_buddy_get_presence(b); status = gaim_presence_get_active_status(presence); if (!gaim_status_is_available(status) && gaim_status_is_online(status)) { if ((userinfo->flags & AIM_FLAG_AWAY) && (userinfo->away_len > 0) && (userinfo->away != NULL) && (userinfo->away_encoding != NULL)) { gchar *charset = oscar_encoding_extract(userinfo->away_encoding); message = oscar_encoding_to_utf8(charset, userinfo->away, userinfo->away_len); g_free(charset); gaim_status_set_attr_string(status, "message", message); g_free(message); } gaim_blist_update_buddy_status(b, status); } return 1; } static int gaim_parse_motd(OscarData *od, FlapConnection *conn, FlapFrame *fr, ...) { char *msg; guint16 id; va_list ap; va_start(ap, fr); id = (guint16) va_arg(ap, unsigned int); msg = va_arg(ap, char *); va_end(ap); gaim_debug_misc("oscar", "MOTD: %s (%hu)\n", msg ? msg : "Unknown", id); if (id < 4) gaim_notify_warning(od->gc, NULL, _("Your AIM connection may be lost."), NULL); return 1; } static int gaim_chatnav_info(OscarData *od, FlapConnection *conn, FlapFrame *fr, ...) { va_list ap; guint16 type; va_start(ap, fr); type = (guint16) va_arg(ap, unsigned int); switch(type) { case 0x0002: { guint8 maxrooms; struct aim_chat_exchangeinfo *exchanges; int exchangecount, i; maxrooms = (guint8) va_arg(ap, unsigned int); exchangecount = va_arg(ap, int); exchanges = va_arg(ap, struct aim_chat_exchangeinfo *); gaim_debug_misc("oscar", "chat info: Chat Rights:\n"); gaim_debug_misc("oscar", "chat info: \tMax Concurrent Rooms: %hhd\n", maxrooms); gaim_debug_misc("oscar", "chat info: \tExchange List: (%d total)\n", exchangecount); for (i = 0; i < exchangecount; i++) gaim_debug_misc("oscar", "chat info: \t\t%hu %s\n", exchanges[i].number, exchanges[i].name ? exchanges[i].name : ""); while (od->create_rooms) { struct create_room *cr = od->create_rooms->data; gaim_debug_info("oscar", "creating room %s\n", cr->name); aim_chatnav_createroom(od, conn, cr->name, cr->exchange); g_free(cr->name); od->create_rooms = g_slist_remove(od->create_rooms, cr); g_free(cr); } } break; case 0x0008: { char *fqcn, *name, *ck; guint16 instance, flags, maxmsglen, maxoccupancy, unknown, exchange; guint8 createperms; guint32 createtime; fqcn = va_arg(ap, char *); instance = (guint16)va_arg(ap, unsigned int); exchange = (guint16)va_arg(ap, unsigned int); flags = (guint16)va_arg(ap, unsigned int); createtime = va_arg(ap, guint32); maxmsglen = (guint16)va_arg(ap, unsigned int); maxoccupancy = (guint16)va_arg(ap, unsigned int); createperms = (guint8)va_arg(ap, unsigned int); unknown = (guint16)va_arg(ap, unsigned int); name = va_arg(ap, char *); ck = va_arg(ap, char *); gaim_debug_misc("oscar", "created room: %s %hu %hu %hu %u %hu %hu %hhu %hu %s %s\n", fqcn, exchange, instance, flags, createtime, maxmsglen, maxoccupancy, createperms, unknown, name, ck); aim_chat_join(od, exchange, ck, instance); } break; default: gaim_debug_warning("oscar", "chatnav info: unknown type (%04hx)\n", type); break; } va_end(ap); return 1; } static int gaim_conv_chat_join(OscarData *od, FlapConnection *conn, FlapFrame *fr, ...) { va_list ap; int count, i; aim_userinfo_t *info; GaimConnection *gc = od->gc; struct chat_connection *c = NULL; va_start(ap, fr); count = va_arg(ap, int); info = va_arg(ap, aim_userinfo_t *); va_end(ap); c = find_oscar_chat_by_conn(gc, conn); if (!c) return 1; for (i = 0; i < count; i++) gaim_conv_chat_add_user(GAIM_CONV_CHAT(c->conv), info[i].sn, NULL, GAIM_CBFLAGS_NONE, TRUE); return 1; } static int gaim_conv_chat_leave(OscarData *od, FlapConnection *conn, FlapFrame *fr, ...) { va_list ap; int count, i; aim_userinfo_t *info; GaimConnection *gc = od->gc; struct chat_connection *c = NULL; va_start(ap, fr); count = va_arg(ap, int); info = va_arg(ap, aim_userinfo_t *); va_end(ap); c = find_oscar_chat_by_conn(gc, conn); if (!c) return 1; for (i = 0; i < count; i++) gaim_conv_chat_remove_user(GAIM_CONV_CHAT(c->conv), info[i].sn, NULL); return 1; } static int gaim_conv_chat_info_update(OscarData *od, FlapConnection *conn, FlapFrame *fr, ...) { va_list ap; aim_userinfo_t *userinfo; struct aim_chat_roominfo *roominfo; char *roomname; int usercount; char *roomdesc; guint16 unknown_c9, unknown_d2, unknown_d5, maxmsglen, maxvisiblemsglen; guint32 creationtime; GaimConnection *gc = od->gc; struct chat_connection *ccon = find_oscar_chat_by_conn(gc, conn); va_start(ap, fr); roominfo = va_arg(ap, struct aim_chat_roominfo *); roomname = va_arg(ap, char *); usercount= va_arg(ap, int); userinfo = va_arg(ap, aim_userinfo_t *); roomdesc = va_arg(ap, char *); unknown_c9 = (guint16)va_arg(ap, unsigned int); creationtime = va_arg(ap, guint32); maxmsglen = (guint16)va_arg(ap, unsigned int); unknown_d2 = (guint16)va_arg(ap, unsigned int); unknown_d5 = (guint16)va_arg(ap, unsigned int); maxvisiblemsglen = (guint16)va_arg(ap, unsigned int); va_end(ap); gaim_debug_misc("oscar", "inside chat_info_update (maxmsglen = %hu, maxvislen = %hu)\n", maxmsglen, maxvisiblemsglen); ccon->maxlen = maxmsglen; ccon->maxvis = maxvisiblemsglen; return 1; } static int gaim_conv_chat_incoming_msg(OscarData *od, FlapConnection *conn, FlapFrame *fr, ...) { GaimConnection *gc = od->gc; struct chat_connection *ccon = find_oscar_chat_by_conn(gc, conn); gchar *utf8; va_list ap; aim_userinfo_t *info; int len; char *msg; char *charset; va_start(ap, fr); info = va_arg(ap, aim_userinfo_t *); len = va_arg(ap, int); msg = va_arg(ap, char *); charset = va_arg(ap, char *); va_end(ap); utf8 = oscar_encoding_to_utf8(charset, msg, len); if (utf8 == NULL) /* The conversion failed! */ utf8 = g_strdup(_("[Unable to display a message from this user because it contained invalid characters.]")); serv_got_chat_in(gc, ccon->id, info->sn, 0, utf8, time((time_t)NULL)); g_free(utf8); return 1; } static int gaim_email_parseupdate(OscarData *od, FlapConnection *conn, FlapFrame *fr, ...) { va_list ap; GaimConnection *gc = od->gc; struct aim_emailinfo *emailinfo; int havenewmail; char *alertitle, *alerturl; va_start(ap, fr); emailinfo = va_arg(ap, struct aim_emailinfo *); havenewmail = va_arg(ap, int); alertitle = va_arg(ap, char *); alerturl = va_arg(ap, char *); va_end(ap); if ((emailinfo != NULL) && gaim_account_get_check_mail(gc->account)) { gchar *to = g_strdup_printf("%s%s%s", gaim_account_get_username(gaim_connection_get_account(gc)), emailinfo->domain ? "@" : "", emailinfo->domain ? emailinfo->domain : ""); if (emailinfo->unread && havenewmail) gaim_notify_emails(gc, emailinfo->nummsgs, FALSE, NULL, NULL, (const char **)&to, (const char **)&emailinfo->url, NULL, NULL); g_free(to); } if (alertitle) gaim_debug_misc("oscar", "Got an alert '%s' %s\n", alertitle, alerturl ? alerturl : ""); return 1; } static int gaim_icon_error(OscarData *od, FlapConnection *conn, FlapFrame *fr, ...) { GaimConnection *gc = od->gc; char *sn; sn = od->requesticon->data; gaim_debug_misc("oscar", "removing %s from hash table\n", sn); od->requesticon = g_slist_remove(od->requesticon, sn); g_free(sn); if (od->icontimer) gaim_timeout_remove(od->icontimer); od->icontimer = gaim_timeout_add(500, gaim_icon_timerfunc, gc); return 1; } static int gaim_icon_parseicon(OscarData *od, FlapConnection *conn, FlapFrame *fr, ...) { GaimConnection *gc = od->gc; GSList *cur; va_list ap; char *sn; guint8 iconcsumtype, *iconcsum, *icon; guint16 iconcsumlen, iconlen; va_start(ap, fr); sn = va_arg(ap, char *); iconcsumtype = va_arg(ap, int); iconcsum = va_arg(ap, guint8 *); iconcsumlen = va_arg(ap, int); icon = va_arg(ap, guint8 *); iconlen = va_arg(ap, int); va_end(ap); /* * Some AIM clients will send a blank GIF image with iconlen 90 when * no icon is set. Ignore these. */ if ((iconlen > 0) && (iconlen != 90)) { char *b16; GaimBuddy *b; gaim_buddy_icons_set_for_user(gaim_connection_get_account(gc), sn, icon, iconlen); b16 = gaim_base16_encode(iconcsum, iconcsumlen); b = gaim_find_buddy(gc->account, sn); if ((b16 != NULL) && (b != NULL)) { gaim_blist_node_set_string((GaimBlistNode*)b, "icon_checksum", b16); g_free(b16); } } cur = od->requesticon; while (cur) { char *cursn = cur->data; if (!aim_sncmp(cursn, sn)) { od->requesticon = g_slist_remove(od->requesticon, cursn); g_free(cursn); cur = od->requesticon; } else cur = cur->next; } if (od->icontimer) gaim_timeout_remove(od->icontimer); od->icontimer = gaim_timeout_add(250, gaim_icon_timerfunc, gc); return 1; } static gboolean gaim_icon_timerfunc(gpointer data) { GaimConnection *gc = data; OscarData *od = gc->proto_data; aim_userinfo_t *userinfo; FlapConnection *conn; conn = flap_connection_getbytype(od, SNAC_FAMILY_BART); if (!conn) { if (!od->iconconnecting) { aim_reqservice(od, SNAC_FAMILY_BART); od->iconconnecting = TRUE; } return FALSE; } if (od->set_icon) { struct stat st; char *iconfile = gaim_buddy_icons_get_full_path(gaim_account_get_buddy_icon(gaim_connection_get_account(gc))); if (iconfile == NULL) { aim_ssi_delicon(od); } else if (!g_stat(iconfile, &st)) { guchar *buf = g_malloc(st.st_size); FILE *file = g_fopen(iconfile, "rb"); if (file) { /* XXX - Use g_file_get_contents()? */ fread(buf, 1, st.st_size, file); fclose(file); gaim_debug_info("oscar", "Uploading icon to icon server\n"); aim_bart_upload(od, buf, st.st_size); } else gaim_debug_error("oscar", "Can't open buddy icon file!\n"); g_free(buf); } else { gaim_debug_error("oscar", "Can't stat buddy icon file!\n"); } g_free(iconfile); od->set_icon = FALSE; } if (!od->requesticon) { gaim_debug_misc("oscar", "no more icons to request\n"); return FALSE; } userinfo = aim_locate_finduserinfo(od, (char *)od->requesticon->data); if ((userinfo != NULL) && (userinfo->iconcsumlen > 0)) { aim_bart_request(od, od->requesticon->data, userinfo->iconcsumtype, userinfo->iconcsum, userinfo->iconcsumlen); return FALSE; } else { gchar *sn = od->requesticon->data; od->requesticon = g_slist_remove(od->requesticon, sn); g_free(sn); } return TRUE; } /* * Recieved in response to an IM sent with the AIM_IMFLAGS_ACK option. */ static int gaim_parse_msgack(OscarData *od, FlapConnection *conn, FlapFrame *fr, ...) { va_list ap; guint16 type; char *sn; va_start(ap, fr); type = (guint16) va_arg(ap, unsigned int); sn = va_arg(ap, char *); va_end(ap); gaim_debug_info("oscar", "Sent message to %s.\n", sn); return 1; } static int gaim_parse_ratechange(OscarData *od, FlapConnection *conn, FlapFrame *fr, ...) { static const char *codes[5] = { "invalid", "change", "warning", "limit", "limit cleared", }; va_list ap; guint16 code, rateclass; guint32 windowsize, clear, alert, limit, disconnect, currentavg, maxavg; va_start(ap, fr); code = (guint16)va_arg(ap, unsigned int); rateclass= (guint16)va_arg(ap, unsigned int); windowsize = va_arg(ap, guint32); clear = va_arg(ap, guint32); alert = va_arg(ap, guint32); limit = va_arg(ap, guint32); disconnect = va_arg(ap, guint32); currentavg = va_arg(ap, guint32); maxavg = va_arg(ap, guint32); va_end(ap); gaim_debug_misc("oscar", "rate %s (param ID 0x%04hx): curavg = %u, maxavg = %u, alert at %u, " "clear warning at %u, limit at %u, disconnect at %u (window size = %u)\n", (code < 5) ? codes[code] : codes[0], rateclass, currentavg, maxavg, alert, clear, limit, disconnect, windowsize); if (code == AIM_RATE_CODE_LIMIT) { gaim_notify_error(od->gc, NULL, _("Rate limiting error."), _("The last action you attempted could not be " "performed because you are over the rate limit. " "Please wait 10 seconds and try again.")); } return 1; } static int gaim_parse_evilnotify(OscarData *od, FlapConnection *conn, FlapFrame *fr, ...) { #ifdef CRAZY_WARNING va_list ap; guint16 newevil; aim_userinfo_t *userinfo; va_start(ap, fr); newevil = (guint16) va_arg(ap, unsigned int); userinfo = va_arg(ap, aim_userinfo_t *); va_end(ap); gaim_prpl_got_account_warning_level(account, (userinfo && userinfo->sn) ? userinfo->sn : NULL, (newevil/10.0) + 0.5); #endif return 1; } static int gaim_selfinfo(OscarData *od, FlapConnection *conn, FlapFrame *fr, ...) { int warning_level; va_list ap; aim_userinfo_t *info; va_start(ap, fr); info = va_arg(ap, aim_userinfo_t *); va_end(ap); /* * What's with the + 0.5? * The 0.5 is basically poor-man's rounding. Normally * casting "13.7" to an int will truncate to "13," but * with 13.7 + 0.5 = 14.2, which becomes "14" when * truncated. */ warning_level = info->warnlevel/10.0 + 0.5; #ifdef CRAZY_WARNING gaim_presence_set_warning_level(presence, warning_level); #endif return 1; } static int gaim_connerr(OscarData *od, FlapConnection *conn, FlapFrame *fr, ...) { GaimConnection *gc = od->gc; va_list ap; guint16 code; char *msg; va_start(ap, fr); code = (guint16)va_arg(ap, int); msg = va_arg(ap, char *); va_end(ap); gaim_debug_info("oscar", "Disconnected. Code is 0x%04x and msg is %s\n", code, (msg != NULL ? msg : "")); g_return_val_if_fail(fr != NULL, 1); g_return_val_if_fail(conn != NULL, 1); if (conn->type == SNAC_FAMILY_LOCATE) { if (code == 0x0001) { gc->wants_to_die = TRUE; gaim_connection_error(gc, _("You have signed on from another location.")); } else { gaim_connection_error(gc, _("You have been signed off for an unknown reason.")); } od->killme = TRUE; } else if (conn->type == SNAC_FAMILY_CHAT) { struct chat_connection *cc; GaimConversation *conv; cc = find_oscar_chat_by_conn(gc, conn); conv = gaim_find_chat(gc, cc->id); if (conv != NULL) { gchar *buf; buf = g_strdup_printf(_("You have been disconnected from chat " "room %s."), cc->name); gaim_conversation_write(conv, NULL, buf, GAIM_MESSAGE_ERROR, time(NULL)); g_free(buf); } oscar_chat_kill(gc, cc); } return 1; } static int gaim_icbm_param_info(OscarData *od, FlapConnection *conn, FlapFrame *fr, ...) { struct aim_icbmparameters *params; va_list ap; va_start(ap, fr); params = va_arg(ap, struct aim_icbmparameters *); va_end(ap); /* XXX - evidently this crashes on solaris. i have no clue why gaim_debug_misc("oscar", "ICBM Parameters: maxchannel = %hu, default flags = 0x%08lx, max msg len = %hu, " "max sender evil = %f, max receiver evil = %f, min msg interval = %u\n", params->maxchan, params->flags, params->maxmsglen, ((float)params->maxsenderwarn)/10.0, ((float)params->maxrecverwarn)/10.0, params->minmsginterval); */ /* Maybe senderwarn and recverwarn should be user preferences... */ params->flags = 0x0000000b; params->maxmsglen = 8000; params->minmsginterval = 0; aim_im_setparams(od, params); return 1; } static int gaim_parse_locaterights(OscarData *od, FlapConnection *conn, FlapFrame *fr, ...) { GaimConnection *gc = od->gc; GaimAccount *account = gaim_connection_get_account(gc); va_list ap; guint16 maxsiglen; va_start(ap, fr); maxsiglen = (guint16) va_arg(ap, int); va_end(ap); gaim_debug_misc("oscar", "locate rights: max sig len = %d\n", maxsiglen); od->rights.maxsiglen = od->rights.maxawaymsglen = (guint)maxsiglen; if (od->icq) aim_locate_setcaps(od, caps_icq); else aim_locate_setcaps(od, caps_aim); oscar_set_info_and_status(account, TRUE, account->user_info, TRUE, gaim_account_get_active_status(account)); return 1; } static int gaim_parse_buddyrights(OscarData *od, FlapConnection *conn, FlapFrame *fr, ...) { va_list ap; guint16 maxbuddies, maxwatchers; va_start(ap, fr); maxbuddies = (guint16) va_arg(ap, unsigned int); maxwatchers = (guint16) va_arg(ap, unsigned int); va_end(ap); gaim_debug_misc("oscar", "buddy list rights: Max buddies = %hu / Max watchers = %hu\n", maxbuddies, maxwatchers); od->rights.maxbuddies = (guint)maxbuddies; od->rights.maxwatchers = (guint)maxwatchers; return 1; } static int gaim_bosrights(OscarData *od, FlapConnection *conn, FlapFrame *fr, ...) { GaimConnection *gc; GaimAccount *account; GaimStatus *status; const char *message; char *tmp; va_list ap; guint16 maxpermits, maxdenies; gc = od->gc; od = (OscarData *)gc->proto_data; account = gaim_connection_get_account(gc); va_start(ap, fr); maxpermits = (guint16) va_arg(ap, unsigned int); maxdenies = (guint16) va_arg(ap, unsigned int); va_end(ap); gaim_debug_misc("oscar", "BOS rights: Max permit = %hu / Max deny = %hu\n", maxpermits, maxdenies); od->rights.maxpermits = (guint)maxpermits; od->rights.maxdenies = (guint)maxdenies; gaim_connection_set_state(gc, GAIM_CONNECTED); gaim_debug_info("oscar", "buddy list loaded\n"); aim_clientready(od, conn); /* Set our available message based on the current status */ status = gaim_account_get_active_status(account); if (gaim_status_is_available(status)) message = gaim_status_get_attr_string(status, "message"); else message = NULL; tmp = gaim_markup_strip_html(message); aim_srv_setstatusmsg(od, tmp); g_free(tmp); aim_srv_setidle(od, 0); if (od->icq) { aim_icq_reqofflinemsgs(od); oscar_set_extendedstatus(gc); aim_icq_setsecurity(od, gaim_account_get_bool(account, "authorization", OSCAR_DEFAULT_AUTHORIZATION), gaim_account_get_bool(account, "web_aware", OSCAR_DEFAULT_WEB_AWARE)); } aim_reqservice(od, SNAC_FAMILY_CHATNAV); if (od->authinfo->email != NULL) aim_reqservice(od, SNAC_FAMILY_ALERT); return 1; } static int gaim_offlinemsg(OscarData *od, FlapConnection *conn, FlapFrame *fr, ...) { va_list ap; struct aim_icq_offlinemsg *msg; struct aim_incomingim_ch4_args args; time_t t; va_start(ap, fr); msg = va_arg(ap, struct aim_icq_offlinemsg *); va_end(ap); gaim_debug_info("oscar", "Received offline message. Converting to channel 4 ICBM...\n"); args.uin = msg->sender; args.type = msg->type; args.flags = msg->flags; args.msglen = msg->msglen; args.msg = msg->msg; t = gaim_time_build(msg->year, msg->month, msg->day, msg->hour, msg->minute, 0); incomingim_chan4(od, conn, NULL, &args, t); return 1; } static int gaim_offlinemsgdone(OscarData *od, FlapConnection *conn, FlapFrame *fr, ...) { aim_icq_ackofflinemsgs(od); return 1; } static int gaim_icqinfo(OscarData *od, FlapConnection *conn, FlapFrame *fr, ...) { GaimConnection *gc = od->gc; GaimBuddy *buddy; struct buddyinfo *bi = NULL; gchar who[16]; GString *str; gchar *utf8; const gchar *alias; va_list ap; struct aim_icq_info *info; va_start(ap, fr); info = va_arg(ap, struct aim_icq_info *); va_end(ap); if (!info->uin) return 0; str = g_string_sized_new(100); g_snprintf(who, sizeof(who), "%u", info->uin); buddy = gaim_find_buddy(gaim_connection_get_account(gc), who); if (buddy != NULL) bi = g_hash_table_lookup(od->buddyinfo, gaim_normalize(buddy->account, buddy->name)); g_string_append_printf(str, "<b>%s:</b> %s", _("UIN"), who); oscar_string_append(gc->account, str, "\n<br>", _("Nick"), info->nick); if ((bi != NULL) && (bi->ipaddr != 0)) { char *tstr = g_strdup_printf("%hhu.%hhu.%hhu.%hhu", (bi->ipaddr & 0xff000000) >> 24, (bi->ipaddr & 0x00ff0000) >> 16, (bi->ipaddr & 0x0000ff00) >> 8, (bi->ipaddr & 0x000000ff)); oscar_string_append(gc->account, str, "\n<br>", _("IP Address"), tstr); g_free(tstr); } oscar_string_append(gc->account, str, "\n<br>", _("First Name"), info->first); oscar_string_append(gc->account, str, "\n<br>", _("Last Name"), info->last); if (info->email && info->email[0] && (utf8 = oscar_utf8_try_convert(gc->account, info->email))) { g_string_append_printf(str, "\n<br><b>%s:</b> <a href=\"mailto:%s\">%s</a>", _("E-Mail Address"), utf8, utf8); g_free(utf8); } if (info->numaddresses && info->email2) { int i; for (i = 0; i < info->numaddresses; i++) { if (info->email2[i] && info->email2[i][0] && (utf8 = oscar_utf8_try_convert(gc->account, info->email2[i]))) { g_string_append_printf(str, "\n<br><b>%s:</b> <a href=\"mailto:%s\">%s</a>", _("E-Mail Address"), utf8, utf8); g_free(utf8); } } } oscar_string_append(gc->account, str, "\n<br>", _("Mobile Phone"), info->mobile); if (info->gender != 0) oscar_string_append(gc->account, str, "\n<br>", _("Gender"), info->gender == 1 ? _("Female") : _("Male")); if ((info->birthyear > 1900) && (info->birthmonth > 0) && (info->birthday > 0)) { struct tm tm; tm.tm_mday = (int)info->birthday; tm.tm_mon = (int)info->birthmonth-1; tm.tm_year = (int)info->birthyear-1900; oscar_string_append(gc->account, str, "\n<br>", _("Birthday"), gaim_date_format_short(&tm)); } if ((info->age > 0) && (info->age < 255)) { char age[5]; snprintf(age, sizeof(age), "%hhd", info->age); oscar_string_append(gc->account, str, "\n<br>", _("Age"), age); } if (info->personalwebpage && info->personalwebpage[0] && (utf8 = oscar_utf8_try_convert(gc->account, info->personalwebpage))) { g_string_append_printf(str, "\n<br><b>%s:</b> <a href=\"%s\">%s</a>", _("Personal Web Page"), utf8, utf8); g_free(utf8); } if (info->info && info->info[0] && (utf8 = oscar_utf8_try_convert(gc->account, info->info))) { g_string_append_printf(str, "<hr><b>%s:</b><br>%s", _("Additional Information"), utf8); g_free(utf8); } g_string_append_printf(str, "<hr>"); if ((info->homeaddr && (info->homeaddr[0])) || (info->homecity && info->homecity[0]) || (info->homestate && info->homestate[0]) || (info->homezip && info->homezip[0])) { g_string_append_printf(str, "<b>%s:</b>", _("Home Address")); oscar_string_append(gc->account, str, "\n<br>", _("Address"), info->homeaddr); oscar_string_append(gc->account, str, "\n<br>", _("City"), info->homecity); oscar_string_append(gc->account, str, "\n<br>", _("State"), info->homestate); oscar_string_append(gc->account, str, "\n<br>", _("Zip Code"), info->homezip); g_string_append_printf(str, "\n<hr>"); } if ((info->workaddr && info->workaddr[0]) || (info->workcity && info->workcity[0]) || (info->workstate && info->workstate[0]) || (info->workzip && info->workzip[0])) { g_string_append_printf(str, "<b>%s:</b>", _("Work Address")); oscar_string_append(gc->account, str, "\n<br>", _("Address"), info->workaddr); oscar_string_append(gc->account, str, "\n<br>", _("City"), info->workcity); oscar_string_append(gc->account, str, "\n<br>", _("State"), info->workstate); oscar_string_append(gc->account, str, "\n<br>", _("Zip Code"), info->workzip); g_string_append_printf(str, "\n<hr>"); } if ((info->workcompany && info->workcompany[0]) || (info->workdivision && info->workdivision[0]) || (info->workposition && info->workposition[0]) || (info->workwebpage && info->workwebpage[0])) { g_string_append_printf(str, "<b>%s:</b>", _("Work Information")); oscar_string_append(gc->account, str, "\n<br>", _("Company"), info->workcompany); oscar_string_append(gc->account, str, "\n<br>", _("Division"), info->workdivision); oscar_string_append(gc->account, str, "\n<br>", _("Position"), info->workposition); if (info->workwebpage && info->workwebpage[0] && (utf8 = oscar_utf8_try_convert(gc->account, info->workwebpage))) { g_string_append_printf(str, "\n<br><b>%s:</b> <a href=\"%s\">%s</a>", _("Web Page"), utf8, utf8); g_free(utf8); } g_string_append_printf(str, "\n<hr>"); } if (buddy != NULL) alias = gaim_buddy_get_alias(buddy); else alias = who; gaim_notify_userinfo(gc, who, str->str, NULL, NULL); g_string_free(str, TRUE); return 1; } static int gaim_icqalias(OscarData *od, FlapConnection *conn, FlapFrame *fr, ...) { GaimConnection *gc = od->gc; GaimAccount *account = gaim_connection_get_account(gc); gchar who[16], *utf8; GaimBuddy *b; va_list ap; struct aim_icq_info *info; va_start(ap, fr); info = va_arg(ap, struct aim_icq_info *); va_end(ap); if (info->uin && info->nick && info->nick[0] && (utf8 = oscar_utf8_try_convert(account, info->nick))) { g_snprintf(who, sizeof(who), "%u", info->uin); serv_got_alias(gc, who, utf8); if ((b = gaim_find_buddy(gc->account, who))) { gaim_blist_node_set_string((GaimBlistNode*)b, "servernick", utf8); } g_free(utf8); } return 1; } static int gaim_popup(OscarData *od, FlapConnection *conn, FlapFrame *fr, ...) { GaimConnection *gc = od->gc; gchar *text; va_list ap; char *msg, *url; guint16 wid, hei, delay; va_start(ap, fr); msg = va_arg(ap, char *); url = va_arg(ap, char *); wid = (guint16) va_arg(ap, int); hei = (guint16) va_arg(ap, int); delay = (guint16) va_arg(ap, int); va_end(ap); text = g_strdup_printf("%s<br><a href=\"%s\">%s</a>", msg, url, url); gaim_notify_formatted(gc, NULL, _("Pop-Up Message"), NULL, text, NULL, NULL); g_free(text); return 1; } static void oscar_searchresults_add_buddy_cb(GaimConnection *gc, GList *row, void *user_data) { gaim_blist_request_add_buddy(gaim_connection_get_account(gc), g_list_nth_data(row, 0), NULL, NULL); } static int gaim_parse_searchreply(OscarData *od, FlapConnection *conn, FlapFrame *fr, ...) { GaimConnection *gc = od->gc; GaimNotifySearchResults *results; GaimNotifySearchColumn *column; gchar *secondary; int i, num; va_list ap; char *email, *SNs; va_start(ap, fr); email = va_arg(ap, char *); num = va_arg(ap, int); SNs = va_arg(ap, char *); va_end(ap); results = gaim_notify_searchresults_new(); if (results == NULL) { gaim_debug_error("oscar", "gaim_parse_searchreply: " "Unable to display the search results.\n"); gaim_notify_error(gc, NULL, _("Unable to display the search results."), NULL); return 1; } secondary = g_strdup_printf( ngettext("The following screen name is associated with %s", "The following screen names are associated with %s", num), email); column = gaim_notify_searchresults_column_new(_("Screen name")); gaim_notify_searchresults_column_add(results, column); for (i = 0; i < num; i++) { GList *row = NULL; row = g_list_append(row, g_strdup(&SNs[i * (MAXSNLEN + 1)])); gaim_notify_searchresults_row_add(results, row); } gaim_notify_searchresults_button_add(results, GAIM_NOTIFY_BUTTON_ADD, oscar_searchresults_add_buddy_cb); gaim_notify_searchresults(gc, NULL, NULL, secondary, results, NULL, NULL); g_free(secondary); return 1; } static int gaim_parse_searcherror(OscarData *od, FlapConnection *conn, FlapFrame *fr, ...) { va_list ap; char *email; char *buf; va_start(ap, fr); email = va_arg(ap, char *); va_end(ap); buf = g_strdup_printf(_("No results found for e-mail address %s"), email); gaim_notify_error(od->gc, NULL, buf, NULL); g_free(buf); return 1; } static int gaim_account_confirm(OscarData *od, FlapConnection *conn, FlapFrame *fr, ...) { GaimConnection *gc = od->gc; guint16 status; va_list ap; char msg[256]; va_start(ap, fr); status = (guint16) va_arg(ap, unsigned int); /* status code of confirmation request */ va_end(ap); gaim_debug_info("oscar", "account confirmation returned status 0x%04x (%s)\n", status, status ? "unknown" : "e-mail sent"); if (!status) { g_snprintf(msg, sizeof(msg), _("You should receive an e-mail asking to confirm %s."), gaim_account_get_username(gaim_connection_get_account(gc))); gaim_notify_info(gc, NULL, _("Account Confirmation Requested"), msg); } return 1; } static int gaim_info_change(OscarData *od, FlapConnection *conn, FlapFrame *fr, ...) { GaimConnection *gc = od->gc; va_list ap; guint16 perms, err; char *url, *sn, *email; int change; va_start(ap, fr); change = va_arg(ap, int); perms = (guint16) va_arg(ap, unsigned int); err = (guint16) va_arg(ap, unsigned int); url = va_arg(ap, char *); sn = va_arg(ap, char *); email = va_arg(ap, char *); va_end(ap); gaim_debug_misc("oscar", "account info: because of %s, perms=0x%04x, err=0x%04x, url=%s, sn=%s, email=%s\n", change ? "change" : "request", perms, err, (url != NULL) ? url : "(null)", (sn != NULL) ? sn : "(null)", (email != NULL) ? email : "(null)"); if ((err > 0) && (url != NULL)) { char *dialog_msg; char *dialog_top = g_strdup_printf(_("Error Changing Account Info")); switch (err) { case 0x0001: { dialog_msg = g_strdup_printf(_("Error 0x%04x: Unable to format screen name because the requested screen name differs from the original."), err); } break; case 0x0006: { dialog_msg = g_strdup_printf(_("Error 0x%04x: Unable to format screen name because it is invalid."), err); } break; case 0x000b: { dialog_msg = g_strdup_printf(_("Error 0x%04x: Unable to format screen name because the requested screen name is too long."), err); } break; case 0x001d: { dialog_msg = g_strdup_printf(_("Error 0x%04x: Unable to change e-mail address because there is already a request pending for this screen name."), err); } break; case 0x0021: { dialog_msg = g_strdup_printf(_("Error 0x%04x: Unable to change e-mail address because the given address has too many screen names associated with it."), err); } break; case 0x0023: { dialog_msg = g_strdup_printf(_("Error 0x%04x: Unable to change e-mail address because the given address is invalid."), err); } break; default: { dialog_msg = g_strdup_printf(_("Error 0x%04x: Unknown error."), err); } break; } gaim_notify_error(gc, NULL, dialog_top, dialog_msg); g_free(dialog_top); g_free(dialog_msg); return 1; } if (sn != NULL) { char *dialog_msg = g_strdup_printf(_("Your screen name is currently formatted as follows:\n%s"), sn); gaim_notify_info(gc, NULL, _("Account Info"), dialog_msg); g_free(dialog_msg); } if (email != NULL) { char *dialog_msg = g_strdup_printf(_("The e-mail address for %s is %s"), gaim_account_get_username(gaim_connection_get_account(gc)), email); gaim_notify_info(gc, NULL, _("Account Info"), dialog_msg); g_free(dialog_msg); } return 1; } static void oscar_keepalive(GaimConnection *gc) { OscarData *od; FlapConnection *conn; od = (OscarData *)gc->proto_data; conn = flap_connection_getbytype(od, SNAC_FAMILY_LOCATE); if (conn != NULL) flap_connection_send_keepalive(od, conn); } static int oscar_send_typing(GaimConnection *gc, const char *name, int typing) { OscarData *od; PeerConnection *conn; od = (OscarData *)gc->proto_data; conn = peer_connection_find_by_type(od, name, OSCAR_CAPABILITY_DIRECTIM); if ((conn != NULL) && (conn->ready)) { peer_odc_send_typing(conn, typing); } else { /* Don't send if this turkey is in our deny list */ GSList *list; for (list=gc->account->deny; (list && aim_sncmp(name, list->data)); list=list->next); if (!list) { struct buddyinfo *bi = g_hash_table_lookup(od->buddyinfo, gaim_normalize(gc->account, name)); if (bi && bi->typingnot) { if (typing == GAIM_TYPING) aim_im_sendmtn(od, 0x0001, name, 0x0002); else if (typing == GAIM_TYPED) aim_im_sendmtn(od, 0x0001, name, 0x0001); else aim_im_sendmtn(od, 0x0001, name, 0x0000); } } } return 0; } /* TODO: Move this into odc.c! */ static void gaim_odc_send_im(PeerConnection *conn, const char *message, GaimMessageFlags imflags) { GString *msg; GString *data; gchar *tmp; int tmplen; guint16 charset, charsubset; GData *attribs; const char *start, *end, *last; int oscar_id = 0; msg = g_string_new("<HTML><BODY>"); data = g_string_new("<BINARY>"); last = message; /* for each valid IMG tag... */ while (last && *last && gaim_markup_find_tag("img", last, &start, &end, &attribs)) { GaimStoredImage *image = NULL; const char *id; if (start - last) { g_string_append_len(msg, last, start - last); } id = g_datalist_get_data(&attribs, "id"); /* ... if it refers to a valid gaim image ... */ if (id && (image = gaim_imgstore_get(atoi(id)))) { /* ... append the message from start to the tag ... */ unsigned long size = gaim_imgstore_get_size(image); const char *filename = gaim_imgstore_get_filename(image); gpointer imgdata = gaim_imgstore_get_data(image); oscar_id++; /* ... insert a new img tag with the oscar id ... */ if (filename) g_string_append_printf(msg, "<IMG SRC=\"%s\" ID=\"%d\" DATASIZE=\"%lu\">", filename, oscar_id, size); else g_string_append_printf(msg, "<IMG ID=\"%d\" DATASIZE=\"%lu\">", oscar_id, size); /* ... and append the data to the binary section ... */ g_string_append_printf(data, "<DATA ID=\"%d\" SIZE=\"%lu\">", oscar_id, size); g_string_append_len(data, imgdata, size); g_string_append(data, "</DATA>"); } /* If the tag is invalid, skip it, thus no else here */ g_datalist_clear(&attribs); /* continue from the end of the tag */ last = end + 1; } /* append any remaining message data */ if (last && *last) g_string_append(msg, last); g_string_append(msg, "</BODY></HTML>"); /* Convert the message to a good encoding */ gaim_plugin_oscar_convert_to_best_encoding(conn->od->gc, conn->sn, msg->str, &tmp, &tmplen, &charset, &charsubset); g_string_free(msg, TRUE); msg = g_string_new_len(tmp, tmplen); /* Append any binary data that we may have */ if (oscar_id) { msg = g_string_append_len(msg, data->str, data->len); msg = g_string_append(msg, "</BINARY>"); } g_string_free(data, TRUE); peer_odc_send_im(conn, msg->str, msg->len, charset, imflags & GAIM_MESSAGE_AUTO_RESP); g_string_free(msg, TRUE); } static int oscar_send_im(GaimConnection *gc, const char *name, const char *message, GaimMessageFlags imflags) { OscarData *od; GaimAccount *account; PeerConnection *conn; int ret; char *iconfile; char *tmp1, *tmp2; od = (OscarData *)gc->proto_data; account = gaim_connection_get_account(gc); ret = 0; iconfile = gaim_buddy_icons_get_full_path(gaim_account_get_buddy_icon(account)); if (imflags & GAIM_MESSAGE_AUTO_RESP) tmp1 = gaim_str_sub_away_formatters(message, name); else tmp1 = g_strdup(message); conn = peer_connection_find_by_type(od, name, OSCAR_CAPABILITY_DIRECTIM); if ((conn != NULL) && (conn->ready)) { /* If we're directly connected, send a direct IM */ gaim_odc_send_im(conn, tmp1, imflags); } else { struct buddyinfo *bi; struct aim_sendimext_args args; struct stat st; gsize len; GaimConversation *conv; conv = gaim_find_conversation_with_account(GAIM_CONV_TYPE_IM, name, account); if (strstr(tmp1, "<IMG ")) gaim_conversation_write(conv, "", _("Your IM Image was not sent. " "You must be Direct Connected to send IM Images."), GAIM_MESSAGE_ERROR, time(NULL)); bi = g_hash_table_lookup(od->buddyinfo, gaim_normalize(account, name)); if (!bi) { bi = g_new0(struct buddyinfo, 1); g_hash_table_insert(od->buddyinfo, g_strdup(gaim_normalize(account, name)), bi); } args.flags = AIM_IMFLAGS_ACK | AIM_IMFLAGS_CUSTOMFEATURES; if (od->icq) { /* We have to present different "features" (whose meaning is unclear and are merely a result of protocol inspection) to offline ICQ buddies. Otherwise, the official ICQ client doesn't treat those messages as being "ANSI- encoded" (and instead, assumes them to be UTF-8). For more details, see SF issue 1179452. */ GaimBuddy *buddy = gaim_find_buddy(gc->account, name); if (buddy && GAIM_BUDDY_IS_ONLINE(buddy)) { args.features = features_icq; args.featureslen = sizeof(features_icq); } else { args.features = features_icq_offline; args.featureslen = sizeof(features_icq_offline); } args.flags |= AIM_IMFLAGS_OFFLINE; } else { args.features = features_aim; args.featureslen = sizeof(features_aim); if (imflags & GAIM_MESSAGE_AUTO_RESP) args.flags |= AIM_IMFLAGS_AWAY; } if (bi->ico_need) { gaim_debug_info("oscar", "Sending buddy icon request with message\n"); args.flags |= AIM_IMFLAGS_BUDDYREQ; bi->ico_need = FALSE; } if (iconfile && !g_stat(iconfile, &st)) { FILE *file = g_fopen(iconfile, "rb"); if (file) { guchar *buf = g_malloc(st.st_size); /* TODO: Use g_file_get_contents()? */ fread(buf, 1, st.st_size, file); fclose(file); args.iconlen = st.st_size; args.iconsum = aimutil_iconsum(buf, st.st_size); args.iconstamp = st.st_mtime; if ((args.iconlen != bi->ico_me_len) || (args.iconsum != bi->ico_me_csum) || (args.iconstamp != bi->ico_me_time)) { bi->ico_informed = FALSE; bi->ico_sent = FALSE; } if (!bi->ico_informed) { gaim_debug_info("oscar", "Claiming to have a buddy icon\n"); args.flags |= AIM_IMFLAGS_HASICON; bi->ico_me_len = args.iconlen; bi->ico_me_csum = args.iconsum; bi->ico_me_time = args.iconstamp; bi->ico_informed = TRUE; } g_free(buf); } } g_free(iconfile); args.destsn = name; /* * If we're IMing an SMS user or an ICQ user from an ICQ account, then strip HTML. */ if (aim_sn_is_sms(name)) { /* Messaging an SMS (mobile) user */ tmp2 = gaim_unescape_html(tmp1); } else if (aim_sn_is_icq(gaim_account_get_username(account))) { if (aim_sn_is_icq(name)) /* From ICQ to ICQ */ tmp2 = gaim_unescape_html(tmp1); else /* From ICQ to AIM */ tmp2 = g_strdup(tmp1); } else { /* From AIM to AIM and AIM to ICQ */ tmp2 = g_strdup(tmp1); } g_free(tmp1); tmp1 = tmp2; len = strlen(tmp1); gaim_plugin_oscar_convert_to_best_encoding(gc, name, tmp1, (char **)&args.msg, &args.msglen, &args.charset, &args.charsubset); gaim_debug_info("oscar", "Sending IM, charset=0x%04hx, charsubset=0x%04hx, length=%d\n", args.charset, args.charsubset, args.msglen); ret = aim_im_sendch1_ext(od, &args); g_free((char *)args.msg); } g_free(tmp1); if (ret >= 0) return 1; return ret; } static void oscar_get_info(GaimConnection *gc, const char *name) { OscarData *od = (OscarData *)gc->proto_data; if (od->icq && aim_sn_is_icq(name)) aim_icq_getallinfo(od, name); else aim_locate_getinfoshort(od, name, 0x00000003); } #if 0 static void oscar_set_dir(GaimConnection *gc, const char *first, const char *middle, const char *last, const char *maiden, const char *city, const char *state, const char *country, int web) { /* XXX - some of these things are wrong, but i'm lazy */ OscarData *od = (OscarData *)gc->proto_data; aim_locate_setdirinfo(od, first, middle, last, maiden, NULL, NULL, city, state, NULL, 0, web); } #endif static void oscar_set_idle(GaimConnection *gc, int time) { OscarData *od = (OscarData *)gc->proto_data; aim_srv_setidle(od, time); } static gchar *gaim_prpl_oscar_convert_to_infotext(const gchar *str, gsize *ret_len, char **encoding) { int charset = 0; char *encoded = NULL; charset = oscar_charset_check(str); if (charset == AIM_CHARSET_UNICODE) { encoded = g_convert(str, strlen(str), "UCS-2BE", "UTF-8", NULL, ret_len, NULL); *encoding = "unicode-2-0"; } else if (charset == AIM_CHARSET_CUSTOM) { encoded = g_convert(str, strlen(str), "ISO-8859-1", "UTF-8", NULL, ret_len, NULL); *encoding = "iso-8859-1"; } else { encoded = g_strdup(str); *ret_len = strlen(str); *encoding = "us-ascii"; } return encoded; } static void oscar_set_info(GaimConnection *gc, const char *rawinfo) { GaimAccount *account; GaimStatus *status; account = gaim_connection_get_account(gc); status = gaim_account_get_active_status(account); oscar_set_info_and_status(account, TRUE, rawinfo, FALSE, status); } static void oscar_set_extendedstatus(GaimConnection *gc) { OscarData *od; GaimAccount *account; GaimStatus *status; const gchar *status_id; guint32 data = 0x00000000; od = gc->proto_data; account = gaim_connection_get_account(gc); status = gaim_account_get_active_status(account); status_id = gaim_status_get_id(status); data |= AIM_ICQ_STATE_HIDEIP; if (gaim_account_get_bool(account, "web_aware", OSCAR_DEFAULT_WEB_AWARE)) data |= AIM_ICQ_STATE_WEBAWARE; if (!strcmp(status_id, OSCAR_STATUS_ID_AVAILABLE) || !strcmp(status_id, OSCAR_STATUS_ID_AVAILABLE)) data |= AIM_ICQ_STATE_NORMAL; else if (!strcmp(status_id, OSCAR_STATUS_ID_AWAY)) data |= AIM_ICQ_STATE_AWAY; else if (!strcmp(status_id, OSCAR_STATUS_ID_DND)) data |= AIM_ICQ_STATE_AWAY | AIM_ICQ_STATE_DND | AIM_ICQ_STATE_BUSY; else if (!strcmp(status_id, OSCAR_STATUS_ID_NA)) data |= AIM_ICQ_STATE_OUT | AIM_ICQ_STATE_AWAY; else if (!strcmp(status_id, OSCAR_STATUS_ID_OCCUPIED)) data |= AIM_ICQ_STATE_AWAY | AIM_ICQ_STATE_BUSY; else if (!strcmp(status_id, OSCAR_STATUS_ID_FREE4CHAT)) data |= AIM_ICQ_STATE_CHAT; else if (!strcmp(status_id, OSCAR_STATUS_ID_INVISIBLE)) data |= AIM_ICQ_STATE_INVISIBLE; else if (!strcmp(status_id, OSCAR_STATUS_ID_CUSTOM)) data |= AIM_ICQ_STATE_OUT | AIM_ICQ_STATE_AWAY; aim_setextstatus(od, data); } static void oscar_set_info_and_status(GaimAccount *account, gboolean setinfo, const char *rawinfo, gboolean setstatus, GaimStatus *status) { GaimConnection *gc = gaim_account_get_connection(account); OscarData *od = gc->proto_data; GaimPresence *presence; GaimStatusType *status_type; GaimStatusPrimitive primitive; gboolean invisible; char *htmlinfo; char *info_encoding = NULL; char *info = NULL; gsize infolen = 0; const char *htmlaway; char *away_encoding = NULL; char *away = NULL; gsize awaylen = 0; status_type = gaim_status_get_type(status); primitive = gaim_status_type_get_primitive(status_type); presence = gaim_account_get_presence(account); invisible = gaim_presence_is_status_primitive_active(presence, GAIM_STATUS_INVISIBLE); if (!setinfo) { /* Do nothing! */ } else if (od->rights.maxsiglen == 0) { gaim_notify_warning(gc, NULL, _("Unable to set AIM profile."), _("You have probably requested to set your " "profile before the login procedure completed. " "Your profile remains unset; try setting it " "again when you are fully connected.")); } else if (rawinfo != NULL) { htmlinfo = gaim_strdup_withhtml(rawinfo); info = gaim_prpl_oscar_convert_to_infotext(htmlinfo, &infolen, &info_encoding); g_free(htmlinfo); if (infolen > od->rights.maxsiglen) { gchar *errstr; errstr = g_strdup_printf(ngettext("The maximum profile length of %d byte " "has been exceeded. Gaim has truncated it for you.", "The maximum profile length of %d bytes " "has been exceeded. Gaim has truncated it for you.", od->rights.maxsiglen), od->rights.maxsiglen); gaim_notify_warning(gc, NULL, _("Profile too long."), errstr); g_free(errstr); } } if (!setstatus) { /* Do nothing! */ } else if (primitive == GAIM_STATUS_AVAILABLE) { const char *status_html; char *status_text; status_html = gaim_status_get_attr_string(status, "message"); if (status_html != NULL) { status_text = gaim_markup_strip_html(status_html); /* If the status_text is longer than 60 character then truncate it */ if (strlen(status_text) > 60) { char *tmp = g_utf8_find_prev_char(status_text, &status_text[58]); strcpy(tmp, "..."); } else { /* User did not specify an available message */ status_text = NULL; } aim_srv_setstatusmsg(od, status_text); g_free(status_text); } /* This is needed for us to un-set any previous away message. */ away = g_strdup(""); } else if (primitive == GAIM_STATUS_AWAY) { htmlaway = gaim_status_get_attr_string(status, "message"); if ((htmlaway == NULL) || (*htmlaway == '\0')) htmlaway = _("Away"); away = gaim_prpl_oscar_convert_to_infotext(htmlaway, &awaylen, &away_encoding); if (awaylen > od->rights.maxawaymsglen) { gchar *errstr; errstr = g_strdup_printf(ngettext("The maximum away message length of %d byte " "has been exceeded. Gaim has truncated it for you.", "The maximum away message length of %d bytes " "has been exceeded. Gaim has truncated it for you.", od->rights.maxawaymsglen), od->rights.maxawaymsglen); gaim_notify_warning(gc, NULL, _("Away message too long."), errstr); g_free(errstr); } } if (setstatus) oscar_set_extendedstatus(gc); aim_locate_setprofile(od, info_encoding, info, MIN(infolen, od->rights.maxsiglen), away_encoding, away, MIN(awaylen, od->rights.maxawaymsglen)); g_free(info); g_free(away); } static void oscar_set_status_icq(GaimAccount *account, GaimStatus *status) { GaimConnection *gc = gaim_account_get_connection(account); OscarData *od = NULL; if (gc) od = (OscarData *)gc->proto_data; if (!od) return; if (gaim_status_type_get_primitive(gaim_status_get_type(status)) == GAIM_STATUS_INVISIBLE) account->perm_deny = GAIM_PRIVACY_ALLOW_USERS; else account->perm_deny = GAIM_PRIVACY_DENY_USERS; if ((od->ssi.received_data) && (aim_ssi_getpermdeny(od->ssi.local) != account->perm_deny)) aim_ssi_setpermdeny(od, account->perm_deny, 0xffffffff); oscar_set_extendedstatus(gc); } static void oscar_set_status(GaimAccount *account, GaimStatus *status) { gaim_debug_info("oscar", "Set status to %s\n", gaim_status_get_name(status)); if (!gaim_status_is_active(status)) return; if (!gaim_account_is_connected(account)) return; /* Set the AIM-style away message for both AIM and ICQ accounts */ oscar_set_info_and_status(account, FALSE, NULL, TRUE, status); /* Set the ICQ status for ICQ accounts only */ if (aim_sn_is_icq(gaim_account_get_username(account))) oscar_set_status_icq(account, status); } #ifdef CRAZY_WARN static void oscar_warn(GaimConnection *gc, const char *name, gboolean anonymous) { OscarData *od = (OscarData *)gc->proto_data; aim_im_warn(od, od->conn, name, anonymous ? AIM_WARN_ANON : 0); } #endif static void oscar_add_buddy(GaimConnection *gc, GaimBuddy *buddy, GaimGroup *group) { OscarData *od = (OscarData *)gc->proto_data; if (!aim_snvalid(buddy->name)) { gchar *buf; buf = g_strdup_printf(_("Could not add the buddy %s because the screen name is invalid. Screen names must either start with a letter and contain only letters, numbers and spaces, or contain only numbers."), buddy->name); if (!gaim_conv_present_error(buddy->name, gaim_connection_get_account(gc), buf)) gaim_notify_error(gc, NULL, _("Unable To Add"), buf); g_free(buf); /* Remove from local list */ gaim_blist_remove_buddy(buddy); return; } if ((od->ssi.received_data) && !(aim_ssi_itemlist_finditem(od->ssi.local, group->name, buddy->name, AIM_SSI_TYPE_BUDDY))) { gaim_debug_info("oscar", "ssi: adding buddy %s to group %s\n", buddy->name, group->name); aim_ssi_addbuddy(od, buddy->name, group->name, gaim_buddy_get_alias_only(buddy), NULL, NULL, 0); } /* XXX - Should this be done from AIM accounts, as well? */ if (od->icq) aim_icq_getalias(od, buddy->name); } static void oscar_remove_buddy(GaimConnection *gc, GaimBuddy *buddy, GaimGroup *group) { OscarData *od = (OscarData *)gc->proto_data; if (od->ssi.received_data) { gaim_debug_info("oscar", "ssi: deleting buddy %s from group %s\n", buddy->name, group->name); aim_ssi_delbuddy(od, buddy->name, group->name); } } static void oscar_move_buddy(GaimConnection *gc, const char *name, const char *old_group, const char *new_group) { OscarData *od = (OscarData *)gc->proto_data; if (od->ssi.received_data && strcmp(old_group, new_group)) { gaim_debug_info("oscar", "ssi: moving buddy %s from group %s to group %s\n", name, old_group, new_group); aim_ssi_movebuddy(od, old_group, new_group, name); } } static void oscar_alias_buddy(GaimConnection *gc, const char *name, const char *alias) { OscarData *od = (OscarData *)gc->proto_data; if (od->ssi.received_data) { char *gname = aim_ssi_itemlist_findparentname(od->ssi.local, name); if (gname) { gaim_debug_info("oscar", "ssi: changing the alias for buddy %s to %s\n", name, alias ? alias : "(none)"); aim_ssi_aliasbuddy(od, gname, name, alias); } } } /* * FYI, the OSCAR SSI code removes empty groups automatically. */ static void oscar_rename_group(GaimConnection *gc, const char *old_name, GaimGroup *group, GList *moved_buddies) { OscarData *od = (OscarData *)gc->proto_data; if (od->ssi.received_data) { if (aim_ssi_itemlist_finditem(od->ssi.local, group->name, NULL, AIM_SSI_TYPE_GROUP)) { GList *cur, *groups = NULL; GaimAccount *account = gaim_connection_get_account(gc); /* Make a list of what the groups each buddy is in */ for (cur = moved_buddies; cur != NULL; cur = cur->next) { GaimBlistNode *node = cur->data; /* node is GaimBuddy, parent is a GaimContact. * We must go two levels up to get the Group */ groups = g_list_append(groups, node->parent->parent); } gaim_account_remove_buddies(account, moved_buddies, groups); gaim_account_add_buddies(account, moved_buddies); g_list_free(groups); gaim_debug_info("oscar", "ssi: moved all buddies from group %s to %s\n", old_name, group->name); } else { aim_ssi_rename_group(od, old_name, group->name); gaim_debug_info("oscar", "ssi: renamed group %s to %s\n", old_name, group->name); } } } static gboolean gaim_ssi_rerequestdata(gpointer data) { OscarData *od = data; aim_ssi_reqdata(od); return TRUE; } static int gaim_ssi_parseerr(OscarData *od, FlapConnection *conn, FlapFrame *fr, ...) { GaimConnection *gc = od->gc; va_list ap; guint16 reason; va_start(ap, fr); reason = (guint16)va_arg(ap, unsigned int); va_end(ap); gaim_debug_error("oscar", "ssi: SNAC error %hu\n", reason); if (reason == 0x0005) { gaim_notify_error(gc, NULL, _("Unable To Retrieve Buddy List"), _("Gaim was temporarily unable to retrieve your buddy list from the AIM servers. Your buddy list is not lost, and will probably become available in a few hours.")); if (od->getblisttimer > 0) gaim_timeout_remove(od->getblisttimer); od->getblisttimer = gaim_timeout_add(30000, gaim_ssi_rerequestdata, od); } oscar_set_extendedstatus(gc); /* Activate SSI */ /* Sending the enable causes other people to be able to see you, and you to see them */ /* Make sure your privacy setting/invisibility is set how you want it before this! */ gaim_debug_info("oscar", "ssi: activating server-stored buddy list\n"); aim_ssi_enable(od); return 1; } static int gaim_ssi_parserights(OscarData *od, FlapConnection *conn, FlapFrame *fr, ...) { int i; va_list ap; int numtypes; guint16 *maxitems; va_start(ap, fr); numtypes = va_arg(ap, int); maxitems = va_arg(ap, guint16 *); va_end(ap); gaim_debug_misc("oscar", "ssi rights:"); for (i=0; i<numtypes; i++) gaim_debug_misc(NULL, " max type 0x%04x=%hd,", i, maxitems[i]); gaim_debug_misc(NULL, "\n"); if (numtypes >= 0) od->rights.maxbuddies = maxitems[0]; if (numtypes >= 1) od->rights.maxgroups = maxitems[1]; if (numtypes >= 2) od->rights.maxpermits = maxitems[2]; if (numtypes >= 3) od->rights.maxdenies = maxitems[3]; return 1; } static int gaim_ssi_parselist(OscarData *od, FlapConnection *conn, FlapFrame *fr, ...) { GaimConnection *gc; GaimAccount *account; GaimGroup *g; GaimBuddy *b; struct aim_ssi_item *curitem; guint32 tmp; va_list ap; guint16 fmtver, numitems; guint32 timestamp; gc = od->gc; od = gc->proto_data; account = gaim_connection_get_account(gc); va_start(ap, fr); fmtver = (guint16)va_arg(ap, int); numitems = (guint16)va_arg(ap, int); timestamp = va_arg(ap, guint32); va_end(ap); /* Don't attempt to re-request our buddy list later */ if (od->getblisttimer != 0) gaim_timeout_remove(od->getblisttimer); od->getblisttimer = 0; gaim_debug_info("oscar", "ssi: syncing local list and server list\n"); if ((timestamp == 0) || (numitems == 0)) { gaim_debug_info("oscar", "Got AIM SSI with a 0 timestamp or 0 numitems--not syncing. This probably means your buddy list is empty.", NULL); return 1; } /* Clean the buddy list */ aim_ssi_cleanlist(od); { /* If not in server list then prune from local list */ GaimBlistNode *gnode, *cnode, *bnode; GaimBuddyList *blist; GSList *cur, *next; /* Buddies */ cur = NULL; if ((blist = gaim_get_blist()) != NULL) { for (gnode = blist->root; gnode; gnode = gnode->next) { if(!GAIM_BLIST_NODE_IS_GROUP(gnode)) continue; g = (GaimGroup *)gnode; for (cnode = gnode->child; cnode; cnode = cnode->next) { if(!GAIM_BLIST_NODE_IS_CONTACT(cnode)) continue; for (bnode = cnode->child; bnode; bnode = bnode->next) { if(!GAIM_BLIST_NODE_IS_BUDDY(bnode)) continue; b = (GaimBuddy *)bnode; if (b->account == gc->account) { if (aim_ssi_itemlist_exists(od->ssi.local, b->name)) { /* If the buddy is an ICQ user then load his nickname */ const char *servernick = gaim_blist_node_get_string((GaimBlistNode*)b, "servernick"); char *alias; if (servernick) serv_got_alias(gc, b->name, servernick); /* Store local alias on server */ alias = aim_ssi_getalias(od->ssi.local, g->name, b->name); if (!alias && b->alias && strlen(b->alias)) aim_ssi_aliasbuddy(od, g->name, b->name, b->alias); g_free(alias); } else { gaim_debug_info("oscar", "ssi: removing buddy %s from local list\n", b->name); /* We can't actually remove now because it will screw up our looping */ cur = g_slist_prepend(cur, b); } } } } } } while (cur != NULL) { b = cur->data; cur = g_slist_remove(cur, b); gaim_blist_remove_buddy(b); } /* Permit list */ if (gc->account->permit) { next = gc->account->permit; while (next != NULL) { cur = next; next = next->next; if (!aim_ssi_itemlist_finditem(od->ssi.local, NULL, cur->data, AIM_SSI_TYPE_PERMIT)) { gaim_debug_info("oscar", "ssi: removing permit %s from local list\n", (const char *)cur->data); gaim_privacy_permit_remove(account, cur->data, TRUE); } } } /* Deny list */ if (gc->account->deny) { next = gc->account->deny; while (next != NULL) { cur = next; next = next->next; if (!aim_ssi_itemlist_finditem(od->ssi.local, NULL, cur->data, AIM_SSI_TYPE_DENY)) { gaim_debug_info("oscar", "ssi: removing deny %s from local list\n", (const char *)cur->data); gaim_privacy_deny_remove(account, cur->data, TRUE); } } } /* Presence settings (idle time visibility) */ if ((tmp = aim_ssi_getpresence(od->ssi.local)) != 0xFFFFFFFF) if (!(tmp & 0x400)) aim_ssi_setpresence(od, tmp | 0x400); } /* end pruning buddies from local list */ /* Add from server list to local list */ for (curitem=od->ssi.local; curitem; curitem=curitem->next) { if ((curitem->name == NULL) || (g_utf8_validate(curitem->name, -1, NULL))) switch (curitem->type) { case 0x0000: { /* Buddy */ if (curitem->name) { char *gname = aim_ssi_itemlist_findparentname(od->ssi.local, curitem->name); char *gname_utf8 = gname ? oscar_utf8_try_convert(gc->account, gname) : NULL; char *alias = aim_ssi_getalias(od->ssi.local, gname, curitem->name); char *alias_utf8 = alias ? oscar_utf8_try_convert(gc->account, alias) : NULL; b = gaim_find_buddy(gc->account, curitem->name); /* Should gname be freed here? -- elb */ /* Not with the current code, but that might be cleaner -- med */ g_free(alias); if (b) { /* Get server stored alias */ if (alias_utf8) { g_free(b->alias); b->alias = g_strdup(alias_utf8); } } else { b = gaim_buddy_new(gc->account, curitem->name, alias_utf8); if (!(g = gaim_find_group(gname_utf8 ? gname_utf8 : _("Orphans")))) { g = gaim_group_new(gname_utf8 ? gname_utf8 : _("Orphans")); gaim_blist_add_group(g, NULL); } gaim_debug_info("oscar", "ssi: adding buddy %s to group %s to local list\n", curitem->name, gname_utf8 ? gname_utf8 : _("Orphans")); gaim_blist_add_buddy(b, NULL, g, NULL); } if (!aim_sncmp(curitem->name, account->username)) { char *comment = aim_ssi_getcomment(od->ssi.local, gname, curitem->name); gaim_check_comment(od, comment); g_free(comment); } g_free(gname_utf8); g_free(alias_utf8); } } break; case 0x0001: { /* Group */ /* Shouldn't add empty groups */ } break; case 0x0002: { /* Permit buddy */ if (curitem->name) { /* if (!find_permdeny_by_name(gc->permit, curitem->name)) { AAA */ GSList *list; for (list=account->permit; (list && aim_sncmp(curitem->name, list->data)); list=list->next); if (!list) { gaim_debug_info("oscar", "ssi: adding permit buddy %s to local list\n", curitem->name); gaim_privacy_permit_add(account, curitem->name, TRUE); } } } break; case 0x0003: { /* Deny buddy */ if (curitem->name) { GSList *list; for (list=account->deny; (list && aim_sncmp(curitem->name, list->data)); list=list->next); if (!list) { gaim_debug_info("oscar", "ssi: adding deny buddy %s to local list\n", curitem->name); gaim_privacy_deny_add(account, curitem->name, TRUE); } } } break; case 0x0004: { /* Permit/deny setting */ if (curitem->data) { guint8 permdeny; if ((permdeny = aim_ssi_getpermdeny(od->ssi.local)) && (permdeny != account->perm_deny)) { gaim_debug_info("oscar", "ssi: changing permdeny from %d to %hhu\n", account->perm_deny, permdeny); account->perm_deny = permdeny; if (od->icq && account->perm_deny == GAIM_PRIVACY_ALLOW_USERS) { gaim_presence_set_status_active(account->presence, OSCAR_STATUS_ID_INVISIBLE, TRUE); } } } } break; case 0x0005: { /* Presence setting */ /* We don't want to change Gaim's setting because it applies to all accounts */ } break; } /* End of switch on curitem->type */ } /* End of for loop */ oscar_set_extendedstatus(gc); /* Activate SSI */ /* Sending the enable causes other people to be able to see you, and you to see them */ /* Make sure your privacy setting/invisibility is set how you want it before this! */ gaim_debug_info("oscar", "ssi: activating server-stored buddy list\n"); aim_ssi_enable(od); return 1; } static int gaim_ssi_parseack(OscarData *od, FlapConnection *conn, FlapFrame *fr, ...) { GaimConnection *gc = od->gc; va_list ap; struct aim_ssi_tmp *retval; va_start(ap, fr); retval = va_arg(ap, struct aim_ssi_tmp *); va_end(ap); while (retval) { gaim_debug_misc("oscar", "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"); if (retval->ack != 0xffff) switch (retval->ack) { case 0x0000: { /* added successfully */ } break; case 0x000c: { /* you are over the limit, the cheat is to the limit, come on fhqwhgads */ gchar *buf; buf = g_strdup_printf(_("Could not 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)"))); if ((retval->name != NULL) && !gaim_conv_present_error(retval->name, gaim_connection_get_account(gc), buf)) gaim_notify_error(gc, NULL, _("Unable To Add"), buf); g_free(buf); } case 0x000e: { /* buddy requires authorization */ if ((retval->action == SNAC_SUBTYPE_FEEDBAG_ADD) && (retval->name)) gaim_auth_sendrequest(gc, retval->name); } break; default: { /* La la la */ gchar *buf; gaim_debug_error("oscar", "ssi: Action 0x%04hx was unsuccessful with error 0x%04hx\n", retval->action, retval->ack); buf = g_strdup_printf(_("Could not add the buddy %s for an unknown reason. The most common reason for this is that you have the maximum number of allowed buddies in your buddy list."), (retval->name ? retval->name : _("(no name)"))); if ((retval->name != NULL) && !gaim_conv_present_error(retval->name, gaim_connection_get_account(gc), buf)) gaim_notify_error(gc, NULL, _("Unable To Add"), buf); g_free(buf); } break; } retval = retval->next; } return 1; } static int gaim_ssi_parseadd(OscarData *od, FlapConnection *conn, FlapFrame *fr, ...) { GaimConnection *gc = od->gc; char *gname, *gname_utf8, *alias, *alias_utf8; GaimBuddy *b; GaimGroup *g; va_list ap; guint16 type; const char *name; va_start(ap, fr); type = (guint16)va_arg(ap, int); name = va_arg(ap, char *); va_end(ap); if ((type != 0x0000) || (name == NULL)) return 1; gname = aim_ssi_itemlist_findparentname(od->ssi.local, name); gname_utf8 = gname ? oscar_utf8_try_convert(gc->account, gname) : NULL; alias = aim_ssi_getalias(od->ssi.local, gname, name); alias_utf8 = alias ? oscar_utf8_try_convert(gc->account, alias) : NULL; b = gaim_find_buddy(gc->account, name); g_free(alias); if (b) { /* Get server stored alias */ if (alias_utf8) { g_free(b->alias); b->alias = g_strdup(alias_utf8); } } else { b = gaim_buddy_new(gc->account, name, alias_utf8); if (!(g = gaim_find_group(gname_utf8 ? gname_utf8 : _("Orphans")))) { g = gaim_group_new(gname_utf8 ? gname_utf8 : _("Orphans")); gaim_blist_add_group(g, NULL); } gaim_debug_info("oscar", "ssi: adding buddy %s to group %s to local list\n", name, gname_utf8 ? gname_utf8 : _("Orphans")); gaim_blist_add_buddy(b, NULL, g, NULL); } g_free(gname_utf8); g_free(alias_utf8); return 1; } static int gaim_ssi_authgiven(OscarData *od, FlapConnection *conn, FlapFrame *fr, ...) { GaimConnection *gc = od->gc; va_list ap; char *sn, *msg; gchar *dialog_msg, *nombre; struct name_data *data; GaimBuddy *buddy; va_start(ap, fr); sn = va_arg(ap, char *); msg = va_arg(ap, char *); va_end(ap); gaim_debug_info("oscar", "ssi: %s has given you permission to add him to your buddy list\n", sn); buddy = gaim_find_buddy(gc->account, sn); if (buddy && (gaim_buddy_get_alias_only(buddy))) nombre = g_strdup_printf("%s (%s)", sn, gaim_buddy_get_alias_only(buddy)); else nombre = g_strdup(sn); dialog_msg = g_strdup_printf(_("The user %s has given you permission to add you to their buddy list. Do you want to add them?"), nombre); data = g_new(struct name_data, 1); data->gc = gc; data->name = g_strdup(sn); data->nick = NULL; gaim_request_yes_no(gc, NULL, _("Authorization Given"), dialog_msg, GAIM_DEFAULT_ACTION_NONE, data, G_CALLBACK(gaim_icq_buddyadd), G_CALLBACK(oscar_free_name_data)); g_free(dialog_msg); g_free(nombre); return 1; } static int gaim_ssi_authrequest(OscarData *od, FlapConnection *conn, FlapFrame *fr, ...) { GaimConnection *gc = od->gc; va_list ap; char *sn; char *msg; GaimAccount *account = gaim_connection_get_account(gc); gchar *nombre; gchar *reason = NULL; gchar *dialog_msg; struct name_data *data; GaimBuddy *buddy; va_start(ap, fr); sn = va_arg(ap, char *); msg = va_arg(ap, char *); va_end(ap); gaim_debug_info("oscar", "ssi: received authorization request from %s\n", sn); buddy = gaim_find_buddy(account, sn); if (buddy && (gaim_buddy_get_alias_only(buddy))) nombre = g_strdup_printf("%s (%s)", sn, gaim_buddy_get_alias_only(buddy)); else nombre = g_strdup(sn); if (msg != NULL) reason = gaim_plugin_oscar_decode_im_part(account, sn, AIM_CHARSET_CUSTOM, 0x0000, msg, strlen(msg)); if (reason == NULL) reason = g_strdup(_("No reason given.")); dialog_msg = g_strdup_printf( _("The user %s wants to add %s to their buddy list for the following reason:\n%s"), nombre, gaim_account_get_username(account), reason); g_free(reason); data = g_new(struct name_data, 1); data->gc = gc; data->name = g_strdup(sn); data->nick = NULL; gaim_request_action(gc, NULL, _("Authorization Request"), dialog_msg, GAIM_DEFAULT_ACTION_NONE, data, 2, _("_Authorize"), G_CALLBACK(gaim_auth_grant), _("_Deny"), G_CALLBACK(gaim_auth_dontgrant_msgprompt)); g_free(dialog_msg); g_free(nombre); return 1; } static int gaim_ssi_authreply(OscarData *od, FlapConnection *conn, FlapFrame *fr, ...) { GaimConnection *gc = od->gc; va_list ap; char *sn, *msg; gchar *dialog_msg, *nombre; guint8 reply; GaimBuddy *buddy; va_start(ap, fr); sn = va_arg(ap, char *); reply = (guint8)va_arg(ap, int); msg = va_arg(ap, char *); va_end(ap); gaim_debug_info("oscar", "ssi: received authorization reply from %s. Reply is 0x%04hhx\n", sn, reply); buddy = gaim_find_buddy(gc->account, sn); if (buddy && (gaim_buddy_get_alias_only(buddy))) nombre = g_strdup_printf("%s (%s)", sn, gaim_buddy_get_alias_only(buddy)); else nombre = g_strdup(sn); if (reply) { /* Granted */ dialog_msg = g_strdup_printf(_("The user %s has granted your request to add them to your buddy list."), nombre); gaim_notify_info(gc, NULL, _("Authorization Granted"), dialog_msg); } else { /* Denied */ 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.")); gaim_notify_info(gc, NULL, _("Authorization Denied"), dialog_msg); } g_free(dialog_msg); g_free(nombre); return 1; } static int gaim_ssi_gotadded(OscarData *od, FlapConnection *conn, FlapFrame *fr, ...) { GaimConnection *gc = od->gc; va_list ap; char *sn; GaimBuddy *buddy; va_start(ap, fr); sn = va_arg(ap, char *); va_end(ap); buddy = gaim_find_buddy(gc->account, sn); gaim_debug_info("oscar", "ssi: %s added you to their buddy list\n", sn); gaim_account_notify_added(gc->account, NULL, sn, (buddy ? gaim_buddy_get_alias_only(buddy) : NULL), NULL); return 1; } static GList *oscar_chat_info(GaimConnection *gc) { GList *m = NULL; struct proto_chat_entry *pce; pce = g_new0(struct proto_chat_entry, 1); pce->label = _("_Room:"); pce->identifier = "room"; pce->required = TRUE; m = g_list_append(m, pce); pce = g_new0(struct proto_chat_entry, 1); pce->label = _("_Exchange:"); pce->identifier = "exchange"; pce->required = TRUE; pce->is_int = TRUE; pce->min = 4; pce->max = 20; m = g_list_append(m, pce); return m; } static GHashTable *oscar_chat_info_defaults(GaimConnection *gc, const char *chat_name) { GHashTable *defaults; defaults = g_hash_table_new_full(g_str_hash, g_str_equal, NULL, g_free); if (chat_name != NULL) g_hash_table_insert(defaults, "room", g_strdup(chat_name)); return defaults; } static char * oscar_get_chat_name(GHashTable *data) { return g_strdup(g_hash_table_lookup(data, "room")); } static void oscar_join_chat(GaimConnection *gc, GHashTable *data) { OscarData *od = (OscarData *)gc->proto_data; FlapConnection *conn; char *name, *exchange; name = g_hash_table_lookup(data, "room"); exchange = g_hash_table_lookup(data, "exchange"); if ((name == NULL) || (*name == '\0')) { gaim_notify_error(gc, NULL, _("Invalid chat name specified."), NULL); return; } gaim_debug_info("oscar", "Attempting to join chat room %s.\n", name); if ((conn = flap_connection_getbytype(od, SNAC_FAMILY_CHATNAV))) { gaim_debug_info("oscar", "chatnav exists, creating room\n"); aim_chatnav_createroom(od, conn, name, atoi(exchange)); } else { /* this gets tricky */ struct create_room *cr = g_new0(struct create_room, 1); gaim_debug_info("oscar", "chatnav does not exist, opening chatnav\n"); cr->exchange = atoi(exchange); cr->name = g_strdup(name); od->create_rooms = g_slist_append(od->create_rooms, cr); aim_reqservice(od, SNAC_FAMILY_CHATNAV); } } static void oscar_chat_invite(GaimConnection *gc, int id, const char *message, const char *name) { OscarData *od = (OscarData *)gc->proto_data; struct chat_connection *ccon = find_oscar_chat(gc, id); if (ccon == NULL) return; aim_im_sendch2_chatinvite(od, name, message ? message : "", ccon->exchange, ccon->name, 0x0); } static void oscar_chat_leave(GaimConnection *gc, int id) { GaimConversation *conv; struct chat_connection *cc; conv = gaim_find_chat(gc, id); g_return_if_fail(conv != NULL); gaim_debug_info("oscar", "Leaving chat room %s\n", conv->name); cc = find_oscar_chat(gc, gaim_conv_chat_get_id(GAIM_CONV_CHAT(conv))); oscar_chat_kill(gc, cc); } static int oscar_send_chat(GaimConnection *gc, int id, const char *message, GaimMessageFlags flags) { OscarData *od = (OscarData *)gc->proto_data; GaimConversation *conv = NULL; struct chat_connection *c = NULL; char *buf, *buf2; guint16 charset, charsubset; char *charsetstr = NULL; int len; if (!(conv = gaim_find_chat(gc, id))) return -EINVAL; if (!(c = find_oscar_chat_by_conv(gc, conv))) return -EINVAL; buf = gaim_strdup_withhtml(message); len = strlen(buf); if (strstr(buf, "<IMG ")) gaim_conversation_write(conv, "", _("Your IM Image was not sent. " "You cannot send IM Images in AIM chats."), GAIM_MESSAGE_ERROR, time(NULL)); gaim_plugin_oscar_convert_to_best_encoding(gc, NULL, buf, &buf2, &len, &charset, &charsubset); /* * Evan S. suggested that maxvis really does mean "number of * visible characters" and not "number of bytes" */ if ((len > c->maxlen) || (len > c->maxvis)) { g_free(buf2); return -E2BIG; } if (charset == AIM_CHARSET_ASCII) charsetstr = "us-ascii"; else if (charset == AIM_CHARSET_UNICODE) charsetstr = "unicode-2-0"; else if (charset == AIM_CHARSET_CUSTOM) charsetstr = "iso-8859-1"; aim_chat_send_im(od, c->conn, 0, buf2, len, charsetstr, "en"); g_free(buf2); return 0; } static const char *oscar_list_icon(GaimAccount *a, GaimBuddy *b) { if ((b == NULL) || (b->name == NULL) || aim_sn_is_sms(b->name)) { if (a != NULL && aim_sn_is_icq(gaim_account_get_username(a))) return "icq"; else return "aim"; } if (aim_sn_is_icq(b->name)) return "icq"; return "aim"; } static void oscar_list_emblems(GaimBuddy *b, const char **se, const char **sw, const char **nw, const char **ne) { GaimConnection *gc = NULL; OscarData *od = NULL; GaimAccount *account = NULL; GaimPresence *presence; GaimStatus *status; const char *status_id; char *emblems[4] = {NULL,NULL,NULL,NULL}; int i = 0; aim_userinfo_t *userinfo = NULL; account = b->account; if (account != NULL) gc = account->gc; if (gc != NULL) od = gc->proto_data; if (od != NULL) userinfo = aim_locate_finduserinfo(od, b->name); presence = gaim_buddy_get_presence(b); status = gaim_presence_get_active_status(presence); status_id = gaim_status_get_id(status); if (gaim_presence_is_online(presence) == FALSE) { char *gname; if ((b->name) && (od) && (od->ssi.received_data) && (gname = aim_ssi_itemlist_findparentname(od->ssi.local, b->name)) && (aim_ssi_waitingforauth(od->ssi.local, gname, b->name))) { emblems[i++] = "notauthorized"; } else { emblems[i++] = "offline"; } } if (b->name && aim_sn_is_icq(b->name)) { if (!strcmp(status_id, OSCAR_STATUS_ID_INVISIBLE)) emblems[i++] = "invisible"; else if (!strcmp(status_id, OSCAR_STATUS_ID_FREE4CHAT)) emblems[i++] = "freeforchat"; else if (!strcmp(status_id, OSCAR_STATUS_ID_DND)) emblems[i++] = "dnd"; else if (!strcmp(status_id, OSCAR_STATUS_ID_NA)) emblems[i++] = "unavailable"; else if (!strcmp(status_id, OSCAR_STATUS_ID_OCCUPIED)) emblems[i++] = "occupied"; else if (!strcmp(status_id, OSCAR_STATUS_ID_AWAY)) emblems[i++] = "away"; } else if (!strcmp(status_id, OSCAR_STATUS_ID_AWAY)) { emblems[i++] = "away"; } if (userinfo != NULL ) { /* if (userinfo->flags & AIM_FLAG_UNCONFIRMED) emblems[i++] = "unconfirmed"; */ if ((i < 4) && userinfo->flags & AIM_FLAG_ADMINISTRATOR) emblems[i++] = "admin"; if ((i < 4) && userinfo->flags & AIM_FLAG_AOL) emblems[i++] = "aol"; if ((i < 4) && userinfo->flags & AIM_FLAG_WIRELESS) emblems[i++] = "wireless"; if ((i < 4) && userinfo->flags & AIM_FLAG_ACTIVEBUDDY) emblems[i++] = "activebuddy"; if ((i < 4) && (userinfo->capabilities & OSCAR_CAPABILITY_HIPTOP)) emblems[i++] = "hiptop"; if ((i < 4) && (userinfo->capabilities & OSCAR_CAPABILITY_SECUREIM)) emblems[i++] = "secure"; } *se = emblems[0]; *sw = emblems[1]; *nw = emblems[2]; *ne = emblems[3]; } static void oscar_tooltip_text(GaimBuddy *b, GString *str, gboolean full) { GaimConnection *gc = b->account->gc; OscarData *od = gc->proto_data; aim_userinfo_t *userinfo = aim_locate_finduserinfo(od, b->name); if (GAIM_BUDDY_IS_ONLINE(b)) { GaimPresence *presence; GaimStatus *status; const char *message; if (full) oscar_string_append_info(gc, str, "\n", b, userinfo); presence = gaim_buddy_get_presence(b); status = gaim_presence_get_active_status(presence); message = gaim_status_get_attr_string(status, "message"); if (gaim_status_is_available(status)) { if (message != NULL) { /* Available status messages are plain text */ gchar *tmp; tmp = g_markup_escape_text(message, -1); g_string_append_printf(str, "\n<b>%s:</b> %s", _("Message"), tmp); g_free(tmp); } } else { if (message != NULL) { /* Away messages are HTML */ gchar *tmp1, *tmp2; tmp2 = gaim_markup_strip_html(message); tmp1 = g_markup_escape_text(tmp2, -1); g_free(tmp2); tmp2 = gaim_str_sub_away_formatters(tmp1, gaim_account_get_username(gaim_connection_get_account(gc))); g_free(tmp1); g_string_append_printf(str, "\n<b>%s:</b> %s", _("Message"), tmp2); g_free(tmp2); } else { g_string_append_printf(str, "\n<b>%s:</b> %s", _("Message"), _("<i>(retrieving)</i>")); } } } } static char *oscar_status_text(GaimBuddy *b) { GaimConnection *gc; GaimAccount *account; OscarData *od; const GaimPresence *presence; const GaimStatus *status; const char *id; const char *message; gchar *ret = NULL; gc = gaim_account_get_connection(gaim_buddy_get_account(b)); account = gaim_connection_get_account(gc); od = gc->proto_data; presence = gaim_buddy_get_presence(b); status = gaim_presence_get_active_status(presence); id = gaim_status_get_id(status); if (!gaim_presence_is_online(presence)) { char *gname = aim_ssi_itemlist_findparentname(od->ssi.local, b->name); if (aim_ssi_waitingforauth(od->ssi.local, gname, b->name)) ret = g_strdup(_("Not Authorized")); else ret = g_strdup(_("Offline")); } else if (gaim_status_is_available(status) && !strcmp(id, OSCAR_STATUS_ID_AVAILABLE)) { /* Available */ message = gaim_status_get_attr_string(status, "message"); if (message != NULL) { ret = g_markup_escape_text(message, -1); gaim_util_chrreplace(ret, '\n', ' '); } } else if (!gaim_status_is_available(status) && !strcmp(id, OSCAR_STATUS_ID_AWAY)) { /* Away */ message = gaim_status_get_attr_string(status, "message"); if (message != NULL) { gchar *tmp1, *tmp2; tmp1 = gaim_markup_strip_html(message); gaim_util_chrreplace(tmp1, '\n', ' '); tmp2 = g_markup_escape_text(tmp1, -1); ret = gaim_str_sub_away_formatters(tmp2, gaim_account_get_username(account)); g_free(tmp1); g_free(tmp2); } else { ret = g_strdup(_("Away")); } } else ret = g_strdup(gaim_status_get_name(status)); return ret; } static int oscar_icon_req(OscarData *od, FlapConnection *conn, FlapFrame *fr, ...) { GaimConnection *gc = od->gc; va_list ap; guint16 type; guint8 flags = 0, length = 0; guchar *md5 = NULL; va_start(ap, fr); type = va_arg(ap, int); switch(type) { case 0x0000: case 0x0001: { flags = va_arg(ap, int); length = va_arg(ap, int); md5 = va_arg(ap, guchar *); if (flags == 0x41) { if (!flap_connection_getbytype(od, SNAC_FAMILY_BART) && !od->iconconnecting) { od->iconconnecting = TRUE; od->set_icon = TRUE; aim_reqservice(od, SNAC_FAMILY_BART); } else { struct stat st; char *iconfile = gaim_buddy_icons_get_full_path(gaim_account_get_buddy_icon(gaim_connection_get_account(gc))); if (iconfile == NULL) { aim_ssi_delicon(od); } else if (!g_stat(iconfile, &st)) { guchar *buf = g_malloc(st.st_size); FILE *file = g_fopen(iconfile, "rb"); if (file) { /* XXX - Use g_file_get_contents()? */ fread(buf, 1, st.st_size, file); fclose(file); gaim_debug_info("oscar", "Uploading icon to icon server\n"); aim_bart_upload(od, buf, st.st_size); } else gaim_debug_error("oscar", "Can't open buddy icon file!\n"); g_free(buf); } else { gaim_debug_error("oscar", "Can't stat buddy icon file!\n"); } g_free(iconfile); } } else if (flags == 0x81) { char *iconfile = gaim_buddy_icons_get_full_path(gaim_account_get_buddy_icon(gaim_connection_get_account(gc))); if (iconfile == NULL) aim_ssi_delicon(od); else { aim_ssi_seticon(od, md5, length); g_free(iconfile); } } } break; case 0x0002: { /* We just set an "available" message? */ } break; } va_end(ap); return 0; } static void oscar_set_permit_deny(GaimConnection *gc) { GaimAccount *account = gaim_connection_get_account(gc); OscarData *od = (OscarData *)gc->proto_data; if (od->ssi.received_data) { switch (account->perm_deny) { case GAIM_PRIVACY_ALLOW_ALL: aim_ssi_setpermdeny(od, 0x01, 0xffffffff); break; case GAIM_PRIVACY_ALLOW_BUDDYLIST: aim_ssi_setpermdeny(od, 0x05, 0xffffffff); break; case GAIM_PRIVACY_ALLOW_USERS: aim_ssi_setpermdeny(od, 0x03, 0xffffffff); break; case GAIM_PRIVACY_DENY_ALL: aim_ssi_setpermdeny(od, 0x02, 0xffffffff); break; case GAIM_PRIVACY_DENY_USERS: aim_ssi_setpermdeny(od, 0x04, 0xffffffff); break; default: aim_ssi_setpermdeny(od, 0x01, 0xffffffff); break; } } } static void oscar_add_permit(GaimConnection *gc, const char *who) { OscarData *od = (OscarData *)gc->proto_data; gaim_debug_info("oscar", "ssi: About to add a permit\n"); if (od->ssi.received_data) aim_ssi_addpermit(od, who); } static void oscar_add_deny(GaimConnection *gc, const char *who) { OscarData *od = (OscarData *)gc->proto_data; gaim_debug_info("oscar", "ssi: About to add a deny\n"); if (od->ssi.received_data) aim_ssi_adddeny(od, who); } static void oscar_rem_permit(GaimConnection *gc, const char *who) { OscarData *od = (OscarData *)gc->proto_data; gaim_debug_info("oscar", "ssi: About to delete a permit\n"); if (od->ssi.received_data) aim_ssi_delpermit(od, who); } static void oscar_rem_deny(GaimConnection *gc, const char *who) { OscarData *od = (OscarData *)gc->proto_data; gaim_debug_info("oscar", "ssi: About to delete a deny\n"); if (od->ssi.received_data) aim_ssi_deldeny(od, who); } static GList * oscar_status_types(GaimAccount *account) { gboolean is_icq; GList *status_types = NULL; GaimStatusType *type; g_return_val_if_fail(account != NULL, NULL); /* Used to flag some statuses as "user settable" or not */ is_icq = aim_sn_is_icq(gaim_account_get_username(account)); /* Common status types */ /* Really the available message should only be settable for AIM accounts */ type = gaim_status_type_new_with_attrs(GAIM_STATUS_AVAILABLE, OSCAR_STATUS_ID_AVAILABLE, NULL, TRUE, TRUE, FALSE, "message", _("Message"), gaim_value_new(GAIM_TYPE_STRING), NULL); status_types = g_list_append(status_types, type); type = gaim_status_type_new_full(GAIM_STATUS_AVAILABLE, OSCAR_STATUS_ID_FREE4CHAT, _("Free For Chat"), TRUE, is_icq, FALSE); status_types = g_list_append(status_types, type); type = gaim_status_type_new_with_attrs(GAIM_STATUS_AWAY, OSCAR_STATUS_ID_AWAY, NULL, TRUE, TRUE, FALSE, "message", _("Message"), gaim_value_new(GAIM_TYPE_STRING), NULL); status_types = g_list_append(status_types, type); type = gaim_status_type_new_full(GAIM_STATUS_INVISIBLE, OSCAR_STATUS_ID_INVISIBLE, NULL, TRUE, TRUE, FALSE); status_types = g_list_append(status_types, type); /* ICQ-specific status types */ type = gaim_status_type_new_full(GAIM_STATUS_UNAVAILABLE, OSCAR_STATUS_ID_OCCUPIED, _("Occupied"), TRUE, is_icq, FALSE); status_types = g_list_append(status_types, type); type = gaim_status_type_new_full(GAIM_STATUS_EXTENDED_AWAY, OSCAR_STATUS_ID_DND, _("Do Not Disturb"), TRUE, is_icq, FALSE); status_types = g_list_append(status_types, type); type = gaim_status_type_new_full(GAIM_STATUS_EXTENDED_AWAY, OSCAR_STATUS_ID_NA, _("Not Available"), TRUE, is_icq, FALSE); status_types = g_list_append(status_types, type); type = gaim_status_type_new_full(GAIM_STATUS_OFFLINE, OSCAR_STATUS_ID_OFFLINE, NULL, TRUE, TRUE, FALSE); status_types = g_list_append(status_types, type); return status_types; } static void oscar_ssi_editcomment(struct name_data *data, const char *text) { GaimConnection *gc = data->gc; OscarData *od = gc->proto_data; GaimBuddy *b; GaimGroup *g; if (!(b = gaim_find_buddy(gaim_connection_get_account(data->gc), data->name))) { oscar_free_name_data(data); return; } if (!(g = gaim_buddy_get_group(b))) { oscar_free_name_data(data); return; } aim_ssi_editcomment(od, g->name, data->name, text); if (!aim_sncmp(data->name, gc->account->username)) gaim_check_comment(od, text); oscar_free_name_data(data); } static void oscar_buddycb_edit_comment(GaimBlistNode *node, gpointer ignore) { GaimBuddy *buddy; GaimConnection *gc; OscarData *od; struct name_data *data; GaimGroup *g; char *comment; gchar *comment_utf8; gchar *title; g_return_if_fail(GAIM_BLIST_NODE_IS_BUDDY(node)); buddy = (GaimBuddy *) node; gc = gaim_account_get_connection(buddy->account); od = gc->proto_data; data = g_new(struct name_data, 1); if (!(g = gaim_buddy_get_group(buddy))) return; comment = aim_ssi_getcomment(od->ssi.local, g->name, buddy->name); comment_utf8 = comment ? oscar_utf8_try_convert(gc->account, comment) : NULL; data->gc = gc; data->name = g_strdup(buddy->name); data->nick = NULL; title = g_strdup_printf(_("Buddy Comment for %s"), data->name); gaim_request_input(gc, title, _("Buddy Comment:"), NULL, comment_utf8, TRUE, FALSE, NULL, _("OK"), G_CALLBACK(oscar_ssi_editcomment), _("Cancel"), G_CALLBACK(oscar_free_name_data), data); g_free(title); g_free(comment); g_free(comment_utf8); } static void oscar_ask_directim_yes_cb(struct oscar_ask_directim_data *data) { peer_connection_propose(data->od, OSCAR_CAPABILITY_DIRECTIM, data->who); g_free(data->who); g_free(data); } static void oscar_ask_directim_no_cb(struct oscar_ask_directim_data *data) { g_free(data->who); g_free(data); } /* This is called from right-click menu on a buddy node. */ static void oscar_ask_directim(gpointer object, gpointer ignored) { GaimBlistNode *node; GaimBuddy *buddy; GaimConnection *gc; gchar *buf; struct oscar_ask_directim_data *data; node = object; g_return_if_fail(GAIM_BLIST_NODE_IS_BUDDY(node)); buddy = (GaimBuddy *)node; gc = gaim_account_get_connection(buddy->account); data = g_new0(struct oscar_ask_directim_data, 1); data->who = g_strdup(buddy->name); data->od = gc->proto_data; buf = g_strdup_printf(_("You have selected to open a Direct IM connection with %s."), buddy->name); gaim_request_action(gc, NULL, buf, _("Because this reveals your IP address, it " "may be considered a security risk. Do you " "wish to continue?"), 0, data, 2, _("_Connect"), G_CALLBACK(oscar_ask_directim_yes_cb), _("Cancel"), G_CALLBACK(oscar_ask_directim_no_cb)); g_free(buf); } static GList *oscar_buddy_menu(GaimBuddy *buddy) { GaimConnection *gc; OscarData *od; GList *m; GaimMenuAction *act; aim_userinfo_t *userinfo; gc = gaim_account_get_connection(buddy->account); od = gc->proto_data; userinfo = aim_locate_finduserinfo(od, buddy->name); m = NULL; act = gaim_menu_action_new(_("Edit Buddy Comment"), GAIM_CALLBACK(oscar_buddycb_edit_comment), NULL, NULL); m = g_list_append(m, act); #if 0 if (od->icq) { act = gaim_menu_action_new(_("Get Status Msg"), GAIM_CALLBACK(oscar_get_icqstatusmsg), NULL, NULL); m = g_list_append(m, act); } #endif if (userinfo && aim_sncmp(gaim_account_get_username(buddy->account), buddy->name) && GAIM_BUDDY_IS_ONLINE(buddy)) { if (userinfo->capabilities & OSCAR_CAPABILITY_DIRECTIM) { act = gaim_menu_action_new(_("Direct IM"), GAIM_CALLBACK(oscar_ask_directim), NULL, NULL); m = g_list_append(m, act); } #if 0 /* TODO: This menu item should be added by the core */ if (userinfo->capabilities & OSCAR_CAPABILITY_GETFILE) { act = gaim_menu_action_new(_("Get File"), GAIM_CALLBACK(oscar_ask_getfile), NULL, NULL); m = g_list_append(m, act); } #endif } if (od->ssi.received_data) { char *gname; gname = aim_ssi_itemlist_findparentname(od->ssi.local, buddy->name); if (gname && aim_ssi_waitingforauth(od->ssi.local, gname, buddy->name)) { act = gaim_menu_action_new(_("Re-request Authorization"), GAIM_CALLBACK(gaim_auth_sendrequest_menu), NULL, NULL); m = g_list_append(m, act); } } return m; } static GList *oscar_blist_node_menu(GaimBlistNode *node) { if(GAIM_BLIST_NODE_IS_BUDDY(node)) { return oscar_buddy_menu((GaimBuddy *) node); } else { return NULL; } } static void oscar_icq_privacy_opts(GaimConnection *gc, GaimRequestFields *fields) { OscarData *od = gc->proto_data; GaimAccount *account = gaim_connection_get_account(gc); GaimRequestField *f; gboolean auth, web_aware; f = gaim_request_fields_get_field(fields, "authorization"); auth = gaim_request_field_bool_get_value(f); f = gaim_request_fields_get_field(fields, "web_aware"); web_aware = gaim_request_field_bool_get_value(f); gaim_account_set_bool(account, "authorization", auth); gaim_account_set_bool(account, "web_aware", web_aware); oscar_set_extendedstatus(gc); aim_icq_setsecurity(od, auth, web_aware); } static void oscar_show_icq_privacy_opts(GaimPluginAction *action) { GaimConnection *gc = (GaimConnection *) action->context; GaimAccount *account = gaim_connection_get_account(gc); GaimRequestFields *fields; GaimRequestFieldGroup *g; GaimRequestField *f; gboolean auth, web_aware; auth = gaim_account_get_bool(account, "authorization", OSCAR_DEFAULT_AUTHORIZATION); web_aware = gaim_account_get_bool(account, "web_aware", OSCAR_DEFAULT_WEB_AWARE); fields = gaim_request_fields_new(); g = gaim_request_field_group_new(NULL); f = gaim_request_field_bool_new("authorization", _("Require authorization"), auth); gaim_request_field_group_add_field(g, f); f = gaim_request_field_bool_new("web_aware", _("Web aware (enabling this will cause you to receive SPAM!)"), web_aware); gaim_request_field_group_add_field(g, f); gaim_request_fields_add_group(fields, g); gaim_request_fields(gc, _("ICQ Privacy Options"), _("ICQ Privacy Options"), NULL, fields, _("OK"), G_CALLBACK(oscar_icq_privacy_opts), _("Cancel"), NULL, gc); } static void oscar_format_screenname(GaimConnection *gc, const char *nick) { OscarData *od = gc->proto_data; if (!aim_sncmp(gaim_account_get_username(gaim_connection_get_account(gc)), nick)) { if (!flap_connection_getbytype(od, SNAC_FAMILY_ADMIN)) { od->setnick = TRUE; od->newsn = g_strdup(nick); aim_reqservice(od, SNAC_FAMILY_ADMIN); } else { aim_admin_setnick(od, flap_connection_getbytype(od, SNAC_FAMILY_ADMIN), nick); } } else { gaim_notify_error(gc, NULL, _("The new formatting is invalid."), _("Screen name formatting can change only capitalization and whitespace.")); } } static void oscar_show_format_screenname(GaimPluginAction *action) { GaimConnection *gc = (GaimConnection *) action->context; gaim_request_input(gc, NULL, _("New screen name formatting:"), NULL, gaim_connection_get_display_name(gc), FALSE, FALSE, NULL, _("OK"), G_CALLBACK(oscar_format_screenname), _("Cancel"), NULL, gc); } static void oscar_confirm_account(GaimPluginAction *action) { GaimConnection *gc; OscarData *od; FlapConnection *conn; gc = (GaimConnection *)action->context; od = gc->proto_data; conn = flap_connection_getbytype(od, SNAC_FAMILY_ADMIN); if (conn != NULL) { aim_admin_reqconfirm(od, conn); } else { od->conf = TRUE; aim_reqservice(od, SNAC_FAMILY_ADMIN); } } static void oscar_show_email(GaimPluginAction *action) { GaimConnection *gc = (GaimConnection *) action->context; OscarData *od = gc->proto_data; FlapConnection *conn = flap_connection_getbytype(od, SNAC_FAMILY_ADMIN); if (conn) { aim_admin_getinfo(od, conn, 0x11); } else { od->reqemail = TRUE; aim_reqservice(od, SNAC_FAMILY_ADMIN); } } static void oscar_change_email(GaimConnection *gc, const char *email) { OscarData *od = gc->proto_data; FlapConnection *conn = flap_connection_getbytype(od, SNAC_FAMILY_ADMIN); if (conn) { aim_admin_setemail(od, conn, email); } else { od->setemail = TRUE; od->email = g_strdup(email); aim_reqservice(od, SNAC_FAMILY_ADMIN); } } static void oscar_show_change_email(GaimPluginAction *action) { GaimConnection *gc = (GaimConnection *) action->context; gaim_request_input(gc, NULL, _("Change Address To:"), NULL, NULL, FALSE, FALSE, NULL, _("OK"), G_CALLBACK(oscar_change_email), _("Cancel"), NULL, gc); } static void oscar_show_awaitingauth(GaimPluginAction *action) { GaimConnection *gc = (GaimConnection *) action->context; OscarData *od = gc->proto_data; gchar *nombre, *text, *tmp; GaimBlistNode *gnode, *cnode, *bnode; int num=0; text = g_strdup(""); for (gnode = gaim_get_blist()->root; gnode; gnode = gnode->next) { GaimGroup *group = (GaimGroup *)gnode; if(!GAIM_BLIST_NODE_IS_GROUP(gnode)) continue; for (cnode = gnode->child; cnode; cnode = cnode->next) { if(!GAIM_BLIST_NODE_IS_CONTACT(cnode)) continue; for (bnode = cnode->child; bnode; bnode = bnode->next) { GaimBuddy *buddy = (GaimBuddy *)bnode; if(!GAIM_BLIST_NODE_IS_BUDDY(bnode)) continue; if (buddy->account == gc->account && aim_ssi_waitingforauth(od->ssi.local, group->name, buddy->name)) { if (gaim_buddy_get_alias_only(buddy)) nombre = g_strdup_printf(" %s (%s)", buddy->name, gaim_buddy_get_alias_only(buddy)); else nombre = g_strdup_printf(" %s", buddy->name); tmp = g_strdup_printf("%s%s<br>", text, nombre); g_free(text); text = tmp; g_free(nombre); num++; } } } } if (!num) { g_free(text); text = g_strdup(_("<i>you are not waiting for authorization</i>")); } gaim_notify_formatted(gc, NULL, _("You are awaiting authorization from " "the following buddies"), _("You can re-request " "authorization from these buddies by " "right-clicking on them and selecting " "\"Re-request Authorization.\""), text, NULL, NULL); g_free(text); } static void search_by_email_cb(GaimConnection *gc, const char *email) { OscarData *od = (OscarData *)gc->proto_data; aim_search_address(od, email); } static void oscar_show_find_email(GaimPluginAction *action) { GaimConnection *gc = (GaimConnection *) action->context; gaim_request_input(gc, _("Find Buddy by E-Mail"), _("Search for a buddy by e-mail address"), _("Type the e-mail address of the buddy you are " "searching for."), NULL, FALSE, FALSE, NULL, _("Search"), G_CALLBACK(search_by_email_cb), _("Cancel"), NULL, gc); } static void oscar_show_set_info(GaimPluginAction *action) { GaimConnection *gc = (GaimConnection *) action->context; gaim_account_request_change_user_info(gaim_connection_get_account(gc)); } static void oscar_show_set_info_icqurl(GaimPluginAction *action) { GaimConnection *gc = (GaimConnection *) action->context; gaim_notify_uri(gc, "http://www.icq.com/whitepages/user_details.php"); } static void oscar_change_pass(GaimPluginAction *action) { GaimConnection *gc = (GaimConnection *) action->context; gaim_account_request_change_password(gaim_connection_get_account(gc)); } static void oscar_show_chpassurl(GaimPluginAction *action) { GaimConnection *gc = (GaimConnection *) action->context; OscarData *od = gc->proto_data; gchar *substituted = gaim_strreplace(od->authinfo->chpassurl, "%s", gaim_account_get_username(gaim_connection_get_account(gc))); gaim_notify_uri(gc, substituted); g_free(substituted); } static void oscar_show_imforwardingurl(GaimPluginAction *action) { GaimConnection *gc = (GaimConnection *) action->context; gaim_notify_uri(gc, "http://mymobile.aol.com/dbreg/register?action=imf&clientID=1"); } static void oscar_set_icon(GaimConnection *gc, const char *iconfile) { OscarData *od = gc->proto_data; FILE *file; struct stat st; if (iconfile == NULL) { aim_ssi_delicon(od); } else if (!g_stat(iconfile, &st)) { guchar *buf = g_malloc(st.st_size); file = g_fopen(iconfile, "rb"); if (file) { GaimCipher *cipher; GaimCipherContext *context; guchar md5[16]; int len; /* XXX - Use g_file_get_contents()? */ len = fread(buf, 1, st.st_size, file); fclose(file); cipher = gaim_ciphers_find_cipher("md5"); context = gaim_cipher_context_new(cipher, NULL); gaim_cipher_context_append(context, buf, len); gaim_cipher_context_digest(context, 16, md5, NULL); gaim_cipher_context_destroy(context); aim_ssi_seticon(od, md5, 16); } else gaim_debug_error("oscar", "Can't open buddy icon file!\n"); g_free(buf); } else gaim_debug_error("oscar", "Can't stat buddy icon file!\n"); } /** * Called by the Gaim core to determine whether or not we're * allowed to send a file to this user. */ static gboolean oscar_can_receive_file(GaimConnection *gc, const char *who) { OscarData *od; GaimAccount *account; od = gc->proto_data; account = gaim_connection_get_account(gc); if (od != NULL) { aim_userinfo_t *userinfo; userinfo = aim_locate_finduserinfo(od, who); /* * Don't allowing sending a file to a user that does not support * file transfer, and don't allow sending to ourselves. */ if ((userinfo != NULL) && (userinfo->capabilities & OSCAR_CAPABILITY_SENDFILE) && aim_sncmp(who, gaim_account_get_username(account))) { return TRUE; } } return FALSE; } static GaimXfer * oscar_new_xfer(GaimConnection *gc, const char *who) { GaimXfer *xfer; OscarData *od; GaimAccount *account; PeerConnection *conn; od = gc->proto_data; account = gaim_connection_get_account(gc); xfer = gaim_xfer_new(account, GAIM_XFER_SEND, who); gaim_xfer_ref(xfer); gaim_xfer_set_init_fnc(xfer, peer_oft_sendcb_init); gaim_xfer_set_cancel_send_fnc(xfer, peer_oft_cb_generic_cancel); gaim_xfer_set_request_denied_fnc(xfer, peer_oft_cb_generic_cancel); gaim_xfer_set_ack_fnc(xfer, peer_oft_sendcb_ack); conn = peer_connection_new(od, OSCAR_CAPABILITY_SENDFILE, who); conn->flags |= PEER_CONNECTION_FLAG_INITIATED_BY_ME; conn->flags |= PEER_CONNECTION_FLAG_APPROVED; aim_icbm_makecookie(conn->cookie); conn->xfer = xfer; xfer->data = conn; return xfer; } /* * Called by the Gaim core when the user indicates that a * file is to be sent to a special someone. */ static void oscar_send_file(GaimConnection *gc, const char *who, const char *file) { GaimXfer *xfer; xfer = oscar_new_xfer(gc, who); if (file != NULL) gaim_xfer_request_accepted(xfer, file); else gaim_xfer_request(xfer); } static GList * oscar_actions(GaimPlugin *plugin, gpointer context) { GaimConnection *gc = (GaimConnection *) context; OscarData *od = gc->proto_data; GList *m = NULL; GaimPluginAction *act; act = gaim_plugin_action_new(_("Set User Info..."), oscar_show_set_info); m = g_list_append(m, act); if (od->icq) { act = gaim_plugin_action_new(_("Set User Info (URL)..."), oscar_show_set_info_icqurl); m = g_list_append(m, act); } act = gaim_plugin_action_new(_("Change Password..."), oscar_change_pass); m = g_list_append(m, act); if (od->authinfo->chpassurl != NULL) { act = gaim_plugin_action_new(_("Change Password (URL)"), oscar_show_chpassurl); m = g_list_append(m, act); act = gaim_plugin_action_new(_("Configure IM Forwarding (URL)"), oscar_show_imforwardingurl); m = g_list_append(m, act); } m = g_list_append(m, NULL); if (od->icq) { /* ICQ actions */ act = gaim_plugin_action_new(_("Set Privacy Options..."), oscar_show_icq_privacy_opts); m = g_list_append(m, act); } else { /* AIM actions */ act = gaim_plugin_action_new(_("Format Screen Name..."), oscar_show_format_screenname); m = g_list_append(m, act); act = gaim_plugin_action_new(_("Confirm Account"), oscar_confirm_account); m = g_list_append(m, act); act = gaim_plugin_action_new(_("Display Currently Registered E-Mail Address"), oscar_show_email); m = g_list_append(m, act); act = gaim_plugin_action_new(_("Change Currently Registered E-Mail Address..."), oscar_show_change_email); m = g_list_append(m, act); } m = g_list_append(m, NULL); act = gaim_plugin_action_new(_("Show Buddies Awaiting Authorization"), oscar_show_awaitingauth); m = g_list_append(m, act); m = g_list_append(m, NULL); act = gaim_plugin_action_new(_("Search for Buddy by E-Mail Address..."), oscar_show_find_email); m = g_list_append(m, act); #if 0 act = gaim_plugin_action_new(_("Search for Buddy by Information"), show_find_info); m = g_list_append(m, act); #endif return m; } static void oscar_change_passwd(GaimConnection *gc, const char *old, const char *new) { OscarData *od = gc->proto_data; if (od->icq) { aim_icq_changepasswd(od, new); } else { FlapConnection *conn; conn = flap_connection_getbytype(od, SNAC_FAMILY_ADMIN); if (conn) { aim_admin_changepasswd(od, conn, new, old); } else { od->chpass = TRUE; od->oldp = g_strdup(old); od->newp = g_strdup(new); aim_reqservice(od, SNAC_FAMILY_ADMIN); } } } static void oscar_convo_closed(GaimConnection *gc, const char *who) { OscarData *od; PeerConnection *conn; od = gc->proto_data; conn = peer_connection_find_by_type(od, who, OSCAR_CAPABILITY_DIRECTIM); if (conn != NULL) { if (!conn->ready) aim_im_sendch2_cancel(conn); peer_connection_destroy(conn, OSCAR_DISCONNECT_LOCAL_CLOSED); } } static void recent_buddies_cb(const char *name, GaimPrefType type, gconstpointer value, gpointer data) { GaimConnection *gc = data; OscarData *od = gc->proto_data; guint32 presence; presence = aim_ssi_getpresence(od->ssi.local); if (value) { presence &= ~AIM_SSI_PRESENCE_FLAG_NORECENTBUDDIES; aim_ssi_setpresence(od, presence); } else { presence |= AIM_SSI_PRESENCE_FLAG_NORECENTBUDDIES; aim_ssi_setpresence(od, presence); } } #ifdef USE_PRPL_PREFERENCES ppref = gaim_plugin_pref_new_with_name_and_label("/plugins/prpl/oscar/recent_buddies", _("Use recent buddies group")); gaim_plugin_pref_frame_add(frame, ppref); ppref = gaim_plugin_pref_new_with_name_and_label("/plugins/prpl/oscar/show_idle", _("Show how long you have been idle")); gaim_plugin_pref_frame_add(frame, ppref); #endif static const char * oscar_normalize(const GaimAccount *account, const char *str) { static char buf[BUF_LEN]; char *tmp1, *tmp2; int i, j; g_return_val_if_fail(str != NULL, NULL); strncpy(buf, str, BUF_LEN); for (i=0, j=0; buf[j]; i++, j++) { while (buf[j] == ' ') j++; buf[i] = buf[j]; } buf[i] = '\0'; tmp1 = g_utf8_strdown(buf, -1); tmp2 = g_utf8_normalize(tmp1, -1, G_NORMALIZE_DEFAULT); g_snprintf(buf, sizeof(buf), "%s", tmp2); g_free(tmp2); g_free(tmp1); return buf; } static gboolean oscar_offline_message(const GaimBuddy *buddy) { OscarData *od; GaimAccount *account; GaimConnection *gc; account = gaim_buddy_get_account(buddy); gc = gaim_account_get_connection(account); od = (OscarData *)gc->proto_data; return (od->icq && aim_sn_is_icq(gaim_account_get_username(account))); } static GaimPluginProtocolInfo prpl_info = { OPT_PROTO_MAIL_CHECK | OPT_PROTO_IM_IMAGE, NULL, /* user_splits */ NULL, /* protocol_options */ {"jpeg,gif,bmp,ico", 48, 48, 50, 50, GAIM_ICON_SCALE_SEND | GAIM_ICON_SCALE_DISPLAY}, /* icon_spec */ oscar_list_icon, /* list_icon */ oscar_list_emblems, /* list_emblems */ oscar_status_text, /* status_text */ oscar_tooltip_text, /* tooltip_text */ oscar_status_types, /* status_types */ oscar_blist_node_menu, /* blist_node_menu */ oscar_chat_info, /* chat_info */ oscar_chat_info_defaults, /* chat_info_defaults */ oscar_login, /* login */ oscar_close, /* close */ oscar_send_im, /* send_im */ oscar_set_info, /* set_info */ oscar_send_typing, /* send_typing */ oscar_get_info, /* get_info */ oscar_set_status, /* set_status */ oscar_set_idle, /* set_idle */ oscar_change_passwd, /* change_passwd */ oscar_add_buddy, /* add_buddy */ NULL, /* add_buddies */ oscar_remove_buddy, /* remove_buddy */ NULL, /* remove_buddies */ oscar_add_permit, /* add_permit */ oscar_add_deny, /* add_deny */ oscar_rem_permit, /* rem_permit */ oscar_rem_deny, /* rem_deny */ oscar_set_permit_deny, /* set_permit_deny */ oscar_join_chat, /* join_chat */ NULL, /* reject_chat */ oscar_get_chat_name, /* get_chat_name */ oscar_chat_invite, /* chat_invite */ oscar_chat_leave, /* chat_leave */ NULL, /* chat_whisper */ oscar_send_chat, /* chat_send */ oscar_keepalive, /* keepalive */ NULL, /* register_user */ NULL, /* get_cb_info */ NULL, /* get_cb_away */ oscar_alias_buddy, /* alias_buddy */ oscar_move_buddy, /* group_buddy */ oscar_rename_group, /* rename_group */ NULL, /* buddy_free */ oscar_convo_closed, /* convo_closed */ oscar_normalize, /* normalize */ oscar_set_icon, /* set_buddy_icon */ NULL, /* remove_group */ NULL, /* get_cb_real_name */ NULL, /* set_chat_topic */ NULL, /* find_blist_chat */ NULL, /* roomlist_get_list */ NULL, /* roomlist_cancel */ NULL, /* roomlist_expand_category */ oscar_can_receive_file, /* can_receive_file */ oscar_send_file, /* send_file */ oscar_new_xfer, /* new_xfer */ oscar_offline_message, /* offline_message */ NULL, /* whiteboard_prpl_ops */ }; static GaimPluginInfo info = { GAIM_PLUGIN_MAGIC, GAIM_MAJOR_VERSION, GAIM_MINOR_VERSION, GAIM_PLUGIN_PROTOCOL, /**< type */ NULL, /**< ui_requirement */ 0, /**< flags */ NULL, /**< dependencies */ GAIM_PRIORITY_DEFAULT, /**< priority */ "prpl-oscar", /**< id */ "AIM/ICQ", /**< name */ VERSION, /**< version */ /** summary */ N_("AIM/ICQ Protocol Plugin"), /** description */ N_("AIM/ICQ Protocol Plugin"), NULL, /**< author */ GAIM_WEBSITE, /**< homepage */ NULL, /**< load */ NULL, /**< unload */ NULL, /**< destroy */ NULL, /**< ui_info */ &prpl_info, /**< extra_info */ NULL, oscar_actions }; static void init_plugin(GaimPlugin *plugin) { GaimAccountOption *option; option = gaim_account_option_string_new(_("Server"), "server", OSCAR_DEFAULT_LOGIN_SERVER); prpl_info.protocol_options = g_list_append(prpl_info.protocol_options, option); option = gaim_account_option_int_new(_("Port"), "port", OSCAR_DEFAULT_LOGIN_PORT); prpl_info.protocol_options = g_list_append(prpl_info.protocol_options, option); option = gaim_account_option_string_new(_("Encoding"), "encoding", OSCAR_DEFAULT_CUSTOM_ENCODING); prpl_info.protocol_options = g_list_append(prpl_info.protocol_options, option); option = gaim_account_option_bool_new( _("Always use AIM/ICQ proxy server\n(slower, but does not reveal your IP address)"), "always_use_rv_proxy", OSCAR_DEFAULT_ALWAYS_USE_RV_PROXY); prpl_info.protocol_options = g_list_append(prpl_info.protocol_options, option); /* Preferences */ gaim_prefs_add_none("/plugins/prpl/oscar"); gaim_prefs_add_bool("/plugins/prpl/oscar/recent_buddies", FALSE); gaim_prefs_add_bool("/plugins/prpl/oscar/show_idle", FALSE); gaim_prefs_remove("/plugins/prpl/oscar/always_use_rv_proxy"); } GAIM_INIT_PLUGIN(oscar, init_plugin, info);