Annotation of ChivanetAimPidgin/oscarprpl/src/c/family_chat.c, revision 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 0x000e - Routines for the Chat service.
! 23: *
! 24: */
! 25:
! 26: #include "oscar.h"
! 27:
! 28: #include <string.h>
! 29:
! 30: /* Stored in the ->internal of chat connections */
! 31: struct chatconnpriv
! 32: {
! 33: guint16 exchange;
! 34: char *name;
! 35: guint16 instance;
! 36: };
! 37:
! 38: void
! 39: flap_connection_destroy_chat(OscarData *od, FlapConnection *conn)
! 40: {
! 41: struct chatconnpriv *ccp = (struct chatconnpriv *)conn->internal;
! 42:
! 43: if (ccp)
! 44: g_free(ccp->name);
! 45: g_free(ccp);
! 46:
! 47: return;
! 48: }
! 49:
! 50: int
! 51: aim_chat_readroominfo(ByteStream *bs, struct aim_chat_roominfo *outinfo)
! 52: {
! 53: if (!bs || !outinfo)
! 54: return 0;
! 55:
! 56: outinfo->exchange = byte_stream_get16(bs);
! 57: outinfo->namelen = byte_stream_get8(bs);
! 58: outinfo->name = (char *)byte_stream_getraw(bs, outinfo->namelen);
! 59: outinfo->instance = byte_stream_get16(bs);
! 60:
! 61: return 0;
! 62: }
! 63:
! 64: /*
! 65: * Subtype 0x0002 - General room information. Lots of stuff.
! 66: *
! 67: * Values I know are in here but I haven't attached
! 68: * them to any of the 'Unknown's:
! 69: * - Language (English)
! 70: *
! 71: */
! 72: static int
! 73: infoupdate(OscarData *od, FlapConnection *conn, aim_module_t *mod, FlapFrame *frame, aim_modsnac_t *snac, ByteStream *bs)
! 74: {
! 75: aim_rxcallback_t userfunc;
! 76: int ret = 0;
! 77: guint8 detaillevel = 0;
! 78: struct aim_chat_roominfo roominfo;
! 79: GSList *tlvlist;
! 80: guint16 maxmsglen, maxvisiblemsglen;
! 81:
! 82: aim_chat_readroominfo(bs, &roominfo);
! 83:
! 84: detaillevel = byte_stream_get8(bs);
! 85:
! 86: if (detaillevel != 0x02) {
! 87: purple_debug_misc("oscar", "faim: chat_roomupdateinfo: detail level %d not supported\n", detaillevel);
! 88: return 1;
! 89: }
! 90:
! 91: byte_stream_get16(bs); /* skip the TLV count */
! 92:
! 93: /*
! 94: * Everything else are TLVs.
! 95: */
! 96: tlvlist = aim_tlvlist_read(bs);
! 97:
! 98: /*
! 99: * Type 0x00d1: Maximum Message Length
! 100: */
! 101: maxmsglen = aim_tlv_get16(tlvlist, 0x00d1, 1);
! 102:
! 103: /*
! 104: * Type 0x00da: Maximum visible message length
! 105: */
! 106: maxvisiblemsglen = aim_tlv_get16(tlvlist, 0x00da, 1);
! 107:
! 108: if ((userfunc = aim_callhandler(od, snac->family, snac->subtype))) {
! 109: ret = userfunc(od, conn, frame, maxmsglen, maxvisiblemsglen);
! 110: }
! 111:
! 112: g_free(roominfo.name);
! 113:
! 114: aim_tlvlist_free(tlvlist);
! 115:
! 116: return ret;
! 117: }
! 118:
! 119: /* Subtypes 0x0003 and 0x0004 */
! 120: static int
! 121: userlistchange(OscarData *od, FlapConnection *conn, aim_module_t *mod, FlapFrame *frame, aim_modsnac_t *snac, ByteStream *bs)
! 122: {
! 123: aim_userinfo_t *userinfo = NULL;
! 124: aim_rxcallback_t userfunc;
! 125: int curcount = 0, ret = 0;
! 126:
! 127: while (byte_stream_bytes_left(bs)) {
! 128: curcount++;
! 129: userinfo = g_realloc(userinfo, curcount * sizeof(aim_userinfo_t));
! 130: aim_info_extract(od, bs, &userinfo[curcount-1]);
! 131: }
! 132:
! 133: if ((userfunc = aim_callhandler(od, snac->family, snac->subtype)))
! 134: ret = userfunc(od, conn, frame, curcount, userinfo);
! 135:
! 136: aim_info_free(userinfo);
! 137: g_free(userinfo);
! 138:
! 139: return ret;
! 140: }
! 141:
! 142: /*
! 143: * Subtype 0x0005 - Send a Chat Message.
! 144: *
! 145: * Possible flags:
! 146: * AIM_CHATFLAGS_NOREFLECT -- Unset the flag that requests messages
! 147: * should be sent to their sender.
! 148: * AIM_CHATFLAGS_AWAY -- Mark the message as an autoresponse
! 149: * (Note that WinAIM does not honor this,
! 150: * and displays the message as normal.)
! 151: *
! 152: * XXX convert this to use tlvchains
! 153: */
! 154: int
! 155: aim_chat_send_im(OscarData *od, FlapConnection *conn, guint16 flags, const gchar *msg, int msglen, const char *encoding, const char *language)
! 156: {
! 157: int i;
! 158: ByteStream bs;
! 159: IcbmCookie *cookie;
! 160: aim_snacid_t snacid;
! 161: guint8 ckstr[8];
! 162: GSList *tlvlist = NULL, *inner_tlvlist = NULL;
! 163:
! 164: if (!od || !conn || !msg || (msglen <= 0))
! 165: return 0;
! 166:
! 167: byte_stream_new(&bs, 1142);
! 168:
! 169: snacid = aim_cachesnac(od, SNAC_FAMILY_CHAT, 0x0005, 0x0000, NULL, 0);
! 170:
! 171: /*
! 172: * Cookie
! 173: *
! 174: * XXX mkcookie should generate the cookie and cache it in one
! 175: * operation to preserve uniqueness.
! 176: */
! 177: for (i = 0; i < 8; i++)
! 178: ckstr[i] = (guint8)rand();
! 179:
! 180: cookie = aim_mkcookie(ckstr, AIM_COOKIETYPE_CHAT, NULL);
! 181: cookie->data = NULL; /* XXX store something useful here */
! 182:
! 183: aim_cachecookie(od, cookie);
! 184:
! 185: /* ICBM Header */
! 186: byte_stream_putraw(&bs, ckstr, 8); /* Cookie */
! 187: byte_stream_put16(&bs, 0x0003); /* Channel */
! 188:
! 189: /*
! 190: * Type 1: Flag meaning this message is destined to the room.
! 191: */
! 192: aim_tlvlist_add_noval(&tlvlist, 0x0001);
! 193:
! 194: /*
! 195: * Type 6: Reflect
! 196: */
! 197: if (!(flags & AIM_CHATFLAGS_NOREFLECT))
! 198: aim_tlvlist_add_noval(&tlvlist, 0x0006);
! 199:
! 200: /*
! 201: * Type 7: Autoresponse
! 202: */
! 203: if (flags & AIM_CHATFLAGS_AWAY)
! 204: aim_tlvlist_add_noval(&tlvlist, 0x0007);
! 205:
! 206: /*
! 207: * SubTLV: Type 1: Message
! 208: */
! 209: aim_tlvlist_add_raw(&inner_tlvlist, 0x0001, msglen, (guchar *)msg);
! 210:
! 211: /*
! 212: * SubTLV: Type 2: Encoding
! 213: */
! 214: if (encoding != NULL)
! 215: aim_tlvlist_add_str(&inner_tlvlist, 0x0002, encoding);
! 216:
! 217: /*
! 218: * SubTLV: Type 3: Language
! 219: */
! 220: if (language != NULL)
! 221: aim_tlvlist_add_str(&inner_tlvlist, 0x0003, language);
! 222:
! 223: /*
! 224: * Type 5: Message block. Contains more TLVs.
! 225: *
! 226: * This could include other information... We just
! 227: * put in a message TLV however.
! 228: *
! 229: */
! 230: aim_tlvlist_add_frozentlvlist(&tlvlist, 0x0005, &inner_tlvlist);
! 231:
! 232: aim_tlvlist_write(&bs, &tlvlist);
! 233:
! 234: aim_tlvlist_free(inner_tlvlist);
! 235: aim_tlvlist_free(tlvlist);
! 236:
! 237: flap_connection_send_snac(od, conn, SNAC_FAMILY_CHAT, 0x0005, snacid, &bs);
! 238:
! 239: byte_stream_destroy(&bs);
! 240:
! 241: return 0;
! 242: }
! 243:
! 244: /*
! 245: * Subtype 0x0006
! 246: *
! 247: * We could probably include this in the normal ICBM parsing
! 248: * code as channel 0x0003, however, since only the start
! 249: * would be the same, we might as well do it here.
! 250: *
! 251: * General outline of this SNAC:
! 252: * snac
! 253: * cookie
! 254: * channel id
! 255: * tlvlist
! 256: * unknown
! 257: * source user info
! 258: * name
! 259: * evility
! 260: * userinfo tlvs
! 261: * online time
! 262: * etc
! 263: * message metatlv
! 264: * message tlv
! 265: * message string
! 266: * possibly others
! 267: *
! 268: */
! 269: static int
! 270: incomingim_ch3(OscarData *od, FlapConnection *conn, aim_module_t *mod, FlapFrame *frame, aim_modsnac_t *snac, ByteStream *bs)
! 271: {
! 272: int ret = 0, i;
! 273: aim_rxcallback_t userfunc;
! 274: aim_userinfo_t userinfo;
! 275: guint8 cookie[8];
! 276: guint16 channel;
! 277: GSList *tlvlist;
! 278: char *msg = NULL;
! 279: int len = 0;
! 280: char *encoding = NULL, *language = NULL;
! 281: IcbmCookie *ck;
! 282: aim_tlv_t *tlv;
! 283: ByteStream tbs;
! 284:
! 285: memset(&userinfo, 0, sizeof(aim_userinfo_t));
! 286:
! 287: /*
! 288: * Read ICBM Cookie.
! 289: */
! 290: for (i = 0; i < 8; i++)
! 291: cookie[i] = byte_stream_get8(bs);
! 292:
! 293: if ((ck = aim_uncachecookie(od, cookie, AIM_COOKIETYPE_CHAT))) {
! 294: g_free(ck->data);
! 295: g_free(ck);
! 296: }
! 297:
! 298: /*
! 299: * Channel ID
! 300: *
! 301: * Channel 0x0003 is used for chat messages.
! 302: *
! 303: */
! 304: channel = byte_stream_get16(bs);
! 305:
! 306: if (channel != 0x0003) {
! 307: purple_debug_misc("oscar", "faim: chat_incoming: unknown channel! (0x%04x)\n", channel);
! 308: return 0;
! 309: }
! 310:
! 311: /*
! 312: * Start parsing TLVs right away.
! 313: */
! 314: tlvlist = aim_tlvlist_read(bs);
! 315:
! 316: /*
! 317: * Type 0x0003: Source User Information
! 318: */
! 319: tlv = aim_tlv_gettlv(tlvlist, 0x0003, 1);
! 320: if (tlv != NULL)
! 321: {
! 322: byte_stream_init(&tbs, tlv->value, tlv->length);
! 323: aim_info_extract(od, &tbs, &userinfo);
! 324: }
! 325:
! 326: /*
! 327: * Type 0x0005: Message Block. Conains more TLVs.
! 328: */
! 329: tlv = aim_tlv_gettlv(tlvlist, 0x0005, 1);
! 330: if (tlv != NULL)
! 331: {
! 332: GSList *inner_tlvlist;
! 333: aim_tlv_t *inner_tlv;
! 334:
! 335: byte_stream_init(&tbs, tlv->value, tlv->length);
! 336: inner_tlvlist = aim_tlvlist_read(&tbs);
! 337:
! 338: /*
! 339: * Type 0x0001: Message.
! 340: */
! 341: inner_tlv = aim_tlv_gettlv(inner_tlvlist, 0x0001, 1);
! 342: if (inner_tlv != NULL)
! 343: {
! 344: len = inner_tlv->length;
! 345: msg = aim_tlv_getvalue_as_string(inner_tlv);
! 346: }
! 347:
! 348: /*
! 349: * Type 0x0002: Encoding.
! 350: */
! 351: encoding = aim_tlv_getstr(inner_tlvlist, 0x0002, 1);
! 352:
! 353: /*
! 354: * Type 0x0003: Language.
! 355: */
! 356: language = aim_tlv_getstr(inner_tlvlist, 0x0003, 1);
! 357:
! 358: aim_tlvlist_free(inner_tlvlist);
! 359: }
! 360:
! 361: if ((userfunc = aim_callhandler(od, snac->family, snac->subtype)))
! 362: ret = userfunc(od, conn, frame, &userinfo, len, msg, encoding, language);
! 363:
! 364: aim_info_free(&userinfo);
! 365: g_free(msg);
! 366: g_free(encoding);
! 367: g_free(language);
! 368: aim_tlvlist_free(tlvlist);
! 369:
! 370: return ret;
! 371: }
! 372:
! 373: static int
! 374: snachandler(OscarData *od, FlapConnection *conn, aim_module_t *mod, FlapFrame *frame, aim_modsnac_t *snac, ByteStream *bs)
! 375: {
! 376: if (snac->subtype == 0x0002)
! 377: return infoupdate(od, conn, mod, frame, snac, bs);
! 378: else if ((snac->subtype == 0x0003) || (snac->subtype == 0x0004))
! 379: return userlistchange(od, conn, mod, frame, snac, bs);
! 380: else if (snac->subtype == 0x0006)
! 381: return incomingim_ch3(od, conn, mod, frame, snac, bs);
! 382:
! 383: return 0;
! 384: }
! 385:
! 386: int
! 387: chat_modfirst(OscarData *od, aim_module_t *mod)
! 388: {
! 389: mod->family = SNAC_FAMILY_CHAT;
! 390: mod->version = 0x0001;
! 391: mod->toolid = 0x0010;
! 392: mod->toolversion = 0x0629;
! 393: mod->flags = 0;
! 394: strncpy(mod->name, "chat", sizeof(mod->name));
! 395: mod->snachandler = snachandler;
! 396:
! 397: return 0;
! 398: }
FreeBSD-CVSweb <freebsd-cvsweb@FreeBSD.org>