Mercurial > pidgin
diff src/protocols/oscar/ssi.c @ 2991:9d11dbb33b6f
[gaim-migrate @ 3004]
KingAnt says: SSI SSI SSI
I say! HI HI HI!
committer: Tailor Script <tailor@pidgin.im>
author | Rob Flynn <gaim@robflynn.com> |
---|---|
date | Sat, 02 Mar 2002 01:36:01 +0000 |
parents | c41030cfed76 |
children | e27517a5c28e |
line wrap: on
line diff
--- a/src/protocols/oscar/ssi.c Tue Feb 26 23:06:53 2002 +0000 +++ b/src/protocols/oscar/ssi.c Sat Mar 02 01:36:01 2002 +0000 @@ -6,6 +6,12 @@ * to be stored on the server, so that they can be accessed from any client. * * This is entirely too complicated. + * You don't know the half of it. + * + * XXX - Make sure moving buddies from group to group moves the buddy in the server list also + * XXX - Test for memory leaks + * XXX - Write a c plugin to manage ssi data + * XXX - Better parsing of rights, and use the rights info to limit adds * */ @@ -13,6 +19,899 @@ #include <aim.h> /* + * Checks if the given screen name is anywhere in our buddy list. + */ +faim_export int aim_ssi_inlist(aim_session_t *sess, aim_conn_t *conn, char *name, fu16_t type) +{ + struct aim_ssi_item *cur; + for (cur=sess->ssi.items; cur; cur=cur->next) + if ((cur->type==type) && (cur->name) && (!aim_sncmp(cur->name, name))) + return 1; + return 0; +} + +/* + * Returns the permit/deny byte + * Should should be removed and the byte should be passed directly to + * the handler + */ +faim_export int aim_ssi_getpermdeny(aim_tlvlist_t *tlvlist) +{ + aim_tlv_t *tlv; + if ((tlv = aim_gettlv(tlvlist, 0x00ca, 1)) && tlv->value) + return tlv->value[0]; + return 0; +} + +/* + * Returns a pointer to an item with the given name and type, or NULL if one does not exist. + */ +static struct aim_ssi_item *get_ssi_item(struct aim_ssi_item *items, char *name, fu16_t type) +{ + struct aim_ssi_item *cur; + if (name) { + for (cur=items; cur; cur=cur->next) + if ((cur->type == type) && (cur->name) && !(aim_sncmp(cur->name, name))) + return cur; + } else { /* they probably want the master group */ + for (cur=items; cur; cur=cur->next) + if ((cur->type == type) && (cur->gid == 0x0000)) + return cur; + } + return NULL; +} + +/* + * Add the given packet to the holding queue. + */ +static int aim_ssi_enqueue(aim_session_t *sess, aim_conn_t *conn, aim_frame_t *fr) +{ + aim_frame_t *cur; + + if (!sess || !conn || !fr) + return -EINVAL; + + fr->next = NULL; + if (sess->ssi.holding_queue == NULL) { + sess->ssi.holding_queue = fr; + if (!sess->ssi.waiting_for_ack) + aim_ssi_modbegin(sess, conn); + } else { + for (cur = sess->ssi.holding_queue; cur->next; cur = cur->next) ; + cur->next = fr; + } + + return 0; +} + +/* + * Send the next SNAC from the holding queue. + */ +static int aim_ssi_dispatch(aim_session_t *sess, aim_conn_t *conn) +{ + aim_frame_t *cur; + + if (!sess || !conn) + return -EINVAL; + + if (!sess->ssi.waiting_for_ack) { + if (sess->ssi.holding_queue) { + sess->ssi.waiting_for_ack = 1; + cur = sess->ssi.holding_queue->next; + sess->ssi.holding_queue->next = NULL; + aim_tx_enqueue(sess, sess->ssi.holding_queue); + sess->ssi.holding_queue = cur; + } else + aim_ssi_modend(sess, conn); + } + + return 0; +} + +/* + * Rebuild the additional data for the given group(s). + */ +static int aim_ssi_rebuildgroup(aim_session_t *sess, aim_conn_t *conn, struct aim_ssi_item *parentgroup) +{ + int newlen; + struct aim_ssi_item *cur; + + if (!sess || !conn || !parentgroup) + return -EINVAL; + + /* Free the old additional data */ + if (parentgroup->data) { + aim_freetlvchain((aim_tlvlist_t **)&parentgroup->data); + parentgroup->data = NULL; + } + + /* Find the length for the new additional data */ + newlen = 0; + if (parentgroup->gid == 0x0000) { + for (cur=sess->ssi.items; cur; cur=cur->next) + if ((cur->gid != 0x0000) && (cur->type == 0x0001)) + newlen += 2; + } else { + for (cur=sess->ssi.items; cur; cur=cur->next) + if ((cur->gid == parentgroup->gid) && (cur->type == 0x0000)) + newlen += 2; + } + + /* Rebuild the additional data */ + if (newlen>0) { + aim_bstream_t tbs; + tbs.len = newlen+4; + tbs.offset = 0; + if (!(tbs.data = (fu8_t *)malloc((tbs.len)*sizeof(fu8_t)))) + return -ENOMEM; + aimbs_put16(&tbs, 0x00c8); + aimbs_put16(&tbs, tbs.len-4); + if (parentgroup->gid == 0x0000) { + for (cur=sess->ssi.items; cur; cur=cur->next) + if ((cur->gid != 0x0000) && (cur->type == 0x0001)) + aimbs_put16(&tbs, cur->gid); + } else { + for (cur=sess->ssi.items; cur; cur=cur->next) + if ((cur->gid == parentgroup->gid) && (cur->type == 0x0000)) + aimbs_put16(&tbs, cur->bid); + } + tbs.offset = 0; + parentgroup->data = (void *)aim_readtlvchain(&tbs); + free(tbs.data); + +/* XXX - WHYYYYYYYYYYYYYYYYYYYYYYYYYYYYYYYYYY does this crash? + fu8_t *newdata; + if (!(newdata = (fu8_t *)malloc((newlen)*sizeof(fu8_t)))) + return -ENOMEM; + newlen = 0; + if (parentgroup->gid == 0x0000) { + for (cur=sess->ssi.items; cur; cur=cur->next) + if ((cur->gid != 0x0000) && (cur->type == 0x0001)) { + memcpy(&newdata[newlen*2], &cur->gid, 2); + newlen += 2; + } + } else { + for (cur=sess->ssi.items; cur; cur=cur->next) + if ((cur->gid == parentgroup->gid) && (cur->type == 0x0000)) { + memcpy(newdata+newlen*2, &cur->bid, 2); + newlen += 2; + } + } + aim_addtlvtochain_raw((aim_tlvlist_t **)&parentgroup->data, 0x00c8, newlen, newdata); + free(newdata); */ + } + + return 0; +} + +/* + * Clears all of our locally stored buddy list information. + */ +static int aim_ssi_freelist(aim_session_t *sess) +{ + struct aim_ssi_item *cur, *delitem; + + cur = sess->ssi.items; + while (cur) { + if (cur->name) free(cur->name); + if (cur->data) aim_freetlvchain((aim_tlvlist_t **)&cur->data); + delitem = cur; + cur = cur->next; + free(delitem); + } + + sess->ssi.revision = 0; + sess->ssi.items = NULL; + sess->ssi.timestamp = (time_t)0; + + return 0; +} + +/* + * This removes all ssi data from server and local copy. + */ +faim_export int aim_ssi_deletelist(aim_session_t *sess, aim_conn_t *conn) +{ + int i, num; + struct aim_ssi_item *cur, **items; + + for (cur=sess->ssi.items, num=0; cur; cur=cur->next) + num++; + + if (!(items = (struct aim_ssi_item **)malloc(num*sizeof(struct aim_ssi_item *)))) + return -ENOMEM; + bzero(items, num*sizeof(struct aim_ssi_item *)); + for (cur=sess->ssi.items, num=0; cur; cur=cur->next) { + items[num] = cur; + num++; + } + + aim_ssi_delitems(sess, conn, items, num); + free(items[num]); + aim_ssi_dispatch(sess, conn); + aim_ssi_freelist(sess); + + return 0; +} + +/* + * This "cleans" the ssi list. It makes sure all buddies are in a group, and + * all groups have additional data for the buddies that are in them. It does + * this by rebuilding the additional data for every group, sending mod SNACs + * as necessary. + */ +faim_export int aim_ssi_cleanlist(aim_session_t *sess, aim_conn_t *conn) +{ + unsigned int num_to_be_fixed; + struct aim_ssi_item *cur, *parentgroup; + + /* Make sure we actually need to clean out the list */ + for (cur=sess->ssi.items, num_to_be_fixed=0; cur; cur=cur->next) { + /* Any buddies directly in the master group */ + if ((cur->type == 0x0000) && (cur->gid == 0x0000)) + num_to_be_fixed++; + } + if (!num_to_be_fixed) + return 0; + + /* Remove all the additional data from all groups and buddies */ + for (cur=sess->ssi.items; cur; cur=cur->next) + if (cur->data && ((cur->type == 0x0000) || (cur->type == 0x0001))) { + aim_freetlvchain((aim_tlvlist_t **)&cur->data); + cur->data = NULL; + } + + /* If there are buddies directly in the master group, make sure */ + /* there is a group to put them in. Any group, any group at all. */ + for (cur=sess->ssi.items; ((cur) && ((cur->type != 0x0000) || (cur->gid != 0x0000))); cur=cur->next); + if (!cur) { + for (parentgroup=sess->ssi.items; ((parentgroup) && ((parentgroup->type!=0x0001) || (parentgroup->gid!=0x0000))); parentgroup=parentgroup->next); + if (!parentgroup) { + char *newgroup; + newgroup = (char*)malloc(strlen("Unknown")*sizeof(char)); + strcpy(newgroup, "Unknown"); + aim_ssi_addgroups(sess, conn, &newgroup, 1); + } + } + + /* Set parentgroup equal to any arbitray group */ + for (parentgroup=sess->ssi.items; parentgroup->gid==0x0000 || parentgroup->type!=0x0001; parentgroup=parentgroup->next); + + /* If there are any buddies directly in the master group, put them in a real group */ + for (cur=sess->ssi.items; cur; cur=cur->next) + if ((cur->type == 0x0000) && (cur->gid == 0x0000)) { + aim_ssi_delitems(sess, conn, &cur, 1); + cur->gid = parentgroup->gid; + aim_ssi_additems(sess, conn, &cur, 1); + } + + /* Rebuild additional data for all groups */ + for (parentgroup=sess->ssi.items; parentgroup; parentgroup=parentgroup->next) + if (parentgroup->type == 0x0001) + aim_ssi_rebuildgroup(sess, conn, parentgroup); + + /* Send a mod snac for all groups */ + for (cur=sess->ssi.items; cur; cur=cur->next) + if (cur->type == 0x0001) + aim_ssi_moditems(sess, conn, &cur, 1); + + /* Begin sending SSI SNACs */ + aim_ssi_dispatch(sess, conn); + + return 0; +} + +/* + * This adds the master group (the group containing all groups) if it doesn't exist. + * It's called by aim_ssi_addgroups, if your list is empty. + */ +faim_export int aim_ssi_addmastergroup(aim_session_t *sess, aim_conn_t *conn) { + struct aim_ssi_item *newitem; + + if (!sess || !conn) + return -EINVAL; + + /* Allocate an array of pointers to each of the new items */ + if (!(newitem = (struct aim_ssi_item *)malloc(sizeof(struct aim_ssi_item)))) + return -ENOMEM; + bzero(newitem, sizeof(struct aim_ssi_item)); + + /* bzero sets most of the vars to what they should be */ + newitem->type = 0x0001; + + /* Add the item to our list */ + newitem->next = sess->ssi.items; + sess->ssi.items = newitem; + + /* If there are any existing groups (technically there shouldn't be, but */ + /* just in case) then add their group ID#'s to the additional data */ + if (sess->ssi.items) + aim_ssi_rebuildgroup(sess, conn, newitem); + + /* Send the add item SNAC */ + aim_ssi_additems(sess, conn, &newitem, 1); + + /* Begin sending SSI SNACs */ + aim_ssi_dispatch(sess, conn); + + return 0; +} + +/* + * The next few functions take data as screen names and groups, + * modifies the ssi item list, and calls the functions to + * add, mod, or del an item or items. + * + * These are what the client should call. The client should + * also make sure it's not adding a buddy that's already in + * the list (using aim_ssi_inlist). + */ +faim_export int aim_ssi_addbuddies(aim_session_t *sess, aim_conn_t *conn, char *gn, char **sn, unsigned int num) +{ + struct aim_ssi_item *cur, *parentgroup, **newitems; + fu16_t i, j; + + if (!sess || !conn || !gn || !sn || !num) + return -EINVAL; + + /* Look up the parent group */ + if (!(parentgroup = get_ssi_item(sess->ssi.items, gn, 0x0001))) { + aim_ssi_addgroups(sess, conn, &gn, 1); + if (!(parentgroup = get_ssi_item(sess->ssi.items, gn, 0x0001))) + return -ENOMEM; + } + + /* Allocate an array of pointers to each of the new items */ + if (!(newitems = (struct aim_ssi_item **)malloc(num*sizeof(struct aim_ssi_item *)))) + return -ENOMEM; + bzero(newitems, num*sizeof(struct aim_ssi_item *)); + + /* The following for loop iterates once per item that needs to be added. */ + /* For each i, create an item and tack it onto the newitems array */ + for (i=0; i<num; i++) { + if (!(newitems[i] = (struct aim_ssi_item *)malloc(sizeof(struct aim_ssi_item)))) { + for (j=0; j<(i-1); j++) { + free(newitems[j]->name); + free(newitems[j]); + } + free(newitems); + return -ENOMEM; + } + if (!(newitems[i]->name = (char *)malloc((strlen(sn[i])+1)*sizeof(char)))) { + for (j=0; j<(i-1); j++) { + free(newitems[j]->name); + free(newitems[j]); + } + free(newitems[i]); + free(newitems); + return -ENOMEM; + } + strcpy(newitems[i]->name, sn[i]); + newitems[i]->gid = parentgroup->gid; + newitems[i]->bid = i ? newitems[i-1]->bid : 0; + do { + newitems[i]->bid += 0x0001; + for (cur=sess->ssi.items, j=0; ((cur) && (!j)); cur=cur->next) + if ((cur->bid == newitems[i]->bid) && (cur->gid == newitems[i]->gid) && (cur->type == 0x0000)) + j=1; + } while (j); + newitems[i]->type = 0x0000; + newitems[i]->data = NULL; + newitems[i]->next = i ? newitems[i-1] : NULL; + } + + /* Add the items to our list */ + newitems[0]->next = sess->ssi.items; + sess->ssi.items = newitems[num-1]; + + /* Send the add item SNAC */ + aim_ssi_additems(sess, conn, newitems, num); + + /* Free the array of pointers to each of the new items */ + free(newitems); + + /* Rebuild the additional data in the parent group */ + aim_ssi_rebuildgroup(sess, conn, parentgroup); + + /* Send the mod item SNAC */ + aim_ssi_moditems(sess, conn, &parentgroup, 1); + + /* Begin sending SSI SNACs */ + aim_ssi_dispatch(sess, conn); + + return 0; +} + +faim_export int aim_ssi_addgroups(aim_session_t *sess, aim_conn_t *conn, char **gn, unsigned int num) +{ + struct aim_ssi_item *cur, *parentgroup, **newitems; + fu16_t i, j; + + if (!sess || !conn || !gn || !num) + return -EINVAL; + + /* Look up the parent group */ + if (!(parentgroup = get_ssi_item(sess->ssi.items, NULL, 0x0001))) { + aim_ssi_addmastergroup(sess, conn); + if (!(parentgroup = get_ssi_item(sess->ssi.items, NULL, 0x0001))) + return -ENOMEM; + } + + /* Allocate an array of pointers to each of the new items */ + if (!(newitems = (struct aim_ssi_item **)malloc(num*sizeof(struct aim_ssi_item *)))) + return -ENOMEM; + bzero(newitems, num*sizeof(struct aim_ssi_item *)); + + /* The following for loop iterates once per item that needs to be added. */ + /* For each i, create an item and tack it onto the newitems array */ + for (i=0; i<num; i++) { + if (!(newitems[i] = (struct aim_ssi_item *)malloc(sizeof(struct aim_ssi_item)))) { + for (j=0; j<(i-1); j++) { + free(newitems[j]->name); + free(newitems[j]); + } + free(newitems); + return -ENOMEM; + } + if (!(newitems[i]->name = (char *)malloc((strlen(gn[i])+1)*sizeof(char)))) { + for (j=0; j<(i-1); j++) { + free(newitems[j]->name); + free(newitems[j]); + } + free(newitems[i]); + free(newitems); + return -ENOMEM; + } + strcpy(newitems[i]->name, gn[i]); + newitems[i]->gid = i ? newitems[i-1]->gid : 0; + do { + newitems[i]->gid += 0x0001; + for (cur=sess->ssi.items, j=0; ((cur) && (!j)); cur=cur->next) + if ((cur->gid == newitems[i]->gid) && (cur->type == 0x0001)) + j=1; + } while (j); + newitems[i]->bid = 0x0000; + newitems[i]->type = 0x0001; + newitems[i]->data = NULL; + newitems[i]->next = i ? newitems[i-1] : NULL; + } + + /* Add the items to our list */ + newitems[0]->next = sess->ssi.items; + sess->ssi.items = newitems[num-1]; + + /* Send the add item SNAC */ + aim_ssi_additems(sess, conn, newitems, num); + + /* Free the array of pointers to each of the new items */ + free(newitems); + + /* Rebuild the additional data in the parent group */ + aim_ssi_rebuildgroup(sess, conn, parentgroup); + + /* Send the mod item SNAC */ + aim_ssi_moditems(sess, conn, &parentgroup, 1); + + /* Begin sending SSI SNACs */ + aim_ssi_dispatch(sess, conn); + + return 0; +} + +faim_export int aim_ssi_addpermits(aim_session_t *sess, aim_conn_t *conn, char **sn, unsigned int num) +{ + struct aim_ssi_item *cur, **newitems; + fu16_t i, j; + + if (!sess || !conn || !sn || !num) + return -EINVAL; + + /* Allocate an array of pointers to each of the new items */ + if (!(newitems = (struct aim_ssi_item **)malloc(num*sizeof(struct aim_ssi_item *)))) + return -ENOMEM; + bzero(newitems, num*sizeof(struct aim_ssi_item *)); + + /* The following for loop iterates once per item that needs to be added. */ + /* For each i, create an item and tack it onto the newitems array */ + for (i=0; i<num; i++) { + if (!(newitems[i] = (struct aim_ssi_item *)malloc(sizeof(struct aim_ssi_item)))) { + for (j=0; j<(i-1); j++) { + free(newitems[j]->name); + free(newitems[j]); + } + free(newitems); + return -ENOMEM; + } + if (!(newitems[i]->name = (char *)malloc((strlen(sn[i])+1)*sizeof(char)))) { + for (j=0; j<(i-1); j++) { + free(newitems[j]->name); + free(newitems[j]); + } + free(newitems[i]); + free(newitems); + return -ENOMEM; + } + strcpy(newitems[i]->name, sn[i]); + newitems[i]->gid = 0x0000; + newitems[i]->bid = i ? newitems[i-1]->bid : 0x0000; + do { + newitems[i]->bid += 0x0001; + for (cur=sess->ssi.items, j=0; ((cur) && (!j)); cur=cur->next) + if ((cur->bid == newitems[i]->bid) && ((cur->type == 0x0002) || (cur->type == 0x0003))) + j=1; + } while (j); + newitems[i]->type = 0x0002; + newitems[i]->data = NULL; + newitems[i]->next = i ? newitems[i-1] : NULL; + } + + /* Add the items to our list */ + newitems[0]->next = sess->ssi.items; + sess->ssi.items = newitems[num-1]; + + /* Send the add item SNAC */ + aim_ssi_additems(sess, conn, newitems, num); + + /* Free the array of pointers to each of the new items */ + free(newitems); + + /* Begin sending SSI SNACs */ + aim_ssi_dispatch(sess, conn); + + return 0; +} + +faim_export int aim_ssi_adddenies(aim_session_t *sess, aim_conn_t *conn, char **sn, unsigned int num) +{ + struct aim_ssi_item *cur, **newitems; + fu16_t i, j; + + if (!sess || !conn || !sn || !num) + return -EINVAL; + + /* Allocate an array of pointers to each of the new items */ + if (!(newitems = (struct aim_ssi_item **)malloc(num*sizeof(struct aim_ssi_item *)))) + return -ENOMEM; + bzero(newitems, num*sizeof(struct aim_ssi_item *)); + + /* The following for loop iterates once per item that needs to be added. */ + /* For each i, create an item and tack it onto the newitems array */ + for (i=0; i<num; i++) { + if (!(newitems[i] = (struct aim_ssi_item *)malloc(sizeof(struct aim_ssi_item)))) { + for (j=0; j<(i-1); j++) { + free(newitems[j]->name); + free(newitems[j]); + } + free(newitems); + return -ENOMEM; + } + if (!(newitems[i]->name = (char *)malloc((strlen(sn[i])+1)*sizeof(char)))) { + for (j=0; j<(i-1); j++) { + free(newitems[j]->name); + free(newitems[j]); + } + free(newitems[i]); + free(newitems); + return -ENOMEM; + } + strcpy(newitems[i]->name, sn[i]); + newitems[i]->gid = 0x0000; + newitems[i]->bid = i ? newitems[i-1]->bid : 0x0000; + do { + newitems[i]->bid += 0x0001; + for (cur=sess->ssi.items, j=0; ((cur) && (!j)); cur=cur->next) + if ((cur->bid == newitems[i]->bid) && ((cur->type == 0x0002) || (cur->type == 0x0003))) + j=1; + } while (j); + newitems[i]->type = 0x0003; + newitems[i]->data = NULL; + newitems[i]->next = i ? newitems[i-1] : NULL; + } + + /* Add the items to our list */ + newitems[0]->next = sess->ssi.items; + sess->ssi.items = newitems[num-1]; + + /* Send the add item SNAC */ + aim_ssi_additems(sess, conn, newitems, num); + + /* Free the array of pointers to each of the new items */ + free(newitems); + + /* Begin sending SSI SNACs */ + aim_ssi_dispatch(sess, conn); + + return 0; +} + +faim_export int aim_ssi_delbuddies(aim_session_t *sess, aim_conn_t *conn, char *gn, char **sn, unsigned int num) +{ + struct aim_ssi_item *cur, *parentgroup, **delitems; + int i; + + if (!sess || !conn || !gn || !sn || !num) + return -EINVAL; + + /* Look up the parent group */ + if (!(parentgroup = get_ssi_item(sess->ssi.items, gn, 0x0001))) + return -EINVAL; + + /* Allocate an array of pointers to each of the items to be deleted */ + delitems = (struct aim_ssi_item **)malloc(num*sizeof(struct aim_ssi_item *)); + bzero(delitems, num*sizeof(struct aim_ssi_item *)); + + /* Make the delitems array a pointer to the aim_ssi_item structs to be deleted */ + for (i=0; i<num; i++) { + if (!(delitems[i] = get_ssi_item(sess->ssi.items, sn[i], 0x0000))) { + free(delitems); + return -EINVAL; + } + + /* Remove the delitems from the item list */ + if (sess->ssi.items == delitems[i]) { + sess->ssi.items = sess->ssi.items->next; + } else { + for (cur=sess->ssi.items; (cur->next && (cur->next!=delitems[i])); cur=cur->next); + if (cur->next) + cur->next = cur->next->next; + } + } + + /* Send the del item SNAC */ + aim_ssi_delitems(sess, conn, delitems, num); + + /* Free the items */ + for (i=0; i<num; i++) { + if (delitems[i]->name) + free(delitems[i]->name); + if (delitems[i]->data) + aim_freetlvchain((aim_tlvlist_t **)&delitems[i]->data); + free(delitems[i]); + } + free(delitems); + + /* Rebuild the additional data in the parent group */ + aim_ssi_rebuildgroup(sess, conn, parentgroup); + + /* Send the mod item SNAC */ + aim_ssi_moditems(sess, conn, &parentgroup, 1); + + /* Delete the group, but only if it's empty */ + if (!parentgroup->data) + aim_ssi_delgroups(sess, conn, &parentgroup->name, 1); + + /* Begin sending SSI SNACs */ + aim_ssi_dispatch(sess, conn); + + return 0; +} + +faim_export int aim_ssi_delgroups(aim_session_t *sess, aim_conn_t *conn, char **gn, unsigned int num) { + struct aim_ssi_item *cur, *parentgroup, **delitems; + int i; + + if (!sess || !conn || !gn || !num) + return -EINVAL; + + /* Look up the parent group */ + if (!(parentgroup = get_ssi_item(sess->ssi.items, NULL, 0x0001))) + return -EINVAL; + + /* Allocate an array of pointers to each of the items to be deleted */ + delitems = (struct aim_ssi_item **)malloc(num*sizeof(struct aim_ssi_item *)); + bzero(delitems, num*sizeof(struct aim_ssi_item *)); + + /* Make the delitems array a pointer to the aim_ssi_item structs to be deleted */ + for (i=0; i<num; i++) { + if (!(delitems[i] = get_ssi_item(sess->ssi.items, gn[i], 0x0001))) { + free(delitems); + return -EINVAL; + } + + /* Remove the delitems from the item list */ + if (sess->ssi.items == delitems[i]) { + sess->ssi.items = sess->ssi.items->next; + } else { + for (cur=sess->ssi.items; (cur->next && (cur->next!=delitems[i])); cur=cur->next); + if (cur->next) + cur->next = cur->next->next; + } + } + + /* Send the del item SNAC */ + aim_ssi_delitems(sess, conn, delitems, num); + + /* Free the items */ + for (i=0; i<num; i++) { + if (delitems[i]->name) + free(delitems[i]->name); + if (delitems[i]->data) + aim_freetlvchain((aim_tlvlist_t **)&delitems[i]->data); + free(delitems[i]); + } + free(delitems); + + /* Rebuild the additional data in the parent group */ + aim_ssi_rebuildgroup(sess, conn, parentgroup); + + /* Send the mod item SNAC */ + aim_ssi_moditems(sess, conn, &parentgroup, 1); + + /* Delete the group, but only if it's empty */ + if (!parentgroup->data) + aim_ssi_delgroups(sess, conn, &parentgroup->name, 1); + + /* Begin sending SSI SNACs */ + aim_ssi_dispatch(sess, conn); + + return 0; +} + +faim_export int aim_ssi_delpermits(aim_session_t *sess, aim_conn_t *conn, char **sn, unsigned int num) { + struct aim_ssi_item *cur, **delitems; + int i; + + if (!sess || !conn || !sn || !num) + return -EINVAL; + + /* Allocate an array of pointers to each of the items to be deleted */ + delitems = (struct aim_ssi_item **)malloc(num*sizeof(struct aim_ssi_item *)); + bzero(delitems, num*sizeof(struct aim_ssi_item *)); + + /* Make the delitems array a pointer to the aim_ssi_item structs to be deleted */ + for (i=0; i<num; i++) { + if (!(delitems[i] = get_ssi_item(sess->ssi.items, sn[i], 0x0002))) { + free(delitems); + return -EINVAL; + } + + /* Remove the delitems from the item list */ + if (sess->ssi.items == delitems[i]) { + sess->ssi.items = sess->ssi.items->next; + } else { + for (cur=sess->ssi.items; (cur->next && (cur->next!=delitems[i])); cur=cur->next); + if (cur->next) + cur->next = cur->next->next; + } + } + + /* Send the del item SNAC */ + aim_ssi_delitems(sess, conn, delitems, num); + + /* Free the items */ + for (i=0; i<num; i++) { + if (delitems[i]->name) + free(delitems[i]->name); + if (delitems[i]->data) + aim_freetlvchain((aim_tlvlist_t **)&delitems[i]->data); + free(delitems[i]); + } + free(delitems); + + /* Begin sending SSI SNACs */ + aim_ssi_dispatch(sess, conn); + + return 0; +} + +faim_export int aim_ssi_deldenies(aim_session_t *sess, aim_conn_t *conn, char **sn, unsigned int num) { + struct aim_ssi_item *cur, **delitems; + int i; + + if (!sess || !conn || !sn || !num) + return -EINVAL; + + /* Allocate an array of pointers to each of the items to be deleted */ + delitems = (struct aim_ssi_item **)malloc(num*sizeof(struct aim_ssi_item *)); + bzero(delitems, num*sizeof(struct aim_ssi_item *)); + + /* Make the delitems array a pointer to the aim_ssi_item structs to be deleted */ + for (i=0; i<num; i++) { + if (!(delitems[i] = get_ssi_item(sess->ssi.items, sn[i], 0x0003))) { + free(delitems); + return -EINVAL; + } + + /* Remove the delitems from the item list */ + if (sess->ssi.items == delitems[i]) { + sess->ssi.items = sess->ssi.items->next; + } else { + for (cur=sess->ssi.items; (cur->next && (cur->next!=delitems[i])); cur=cur->next); + if (cur->next) + cur->next = cur->next->next; + } + } + + /* Send the del item SNAC */ + aim_ssi_delitems(sess, conn, delitems, num); + + /* Free the items */ + for (i=0; i<num; i++) { + if (delitems[i]->name) + free(delitems[i]->name); + if (delitems[i]->data) + aim_freetlvchain((aim_tlvlist_t **)&delitems[i]->data); + free(delitems[i]); + } + free(delitems); + + /* Begin sending SSI SNACs */ + aim_ssi_dispatch(sess, conn); + + return 0; +} + +/* + * Stores your permit/deny setting on the server, and starts using it. + * The permit/deny byte should be: + * 1 - Allow all users + * 2 - Block all users + * 3 - Allow only the users below + * 4 - Block the users below + * 5 - Allow only users on my buddy list + * To block AIM users, change the x00cb tlv from xffff to x0004 + */ +faim_export int aim_ssi_setpermdeny(aim_session_t *sess, aim_conn_t *conn, int permdeny) { + struct aim_ssi_item *cur, *tmp; + fu16_t i, j; + aim_tlv_t *tlv; + + if (!sess || !conn) + return -EINVAL; + + /* Look up the permit/deny settings item */ + for (cur=sess->ssi.items; (cur && (cur->type!=0x0004)); cur=cur->next); + + if (cur) { + /* The permit/deny item exists */ + if (cur->data && (tlv = aim_gettlv(cur->data, 0x00ca, 1))) { + /* Just change the value of the x00ca TLV */ + if (tlv->length != 1) { + tlv->length = 1; + free(tlv->value); + tlv->value = (fu8_t *)malloc(sizeof(fu8_t)); + } + memcpy(tlv->value, &permdeny, 1); + } else { + /* Need to add the x00ca TLV to the TLV chain */ + aim_addtlvtochain8((aim_tlvlist_t**)&cur->data, 0x00ca, permdeny); + } + + /* Send the mod item SNAC */ + aim_ssi_moditems(sess, conn, &cur, 1); + } else { + /* Need to add the permit/deny item */ + if (!(cur = (struct aim_ssi_item *)malloc(sizeof(struct aim_ssi_item)))) + return -ENOMEM; + cur->name = NULL; + cur->gid = 0x0000; + cur->bid = 0x007a; /* XXX - Is this number significant? */ + do { + cur->bid += 0x0001; + for (tmp=sess->ssi.items, j=0; ((tmp) && (!j)); tmp=tmp->next) + if (tmp->bid == cur->bid) + j=1; + } while (j); + cur->type = 0x0004; + cur->data = NULL; + aim_addtlvtochain8((aim_tlvlist_t**)&cur->data, 0x00ca, permdeny); + aim_addtlvtochain32((aim_tlvlist_t**)&cur->data, 0x00cb, 0xffffffff); + + /* Add the item to our list */ + cur->next = sess->ssi.items; + sess->ssi.items = cur; + + /* Send the add item SNAC */ + aim_ssi_additems(sess, conn, &cur, 1); + } + + /* Begin sending SSI SNACs */ + aim_ssi_dispatch(sess, conn); + + return 0; +} + +/* * Request SSI Rights. */ faim_export int aim_ssi_reqrights(aim_session_t *sess, aim_conn_t *conn) @@ -72,57 +971,54 @@ { int ret = 0; aim_rxcallback_t userfunc; - struct aim_ssi_item *list = NULL; + struct aim_ssi_item *cur = NULL; fu8_t fmtver; /* guess */ - fu16_t itemcount; - fu32_t stamp; + fu16_t revision; + fu32_t timestamp; - fmtver = aimbs_get8(bs); - itemcount = aimbs_get16(bs); + fmtver = aimbs_get8(bs); /* Version of ssi data. Should be 0x00 */ + revision = aimbs_get16(bs); /* # of times ssi data has been modified */ + if (revision != 0) + sess->ssi.revision = revision; + + for (cur = sess->ssi.items; cur && cur->next; cur=cur->next) ; while (aim_bstream_empty(bs) > 4) { /* last four bytes are stamp */ fu16_t namelen, tbslen; - struct aim_ssi_item *nl, *el; - if (!(nl = malloc(sizeof(struct aim_ssi_item)))) - break; - memset(nl, 0, sizeof(struct aim_ssi_item)); + if (!sess->ssi.items) { + if (!(sess->ssi.items = malloc(sizeof(struct aim_ssi_item)))) + return -ENOMEM; + cur = sess->ssi.items; + } else { + if (!(cur->next = malloc(sizeof(struct aim_ssi_item)))) + return -ENOMEM; + cur = cur->next; + } + memset(cur, 0, sizeof(struct aim_ssi_item)); if ((namelen = aimbs_get16(bs))) - nl->name = aimbs_getstr(bs, namelen); - nl->gid = aimbs_get16(bs); - nl->bid = aimbs_get16(bs); - nl->type = aimbs_get16(bs); + cur->name = aimbs_getstr(bs, namelen); + cur->gid = aimbs_get16(bs); + cur->bid = aimbs_get16(bs); + cur->type = aimbs_get16(bs); if ((tbslen = aimbs_get16(bs))) { aim_bstream_t tbs; aim_bstream_init(&tbs, bs->data + bs->offset /* XXX */, tbslen); - nl->data = (void *)aim_readtlvchain(&tbs); + cur->data = (void *)aim_readtlvchain(&tbs); aim_bstream_advance(bs, tbslen); } - - for (el = list; el && el->next; el = el->next) - ; - if (el) - el->next = nl; - else - list = nl; } - stamp = aimbs_get32(bs); + timestamp = aimbs_get32(bs); + if (timestamp != 0) + sess->ssi.timestamp = timestamp; + sess->ssi.received_data = 1; if ((userfunc = aim_callhandler(sess, rx->conn, snac->family, snac->subtype))) - ret = userfunc(sess, rx, fmtver, itemcount, stamp, list); - - while (list) { - struct aim_ssi_item *tmp; - - tmp = list->next; - aim_freetlvchain((aim_tlvlist_t **)&list->data); - free(list); - list = tmp; - } + ret = userfunc(sess, rx, fmtver, sess->ssi.revision, sess->ssi.timestamp, sess->ssi.items); return ret; } @@ -142,6 +1038,170 @@ } /* + * SSI Add Item. + * + * Adds an item to the data stored on the AIM server. An item could + * be a group, buddy, or permit or deny buddy. + * + */ +faim_export int aim_ssi_additems(aim_session_t *sess, aim_conn_t *conn, struct aim_ssi_item **items, unsigned int num) +{ + aim_frame_t *fr; + aim_snacid_t snacid; + int i, snaclen; + + if (!sess || !conn || !items || !num) + return -EINVAL; + + snaclen = 10; /* For family, subtype, flags, and SNAC ID */ + for (i=0; i<num; i++) { + snaclen += 10; /* For length, GID, BID, type, and length */ + if (items[i]->name) + snaclen += strlen(items[i]->name); + if (items[i]->data) + snaclen += aim_sizetlvchain((aim_tlvlist_t **)&items[i]->data); + } + + if (!(fr = aim_tx_new(sess, conn, AIM_FRAMETYPE_FLAP, 0x02, snaclen))) + return -ENOMEM; + + snacid = aim_cachesnac(sess, AIM_CB_FAM_SSI, AIM_CB_SSI_ADD, 0x0000, NULL, 0); + aim_putsnac(&fr->data, AIM_CB_FAM_SSI, AIM_CB_SSI_ADD, 0x0000, snacid); + + for (i=0; i<num; i++) { + aimbs_put16(&fr->data, items[i]->name ? strlen(items[i]->name) : 0); + if (items[i]->name) + aimbs_putraw(&fr->data, items[i]->name, strlen(items[i]->name)); + aimbs_put16(&fr->data, items[i]->gid); + aimbs_put16(&fr->data, items[i]->bid); + aimbs_put16(&fr->data, items[i]->type); + aimbs_put16(&fr->data, items[i]->data ? aim_sizetlvchain((aim_tlvlist_t **)&items[i]->data) : 0); + if (items[i]->data) + aim_writetlvchain(&fr->data, (aim_tlvlist_t **)&items[i]->data); + } + + aim_ssi_enqueue(sess, conn, fr); + + return 0; +} + +/* + * SSI Mod Item. + * + * Modifies an item in the data stored on the AIM server. An item could + * be a group, buddy, or permit or deny buddy. + * + */ +faim_export int aim_ssi_moditems(aim_session_t *sess, aim_conn_t *conn, struct aim_ssi_item **items, unsigned int num) +{ + aim_frame_t *fr; + aim_snacid_t snacid; + int i, snaclen; + + if (!sess || !conn || !items || !num) + return -EINVAL; + + snaclen = 10; /* For family, subtype, flags, and SNAC ID */ + for (i=0; i<num; i++) { + snaclen += 10; /* For length, GID, BID, type, and length */ + if (items[i]->name) + snaclen += strlen(items[i]->name); + if (items[i]->data) + snaclen += aim_sizetlvchain((aim_tlvlist_t **)&items[i]->data); + } + + if (!(fr = aim_tx_new(sess, conn, AIM_FRAMETYPE_FLAP, 0x02, snaclen))) + return -ENOMEM; + + snacid = aim_cachesnac(sess, AIM_CB_FAM_SSI, AIM_CB_SSI_MOD, 0x0000, NULL, 0); + aim_putsnac(&fr->data, AIM_CB_FAM_SSI, AIM_CB_SSI_MOD, 0x0000, snacid); + + for (i=0; i<num; i++) { + aimbs_put16(&fr->data, items[i]->name ? strlen(items[i]->name) : 0); + if (items[i]->name) + aimbs_putraw(&fr->data, items[i]->name, strlen(items[i]->name)); + aimbs_put16(&fr->data, items[i]->gid); + aimbs_put16(&fr->data, items[i]->bid); + aimbs_put16(&fr->data, items[i]->type); + aimbs_put16(&fr->data, items[i]->data ? aim_sizetlvchain((aim_tlvlist_t **)&items[i]->data) : 0); + if (items[i]->data) + aim_writetlvchain(&fr->data, (aim_tlvlist_t **)&items[i]->data); + } + + aim_ssi_enqueue(sess, conn, fr); + + return 0; +} + +/* + * SSI Del Item. + * + * Deletes an item from the data stored on the AIM server. An item + * could be a group, buddy, or permit or deny buddy. + * + */ +faim_export int aim_ssi_delitems(aim_session_t *sess, aim_conn_t *conn, struct aim_ssi_item **items, unsigned int num) +{ + aim_frame_t *fr; + aim_snacid_t snacid; + int i, snaclen; + + if (!sess || !conn || !items || !num) + return -EINVAL; + + snaclen = 10; /* For family, subtype, flags, and SNAC ID */ + for (i=0; i<num; i++) { + snaclen += 10; /* For length, GID, BID, type, and length */ + if (items[i]->name) + snaclen += strlen(items[i]->name); + if (items[i]->data) + snaclen += aim_sizetlvchain((aim_tlvlist_t **)&items[i]->data); + } + + if (!(fr = aim_tx_new(sess, conn, AIM_FRAMETYPE_FLAP, 0x02, snaclen))) + return -ENOMEM; + + snacid = aim_cachesnac(sess, AIM_CB_FAM_SSI, AIM_CB_SSI_DEL, 0x0000, NULL, 0); + aim_putsnac(&fr->data, AIM_CB_FAM_SSI, AIM_CB_SSI_DEL, 0x0000, snacid); + + for (i=0; i<num; i++) { + aimbs_put16(&fr->data, items[i]->name ? strlen(items[i]->name) : 0); + if (items[i]->name) + aimbs_putraw(&fr->data, items[i]->name, strlen(items[i]->name)); + aimbs_put16(&fr->data, items[i]->gid); + aimbs_put16(&fr->data, items[i]->bid); + aimbs_put16(&fr->data, items[i]->type); + aimbs_put16(&fr->data, items[i]->data ? aim_sizetlvchain((aim_tlvlist_t **)&items[i]->data) : 0); + if (items[i]->data) + aim_writetlvchain(&fr->data, (aim_tlvlist_t **)&items[i]->data); + } + + aim_ssi_enqueue(sess, conn, fr); + + return 0; +} + +/* + * SSI Add/Mod/Del Ack. + * + * Response to aim_ssi_additem(), aim_ssi_moditem(), or aim_ssi_delitem(). + * + */ +static int parseack(aim_session_t *sess, aim_module_t *mod, aim_frame_t *rx, aim_modsnac_t *snac, aim_bstream_t *bs) +{ + int ret = 0; + aim_rxcallback_t userfunc; + + sess->ssi.waiting_for_ack = 0; + aim_ssi_dispatch(sess, rx->conn); + + if ((userfunc = aim_callhandler(sess, rx->conn, snac->family, snac->subtype))) + ret = userfunc(sess, rx); + + return ret; +} + +/* * SSI Begin Data Modification. * * Tells the server you're going to start modifying data. @@ -175,6 +1235,8 @@ int ret = 0; aim_rxcallback_t userfunc; + sess->ssi.received_data = 1; + if ((userfunc = aim_callhandler(sess, rx->conn, snac->family, snac->subtype))) ret = userfunc(sess, rx); @@ -188,12 +1250,21 @@ return parserights(sess, mod, rx, snac, bs); else if (snac->subtype == 0x006) return parsedata(sess, mod, rx, snac, bs); + else if (snac->subtype == AIM_CB_SSI_SRVACK) + return parseack(sess, mod, rx, snac, bs); else if (snac->subtype == 0x00f) return parsedataunchanged(sess, mod, rx, snac, bs); return 0; } +static void ssi_shutdown(aim_session_t *sess, aim_module_t *mod) +{ + aim_ssi_freelist(sess); + + return; +} + faim_internal int ssi_modfirst(aim_session_t *sess, aim_module_t *mod) { @@ -204,8 +1275,7 @@ mod->flags = 0; strncpy(mod->name, "ssi", sizeof(mod->name)); mod->snachandler = snachandler; + mod->shutdown = ssi_shutdown; return 0; } - -