Annotation of ChivanetAimPidgin/oscarprpl/src/c/family_icq.c, revision 1.1.1.1

1.1       snw         1: /*
                      2:  * Purple's oscar protocol plugin
                      3:  * This file is the legal property of its developers.
                      4:  * Please see the AUTHORS file distributed alongside this file.
                      5:  *
                      6:  * This library is free software; you can redistribute it and/or
                      7:  * modify it under the terms of the GNU Lesser General Public
                      8:  * License as published by the Free Software Foundation; either
                      9:  * version 2 of the License, or (at your option) any later version.
                     10:  *
                     11:  * This library is distributed in the hope that it will be useful,
                     12:  * but WITHOUT ANY WARRANTY; without even the implied warranty of
                     13:  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
                     14:  * Lesser General Public License for more details.
                     15:  *
                     16:  * You should have received a copy of the GNU Lesser General Public
                     17:  * License along with this library; if not, write to the Free Software
                     18:  * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA  02111-1301  USA
                     19: */
                     20: 
                     21: /*
                     22:  * Family 0x0015 - Encapsulated ICQ.
                     23:  *
                     24:  */
                     25: 
                     26: #include "encoding.h"
                     27: #include "oscar.h"
                     28: 
                     29: #define AIM_ICQ_INFO_REQUEST 0x04b2
                     30: #define AIM_ICQ_ALIAS_REQUEST 0x04ba
                     31: 
                     32: static
                     33: int compare_icq_infos(gconstpointer a, gconstpointer b)
                     34: {
                     35:        const struct aim_icq_info* aa = a;
                     36:        const guint16* bb = b;
                     37:        return aa->reqid - *bb;
                     38: }
                     39: 
                     40: static void aim_icq_freeinfo(struct aim_icq_info *info) {
                     41:        int i;
                     42: 
                     43:        if (!info)
                     44:                return;
                     45:        g_free(info->nick);
                     46:        g_free(info->first);
                     47:        g_free(info->last);
                     48:        g_free(info->email);
                     49:        g_free(info->homecity);
                     50:        g_free(info->homestate);
                     51:        g_free(info->homephone);
                     52:        g_free(info->homefax);
                     53:        g_free(info->homeaddr);
                     54:        g_free(info->mobile);
                     55:        g_free(info->homezip);
                     56:        g_free(info->personalwebpage);
                     57:        if (info->email2)
                     58:                for (i = 0; i < info->numaddresses; i++)
                     59:                        g_free(info->email2[i]);
                     60:        g_free(info->email2);
                     61:        g_free(info->workcity);
                     62:        g_free(info->workstate);
                     63:        g_free(info->workphone);
                     64:        g_free(info->workfax);
                     65:        g_free(info->workaddr);
                     66:        g_free(info->workzip);
                     67:        g_free(info->workcompany);
                     68:        g_free(info->workdivision);
                     69:        g_free(info->workposition);
                     70:        g_free(info->workwebpage);
                     71:        g_free(info->info);
                     72:        g_free(info->status_note_title);
                     73:        g_free(info->auth_request_reason);
                     74: }
                     75: 
                     76: static
                     77: int error(OscarData *od, aim_modsnac_t *error_snac, ByteStream *bs)
                     78: {
                     79:        aim_snac_t *original_snac = aim_remsnac(od, error_snac->id);
                     80:        guint16 *request_type;
                     81:        GSList *original_info_ptr;
                     82:        struct aim_icq_info *original_info;
                     83:        guint16 reason;
                     84:        gchar *uin;
                     85: 
                     86:        if (!original_snac || (original_snac->family != SNAC_FAMILY_ICQ) || !original_snac->data) {
                     87:                purple_debug_misc("oscar", "icq: the original snac for the error packet was not found");
                     88:                g_free(original_snac);
                     89:                return 0;
                     90:        }
                     91: 
                     92:        request_type = original_snac->data;
                     93:        original_info_ptr = g_slist_find_custom(od->icq_info, &original_snac->id, compare_icq_infos);
                     94: 
                     95:        if (!original_info_ptr) {
                     96:                purple_debug_misc("oscar", "icq: the request info for the error packet was not found");
                     97:                g_free(original_snac);
                     98:                return 0;
                     99:        }
                    100: 
                    101:        original_info = original_info_ptr->data;
                    102: 
                    103:        reason = byte_stream_get16(bs);
                    104:        uin = g_strdup_printf("%u", original_info->uin);
                    105:        switch (*request_type) {
                    106:                case AIM_ICQ_INFO_REQUEST:
                    107:                        oscar_user_info_display_error(od, reason, uin);
                    108:                        break;
                    109:                case AIM_ICQ_ALIAS_REQUEST:
                    110:                        /* Couldn't retrieve an alias for the buddy requesting authorization; have to make do with UIN only. */
                    111:                        if (original_info->for_auth_request)
                    112:                                oscar_auth_recvrequest(od->gc, uin, NULL, original_info->auth_request_reason);
                    113:                        break;
                    114:                default:
                    115:                        purple_debug_misc("oscar", "icq: got an error packet with unknown request type %u", *request_type);
                    116:                        break;
                    117:        }
                    118: 
                    119:        aim_icq_freeinfo(original_info);
                    120:        od->icq_info = g_slist_remove(od->icq_info, original_info_ptr);
                    121:        g_free(original_snac->data);
                    122:        g_free(original_snac);
                    123:        return 1;
                    124: }
                    125: 
                    126: int
                    127: aim_icq_setsecurity(OscarData *od, gboolean auth_required, gboolean webaware)
                    128: {
                    129:        FlapConnection *conn;
                    130:        ByteStream bs;
                    131:        aim_snacid_t snacid;
                    132:        int bslen;
                    133: 
                    134:        if (!od || !(conn = flap_connection_findbygroup(od, SNAC_FAMILY_ICQ)))
                    135:                return -EINVAL;
                    136: 
                    137:        bslen = 2+4+2+2+2+2+2+1+1+1+1+1+1;
                    138: 
                    139:        byte_stream_new(&bs, 4 + bslen);
                    140: 
                    141:        snacid = aim_cachesnac(od, SNAC_FAMILY_ICQ, 0x0002, 0x0000, NULL, 0);
                    142: 
                    143:        /* For simplicity, don't bother using a tlvlist */
                    144:        byte_stream_put16(&bs, 0x0001);
                    145:        byte_stream_put16(&bs, bslen);
                    146: 
                    147:        byte_stream_putle16(&bs, bslen - 2);
                    148:        byte_stream_putuid(&bs, od);
                    149:        byte_stream_putle16(&bs, 0x07d0); /* I command thee. */
                    150:        byte_stream_putle16(&bs, snacid); /* eh. */
                    151:        byte_stream_putle16(&bs, 0x0c3a); /* shrug. */
                    152:        byte_stream_putle16(&bs, 0x030c);
                    153:        byte_stream_putle16(&bs, 0x0001);
                    154:        byte_stream_putle8(&bs, webaware);
                    155:        byte_stream_putle8(&bs, 0xf8);
                    156:        byte_stream_putle8(&bs, 0x02);
                    157:        byte_stream_putle8(&bs, 0x01);
                    158:        byte_stream_putle8(&bs, 0x00);
                    159:        byte_stream_putle8(&bs, !auth_required);
                    160: 
                    161:        flap_connection_send_snac(od, conn, SNAC_FAMILY_ICQ, 0x0002, snacid, &bs);
                    162: 
                    163:        byte_stream_destroy(&bs);
                    164: 
                    165:        return 0;
                    166: }
                    167: 
                    168: /**
                    169:  * Change your ICQ password.
                    170:  *
                    171:  * @param od The oscar session
                    172:  * @param passwd The new password.  If this is longer than 8 characters it
                    173:  *        will be truncated.
                    174:  * @return Return 0 if no errors, otherwise return the error number.
                    175:  */
                    176: int aim_icq_changepasswd(OscarData *od, const char *passwd)
                    177: {
                    178:        FlapConnection *conn;
                    179:        ByteStream bs;
                    180:        aim_snacid_t snacid;
                    181:        int bslen, passwdlen;
                    182: 
                    183:        if (!passwd)
                    184:                return -EINVAL;
                    185: 
                    186:        if (!od || !(conn = flap_connection_findbygroup(od, SNAC_FAMILY_ICQ)))
                    187:                return -EINVAL;
                    188: 
                    189:        passwdlen = strlen(passwd);
                    190:        if (passwdlen > MAXICQPASSLEN)
                    191:                passwdlen = MAXICQPASSLEN;
                    192:        bslen = 2+4+2+2+2+2+passwdlen+1;
                    193: 
                    194:        byte_stream_new(&bs, 4 + bslen);
                    195: 
                    196:        snacid = aim_cachesnac(od, SNAC_FAMILY_ICQ, 0x0002, 0x0000, NULL, 0);
                    197: 
                    198:        /* For simplicity, don't bother using a tlvlist */
                    199:        byte_stream_put16(&bs, 0x0001);
                    200:        byte_stream_put16(&bs, bslen);
                    201: 
                    202:        byte_stream_putle16(&bs, bslen - 2);
                    203:        byte_stream_putuid(&bs, od);
                    204:        byte_stream_putle16(&bs, 0x07d0); /* I command thee. */
                    205:        byte_stream_putle16(&bs, snacid); /* eh. */
                    206:        byte_stream_putle16(&bs, 0x042e); /* shrug. */
                    207:        byte_stream_putle16(&bs, passwdlen+1);
                    208:        byte_stream_putraw(&bs, (const guint8 *)passwd, passwdlen);
                    209:        byte_stream_putle8(&bs, '\0');
                    210: 
                    211:        flap_connection_send_snac(od, conn, SNAC_FAMILY_ICQ, 0x0002, snacid, &bs);
                    212: 
                    213:        byte_stream_destroy(&bs);
                    214: 
                    215:        return 0;
                    216: }
                    217: 
                    218: int aim_icq_getallinfo(OscarData *od, const char *uin)
                    219: {
                    220:        FlapConnection *conn;
                    221:        ByteStream bs;
                    222:        aim_snacid_t snacid;
                    223:        int bslen;
                    224:        struct aim_icq_info *info;
                    225:        guint16 request_type = AIM_ICQ_INFO_REQUEST;
                    226: 
                    227:        if (!uin || uin[0] < '0' || uin[0] > '9')
                    228:                return -EINVAL;
                    229: 
                    230:        if (!od || !(conn = flap_connection_findbygroup(od, SNAC_FAMILY_ICQ)))
                    231:                return -EINVAL;
                    232: 
                    233:        bslen = 2 + 4 + 2 + 2 + 2 + 4;
                    234: 
                    235:        byte_stream_new(&bs, 4 + bslen);
                    236: 
                    237:        snacid = aim_cachesnac(od, SNAC_FAMILY_ICQ, 0x0002, 0x0000, &request_type, sizeof(request_type));
                    238: 
                    239:        /* For simplicity, don't bother using a tlvlist */
                    240:        byte_stream_put16(&bs, 0x0001);
                    241:        byte_stream_put16(&bs, bslen);
                    242: 
                    243:        byte_stream_putle16(&bs, bslen - 2);
                    244:        byte_stream_putuid(&bs, od);
                    245:        byte_stream_putle16(&bs, 0x07d0); /* I command thee. */
                    246:        byte_stream_putle16(&bs, snacid); /* eh. */
                    247:        byte_stream_putle16(&bs, request_type); /* shrug. */
                    248:        byte_stream_putle32(&bs, atoi(uin));
                    249: 
                    250:        flap_connection_send_snac_with_priority(od, conn, SNAC_FAMILY_ICQ, 0x0002, snacid, &bs, FALSE);
                    251: 
                    252:        byte_stream_destroy(&bs);
                    253: 
                    254:        /* Keep track of this request and the ICQ number and request ID */
                    255:        info = (struct aim_icq_info *)g_new0(struct aim_icq_info, 1);
                    256:        info->reqid = snacid;
                    257:        info->uin = atoi(uin);
                    258:        od->icq_info = g_slist_prepend(od->icq_info, info);
                    259: 
                    260:        return 0;
                    261: }
                    262: 
                    263: int aim_icq_getalias(OscarData *od, const char *uin, gboolean for_auth_request, char *auth_request_reason)
                    264: {
                    265:        FlapConnection *conn;
                    266:        ByteStream bs;
                    267:        aim_snacid_t snacid;
                    268:        int bslen;
                    269:        struct aim_icq_info *info;
                    270:        guint16 request_type = AIM_ICQ_ALIAS_REQUEST;
                    271: 
                    272:        if (!uin || uin[0] < '0' || uin[0] > '9')
                    273:                return -EINVAL;
                    274: 
                    275:        if (!od || !(conn = flap_connection_findbygroup(od, SNAC_FAMILY_ICQ)))
                    276:                return -EINVAL;
                    277: 
                    278:        purple_debug_info("oscar", "Requesting ICQ alias for %s\n", uin);
                    279: 
                    280:        bslen = 2 + 4 + 2 + 2 + 2 + 4;
                    281: 
                    282:        byte_stream_new(&bs, 4 + bslen);
                    283: 
                    284:        snacid = aim_cachesnac(od, SNAC_FAMILY_ICQ, 0x0002, 0x0000, &request_type, sizeof(request_type));
                    285: 
                    286:        /* For simplicity, don't bother using a tlvlist */
                    287:        byte_stream_put16(&bs, 0x0001);
                    288:        byte_stream_put16(&bs, bslen);
                    289: 
                    290:        byte_stream_putle16(&bs, bslen - 2);
                    291:        byte_stream_putuid(&bs, od);
                    292:        byte_stream_putle16(&bs, 0x07d0); /* I command thee. */
                    293:        byte_stream_putle16(&bs, snacid); /* eh. */
                    294:        byte_stream_putle16(&bs, request_type); /* shrug. */
                    295:        byte_stream_putle32(&bs, atoi(uin));
                    296: 
                    297:        flap_connection_send_snac_with_priority(od, conn, SNAC_FAMILY_ICQ, 0x0002, snacid, &bs, FALSE);
                    298: 
                    299:        byte_stream_destroy(&bs);
                    300: 
                    301:        /* Keep track of this request and the ICQ number and request ID */
                    302:        info = (struct aim_icq_info *)g_new0(struct aim_icq_info, 1);
                    303:        info->reqid = snacid;
                    304:        info->uin = atoi(uin);
                    305:        info->for_auth_request = for_auth_request;
                    306:        info->auth_request_reason = g_strdup(auth_request_reason);
                    307:        od->icq_info = g_slist_prepend(od->icq_info, info);
                    308: 
                    309:        return 0;
                    310: }
                    311: 
                    312: /*
                    313:  * Send an SMS message.  This is the non-US way.  The US-way is to IM
                    314:  * their cell phone number (+19195551234).
                    315:  *
                    316:  * We basically construct and send an XML message.  The format is:
                    317:  * <icq_sms_message>
                    318:  *   <destination>full_phone_without_leading_+</destination>
                    319:  *   <text>message</text>
                    320:  *   <codepage>1252</codepage>
                    321:  *   <senders_UIN>self_uin</senders_UIN>
                    322:  *   <senders_name>self_name</senders_name>
                    323:  *   <delivery_receipt>Yes|No</delivery_receipt>
                    324:  *   <time>Wkd, DD Mmm YYYY HH:MM:SS TMZ</time>
                    325:  * </icq_sms_message>
                    326:  *
                    327:  * Yeah hi Peter, whaaaat's happening.  If there's any way to use
                    328:  * a codepage other than 1252 that would be great.  Thaaaanks.
                    329:  */
                    330: int aim_icq_sendsms(OscarData *od, const char *name, const char *msg, const char *alias)
                    331: {
                    332:        FlapConnection *conn;
                    333:        PurpleAccount *account;
                    334:        ByteStream bs;
                    335:        aim_snacid_t snacid;
                    336:        int bslen, xmllen;
                    337:        char *xml;
                    338:        const char *timestr, *username;
                    339:        time_t t;
                    340:        struct tm *tm;
                    341:        gchar *stripped;
                    342: 
                    343:        if (!od || !(conn = flap_connection_findbygroup(od, SNAC_FAMILY_ICQ)))
                    344:                return -EINVAL;
                    345: 
                    346:        if (!name || !msg || !alias)
                    347:                return -EINVAL;
                    348: 
                    349:        account = purple_connection_get_account(od->gc);
                    350:        username = purple_account_get_username(account);
                    351: 
                    352:        time(&t);
                    353:        tm = gmtime(&t);
                    354:        timestr = purple_utf8_strftime("%a, %d %b %Y %T %Z", tm);
                    355: 
                    356:        stripped = purple_markup_strip_html(msg);
                    357: 
                    358:        /* The length of xml included the null terminating character */
                    359:        xmllen = 209 + strlen(name) + strlen(stripped) + strlen(username) + strlen(alias) + strlen(timestr) + 1;
                    360: 
                    361:        xml = g_new(char, xmllen);
                    362:        snprintf(xml, xmllen, "<icq_sms_message>"
                    363:                "<destination>%s</destination>"
                    364:                "<text>%s</text>"
                    365:                "<codepage>1252</codepage>"
                    366:                "<senders_UIN>%s</senders_UIN>"
                    367:                "<senders_name>%s</senders_name>"
                    368:                "<delivery_receipt>Yes</delivery_receipt>"
                    369:                "<time>%s</time>"
                    370:                "</icq_sms_message>",
                    371:                name, stripped, username, alias, timestr);
                    372: 
                    373:        bslen = 36 + xmllen;
                    374: 
                    375:        byte_stream_new(&bs, 4 + bslen);
                    376: 
                    377:        snacid = aim_cachesnac(od, SNAC_FAMILY_ICQ, 0x0002, 0x0000, NULL, 0);
                    378: 
                    379:        /* For simplicity, don't bother using a tlvlist */
                    380:        byte_stream_put16(&bs, 0x0001);
                    381:        byte_stream_put16(&bs, bslen);
                    382: 
                    383:        byte_stream_putle16(&bs, bslen - 2);
                    384:        byte_stream_putuid(&bs, od);
                    385:        byte_stream_putle16(&bs, 0x07d0); /* I command thee. */
                    386:        byte_stream_putle16(&bs, snacid); /* eh. */
                    387: 
                    388:        /* From libicq200-0.3.2/src/SNAC-SRV.cpp */
                    389:        byte_stream_putle16(&bs, 0x1482);
                    390:        byte_stream_put16(&bs, 0x0001);
                    391:        byte_stream_put16(&bs, 0x0016);
                    392:        byte_stream_put32(&bs, 0x00000000);
                    393:        byte_stream_put32(&bs, 0x00000000);
                    394:        byte_stream_put32(&bs, 0x00000000);
                    395:        byte_stream_put32(&bs, 0x00000000);
                    396: 
                    397:        byte_stream_put16(&bs, 0x0000);
                    398:        byte_stream_put16(&bs, xmllen);
                    399:        byte_stream_putstr(&bs, xml);
                    400:        byte_stream_put8(&bs, 0x00);
                    401: 
                    402:        flap_connection_send_snac(od, conn, SNAC_FAMILY_ICQ, 0x0002, snacid, &bs);
                    403: 
                    404:        byte_stream_destroy(&bs);
                    405: 
                    406:        g_free(xml);
                    407:        g_free(stripped);
                    408: 
                    409:        return 0;
                    410: }
                    411: 
                    412: static void
                    413: gotalias(OscarData *od, struct aim_icq_info *info)
                    414: {
                    415:        PurpleConnection *gc = od->gc;
                    416:        PurpleAccount *account = purple_connection_get_account(gc);
                    417:        PurpleBuddy *b;
                    418:        gchar *utf8 = oscar_utf8_try_convert(account, od, info->nick);
                    419: 
                    420:        if (info->for_auth_request) {
                    421:                oscar_auth_recvrequest(gc, g_strdup_printf("%u", info->uin), utf8, info->auth_request_reason);
                    422:        } else {
                    423:                if (utf8 && *utf8) {
                    424:                        gchar who[16];
                    425:                        g_snprintf(who, sizeof(who), "%u", info->uin);
                    426:                        serv_got_alias(gc, who, utf8);
                    427:                        if ((b = purple_find_buddy(account, who))) {
                    428:                                purple_blist_node_set_string((PurpleBlistNode*)b, "servernick", utf8);
                    429:                        }
                    430:                }
                    431:                g_free(utf8);
                    432:        }
                    433: }
                    434: 
                    435: /**
                    436:  * Subtype 0x0003 - Response to SNAC_FAMILY_ICQ/0x002, contains an ICQesque packet.
                    437:  */
                    438: static int
                    439: icqresponse(OscarData *od, aim_modsnac_t *snac, ByteStream *bs)
                    440: {
                    441:        GSList *tlvlist;
                    442:        aim_tlv_t *datatlv;
                    443:        ByteStream qbs;
                    444:        guint32 ouruin;
                    445:        guint16 cmdlen, cmd, reqid;
                    446: 
                    447:        if (!(tlvlist = aim_tlvlist_read(bs)) || !(datatlv = aim_tlv_gettlv(tlvlist, 0x0001, 1))) {
                    448:                aim_tlvlist_free(tlvlist);
                    449:                purple_debug_misc("oscar", "corrupt ICQ response\n");
                    450:                return 0;
                    451:        }
                    452: 
                    453:        byte_stream_init(&qbs, datatlv->value, datatlv->length);
                    454: 
                    455:        cmdlen = byte_stream_getle16(&qbs);
                    456:        ouruin = byte_stream_getle32(&qbs);
                    457:        cmd = byte_stream_getle16(&qbs);
                    458:        reqid = byte_stream_getle16(&qbs);
                    459: 
                    460:        purple_debug_misc("oscar", "icq response: %d bytes, %u, 0x%04x, 0x%04x\n", cmdlen, ouruin, cmd, reqid);
                    461: 
                    462:        if (cmd == 0x07da) { /* information */
                    463:                guint16 subtype;
                    464:                GSList *info_ptr;
                    465:                struct aim_icq_info *info;
                    466: 
                    467:                subtype = byte_stream_getle16(&qbs);
                    468:                byte_stream_advance(&qbs, 1); /* 0x0a */
                    469: 
                    470:                /* find other data from the same request */
                    471:                info_ptr = g_slist_find_custom(od->icq_info, &reqid, compare_icq_infos);
                    472:                if (!info_ptr) {
                    473:                        struct aim_icq_info *new_info = (struct aim_icq_info *)g_new0(struct aim_icq_info, 1);
                    474:                        new_info->reqid = reqid;
                    475:                        info_ptr = od->icq_info = g_slist_prepend(od->icq_info, new_info);
                    476:                }
                    477: 
                    478:                info = info_ptr->data;
                    479:                switch (subtype) {
                    480:                case 0x00a0: { /* hide ip status */
                    481:                        /* nothing */
                    482:                } break;
                    483: 
                    484:                case 0x00aa: { /* password change status */
                    485:                        /* nothing */
                    486:                } break;
                    487: 
                    488:                case 0x00c8: { /* general and "home" information */
                    489:                        info->nick = byte_stream_getstr(&qbs, byte_stream_getle16(&qbs));
                    490:                        info->first = byte_stream_getstr(&qbs, byte_stream_getle16(&qbs));
                    491:                        info->last = byte_stream_getstr(&qbs, byte_stream_getle16(&qbs));
                    492:                        info->email = byte_stream_getstr(&qbs, byte_stream_getle16(&qbs));
                    493:                        info->homecity = byte_stream_getstr(&qbs, byte_stream_getle16(&qbs));
                    494:                        info->homestate = byte_stream_getstr(&qbs, byte_stream_getle16(&qbs));
                    495:                        info->homephone = byte_stream_getstr(&qbs, byte_stream_getle16(&qbs));
                    496:                        info->homefax = byte_stream_getstr(&qbs, byte_stream_getle16(&qbs));
                    497:                        info->homeaddr = byte_stream_getstr(&qbs, byte_stream_getle16(&qbs));
                    498:                        info->mobile = byte_stream_getstr(&qbs, byte_stream_getle16(&qbs));
                    499:                        info->homezip = byte_stream_getstr(&qbs, byte_stream_getle16(&qbs));
                    500:                        info->homecountry = byte_stream_getle16(&qbs);
                    501:                        /* 0x0a 00 02 00 */
                    502:                        /* 1 byte timezone? */
                    503:                        /* 1 byte hide email flag? */
                    504:                } break;
                    505: 
                    506:                case 0x00dc: { /* personal information */
                    507:                        info->age = byte_stream_getle8(&qbs);
                    508:                        info->unknown = byte_stream_getle8(&qbs);
                    509:                        info->gender = byte_stream_getle8(&qbs); /* Not specified=0x00, Female=0x01, Male=0x02 */
                    510:                        info->personalwebpage = byte_stream_getstr(&qbs, byte_stream_getle16(&qbs));
                    511:                        info->birthyear = byte_stream_getle16(&qbs);
                    512:                        info->birthmonth = byte_stream_getle8(&qbs);
                    513:                        info->birthday = byte_stream_getle8(&qbs);
                    514:                        info->language1 = byte_stream_getle8(&qbs);
                    515:                        info->language2 = byte_stream_getle8(&qbs);
                    516:                        info->language3 = byte_stream_getle8(&qbs);
                    517:                        /* 0x00 00 01 00 00 01 00 00 00 00 00 */
                    518:                } break;
                    519: 
                    520:                case 0x00d2: { /* work information */
                    521:                        info->workcity = byte_stream_getstr(&qbs, byte_stream_getle16(&qbs));
                    522:                        info->workstate = byte_stream_getstr(&qbs, byte_stream_getle16(&qbs));
                    523:                        info->workphone = byte_stream_getstr(&qbs, byte_stream_getle16(&qbs));
                    524:                        info->workfax = byte_stream_getstr(&qbs, byte_stream_getle16(&qbs));
                    525:                        info->workaddr = byte_stream_getstr(&qbs, byte_stream_getle16(&qbs));
                    526:                        info->workzip = byte_stream_getstr(&qbs, byte_stream_getle16(&qbs));
                    527:                        info->workcountry = byte_stream_getle16(&qbs);
                    528:                        info->workcompany = byte_stream_getstr(&qbs, byte_stream_getle16(&qbs));
                    529:                        info->workdivision = byte_stream_getstr(&qbs, byte_stream_getle16(&qbs));
                    530:                        info->workposition = byte_stream_getstr(&qbs, byte_stream_getle16(&qbs));
                    531:                        byte_stream_advance(&qbs, 2); /* 0x01 00 */
                    532:                        info->workwebpage = byte_stream_getstr(&qbs, byte_stream_getle16(&qbs));
                    533:                } break;
                    534: 
                    535:                case 0x00e6: { /* additional personal information */
                    536:                        info->info = byte_stream_getstr(&qbs, byte_stream_getle16(&qbs)-1);
                    537:                } break;
                    538: 
                    539:                case 0x00eb: { /* email address(es) */
                    540:                        int i;
                    541:                        info->numaddresses = byte_stream_getle16(&qbs);
                    542:                        info->email2 = (char **)g_new0(char *, info->numaddresses);
                    543:                        for (i = 0; i < info->numaddresses; i++) {
                    544:                                info->email2[i] = byte_stream_getstr(&qbs, byte_stream_getle16(&qbs));
                    545:                                if (i+1 != info->numaddresses)
                    546:                                        byte_stream_advance(&qbs, 1); /* 0x00 */
                    547:                        }
                    548:                } break;
                    549: 
                    550:                case 0x00f0: { /* personal interests */
                    551:                } break;
                    552: 
                    553:                case 0x00fa: { /* past background and current organizations */
                    554:                } break;
                    555: 
                    556:                case 0x0104: { /* alias info */
                    557:                        info->nick = byte_stream_getstr(&qbs, byte_stream_getle16(&qbs));
                    558:                        info->first = byte_stream_getstr(&qbs, byte_stream_getle16(&qbs));
                    559:                        info->last = byte_stream_getstr(&qbs, byte_stream_getle16(&qbs));
                    560:                        byte_stream_advance(&qbs, byte_stream_getle16(&qbs)); /* email address? */
                    561:                        /* Then 0x00 02 00 */
                    562:                } break;
                    563: 
                    564:                case 0x010e: { /* unknown */
                    565:                        /* 0x00 00 */
                    566:                } break;
                    567: 
                    568:                case 0x019a: { /* simple info */
                    569:                        byte_stream_advance(&qbs, 2);
                    570:                        info->uin = byte_stream_getle32(&qbs);
                    571:                        info->nick = byte_stream_getstr(&qbs, byte_stream_getle16(&qbs));
                    572:                        info->first = byte_stream_getstr(&qbs, byte_stream_getle16(&qbs));
                    573:                        info->last = byte_stream_getstr(&qbs, byte_stream_getle16(&qbs));
                    574:                        info->email = byte_stream_getstr(&qbs, byte_stream_getle16(&qbs));
                    575:                        /* Then 0x00 02 00 00 00 00 00 */
                    576:                } break;
                    577: 
                    578:                /* status note title and send request for status note text */
                    579:                case 0x0fb4: {
                    580:                        GSList *tlvlist;
                    581:                        aim_tlv_t *tlv;
                    582:                        FlapConnection *conn;
                    583:                        char *uin = NULL;
                    584:                        char *status_note_title = NULL;
                    585: 
                    586:                        conn = flap_connection_findbygroup(od, 0x0004);
                    587:                        if (conn == NULL)
                    588:                        {
                    589:                                purple_debug_misc("oscar", "icq/0x0fb4: flap connection was not found.\n");
                    590:                                break;
                    591:                        }
                    592: 
                    593:                        byte_stream_advance(&qbs, 0x02); /* length */
                    594:                        byte_stream_advance(&qbs, 0x2f); /* unknown stuff */
                    595: 
                    596:                        tlvlist = aim_tlvlist_read(&qbs);
                    597: 
                    598:                        tlv = aim_tlv_gettlv(tlvlist, 0x0032, 1);
                    599:                        if (tlv != NULL)
                    600:                                /* Get user number */
                    601:                                uin = aim_tlv_getvalue_as_string(tlv);
                    602: 
                    603:                        tlv = aim_tlv_gettlv(tlvlist, 0x0226, 1);
                    604:                        if (tlv != NULL)
                    605:                                /* Get status note title */
                    606:                                status_note_title = aim_tlv_getvalue_as_string(tlv);
                    607: 
                    608:                        aim_tlvlist_free(tlvlist);
                    609: 
                    610:                        if (uin == NULL || status_note_title == NULL)
                    611:                        {
                    612:                                purple_debug_misc("oscar", "icq/0x0fb4: uin or "
                    613:                                                "status_note_title was not found\n");
                    614:                                g_free(uin);
                    615:                                g_free(status_note_title);
                    616:                                break;
                    617:                        }
                    618: 
                    619:                        if (status_note_title[0] == '\0')
                    620:                        {
                    621:                                PurpleAccount *account;
                    622:                                PurpleBuddy *buddy;
                    623:                                PurplePresence *presence;
                    624:                                PurpleStatus *status;
                    625: 
                    626:                                account = purple_connection_get_account(od->gc);
                    627:                                buddy = purple_find_buddy(account, uin);
                    628:                                presence = purple_buddy_get_presence(buddy);
                    629:                                status = purple_presence_get_active_status(presence);
                    630: 
                    631:                                purple_prpl_got_user_status(account, uin,
                    632:                                                purple_status_get_id(status),
                    633:                                                "message", NULL, NULL);
                    634: 
                    635:                                g_free(status_note_title);
                    636:                        }
                    637:                        else
                    638:                        {
                    639:                                struct aim_icq_info *info;
                    640:                                ByteStream bs;
                    641:                                guint32 bslen;
                    642:                                aim_snacid_t snacid;
                    643:                                guchar cookie[8];
                    644: 
                    645:                                info = g_new0(struct aim_icq_info, 1);
                    646: 
                    647:                                bslen = 13 + strlen(uin) + 30 + 6 + 4 + 55 + 85 + 4;
                    648:                                byte_stream_new(&bs, 4 + bslen);
                    649: 
                    650:                                snacid = aim_cachesnac(od, 0x0004, 0x0006, 0x0000, NULL, 0);
                    651: 
                    652:                                aim_icbm_makecookie(cookie);
                    653: 
                    654:                                byte_stream_putraw(&bs, cookie, 8); /* ICBM cookie */
                    655:                                byte_stream_put16(&bs, 0x0002); /* message channel */
                    656:                                byte_stream_put8(&bs, strlen(uin)); /* uin */
                    657:                                byte_stream_putstr(&bs, uin);
                    658: 
                    659:                                byte_stream_put16(&bs, 0x0005); /* rendez vous data */
                    660:                                byte_stream_put16(&bs, 0x00b2);
                    661:                                byte_stream_put16(&bs, 0x0000); /* request */
                    662:                                byte_stream_putraw(&bs, cookie, 8); /* ICBM cookie */
                    663:                                byte_stream_put32(&bs, 0x09461349); /* ICQ server relaying */
                    664:                                byte_stream_put16(&bs, 0x4c7f);
                    665:                                byte_stream_put16(&bs, 0x11d1);
                    666:                                byte_stream_put32(&bs, 0x82224445);
                    667:                                byte_stream_put32(&bs, 0x53540000);
                    668: 
                    669:                                byte_stream_put16(&bs, 0x000a); /* unknown TLV */
                    670:                                byte_stream_put16(&bs, 0x0002);
                    671:                                byte_stream_put16(&bs, 0x0001);
                    672: 
                    673:                                byte_stream_put16(&bs, 0x000f); /* unknown TLV */
                    674:                                byte_stream_put16(&bs, 0x0000);
                    675: 
                    676:                                byte_stream_put16(&bs, 0x2711); /* extended data */
                    677:                                byte_stream_put16(&bs, 0x008a);
                    678:                                byte_stream_putle16(&bs, 0x001b); /* length */
                    679:                                byte_stream_putle16(&bs, 0x0009); /* version */
                    680:                                byte_stream_putle32(&bs, 0x00000000); /* plugin: none */
                    681:                                byte_stream_putle32(&bs, 0x00000000);
                    682:                                byte_stream_putle32(&bs, 0x00000000);
                    683:                                byte_stream_putle32(&bs, 0x00000000);
                    684:                                byte_stream_putle16(&bs, 0x0000); /* unknown */
                    685:                                byte_stream_putle32(&bs, 0x00000000); /* client capabilities flags */
                    686:                                byte_stream_put8(&bs, 0x00); /* unknown */
                    687:                                byte_stream_putle16(&bs, 0x0064); /* downcounter? */
                    688:                                byte_stream_putle16(&bs, 0x000e); /* length */
                    689:                                byte_stream_putle16(&bs, 0x0064); /* downcounter? */
                    690:                                byte_stream_putle32(&bs, 0x00000000); /* unknown */
                    691:                                byte_stream_putle32(&bs, 0x00000000);
                    692:                                byte_stream_putle32(&bs, 0x00000000);
                    693:                                byte_stream_put8(&bs, 0x1a); /* message type: plugin message descibed by text string */
                    694:                                byte_stream_put8(&bs, 0x00); /* message flags */
                    695:                                byte_stream_putle16(&bs, 0x0000); /* status code */
                    696:                                byte_stream_putle16(&bs, 0x0001); /* priority code */
                    697:                                byte_stream_putle16(&bs, 0x0000); /* text length */
                    698: 
                    699:                                byte_stream_put8(&bs, 0x3a); /* message dump */
                    700:                                byte_stream_put32(&bs, 0x00811a18);
                    701:                                byte_stream_put32(&bs, 0xbc0e6c18);
                    702:                                byte_stream_put32(&bs, 0x47a5916f);
                    703:                                byte_stream_put32(&bs, 0x18dcc76f);
                    704:                                byte_stream_put32(&bs, 0x1a010013);
                    705:                                byte_stream_put32(&bs, 0x00000041);
                    706:                                byte_stream_put32(&bs, 0x77617920);
                    707:                                byte_stream_put32(&bs, 0x53746174);
                    708:                                byte_stream_put32(&bs, 0x7573204d);
                    709:                                byte_stream_put32(&bs, 0x65737361);
                    710:                                byte_stream_put32(&bs, 0x67650100);
                    711:                                byte_stream_put32(&bs, 0x00000000);
                    712:                                byte_stream_put32(&bs, 0x00000000);
                    713:                                byte_stream_put32(&bs, 0x00000000);
                    714:                                byte_stream_put32(&bs, 0x00000015);
                    715:                                byte_stream_put32(&bs, 0x00000000);
                    716:                                byte_stream_put32(&bs, 0x0000000d);
                    717:                                byte_stream_put32(&bs, 0x00000074);
                    718:                                byte_stream_put32(&bs, 0x6578742f);
                    719:                                byte_stream_put32(&bs, 0x782d616f);
                    720:                                byte_stream_put32(&bs, 0x6c727466);
                    721: 
                    722:                                byte_stream_put16(&bs, 0x0003); /* server ACK requested */
                    723:                                byte_stream_put16(&bs, 0x0000);
                    724: 
                    725:                                info->uin = atoi(uin);
                    726:                                info->status_note_title = status_note_title;
                    727: 
                    728:                                memcpy(&info->icbm_cookie, cookie, 8);
                    729: 
                    730:                                od->icq_info = g_slist_prepend(od->icq_info, info);
                    731: 
                    732:                                flap_connection_send_snac_with_priority(od, conn, 0x0004, 0x0006, snacid, &bs, FALSE);
                    733: 
                    734:                                byte_stream_destroy(&bs);
                    735:                        }
                    736: 
                    737:                        g_free(uin);
                    738: 
                    739:                } break;
                    740: 
                    741:                } /* End switch statement */
                    742: 
                    743:                if (!(snac->flags & 0x0001)) {
                    744:                        if (subtype != 0x0104)
                    745:                                oscar_user_info_display_icq(od, info);
                    746: 
                    747:                        if (info->uin && info->nick)
                    748:                                gotalias(od, info);
                    749: 
                    750:                        aim_icq_freeinfo(info);
                    751:                        od->icq_info = g_slist_remove(od->icq_info, info);
                    752:                }
                    753:        }
                    754: 
                    755:        aim_tlvlist_free(tlvlist);
                    756: 
                    757:        return 1;
                    758: }
                    759: 
                    760: static int
                    761: snachandler(OscarData *od, FlapConnection *conn, aim_module_t *mod, FlapFrame *frame, aim_modsnac_t *snac, ByteStream *bs)
                    762: {
                    763:        if (snac->subtype == 0x0001)
                    764:                return error(od, snac, bs);
                    765:        else if (snac->subtype == 0x0003)
                    766:                return icqresponse(od, snac, bs);
                    767: 
                    768:        return 0;
                    769: }
                    770: 
                    771: static void
                    772: icq_shutdown(OscarData *od, aim_module_t *mod)
                    773: {
                    774:        GSList *cur;
                    775:        for (cur = od->icq_info; cur; cur = cur->next)
                    776:                aim_icq_freeinfo(cur->data);
                    777:        g_slist_free(od->icq_info);
                    778: }
                    779: 
                    780: int
                    781: icq_modfirst(OscarData *od, aim_module_t *mod)
                    782: {
                    783:        mod->family = SNAC_FAMILY_ICQ;
                    784:        mod->version = 0x0001;
                    785:        mod->toolid = 0x0110;
                    786:        mod->toolversion = 0x047c;
                    787:        mod->flags = 0;
                    788:        strncpy(mod->name, "icq", sizeof(mod->name));
                    789:        mod->snachandler = snachandler;
                    790:        mod->shutdown = icq_shutdown;
                    791: 
                    792:        return 0;
                    793: }

FreeBSD-CVSweb <freebsd-cvsweb@FreeBSD.org>