pidgin/pidgin

Add support for osslsigncode and newer versions of nsis
release-2.x.y
2021-03-30, Gary Kramlich
cbb6d1c52d6e
Add support for osslsigncode and newer versions of nsis

Testing Done:
Built locally on windows with `make -f Makefile.mingw installers`

Ran the installer in Russian to verify that that issue was fixed as well.

Bugs closed: PIDGIN-17325, PIDGIN-17402

Reviewed at https://reviews.imfreedom.org/r/588/
/*
* Purple's oscar protocol plugin
* This file is the legal property of its developers.
* Please see the AUTHORS file distributed alongside this file.
*
* This library is free software; you can redistribute it and/or
* modify it under the terms of the GNU Lesser General Public
* License as published by the Free Software Foundation; either
* version 2 of the License, or (at your option) any later version.
*
* This library 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
* Lesser General Public License for more details.
*
* You should have received a copy of the GNU Lesser General Public
* License along with this library; if not, write to the Free Software
* Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02111-1301 USA
*/
/*
* Family 0x0015 - Encapsulated ICQ.
*
*/
#include "encoding.h"
#include "oscar.h"
#define AIM_ICQ_INFO_REQUEST 0x04b2
#define AIM_ICQ_ALIAS_REQUEST 0x04ba
static
int compare_icq_infos(gconstpointer a, gconstpointer b)
{
const struct aim_icq_info* aa = a;
const guint16* bb = b;
return aa->reqid - *bb;
}
static void aim_icq_freeinfo(struct aim_icq_info *info) {
int i;
if (!info)
return;
g_free(info->nick);
g_free(info->first);
g_free(info->last);
g_free(info->email);
g_free(info->homecity);
g_free(info->homestate);
g_free(info->homephone);
g_free(info->homefax);
g_free(info->homeaddr);
g_free(info->mobile);
g_free(info->homezip);
g_free(info->personalwebpage);
if (info->email2)
for (i = 0; i < info->numaddresses; i++)
g_free(info->email2[i]);
g_free(info->email2);
g_free(info->workcity);
g_free(info->workstate);
g_free(info->workphone);
g_free(info->workfax);
g_free(info->workaddr);
g_free(info->workzip);
g_free(info->workcompany);
g_free(info->workdivision);
g_free(info->workposition);
g_free(info->workwebpage);
g_free(info->info);
g_free(info->status_note_title);
g_free(info->auth_request_reason);
}
static
int error(OscarData *od, aim_modsnac_t *error_snac, ByteStream *bs)
{
aim_snac_t *original_snac = aim_remsnac(od, error_snac->id);
guint16 *request_type;
GSList *original_info_ptr;
struct aim_icq_info *original_info;
guint16 reason;
gchar *uin;
if (!original_snac || (original_snac->family != SNAC_FAMILY_ICQ) || !original_snac->data) {
purple_debug_misc("oscar", "icq: the original snac for the error packet was not found");
g_free(original_snac);
return 0;
}
request_type = original_snac->data;
original_info_ptr = g_slist_find_custom(od->icq_info, &original_snac->id, compare_icq_infos);
if (!original_info_ptr) {
purple_debug_misc("oscar", "icq: the request info for the error packet was not found");
g_free(original_snac);
return 0;
}
original_info = original_info_ptr->data;
reason = byte_stream_get16(bs);
uin = g_strdup_printf("%u", original_info->uin);
switch (*request_type) {
case AIM_ICQ_INFO_REQUEST:
oscar_user_info_display_error(od, reason, uin);
break;
case AIM_ICQ_ALIAS_REQUEST:
/* Couldn't retrieve an alias for the buddy requesting authorization; have to make do with UIN only. */
if (original_info->for_auth_request)
oscar_auth_recvrequest(od->gc, uin, NULL, original_info->auth_request_reason);
break;
default:
purple_debug_misc("oscar", "icq: got an error packet with unknown request type %u", *request_type);
break;
}
aim_icq_freeinfo(original_info);
od->icq_info = g_slist_remove(od->icq_info, original_info_ptr);
g_free(original_snac->data);
g_free(original_snac);
return 1;
}
int
aim_icq_setsecurity(OscarData *od, gboolean auth_required, gboolean webaware)
{
FlapConnection *conn;
ByteStream bs;
aim_snacid_t snacid;
int bslen;
if (!od || !(conn = flap_connection_findbygroup(od, SNAC_FAMILY_ICQ)))
return -EINVAL;
bslen = 2+4+2+2+2+2+2+1+1+1+1+1+1;
byte_stream_new(&bs, 4 + bslen);
snacid = aim_cachesnac(od, SNAC_FAMILY_ICQ, 0x0002, 0x0000, NULL, 0);
/* For simplicity, don't bother using a tlvlist */
byte_stream_put16(&bs, 0x0001);
byte_stream_put16(&bs, bslen);
byte_stream_putle16(&bs, bslen - 2);
byte_stream_putuid(&bs, od);
byte_stream_putle16(&bs, 0x07d0); /* I command thee. */
byte_stream_putle16(&bs, snacid); /* eh. */
byte_stream_putle16(&bs, 0x0c3a); /* shrug. */
byte_stream_putle16(&bs, 0x030c);
byte_stream_putle16(&bs, 0x0001);
byte_stream_putle8(&bs, webaware);
byte_stream_putle8(&bs, 0xf8);
byte_stream_putle8(&bs, 0x02);
byte_stream_putle8(&bs, 0x01);
byte_stream_putle8(&bs, 0x00);
byte_stream_putle8(&bs, !auth_required);
flap_connection_send_snac(od, conn, SNAC_FAMILY_ICQ, 0x0002, snacid, &bs);
byte_stream_destroy(&bs);
return 0;
}
/**
* Change your ICQ password.
*
* @param od The oscar session
* @param passwd The new password. If this is longer than 8 characters it
* will be truncated.
* @return Return 0 if no errors, otherwise return the error number.
*/
int aim_icq_changepasswd(OscarData *od, const char *passwd)
{
FlapConnection *conn;
ByteStream bs;
aim_snacid_t snacid;
int bslen, passwdlen;
if (!passwd)
return -EINVAL;
if (!od || !(conn = flap_connection_findbygroup(od, SNAC_FAMILY_ICQ)))
return -EINVAL;
passwdlen = strlen(passwd);
if (passwdlen > MAXICQPASSLEN)
passwdlen = MAXICQPASSLEN;
bslen = 2+4+2+2+2+2+passwdlen+1;
byte_stream_new(&bs, 4 + bslen);
snacid = aim_cachesnac(od, SNAC_FAMILY_ICQ, 0x0002, 0x0000, NULL, 0);
/* For simplicity, don't bother using a tlvlist */
byte_stream_put16(&bs, 0x0001);
byte_stream_put16(&bs, bslen);
byte_stream_putle16(&bs, bslen - 2);
byte_stream_putuid(&bs, od);
byte_stream_putle16(&bs, 0x07d0); /* I command thee. */
byte_stream_putle16(&bs, snacid); /* eh. */
byte_stream_putle16(&bs, 0x042e); /* shrug. */
byte_stream_putle16(&bs, passwdlen+1);
byte_stream_putraw(&bs, (const guint8 *)passwd, passwdlen);
byte_stream_putle8(&bs, '\0');
flap_connection_send_snac(od, conn, SNAC_FAMILY_ICQ, 0x0002, snacid, &bs);
byte_stream_destroy(&bs);
return 0;
}
int aim_icq_getallinfo(OscarData *od, const char *uin)
{
FlapConnection *conn;
ByteStream bs;
aim_snacid_t snacid;
int bslen;
struct aim_icq_info *info;
guint16 request_type = AIM_ICQ_INFO_REQUEST;
if (!uin || uin[0] < '0' || uin[0] > '9')
return -EINVAL;
if (!od || !(conn = flap_connection_findbygroup(od, SNAC_FAMILY_ICQ)))
return -EINVAL;
bslen = 2 + 4 + 2 + 2 + 2 + 4;
byte_stream_new(&bs, 4 + bslen);
snacid = aim_cachesnac(od, SNAC_FAMILY_ICQ, 0x0002, 0x0000, &request_type, sizeof(request_type));
/* For simplicity, don't bother using a tlvlist */
byte_stream_put16(&bs, 0x0001);
byte_stream_put16(&bs, bslen);
byte_stream_putle16(&bs, bslen - 2);
byte_stream_putuid(&bs, od);
byte_stream_putle16(&bs, 0x07d0); /* I command thee. */
byte_stream_putle16(&bs, snacid); /* eh. */
byte_stream_putle16(&bs, request_type); /* shrug. */
byte_stream_putle32(&bs, atoi(uin));
flap_connection_send_snac_with_priority(od, conn, SNAC_FAMILY_ICQ, 0x0002, snacid, &bs, FALSE);
byte_stream_destroy(&bs);
/* Keep track of this request and the ICQ number and request ID */
info = (struct aim_icq_info *)g_new0(struct aim_icq_info, 1);
info->reqid = snacid;
info->uin = atoi(uin);
od->icq_info = g_slist_prepend(od->icq_info, info);
return 0;
}
int aim_icq_getalias(OscarData *od, const char *uin, gboolean for_auth_request, char *auth_request_reason)
{
FlapConnection *conn;
ByteStream bs;
aim_snacid_t snacid;
int bslen;
struct aim_icq_info *info;
guint16 request_type = AIM_ICQ_ALIAS_REQUEST;
if (!uin || uin[0] < '0' || uin[0] > '9')
return -EINVAL;
if (!od || !(conn = flap_connection_findbygroup(od, SNAC_FAMILY_ICQ)))
return -EINVAL;
purple_debug_info("oscar", "Requesting ICQ alias for %s\n", uin);
bslen = 2 + 4 + 2 + 2 + 2 + 4;
byte_stream_new(&bs, 4 + bslen);
snacid = aim_cachesnac(od, SNAC_FAMILY_ICQ, 0x0002, 0x0000, &request_type, sizeof(request_type));
/* For simplicity, don't bother using a tlvlist */
byte_stream_put16(&bs, 0x0001);
byte_stream_put16(&bs, bslen);
byte_stream_putle16(&bs, bslen - 2);
byte_stream_putuid(&bs, od);
byte_stream_putle16(&bs, 0x07d0); /* I command thee. */
byte_stream_putle16(&bs, snacid); /* eh. */
byte_stream_putle16(&bs, request_type); /* shrug. */
byte_stream_putle32(&bs, atoi(uin));
flap_connection_send_snac_with_priority(od, conn, SNAC_FAMILY_ICQ, 0x0002, snacid, &bs, FALSE);
byte_stream_destroy(&bs);
/* Keep track of this request and the ICQ number and request ID */
info = (struct aim_icq_info *)g_new0(struct aim_icq_info, 1);
info->reqid = snacid;
info->uin = atoi(uin);
info->for_auth_request = for_auth_request;
info->auth_request_reason = g_strdup(auth_request_reason);
od->icq_info = g_slist_prepend(od->icq_info, info);
return 0;
}
/*
* Send an SMS message. This is the non-US way. The US-way is to IM
* their cell phone number (+19195551234).
*
* We basically construct and send an XML message. The format is:
* <icq_sms_message>
* <destination>full_phone_without_leading_+</destination>
* <text>message</text>
* <codepage>1252</codepage>
* <senders_UIN>self_uin</senders_UIN>
* <senders_name>self_name</senders_name>
* <delivery_receipt>Yes|No</delivery_receipt>
* <time>Wkd, DD Mmm YYYY HH:MM:SS TMZ</time>
* </icq_sms_message>
*
* Yeah hi Peter, whaaaat's happening. If there's any way to use
* a codepage other than 1252 that would be great. Thaaaanks.
*/
int aim_icq_sendsms(OscarData *od, const char *name, const char *msg, const char *alias)
{
FlapConnection *conn;
PurpleAccount *account;
ByteStream bs;
aim_snacid_t snacid;
int bslen, xmllen;
char *xml;
const char *timestr, *username;
time_t t;
struct tm *tm;
gchar *stripped;
if (!od || !(conn = flap_connection_findbygroup(od, SNAC_FAMILY_ICQ)))
return -EINVAL;
if (!name || !msg || !alias)
return -EINVAL;
account = purple_connection_get_account(od->gc);
username = purple_account_get_username(account);
time(&t);
tm = gmtime(&t);
timestr = purple_utf8_strftime("%a, %d %b %Y %T %Z", tm);
stripped = purple_markup_strip_html(msg);
/* The length of xml included the null terminating character */
xmllen = 209 + strlen(name) + strlen(stripped) + strlen(username) + strlen(alias) + strlen(timestr) + 1;
xml = g_new(char, xmllen);
snprintf(xml, xmllen, "<icq_sms_message>"
"<destination>%s</destination>"
"<text>%s</text>"
"<codepage>1252</codepage>"
"<senders_UIN>%s</senders_UIN>"
"<senders_name>%s</senders_name>"
"<delivery_receipt>Yes</delivery_receipt>"
"<time>%s</time>"
"</icq_sms_message>",
name, stripped, username, alias, timestr);
bslen = 36 + xmllen;
byte_stream_new(&bs, 4 + bslen);
snacid = aim_cachesnac(od, SNAC_FAMILY_ICQ, 0x0002, 0x0000, NULL, 0);
/* For simplicity, don't bother using a tlvlist */
byte_stream_put16(&bs, 0x0001);
byte_stream_put16(&bs, bslen);
byte_stream_putle16(&bs, bslen - 2);
byte_stream_putuid(&bs, od);
byte_stream_putle16(&bs, 0x07d0); /* I command thee. */
byte_stream_putle16(&bs, snacid); /* eh. */
/* From libicq200-0.3.2/src/SNAC-SRV.cpp */
byte_stream_putle16(&bs, 0x1482);
byte_stream_put16(&bs, 0x0001);
byte_stream_put16(&bs, 0x0016);
byte_stream_put32(&bs, 0x00000000);
byte_stream_put32(&bs, 0x00000000);
byte_stream_put32(&bs, 0x00000000);
byte_stream_put32(&bs, 0x00000000);
byte_stream_put16(&bs, 0x0000);
byte_stream_put16(&bs, xmllen);
byte_stream_putstr(&bs, xml);
byte_stream_put8(&bs, 0x00);
flap_connection_send_snac(od, conn, SNAC_FAMILY_ICQ, 0x0002, snacid, &bs);
byte_stream_destroy(&bs);
g_free(xml);
g_free(stripped);
return 0;
}
static void
gotalias(OscarData *od, struct aim_icq_info *info)
{
PurpleConnection *gc = od->gc;
PurpleAccount *account = purple_connection_get_account(gc);
PurpleBuddy *b;
gchar *utf8 = oscar_utf8_try_convert(account, od, info->nick);
if (info->for_auth_request) {
oscar_auth_recvrequest(gc, g_strdup_printf("%u", info->uin), utf8, info->auth_request_reason);
} else {
if (utf8 && *utf8) {
gchar who[16];
g_snprintf(who, sizeof(who), "%u", info->uin);
serv_got_alias(gc, who, utf8);
if ((b = purple_find_buddy(account, who))) {
purple_blist_node_set_string((PurpleBlistNode*)b, "servernick", utf8);
}
}
g_free(utf8);
}
}
/**
* Subtype 0x0003 - Response to SNAC_FAMILY_ICQ/0x002, contains an ICQesque packet.
*/
static int
icqresponse(OscarData *od, aim_modsnac_t *snac, ByteStream *bs)
{
GSList *tlvlist;
aim_tlv_t *datatlv;
ByteStream qbs;
guint32 ouruin;
guint16 cmdlen, cmd, reqid;
if (!(tlvlist = aim_tlvlist_read(bs)) || !(datatlv = aim_tlv_gettlv(tlvlist, 0x0001, 1))) {
aim_tlvlist_free(tlvlist);
purple_debug_misc("oscar", "corrupt ICQ response\n");
return 0;
}
byte_stream_init(&qbs, datatlv->value, datatlv->length);
cmdlen = byte_stream_getle16(&qbs);
ouruin = byte_stream_getle32(&qbs);
cmd = byte_stream_getle16(&qbs);
reqid = byte_stream_getle16(&qbs);
purple_debug_misc("oscar", "icq response: %d bytes, %u, 0x%04x, 0x%04x\n", cmdlen, ouruin, cmd, reqid);
if (cmd == 0x07da) { /* information */
guint16 subtype;
GSList *info_ptr;
struct aim_icq_info *info;
subtype = byte_stream_getle16(&qbs);
byte_stream_advance(&qbs, 1); /* 0x0a */
/* find other data from the same request */
info_ptr = g_slist_find_custom(od->icq_info, &reqid, compare_icq_infos);
if (!info_ptr) {
struct aim_icq_info *new_info = (struct aim_icq_info *)g_new0(struct aim_icq_info, 1);
new_info->reqid = reqid;
info_ptr = od->icq_info = g_slist_prepend(od->icq_info, new_info);
}
info = info_ptr->data;
switch (subtype) {
case 0x00a0: { /* hide ip status */
/* nothing */
} break;
case 0x00aa: { /* password change status */
/* nothing */
} break;
case 0x00c8: { /* general and "home" information */
info->nick = byte_stream_getstr(&qbs, byte_stream_getle16(&qbs));
info->first = byte_stream_getstr(&qbs, byte_stream_getle16(&qbs));
info->last = byte_stream_getstr(&qbs, byte_stream_getle16(&qbs));
info->email = byte_stream_getstr(&qbs, byte_stream_getle16(&qbs));
info->homecity = byte_stream_getstr(&qbs, byte_stream_getle16(&qbs));
info->homestate = byte_stream_getstr(&qbs, byte_stream_getle16(&qbs));
info->homephone = byte_stream_getstr(&qbs, byte_stream_getle16(&qbs));
info->homefax = byte_stream_getstr(&qbs, byte_stream_getle16(&qbs));
info->homeaddr = byte_stream_getstr(&qbs, byte_stream_getle16(&qbs));
info->mobile = byte_stream_getstr(&qbs, byte_stream_getle16(&qbs));
info->homezip = byte_stream_getstr(&qbs, byte_stream_getle16(&qbs));
info->homecountry = byte_stream_getle16(&qbs);
/* 0x0a 00 02 00 */
/* 1 byte timezone? */
/* 1 byte hide email flag? */
} break;
case 0x00dc: { /* personal information */
info->age = byte_stream_getle8(&qbs);
info->unknown = byte_stream_getle8(&qbs);
info->gender = byte_stream_getle8(&qbs); /* Not specified=0x00, Female=0x01, Male=0x02 */
info->personalwebpage = byte_stream_getstr(&qbs, byte_stream_getle16(&qbs));
info->birthyear = byte_stream_getle16(&qbs);
info->birthmonth = byte_stream_getle8(&qbs);
info->birthday = byte_stream_getle8(&qbs);
info->language1 = byte_stream_getle8(&qbs);
info->language2 = byte_stream_getle8(&qbs);
info->language3 = byte_stream_getle8(&qbs);
/* 0x00 00 01 00 00 01 00 00 00 00 00 */
} break;
case 0x00d2: { /* work information */
info->workcity = byte_stream_getstr(&qbs, byte_stream_getle16(&qbs));
info->workstate = byte_stream_getstr(&qbs, byte_stream_getle16(&qbs));
info->workphone = byte_stream_getstr(&qbs, byte_stream_getle16(&qbs));
info->workfax = byte_stream_getstr(&qbs, byte_stream_getle16(&qbs));
info->workaddr = byte_stream_getstr(&qbs, byte_stream_getle16(&qbs));
info->workzip = byte_stream_getstr(&qbs, byte_stream_getle16(&qbs));
info->workcountry = byte_stream_getle16(&qbs);
info->workcompany = byte_stream_getstr(&qbs, byte_stream_getle16(&qbs));
info->workdivision = byte_stream_getstr(&qbs, byte_stream_getle16(&qbs));
info->workposition = byte_stream_getstr(&qbs, byte_stream_getle16(&qbs));
byte_stream_advance(&qbs, 2); /* 0x01 00 */
info->workwebpage = byte_stream_getstr(&qbs, byte_stream_getle16(&qbs));
} break;
case 0x00e6: { /* additional personal information */
info->info = byte_stream_getstr(&qbs, byte_stream_getle16(&qbs)-1);
} break;
case 0x00eb: { /* email address(es) */
int i;
info->numaddresses = byte_stream_getle16(&qbs);
info->email2 = (char **)g_new0(char *, info->numaddresses);
for (i = 0; i < info->numaddresses; i++) {
info->email2[i] = byte_stream_getstr(&qbs, byte_stream_getle16(&qbs));
if (i+1 != info->numaddresses)
byte_stream_advance(&qbs, 1); /* 0x00 */
}
} break;
case 0x00f0: { /* personal interests */
} break;
case 0x00fa: { /* past background and current organizations */
} break;
case 0x0104: { /* alias info */
info->nick = byte_stream_getstr(&qbs, byte_stream_getle16(&qbs));
info->first = byte_stream_getstr(&qbs, byte_stream_getle16(&qbs));
info->last = byte_stream_getstr(&qbs, byte_stream_getle16(&qbs));
byte_stream_advance(&qbs, byte_stream_getle16(&qbs)); /* email address? */
/* Then 0x00 02 00 */
} break;
case 0x010e: { /* unknown */
/* 0x00 00 */
} break;
case 0x019a: { /* simple info */
byte_stream_advance(&qbs, 2);
info->uin = byte_stream_getle32(&qbs);
info->nick = byte_stream_getstr(&qbs, byte_stream_getle16(&qbs));
info->first = byte_stream_getstr(&qbs, byte_stream_getle16(&qbs));
info->last = byte_stream_getstr(&qbs, byte_stream_getle16(&qbs));
info->email = byte_stream_getstr(&qbs, byte_stream_getle16(&qbs));
/* Then 0x00 02 00 00 00 00 00 */
} break;
/* status note title and send request for status note text */
case 0x0fb4: {
GSList *tlvlist;
aim_tlv_t *tlv;
FlapConnection *conn;
char *uin = NULL;
char *status_note_title = NULL;
conn = flap_connection_findbygroup(od, 0x0004);
if (conn == NULL)
{
purple_debug_misc("oscar", "icq/0x0fb4: flap connection was not found.\n");
break;
}
byte_stream_advance(&qbs, 0x02); /* length */
byte_stream_advance(&qbs, 0x2f); /* unknown stuff */
tlvlist = aim_tlvlist_read(&qbs);
tlv = aim_tlv_gettlv(tlvlist, 0x0032, 1);
if (tlv != NULL)
/* Get user number */
uin = aim_tlv_getvalue_as_string(tlv);
tlv = aim_tlv_gettlv(tlvlist, 0x0226, 1);
if (tlv != NULL)
/* Get status note title */
status_note_title = aim_tlv_getvalue_as_string(tlv);
aim_tlvlist_free(tlvlist);
if (uin == NULL || status_note_title == NULL)
{
purple_debug_misc("oscar", "icq/0x0fb4: uin or "
"status_note_title was not found\n");
g_free(uin);
g_free(status_note_title);
break;
}
if (status_note_title[0] == '\0')
{
PurpleAccount *account;
PurpleBuddy *buddy;
PurplePresence *presence;
PurpleStatus *status;
account = purple_connection_get_account(od->gc);
buddy = purple_find_buddy(account, uin);
presence = purple_buddy_get_presence(buddy);
status = purple_presence_get_active_status(presence);
purple_prpl_got_user_status(account, uin,
purple_status_get_id(status),
"message", NULL, NULL);
g_free(status_note_title);
}
else
{
struct aim_icq_info *info;
ByteStream bs;
guint32 bslen;
aim_snacid_t snacid;
guchar cookie[8];
info = g_new0(struct aim_icq_info, 1);
bslen = 13 + strlen(uin) + 30 + 6 + 4 + 55 + 85 + 4;
byte_stream_new(&bs, 4 + bslen);
snacid = aim_cachesnac(od, 0x0004, 0x0006, 0x0000, NULL, 0);
aim_icbm_makecookie(cookie);
byte_stream_putraw(&bs, cookie, 8); /* ICBM cookie */
byte_stream_put16(&bs, 0x0002); /* message channel */
byte_stream_put8(&bs, strlen(uin)); /* uin */
byte_stream_putstr(&bs, uin);
byte_stream_put16(&bs, 0x0005); /* rendez vous data */
byte_stream_put16(&bs, 0x00b2);
byte_stream_put16(&bs, 0x0000); /* request */
byte_stream_putraw(&bs, cookie, 8); /* ICBM cookie */
byte_stream_put32(&bs, 0x09461349); /* ICQ server relaying */
byte_stream_put16(&bs, 0x4c7f);
byte_stream_put16(&bs, 0x11d1);
byte_stream_put32(&bs, 0x82224445);
byte_stream_put32(&bs, 0x53540000);
byte_stream_put16(&bs, 0x000a); /* unknown TLV */
byte_stream_put16(&bs, 0x0002);
byte_stream_put16(&bs, 0x0001);
byte_stream_put16(&bs, 0x000f); /* unknown TLV */
byte_stream_put16(&bs, 0x0000);
byte_stream_put16(&bs, 0x2711); /* extended data */
byte_stream_put16(&bs, 0x008a);
byte_stream_putle16(&bs, 0x001b); /* length */
byte_stream_putle16(&bs, 0x0009); /* version */
byte_stream_putle32(&bs, 0x00000000); /* plugin: none */
byte_stream_putle32(&bs, 0x00000000);
byte_stream_putle32(&bs, 0x00000000);
byte_stream_putle32(&bs, 0x00000000);
byte_stream_putle16(&bs, 0x0000); /* unknown */
byte_stream_putle32(&bs, 0x00000000); /* client capabilities flags */
byte_stream_put8(&bs, 0x00); /* unknown */
byte_stream_putle16(&bs, 0x0064); /* downcounter? */
byte_stream_putle16(&bs, 0x000e); /* length */
byte_stream_putle16(&bs, 0x0064); /* downcounter? */
byte_stream_putle32(&bs, 0x00000000); /* unknown */
byte_stream_putle32(&bs, 0x00000000);
byte_stream_putle32(&bs, 0x00000000);
byte_stream_put8(&bs, 0x1a); /* message type: plugin message descibed by text string */
byte_stream_put8(&bs, 0x00); /* message flags */
byte_stream_putle16(&bs, 0x0000); /* status code */
byte_stream_putle16(&bs, 0x0001); /* priority code */
byte_stream_putle16(&bs, 0x0000); /* text length */
byte_stream_put8(&bs, 0x3a); /* message dump */
byte_stream_put32(&bs, 0x00811a18);
byte_stream_put32(&bs, 0xbc0e6c18);
byte_stream_put32(&bs, 0x47a5916f);
byte_stream_put32(&bs, 0x18dcc76f);
byte_stream_put32(&bs, 0x1a010013);
byte_stream_put32(&bs, 0x00000041);
byte_stream_put32(&bs, 0x77617920);
byte_stream_put32(&bs, 0x53746174);
byte_stream_put32(&bs, 0x7573204d);
byte_stream_put32(&bs, 0x65737361);
byte_stream_put32(&bs, 0x67650100);
byte_stream_put32(&bs, 0x00000000);
byte_stream_put32(&bs, 0x00000000);
byte_stream_put32(&bs, 0x00000000);
byte_stream_put32(&bs, 0x00000015);
byte_stream_put32(&bs, 0x00000000);
byte_stream_put32(&bs, 0x0000000d);
byte_stream_put32(&bs, 0x00000074);
byte_stream_put32(&bs, 0x6578742f);
byte_stream_put32(&bs, 0x782d616f);
byte_stream_put32(&bs, 0x6c727466);
byte_stream_put16(&bs, 0x0003); /* server ACK requested */
byte_stream_put16(&bs, 0x0000);
info->uin = atoi(uin);
info->status_note_title = status_note_title;
memcpy(&info->icbm_cookie, cookie, 8);
od->icq_info = g_slist_prepend(od->icq_info, info);
flap_connection_send_snac_with_priority(od, conn, 0x0004, 0x0006, snacid, &bs, FALSE);
byte_stream_destroy(&bs);
}
g_free(uin);
} break;
} /* End switch statement */
if (!(snac->flags & 0x0001)) {
if (subtype != 0x0104)
oscar_user_info_display_icq(od, info);
if (info->uin && info->nick)
gotalias(od, info);
aim_icq_freeinfo(info);
od->icq_info = g_slist_remove(od->icq_info, info);
}
}
aim_tlvlist_free(tlvlist);
return 1;
}
static int
snachandler(OscarData *od, FlapConnection *conn, aim_module_t *mod, FlapFrame *frame, aim_modsnac_t *snac, ByteStream *bs)
{
if (snac->subtype == 0x0001)
return error(od, snac, bs);
else if (snac->subtype == 0x0003)
return icqresponse(od, snac, bs);
return 0;
}
static void
icq_shutdown(OscarData *od, aim_module_t *mod)
{
GSList *cur;
for (cur = od->icq_info; cur; cur = cur->next)
aim_icq_freeinfo(cur->data);
g_slist_free(od->icq_info);
}
int
icq_modfirst(OscarData *od, aim_module_t *mod)
{
mod->family = SNAC_FAMILY_ICQ;
mod->version = 0x0001;
mod->toolid = 0x0110;
mod->toolversion = 0x047c;
mod->flags = 0;
strncpy(mod->name, "icq", sizeof(mod->name));
mod->snachandler = snachandler;
mod->shutdown = icq_shutdown;
return 0;
}