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.
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.
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.
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
22 * Family 0x0015 - Encapsulated ICQ.
29 #define AIM_ICQ_INFO_REQUEST 0x04b2
30 #define AIM_ICQ_ALIAS_REQUEST 0x04ba
33 int compare_icq_infos(gconstpointer a
, gconstpointer b
)
35 const struct aim_icq_info
* aa
= a
;
36 const guint16
* bb
= b
;
37 return aa
->reqid
- *bb
;
40 static void aim_icq_freeinfo(struct aim_icq_info
*info
) {
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
);
55 g_free(info
->homezip
);
56 g_free(info
->personalwebpage
);
58 for (i
= 0; i
< info
->numaddresses
; i
++)
59 g_free(info
->email2
[i
]);
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
);
72 g_free(info
->status_note_title
);
73 g_free(info
->auth_request_reason
);
77 int error(OscarData
*od
, aim_modsnac_t
*error_snac
, ByteStream
*bs
)
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
;
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
);
92 request_type
= original_snac
->data
;
93 original_info_ptr
= g_slist_find_custom(od
->icq_info
, &original_snac
->id
, compare_icq_infos
);
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
);
101 original_info
= original_info_ptr
->data
;
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
);
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
);
115 purple_debug_misc("oscar", "icq: got an error packet with unknown request type %u", *request_type
);
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
);
127 aim_icq_setsecurity(OscarData
*od
, gboolean auth_required
, gboolean webaware
)
129 FlapConnection
*conn
;
134 if (!od
|| !(conn
= flap_connection_findbygroup(od
, SNAC_FAMILY_ICQ
)))
137 bslen
= 2+4+2+2+2+2+2+1+1+1+1+1+1;
139 byte_stream_new(&bs
, 4 + bslen
);
141 snacid
= aim_cachesnac(od
, SNAC_FAMILY_ICQ
, 0x0002, 0x0000, NULL
, 0);
143 /* For simplicity, don't bother using a tlvlist */
144 byte_stream_put16(&bs
, 0x0001);
145 byte_stream_put16(&bs
, bslen
);
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
);
161 flap_connection_send_snac(od
, conn
, SNAC_FAMILY_ICQ
, 0x0002, snacid
, &bs
);
163 byte_stream_destroy(&bs
);
169 * Change your ICQ password.
171 * @param od The oscar session
172 * @param passwd The new password. If this is longer than 8 characters it
174 * @return Return 0 if no errors, otherwise return the error number.
176 int aim_icq_changepasswd(OscarData
*od
, const char *passwd
)
178 FlapConnection
*conn
;
181 int bslen
, passwdlen
;
186 if (!od
|| !(conn
= flap_connection_findbygroup(od
, SNAC_FAMILY_ICQ
)))
189 passwdlen
= strlen(passwd
);
190 if (passwdlen
> MAXICQPASSLEN
)
191 passwdlen
= MAXICQPASSLEN
;
192 bslen
= 2+4+2+2+2+2+passwdlen
+1;
194 byte_stream_new(&bs
, 4 + bslen
);
196 snacid
= aim_cachesnac(od
, SNAC_FAMILY_ICQ
, 0x0002, 0x0000, NULL
, 0);
198 /* For simplicity, don't bother using a tlvlist */
199 byte_stream_put16(&bs
, 0x0001);
200 byte_stream_put16(&bs
, bslen
);
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');
211 flap_connection_send_snac(od
, conn
, SNAC_FAMILY_ICQ
, 0x0002, snacid
, &bs
);
213 byte_stream_destroy(&bs
);
218 int aim_icq_getallinfo(OscarData
*od
, const char *uin
)
220 FlapConnection
*conn
;
224 struct aim_icq_info
*info
;
225 guint16 request_type
= AIM_ICQ_INFO_REQUEST
;
227 if (!uin
|| uin
[0] < '0' || uin
[0] > '9')
230 if (!od
|| !(conn
= flap_connection_findbygroup(od
, SNAC_FAMILY_ICQ
)))
233 bslen
= 2 + 4 + 2 + 2 + 2 + 4;
235 byte_stream_new(&bs
, 4 + bslen
);
237 snacid
= aim_cachesnac(od
, SNAC_FAMILY_ICQ
, 0x0002, 0x0000, &request_type
, sizeof(request_type
));
239 /* For simplicity, don't bother using a tlvlist */
240 byte_stream_put16(&bs
, 0x0001);
241 byte_stream_put16(&bs
, bslen
);
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
));
250 flap_connection_send_snac_with_priority(od
, conn
, SNAC_FAMILY_ICQ
, 0x0002, snacid
, &bs
, FALSE
);
252 byte_stream_destroy(&bs
);
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
);
263 int aim_icq_getalias(OscarData
*od
, const char *uin
, gboolean for_auth_request
, char *auth_request_reason
)
265 FlapConnection
*conn
;
269 struct aim_icq_info
*info
;
270 guint16 request_type
= AIM_ICQ_ALIAS_REQUEST
;
272 if (!uin
|| uin
[0] < '0' || uin
[0] > '9')
275 if (!od
|| !(conn
= flap_connection_findbygroup(od
, SNAC_FAMILY_ICQ
)))
278 purple_debug_info("oscar", "Requesting ICQ alias for %s\n", uin
);
280 bslen
= 2 + 4 + 2 + 2 + 2 + 4;
282 byte_stream_new(&bs
, 4 + bslen
);
284 snacid
= aim_cachesnac(od
, SNAC_FAMILY_ICQ
, 0x0002, 0x0000, &request_type
, sizeof(request_type
));
286 /* For simplicity, don't bother using a tlvlist */
287 byte_stream_put16(&bs
, 0x0001);
288 byte_stream_put16(&bs
, bslen
);
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
));
297 flap_connection_send_snac_with_priority(od
, conn
, SNAC_FAMILY_ICQ
, 0x0002, snacid
, &bs
, FALSE
);
299 byte_stream_destroy(&bs
);
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
);
313 * Send an SMS message. This is the non-US way. The US-way is to IM
314 * their cell phone number (+19195551234).
316 * We basically construct and send an XML message. The format is:
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>
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.
330 int aim_icq_sendsms(OscarData
*od
, const char *name
, const char *msg
, const char *alias
)
332 FlapConnection
*conn
;
333 PurpleAccount
*account
;
338 const char *timestr
, *username
;
343 if (!od
|| !(conn
= flap_connection_findbygroup(od
, SNAC_FAMILY_ICQ
)))
346 if (!name
|| !msg
|| !alias
)
349 account
= purple_connection_get_account(od
->gc
);
350 username
= purple_account_get_username(account
);
354 timestr
= purple_utf8_strftime("%a, %d %b %Y %T %Z", tm
);
356 stripped
= purple_markup_strip_html(msg
);
358 /* The length of xml included the null terminating character */
359 xmllen
= 209 + strlen(name
) + strlen(stripped
) + strlen(username
) + strlen(alias
) + strlen(timestr
) + 1;
361 xml
= g_new(char, xmllen
);
362 snprintf(xml
, xmllen
, "<icq_sms_message>"
363 "<destination>%s</destination>"
365 "<codepage>1252</codepage>"
366 "<senders_UIN>%s</senders_UIN>"
367 "<senders_name>%s</senders_name>"
368 "<delivery_receipt>Yes</delivery_receipt>"
370 "</icq_sms_message>",
371 name
, stripped
, username
, alias
, timestr
);
375 byte_stream_new(&bs
, 4 + bslen
);
377 snacid
= aim_cachesnac(od
, SNAC_FAMILY_ICQ
, 0x0002, 0x0000, NULL
, 0);
379 /* For simplicity, don't bother using a tlvlist */
380 byte_stream_put16(&bs
, 0x0001);
381 byte_stream_put16(&bs
, bslen
);
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. */
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);
397 byte_stream_put16(&bs
, 0x0000);
398 byte_stream_put16(&bs
, xmllen
);
399 byte_stream_putstr(&bs
, xml
);
400 byte_stream_put8(&bs
, 0x00);
402 flap_connection_send_snac(od
, conn
, SNAC_FAMILY_ICQ
, 0x0002, snacid
, &bs
);
404 byte_stream_destroy(&bs
);
413 gotalias(OscarData
*od
, struct aim_icq_info
*info
)
415 PurpleConnection
*gc
= od
->gc
;
416 PurpleAccount
*account
= purple_connection_get_account(gc
);
418 gchar
*utf8
= oscar_utf8_try_convert(account
, od
, info
->nick
);
420 if (info
->for_auth_request
) {
421 oscar_auth_recvrequest(gc
, g_strdup_printf("%u", info
->uin
), utf8
, info
->auth_request_reason
);
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
);
436 * Subtype 0x0003 - Response to SNAC_FAMILY_ICQ/0x002, contains an ICQesque packet.
439 icqresponse(OscarData
*od
, aim_modsnac_t
*snac
, ByteStream
*bs
)
445 guint16 cmdlen
, cmd
, reqid
;
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");
453 byte_stream_init(&qbs
, datatlv
->value
, datatlv
->length
);
455 cmdlen
= byte_stream_getle16(&qbs
);
456 ouruin
= byte_stream_getle32(&qbs
);
457 cmd
= byte_stream_getle16(&qbs
);
458 reqid
= byte_stream_getle16(&qbs
);
460 purple_debug_misc("oscar", "icq response: %d bytes, %u, 0x%04x, 0x%04x\n", cmdlen
, ouruin
, cmd
, reqid
);
462 if (cmd
== 0x07da) { /* information */
465 struct aim_icq_info
*info
;
467 subtype
= byte_stream_getle16(&qbs
);
468 byte_stream_advance(&qbs
, 1); /* 0x0a */
470 /* find other data from the same request */
471 info_ptr
= g_slist_find_custom(od
->icq_info
, &reqid
, compare_icq_infos
);
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
);
478 info
= info_ptr
->data
;
480 case 0x00a0: { /* hide ip status */
484 case 0x00aa: { /* password change status */
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
);
502 /* 1 byte timezone? */
503 /* 1 byte hide email flag? */
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 */
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
));
535 case 0x00e6: { /* additional personal information */
536 info
->info
= byte_stream_getstr(&qbs
, byte_stream_getle16(&qbs
)-1);
539 case 0x00eb: { /* email address(es) */
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 */
550 case 0x00f0: { /* personal interests */
553 case 0x00fa: { /* past background and current organizations */
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 */
564 case 0x010e: { /* unknown */
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 */
578 /* status note title and send request for status note text */
582 FlapConnection
*conn
;
584 char *status_note_title
= NULL
;
586 conn
= flap_connection_findbygroup(od
, 0x0004);
589 purple_debug_misc("oscar", "icq/0x0fb4: flap connection was not found.\n");
593 byte_stream_advance(&qbs
, 0x02); /* length */
594 byte_stream_advance(&qbs
, 0x2f); /* unknown stuff */
596 tlvlist
= aim_tlvlist_read(&qbs
);
598 tlv
= aim_tlv_gettlv(tlvlist
, 0x0032, 1);
600 /* Get user number */
601 uin
= aim_tlv_getvalue_as_string(tlv
);
603 tlv
= aim_tlv_gettlv(tlvlist
, 0x0226, 1);
605 /* Get status note title */
606 status_note_title
= aim_tlv_getvalue_as_string(tlv
);
608 aim_tlvlist_free(tlvlist
);
610 if (uin
== NULL
|| status_note_title
== NULL
)
612 purple_debug_misc("oscar", "icq/0x0fb4: uin or "
613 "status_note_title was not found\n");
615 g_free(status_note_title
);
619 if (status_note_title
[0] == '\0')
621 PurpleAccount
*account
;
623 PurplePresence
*presence
;
624 PurpleStatus
*status
;
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
);
631 purple_prpl_got_user_status(account
, uin
,
632 purple_status_get_id(status
),
633 "message", NULL
, NULL
);
635 g_free(status_note_title
);
639 struct aim_icq_info
*info
;
645 info
= g_new0(struct aim_icq_info
, 1);
647 bslen
= 13 + strlen(uin
) + 30 + 6 + 4 + 55 + 85 + 4;
648 byte_stream_new(&bs
, 4 + bslen
);
650 snacid
= aim_cachesnac(od
, 0x0004, 0x0006, 0x0000, NULL
, 0);
652 aim_icbm_makecookie(cookie
);
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
);
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);
669 byte_stream_put16(&bs
, 0x000a); /* unknown TLV */
670 byte_stream_put16(&bs
, 0x0002);
671 byte_stream_put16(&bs
, 0x0001);
673 byte_stream_put16(&bs
, 0x000f); /* unknown TLV */
674 byte_stream_put16(&bs
, 0x0000);
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 */
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);
722 byte_stream_put16(&bs
, 0x0003); /* server ACK requested */
723 byte_stream_put16(&bs
, 0x0000);
725 info
->uin
= atoi(uin
);
726 info
->status_note_title
= status_note_title
;
728 memcpy(&info
->icbm_cookie
, cookie
, 8);
730 od
->icq_info
= g_slist_prepend(od
->icq_info
, info
);
732 flap_connection_send_snac_with_priority(od
, conn
, 0x0004, 0x0006, snacid
, &bs
, FALSE
);
734 byte_stream_destroy(&bs
);
741 } /* End switch statement */
743 if (!(snac
->flags
& 0x0001)) {
744 if (subtype
!= 0x0104)
745 oscar_user_info_display_icq(od
, info
);
747 if (info
->uin
&& info
->nick
)
750 aim_icq_freeinfo(info
);
751 od
->icq_info
= g_slist_remove(od
->icq_info
, info
);
755 aim_tlvlist_free(tlvlist
);
761 snachandler(OscarData
*od
, FlapConnection
*conn
, aim_module_t
*mod
, FlapFrame
*frame
, aim_modsnac_t
*snac
, ByteStream
*bs
)
763 if (snac
->subtype
== 0x0001)
764 return error(od
, snac
, bs
);
765 else if (snac
->subtype
== 0x0003)
766 return icqresponse(od
, snac
, bs
);
772 icq_shutdown(OscarData
*od
, aim_module_t
*mod
)
775 for (cur
= od
->icq_info
; cur
; cur
= cur
->next
)
776 aim_icq_freeinfo(cur
->data
);
777 g_slist_free(od
->icq_info
);
781 icq_modfirst(OscarData
*od
, aim_module_t
*mod
)
783 mod
->family
= SNAC_FAMILY_ICQ
;
784 mod
->version
= 0x0001;
785 mod
->toolid
= 0x0110;
786 mod
->toolversion
= 0x047c;
788 strncpy(mod
->name
, "icq", sizeof(mod
->name
));
789 mod
->snachandler
= snachandler
;
790 mod
->shutdown
= icq_shutdown
;