comparison libpurple/protocols/oscar/family_icq.c @ 15374:5fe8042783c1

Rename gtk/ and libgaim/ to pidgin/ and libpurple/
author Sean Egan <seanegan@gmail.com>
date Sat, 20 Jan 2007 02:32:10 +0000
parents
children 32c366eeeb99
comparison
equal deleted inserted replaced
15373:f79e0f4df793 15374:5fe8042783c1
1 /*
2 * Gaim'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., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
19 */
20
21 /*
22 * Family 0x0015 - Encapsulated ICQ.
23 *
24 */
25
26 #include "oscar.h"
27
28 int aim_icq_reqofflinemsgs(OscarData *od)
29 {
30 FlapConnection *conn;
31 FlapFrame *frame;
32 aim_snacid_t snacid;
33 int bslen;
34
35 if (!od || !(conn = flap_connection_findbygroup(od, 0x0015)))
36 return -EINVAL;
37
38 bslen = 2 + 4 + 2 + 2;
39
40 frame = flap_frame_new(od, 0x02, 10 + 4 + bslen);
41
42 snacid = aim_cachesnac(od, 0x0015, 0x0002, 0x0000, NULL, 0);
43 aim_putsnac(&frame->data, 0x0015, 0x0002, 0x0000, snacid);
44
45 /* For simplicity, don't bother using a tlvlist */
46 byte_stream_put16(&frame->data, 0x0001);
47 byte_stream_put16(&frame->data, bslen);
48
49 byte_stream_putle16(&frame->data, bslen - 2);
50 byte_stream_putle32(&frame->data, atoi(od->sn));
51 byte_stream_putle16(&frame->data, 0x003c); /* I command thee. */
52 byte_stream_putle16(&frame->data, snacid); /* eh. */
53
54 flap_connection_send(conn, frame);
55
56 return 0;
57 }
58
59 int aim_icq_ackofflinemsgs(OscarData *od)
60 {
61 FlapConnection *conn;
62 FlapFrame *frame;
63 aim_snacid_t snacid;
64 int bslen;
65
66 if (!od || !(conn = flap_connection_findbygroup(od, 0x0015)))
67 return -EINVAL;
68
69 bslen = 2 + 4 + 2 + 2;
70
71 frame = flap_frame_new(od, 0x02, 10 + 4 + bslen);
72
73 snacid = aim_cachesnac(od, 0x0015, 0x0002, 0x0000, NULL, 0);
74 aim_putsnac(&frame->data, 0x0015, 0x0002, 0x0000, snacid);
75
76 /* For simplicity, don't bother using a tlvlist */
77 byte_stream_put16(&frame->data, 0x0001);
78 byte_stream_put16(&frame->data, bslen);
79
80 byte_stream_putle16(&frame->data, bslen - 2);
81 byte_stream_putle32(&frame->data, atoi(od->sn));
82 byte_stream_putle16(&frame->data, 0x003e); /* I command thee. */
83 byte_stream_putle16(&frame->data, snacid); /* eh. */
84
85 flap_connection_send(conn, frame);
86
87 return 0;
88 }
89
90 int
91 aim_icq_setsecurity(OscarData *od, gboolean auth_required, gboolean webaware)
92 {
93 FlapConnection *conn;
94 FlapFrame *frame;
95 aim_snacid_t snacid;
96 int bslen;
97
98 if (!od || !(conn = flap_connection_findbygroup(od, 0x0015)))
99 return -EINVAL;
100
101 bslen = 2+4+2+2+2+2+2+1+1+1+1+1+1;
102
103 frame = flap_frame_new(od, 0x02, 10 + 4 + bslen);
104
105 snacid = aim_cachesnac(od, 0x0015, 0x0002, 0x0000, NULL, 0);
106 aim_putsnac(&frame->data, 0x0015, 0x0002, 0x0000, snacid);
107
108 /* For simplicity, don't bother using a tlvlist */
109 byte_stream_put16(&frame->data, 0x0001);
110 byte_stream_put16(&frame->data, bslen);
111
112 byte_stream_putle16(&frame->data, bslen - 2);
113 byte_stream_putle32(&frame->data, atoi(od->sn));
114 byte_stream_putle16(&frame->data, 0x07d0); /* I command thee. */
115 byte_stream_putle16(&frame->data, snacid); /* eh. */
116 byte_stream_putle16(&frame->data, 0x0c3a); /* shrug. */
117 byte_stream_putle16(&frame->data, 0x030c);
118 byte_stream_putle16(&frame->data, 0x0001);
119 byte_stream_putle8(&frame->data, webaware);
120 byte_stream_putle8(&frame->data, 0xf8);
121 byte_stream_putle8(&frame->data, 0x02);
122 byte_stream_putle8(&frame->data, 0x01);
123 byte_stream_putle8(&frame->data, 0x00);
124 byte_stream_putle8(&frame->data, !auth_required);
125
126 flap_connection_send(conn, frame);
127
128 return 0;
129 }
130
131 /**
132 * Change your ICQ password.
133 *
134 * @param od The oscar session
135 * @param passwd The new password. If this is longer than 8 characters it
136 * will be truncated.
137 * @return Return 0 if no errors, otherwise return the error number.
138 */
139 int aim_icq_changepasswd(OscarData *od, const char *passwd)
140 {
141 FlapConnection *conn;
142 FlapFrame *frame;
143 aim_snacid_t snacid;
144 int bslen, passwdlen;
145
146 if (!passwd)
147 return -EINVAL;
148
149 if (!od || !(conn = flap_connection_findbygroup(od, 0x0015)))
150 return -EINVAL;
151
152 passwdlen = strlen(passwd);
153 if (passwdlen > MAXICQPASSLEN)
154 passwdlen = MAXICQPASSLEN;
155 bslen = 2+4+2+2+2+2+passwdlen+1;
156
157 frame = flap_frame_new(od, 0x02, 10 + 4 + bslen);
158
159 snacid = aim_cachesnac(od, 0x0015, 0x0002, 0x0000, NULL, 0);
160 aim_putsnac(&frame->data, 0x0015, 0x0002, 0x0000, snacid);
161
162 /* For simplicity, don't bother using a tlvlist */
163 byte_stream_put16(&frame->data, 0x0001);
164 byte_stream_put16(&frame->data, bslen);
165
166 byte_stream_putle16(&frame->data, bslen - 2);
167 byte_stream_putle32(&frame->data, atoi(od->sn));
168 byte_stream_putle16(&frame->data, 0x07d0); /* I command thee. */
169 byte_stream_putle16(&frame->data, snacid); /* eh. */
170 byte_stream_putle16(&frame->data, 0x042e); /* shrug. */
171 byte_stream_putle16(&frame->data, passwdlen+1);
172 byte_stream_putstr(&frame->data, passwd);
173 byte_stream_putle8(&frame->data, '\0');
174
175 flap_connection_send(conn, frame);
176
177 return 0;
178 }
179
180 int aim_icq_getallinfo(OscarData *od, const char *uin)
181 {
182 FlapConnection *conn;
183 FlapFrame *frame;
184 aim_snacid_t snacid;
185 int bslen;
186 struct aim_icq_info *info;
187
188 if (!uin || uin[0] < '0' || uin[0] > '9')
189 return -EINVAL;
190
191 if (!od || !(conn = flap_connection_findbygroup(od, 0x0015)))
192 return -EINVAL;
193
194 bslen = 2 + 4 + 2 + 2 + 2 + 4;
195
196 frame = flap_frame_new(od, 0x02, 10 + 4 + bslen);
197
198 snacid = aim_cachesnac(od, 0x0015, 0x0002, 0x0000, NULL, 0);
199 aim_putsnac(&frame->data, 0x0015, 0x0002, 0x0000, snacid);
200
201 /* For simplicity, don't bother using a tlvlist */
202 byte_stream_put16(&frame->data, 0x0001);
203 byte_stream_put16(&frame->data, bslen);
204
205 byte_stream_putle16(&frame->data, bslen - 2);
206 byte_stream_putle32(&frame->data, atoi(od->sn));
207 byte_stream_putle16(&frame->data, 0x07d0); /* I command thee. */
208 byte_stream_putle16(&frame->data, snacid); /* eh. */
209 byte_stream_putle16(&frame->data, 0x04b2); /* shrug. */
210 byte_stream_putle32(&frame->data, atoi(uin));
211
212 flap_connection_send(conn, frame);
213
214 /* Keep track of this request and the ICQ number and request ID */
215 info = (struct aim_icq_info *)calloc(1, sizeof(struct aim_icq_info));
216 info->reqid = snacid;
217 info->uin = atoi(uin);
218 info->next = od->icq_info;
219 od->icq_info = info;
220
221 return 0;
222 }
223
224 int aim_icq_getalias(OscarData *od, const char *uin)
225 {
226 FlapConnection *conn;
227 FlapFrame *frame;
228 aim_snacid_t snacid;
229 int bslen;
230 struct aim_icq_info *info;
231
232 if (!uin || uin[0] < '0' || uin[0] > '9')
233 return -EINVAL;
234
235 if (!od || !(conn = flap_connection_findbygroup(od, 0x0015)))
236 return -EINVAL;
237
238 bslen = 2 + 4 + 2 + 2 + 2 + 4;
239
240 frame = flap_frame_new(od, 0x02, 10 + 4 + bslen);
241
242 snacid = aim_cachesnac(od, 0x0015, 0x0002, 0x0000, NULL, 0);
243 aim_putsnac(&frame->data, 0x0015, 0x0002, 0x0000, snacid);
244
245 /* For simplicity, don't bother using a tlvlist */
246 byte_stream_put16(&frame->data, 0x0001);
247 byte_stream_put16(&frame->data, bslen);
248
249 byte_stream_putle16(&frame->data, bslen - 2);
250 byte_stream_putle32(&frame->data, atoi(od->sn));
251 byte_stream_putle16(&frame->data, 0x07d0); /* I command thee. */
252 byte_stream_putle16(&frame->data, snacid); /* eh. */
253 byte_stream_putle16(&frame->data, 0x04ba); /* shrug. */
254 byte_stream_putle32(&frame->data, atoi(uin));
255
256 flap_connection_send(conn, frame);
257
258 /* Keep track of this request and the ICQ number and request ID */
259 info = (struct aim_icq_info *)calloc(1, sizeof(struct aim_icq_info));
260 info->reqid = snacid;
261 info->uin = atoi(uin);
262 info->next = od->icq_info;
263 od->icq_info = info;
264
265 return 0;
266 }
267
268 int aim_icq_getsimpleinfo(OscarData *od, const char *uin)
269 {
270 FlapConnection *conn;
271 FlapFrame *frame;
272 aim_snacid_t snacid;
273 int bslen;
274
275 if (!uin || uin[0] < '0' || uin[0] > '9')
276 return -EINVAL;
277
278 if (!od || !(conn = flap_connection_findbygroup(od, 0x0015)))
279 return -EINVAL;
280
281 bslen = 2 + 4 + 2 + 2 + 2 + 4;
282
283 frame = flap_frame_new(od, 0x02, 10 + 4 + bslen);
284
285 snacid = aim_cachesnac(od, 0x0015, 0x0002, 0x0000, NULL, 0);
286 aim_putsnac(&frame->data, 0x0015, 0x0002, 0x0000, snacid);
287
288 /* For simplicity, don't bother using a tlvlist */
289 byte_stream_put16(&frame->data, 0x0001);
290 byte_stream_put16(&frame->data, bslen);
291
292 byte_stream_putle16(&frame->data, bslen - 2);
293 byte_stream_putle32(&frame->data, atoi(od->sn));
294 byte_stream_putle16(&frame->data, 0x07d0); /* I command thee. */
295 byte_stream_putle16(&frame->data, snacid); /* eh. */
296 byte_stream_putle16(&frame->data, 0x051f); /* shrug. */
297 byte_stream_putle32(&frame->data, atoi(uin));
298
299 flap_connection_send(conn, frame);
300
301 return 0;
302 }
303
304 #if 0
305 int aim_icq_sendxmlreq(OscarData *od, const char *xml)
306 {
307 FlapConnection *conn;
308 FlapFrame *frame;
309 aim_snacid_t snacid;
310 int bslen;
311
312 if (!xml || !strlen(xml))
313 return -EINVAL;
314
315 if (!od || !(conn = flap_connection_findbygroup(od, 0x0015)))
316 return -EINVAL;
317
318 bslen = 2 + 10 + 2 + strlen(xml) + 1;
319
320 frame = flap_frame_new(od, 0x02, 10 + 4 + bslen);
321
322 snacid = aim_cachesnac(od, 0x0015, 0x0002, 0x0000, NULL, 0);
323 aim_putsnac(&frame->data, 0x0015, 0x0002, 0x0000, snacid);
324
325 /* For simplicity, don't bother using a tlvlist */
326 byte_stream_put16(&frame->data, 0x0001);
327 byte_stream_put16(&frame->data, bslen);
328
329 byte_stream_putle16(&frame->data, bslen - 2);
330 byte_stream_putle32(&frame->data, atoi(od->sn));
331 byte_stream_putle16(&frame->data, 0x07d0); /* I command thee. */
332 byte_stream_putle16(&frame->data, snacid); /* eh. */
333 byte_stream_putle16(&frame->data, 0x0998); /* shrug. */
334 byte_stream_putle16(&frame->data, strlen(xml) + 1);
335 byte_stream_putraw(&frame->data, (guint8 *)xml, strlen(xml) + 1);
336
337 flap_connection_send(conn, frame);
338
339 return 0;
340 }
341 #endif
342
343 #if 0
344 /*
345 * Send an SMS message. This is the non-US way. The US-way is to IM
346 * their cell phone number (+19195551234).
347 *
348 * We basically construct and send an XML message. The format is:
349 * <icq_sms_message>
350 * <destination>full_phone_without_leading_+</destination>
351 * <text>message</text>
352 * <codepage>1252</codepage>
353 * <senders_UIN>self_uin</senders_UIN>
354 * <senders_name>self_name</senders_name>
355 * <delivery_receipt>Yes|No</delivery_receipt>
356 * <time>Wkd, DD Mmm YYYY HH:MM:SS TMZ</time>
357 * </icq_sms_message>
358 *
359 * Yeah hi Peter, whaaaat's happening. If there's any way to use
360 * a codepage other than 1252 that would be great. Thaaaanks.
361 */
362 int aim_icq_sendsms(OscarData *od, const char *name, const char *msg, const char *alias)
363 {
364 FlapConnection *conn;
365 FlapFrame *frame;
366 aim_snacid_t snacid;
367 int bslen, xmllen;
368 char *xml;
369 const char *timestr;
370 time_t t;
371 struct tm *tm;
372
373 if (!od || !(conn = flap_connection_findbygroup(od, 0x0015)))
374 return -EINVAL;
375
376 if (!name || !msg || !alias)
377 return -EINVAL;
378
379 time(&t);
380 tm = gmtime(&t);
381 timestr = gaim_utf8_strftime("%a, %d %b %Y %T %Z", tm);
382
383 /* The length of xml included the null terminating character */
384 xmllen = 225 + strlen(name) + strlen(msg) + strlen(od->sn) + strlen(alias) + strlen(timestr) + 1;
385
386 xml = g_new(char, xmllen);
387 snprintf(xml, xmllen, "<icq_sms_message>\n"
388 "\t<destination>%s</destination>\n"
389 "\t<text>%s</text>\n"
390 "\t<codepage>1252</codepage>\n"
391 "\t<senders_UIN>%s</senders_UIN>\n"
392 "\t<senders_name>%s</senders_name>\n"
393 "\t<delivery_receipt>Yes</delivery_receipt>\n"
394 "\t<time>%s</time>\n"
395 "</icq_sms_message>\n",
396 name, msg, od->sn, alias, timestr);
397
398 bslen = 37 + xmllen;
399
400 frame = flap_frame_new(od, 0x02, 10 + 4 + bslen);
401
402 snacid = aim_cachesnac(od, 0x0015, 0x0002, 0x0000, NULL, 0);
403 aim_putsnac(&frame->data, 0x0015, 0x0002, 0x0000, snacid);
404
405 /* For simplicity, don't bother using a tlvlist */
406 byte_stream_put16(&frame->data, 0x0001);
407 byte_stream_put16(&frame->data, bslen);
408
409 byte_stream_putle16(&frame->data, bslen - 2);
410 byte_stream_putle32(&frame->data, atoi(od->sn));
411 byte_stream_putle16(&frame->data, 0x07d0); /* I command thee. */
412 byte_stream_putle16(&frame->data, snacid); /* eh. */
413
414 /* From libicq200-0.3.2/src/SNAC-SRV.cpp */
415 byte_stream_putle16(&frame->data, 0x8214);
416 byte_stream_put16(&frame->data, 0x0001);
417 byte_stream_put16(&frame->data, 0x0016);
418 byte_stream_put32(&frame->data, 0x00000000);
419 byte_stream_put32(&frame->data, 0x00000000);
420 byte_stream_put32(&frame->data, 0x00000000);
421 byte_stream_put32(&frame->data, 0x00000000);
422
423 byte_stream_put16(&frame->data, 0x0000);
424 byte_stream_put16(&frame->data, xmllen);
425 byte_stream_putstr(&frame->data, xml);
426
427 flap_connection_send(conn, frame);
428
429 free(xml);
430
431 return 0;
432 }
433 #endif
434
435 static void aim_icq_freeinfo(struct aim_icq_info *info) {
436 int i;
437
438 if (!info)
439 return;
440 free(info->nick);
441 free(info->first);
442 free(info->last);
443 free(info->email);
444 free(info->homecity);
445 free(info->homestate);
446 free(info->homephone);
447 free(info->homefax);
448 free(info->homeaddr);
449 free(info->mobile);
450 free(info->homezip);
451 free(info->personalwebpage);
452 if (info->email2)
453 for (i = 0; i < info->numaddresses; i++)
454 free(info->email2[i]);
455 free(info->email2);
456 free(info->workcity);
457 free(info->workstate);
458 free(info->workphone);
459 free(info->workfax);
460 free(info->workaddr);
461 free(info->workzip);
462 free(info->workcompany);
463 free(info->workdivision);
464 free(info->workposition);
465 free(info->workwebpage);
466 free(info->info);
467 free(info);
468 }
469
470 /**
471 * Subtype 0x0003 - Response to 0x0015/0x002, contains an ICQesque packet.
472 */
473 static int
474 icqresponse(OscarData *od, FlapConnection *conn, aim_module_t *mod, FlapFrame *frame, aim_modsnac_t *snac, ByteStream *bs)
475 {
476 int ret = 0;
477 aim_tlvlist_t *tl;
478 aim_tlv_t *datatlv;
479 ByteStream qbs;
480 guint32 ouruin;
481 guint16 cmdlen, cmd, reqid;
482
483 if (!(tl = aim_tlvlist_read(bs)) || !(datatlv = aim_tlv_gettlv(tl, 0x0001, 1))) {
484 aim_tlvlist_free(&tl);
485 gaim_debug_misc("oscar", "corrupt ICQ response\n");
486 return 0;
487 }
488
489 byte_stream_init(&qbs, datatlv->value, datatlv->length);
490
491 cmdlen = byte_stream_getle16(&qbs);
492 ouruin = byte_stream_getle32(&qbs);
493 cmd = byte_stream_getle16(&qbs);
494 reqid = byte_stream_getle16(&qbs);
495
496 gaim_debug_misc("oscar", "icq response: %d bytes, %ld, 0x%04x, 0x%04x\n", cmdlen, ouruin, cmd, reqid);
497
498 if (cmd == 0x0041) { /* offline message */
499 struct aim_icq_offlinemsg msg;
500 aim_rxcallback_t userfunc;
501
502 memset(&msg, 0, sizeof(msg));
503
504 msg.sender = byte_stream_getle32(&qbs);
505 msg.year = byte_stream_getle16(&qbs);
506 msg.month = byte_stream_getle8(&qbs);
507 msg.day = byte_stream_getle8(&qbs);
508 msg.hour = byte_stream_getle8(&qbs);
509 msg.minute = byte_stream_getle8(&qbs);
510 msg.type = byte_stream_getle8(&qbs);
511 msg.flags = byte_stream_getle8(&qbs);
512 msg.msglen = byte_stream_getle16(&qbs);
513 msg.msg = byte_stream_getstr(&qbs, msg.msglen);
514
515 if ((userfunc = aim_callhandler(od, SNAC_FAMILY_ICQ, SNAC_SUBTYPE_ICQ_OFFLINEMSG)))
516 ret = userfunc(od, conn, frame, &msg);
517
518 free(msg.msg);
519
520 } else if (cmd == 0x0042) {
521 aim_rxcallback_t userfunc;
522
523 if ((userfunc = aim_callhandler(od, SNAC_FAMILY_ICQ, SNAC_SUBTYPE_ICQ_OFFLINEMSGCOMPLETE)))
524 ret = userfunc(od, conn, frame);
525
526 } else if (cmd == 0x07da) { /* information */
527 guint16 subtype;
528 struct aim_icq_info *info;
529 aim_rxcallback_t userfunc;
530
531 subtype = byte_stream_getle16(&qbs);
532 byte_stream_advance(&qbs, 1); /* 0x0a */
533
534 /* find other data from the same request */
535 for (info = od->icq_info; info && (info->reqid != reqid); info = info->next);
536 if (!info) {
537 info = (struct aim_icq_info *)calloc(1, sizeof(struct aim_icq_info));
538 info->reqid = reqid;
539 info->next = od->icq_info;
540 od->icq_info = info;
541 }
542
543 switch (subtype) {
544 case 0x00a0: { /* hide ip status */
545 /* nothing */
546 } break;
547
548 case 0x00aa: { /* password change status */
549 /* nothing */
550 } break;
551
552 case 0x00c8: { /* general and "home" information */
553 info->nick = byte_stream_getstr(&qbs, byte_stream_getle16(&qbs));
554 info->first = byte_stream_getstr(&qbs, byte_stream_getle16(&qbs));
555 info->last = byte_stream_getstr(&qbs, byte_stream_getle16(&qbs));
556 info->email = byte_stream_getstr(&qbs, byte_stream_getle16(&qbs));
557 info->homecity = byte_stream_getstr(&qbs, byte_stream_getle16(&qbs));
558 info->homestate = byte_stream_getstr(&qbs, byte_stream_getle16(&qbs));
559 info->homephone = byte_stream_getstr(&qbs, byte_stream_getle16(&qbs));
560 info->homefax = byte_stream_getstr(&qbs, byte_stream_getle16(&qbs));
561 info->homeaddr = byte_stream_getstr(&qbs, byte_stream_getle16(&qbs));
562 info->mobile = byte_stream_getstr(&qbs, byte_stream_getle16(&qbs));
563 info->homezip = byte_stream_getstr(&qbs, byte_stream_getle16(&qbs));
564 info->homecountry = byte_stream_getle16(&qbs);
565 /* 0x0a 00 02 00 */
566 /* 1 byte timezone? */
567 /* 1 byte hide email flag? */
568 } break;
569
570 case 0x00dc: { /* personal information */
571 info->age = byte_stream_getle8(&qbs);
572 info->unknown = byte_stream_getle8(&qbs);
573 info->gender = byte_stream_getle8(&qbs); /* Not specified=0x00, Female=0x01, Male=0x02 */
574 info->personalwebpage = byte_stream_getstr(&qbs, byte_stream_getle16(&qbs));
575 info->birthyear = byte_stream_getle16(&qbs);
576 info->birthmonth = byte_stream_getle8(&qbs);
577 info->birthday = byte_stream_getle8(&qbs);
578 info->language1 = byte_stream_getle8(&qbs);
579 info->language2 = byte_stream_getle8(&qbs);
580 info->language3 = byte_stream_getle8(&qbs);
581 /* 0x00 00 01 00 00 01 00 00 00 00 00 */
582 } break;
583
584 case 0x00d2: { /* work information */
585 info->workcity = byte_stream_getstr(&qbs, byte_stream_getle16(&qbs));
586 info->workstate = byte_stream_getstr(&qbs, byte_stream_getle16(&qbs));
587 info->workphone = byte_stream_getstr(&qbs, byte_stream_getle16(&qbs));
588 info->workfax = byte_stream_getstr(&qbs, byte_stream_getle16(&qbs));
589 info->workaddr = byte_stream_getstr(&qbs, byte_stream_getle16(&qbs));
590 info->workzip = byte_stream_getstr(&qbs, byte_stream_getle16(&qbs));
591 info->workcountry = byte_stream_getle16(&qbs);
592 info->workcompany = byte_stream_getstr(&qbs, byte_stream_getle16(&qbs));
593 info->workdivision = byte_stream_getstr(&qbs, byte_stream_getle16(&qbs));
594 info->workposition = byte_stream_getstr(&qbs, byte_stream_getle16(&qbs));
595 byte_stream_advance(&qbs, 2); /* 0x01 00 */
596 info->workwebpage = byte_stream_getstr(&qbs, byte_stream_getle16(&qbs));
597 } break;
598
599 case 0x00e6: { /* additional personal information */
600 info->info = byte_stream_getstr(&qbs, byte_stream_getle16(&qbs)-1);
601 } break;
602
603 case 0x00eb: { /* email address(es) */
604 int i;
605 info->numaddresses = byte_stream_getle16(&qbs);
606 info->email2 = (char **)calloc(info->numaddresses, sizeof(char *));
607 for (i = 0; i < info->numaddresses; i++) {
608 info->email2[i] = byte_stream_getstr(&qbs, byte_stream_getle16(&qbs));
609 if (i+1 != info->numaddresses)
610 byte_stream_advance(&qbs, 1); /* 0x00 */
611 }
612 } break;
613
614 case 0x00f0: { /* personal interests */
615 } break;
616
617 case 0x00fa: { /* past background and current organizations */
618 } break;
619
620 case 0x0104: { /* alias info */
621 info->nick = byte_stream_getstr(&qbs, byte_stream_getle16(&qbs));
622 info->first = byte_stream_getstr(&qbs, byte_stream_getle16(&qbs));
623 info->last = byte_stream_getstr(&qbs, byte_stream_getle16(&qbs));
624 byte_stream_advance(&qbs, byte_stream_getle16(&qbs)); /* email address? */
625 /* Then 0x00 02 00 */
626 } break;
627
628 case 0x010e: { /* unknown */
629 /* 0x00 00 */
630 } break;
631
632 case 0x019a: { /* simple info */
633 byte_stream_advance(&qbs, 2);
634 info->uin = byte_stream_getle32(&qbs);
635 info->nick = byte_stream_getstr(&qbs, byte_stream_getle16(&qbs));
636 info->first = byte_stream_getstr(&qbs, byte_stream_getle16(&qbs));
637 info->last = byte_stream_getstr(&qbs, byte_stream_getle16(&qbs));
638 info->email = byte_stream_getstr(&qbs, byte_stream_getle16(&qbs));
639 /* Then 0x00 02 00 00 00 00 00 */
640 } break;
641 } /* End switch statement */
642
643 if (!(snac->flags & 0x0001)) {
644 if (subtype != 0x0104)
645 if ((userfunc = aim_callhandler(od, SNAC_FAMILY_ICQ, SNAC_SUBTYPE_ICQ_INFO)))
646 ret = userfunc(od, conn, frame, info);
647
648 if (info->uin && info->nick)
649 if ((userfunc = aim_callhandler(od, SNAC_FAMILY_ICQ, SNAC_SUBTYPE_ICQ_ALIAS)))
650 ret = userfunc(od, conn, frame, info);
651
652 if (od->icq_info == info) {
653 od->icq_info = info->next;
654 } else {
655 struct aim_icq_info *cur;
656 for (cur=od->icq_info; (cur->next && (cur->next!=info)); cur=cur->next);
657 if (cur->next)
658 cur->next = cur->next->next;
659 }
660 aim_icq_freeinfo(info);
661 }
662 }
663
664 aim_tlvlist_free(&tl);
665
666 return ret;
667 }
668
669 static int
670 snachandler(OscarData *od, FlapConnection *conn, aim_module_t *mod, FlapFrame *frame, aim_modsnac_t *snac, ByteStream *bs)
671 {
672 if (snac->subtype == 0x0003)
673 return icqresponse(od, conn, mod, frame, snac, bs);
674
675 return 0;
676 }
677
678 static void
679 icq_shutdown(OscarData *od, aim_module_t *mod)
680 {
681 struct aim_icq_info *del;
682
683 while (od->icq_info) {
684 del = od->icq_info;
685 od->icq_info = od->icq_info->next;
686 aim_icq_freeinfo(del);
687 }
688
689 return;
690 }
691
692 int
693 icq_modfirst(OscarData *od, aim_module_t *mod)
694 {
695 mod->family = 0x0015;
696 mod->version = 0x0001;
697 mod->toolid = 0x0110;
698 mod->toolversion = 0x047c;
699 mod->flags = 0;
700 strncpy(mod->name, "icq", sizeof(mod->name));
701 mod->snachandler = snachandler;
702 mod->shutdown = icq_shutdown;
703
704 return 0;
705 }