From: Sean E. <sea...@us...> - 2003-06-12 22:19:13
|
Update of /cvsroot/gaim/gaim/src/protocols/oscar In directory sc8-pr-cvs1:/tmp/cvs-serv10317/src/protocols/oscar Modified Files: aim.h icon.c oscar.c service.c ssi.c Log Message: Another reason not to use CVS. This saves your buddy icons to the OSCAR servers (I hope). It's not fully tested and it may screw things up, but I wanted to get it in CVS so other people can try to break this, and KingAnt can review my work (that libfaim wizard that he is.) Don't use CVS. I'd like to thank Christian Hammond, for his work on the account and prpl APIs that made it so easy to add the hook into buddy icon changing. I'd like to thank Mark Doliner for laying most of the groundwork and for his kind words of encouragement. I'd like to thank myself for making the buddy icon selector real sexy-like. I'd like to thank the Lord almighty for giving me the strength to do this. I promised myself I wouldn't cry. I never imagined I'd be committing this. Index: aim.h =================================================================== RCS file: /cvsroot/gaim/gaim/src/protocols/oscar/aim.h,v retrieving revision 1.95 retrieving revision 1.96 diff -u -d -r1.95 -r1.96 --- aim.h 12 Jun 2003 03:27:58 -0000 1.95 +++ aim.h 12 Jun 2003 22:19:05 -0000 1.96 @@ -1107,7 +1107,7 @@ /* 0x0010 - icon.c */ -faim_export int aim_icon_upload(aim_session_t *sess, int num, const fu8_t *icon, fu16_t iconlen); +faim_export int aim_icon_upload(aim_session_t *sess, aim_conn_t *conn, const fu8_t *icon, fu16_t iconlen); faim_export int aim_icon_request(aim_session_t *sess, const char *sn, const fu8_t *iconstr, fu16_t iconstrlen); Index: icon.c =================================================================== RCS file: /cvsroot/gaim/gaim/src/protocols/oscar/icon.c,v retrieving revision 1.6 retrieving revision 1.7 diff -u -d -r1.6 -r1.7 --- icon.c 25 Mar 2003 05:00:59 -0000 1.6 +++ icon.c 12 Jun 2003 22:19:05 -0000 1.7 @@ -14,18 +14,17 @@ * * @param sess The oscar session. * @param conn The icon connection for this session. - * @param num The reference number of the icon you are uploading. * @param icon The raw data of the icon image file. * @param iconlen Length of the raw data of the icon image file. * @return Return 0 if no errors, otherwise return the error number. */ -faim_export int aim_icon_upload(aim_session_t *sess, int num, const fu8_t *icon, fu16_t iconlen) +faim_export int aim_icon_upload(aim_session_t *sess, aim_conn_t *conn, const fu8_t *icon, fu16_t iconlen) { aim_conn_t *conn; aim_frame_t *fr; aim_snacid_t snacid; - if (!sess || !(conn = aim_conn_findbygroup(sess, 0x0010)) || !num || !icon || !iconlen) + if (!sess || !(conn = aim_conn_findbygroup(sess, 0x0010)) || !icon || !iconlen) return -EINVAL; if (!(fr = aim_tx_new(sess, conn, AIM_FRAMETYPE_FLAP, 0x02, 10 + 2 + 2+iconlen))) @@ -34,7 +33,7 @@ aim_putsnac(&fr->data, 0x0010, 0x0002, 0x0000, snacid); /* The reference number for the icon */ - aimbs_put16(&fr->data, num); + aimbs_put16(&fr->data, 1); /* The icon */ aimbs_put16(&fr->data, iconlen); Index: oscar.c =================================================================== RCS file: /cvsroot/gaim/gaim/src/protocols/oscar/oscar.c,v retrieving revision 1.385 retrieving revision 1.386 diff -u -d -r1.385 -r1.386 --- oscar.c 12 Jun 2003 04:20:07 -0000 1.385 +++ oscar.c 12 Jun 2003 22:19:05 -0000 1.386 @@ -51,6 +51,7 @@ #include "core.h" #include "proxy.h" #include "aim.h" +#include "md5.h" #ifdef _WIN32 #include "win32dep.h" @@ -84,6 +85,7 @@ guint icopa; gboolean iconconnecting; + gboolean set_icon; GSList *create_rooms; @@ -96,7 +98,7 @@ gboolean chpass; char *oldp; char *newp; - + GSList *oscar_chats; GSList *direct_ims; GSList *file_transfers; @@ -281,6 +283,8 @@ static int oscar_sendfile_ack (aim_session_t *, aim_frame_t *, ...); static int oscar_sendfile_done (aim_session_t *, aim_frame_t *, ...); +static int gaim_buddyiconreq (aim_session_t *, aim_frame_t *, ...); + /* for icons */ static gboolean gaim_icon_timerfunc(gpointer data); @@ -1143,6 +1147,7 @@ aim_conn_addhandler(sess, bosconn, 0x0009, 0x0001, gaim_parse_genericerr, 0); aim_conn_addhandler(sess, bosconn, 0x0001, 0x001f, gaim_memrequest, 0); aim_conn_addhandler(sess, bosconn, 0x0001, 0x000f, gaim_selfinfo, 0); + aim_conn_addhandler(sess, bosconn, 0x0001, 0x0021, gaim_buddyiconreq,0); aim_conn_addhandler(sess, bosconn, AIM_CB_FAM_ICQ, AIM_CB_ICQ_OFFLINEMSG, gaim_offlinemsg, 0); aim_conn_addhandler(sess, bosconn, AIM_CB_FAM_ICQ, AIM_CB_ICQ_OFFLINEMSGCOMPLETE, gaim_offlinemsgdone, 0); aim_conn_addhandler(sess, bosconn, AIM_CB_FAM_POP, 0x0002, gaim_popup, 0); @@ -3532,12 +3537,6 @@ struct buddyinfo *bi; aim_conn_t *conn; - if (!od->requesticon) { - gaim_debug(GAIM_DEBUG_MISC, "oscar", - "no more icons to request\n"); - return FALSE; - } - conn = aim_getconn_type(od->sess, AIM_CONN_TYPE_ICON); if (!conn && !od->iconconnecting) { aim_reqservice(od->sess, od->conn, AIM_CONN_TYPE_ICON); @@ -3545,6 +3544,38 @@ return FALSE; } + if (od->set_icon) { + const char *iconfile; + if ((iconfile = gaim_account_get_buddy_icon(gaim_connection_get_account(gc)))) { + FILE *file; + struct stat st; + gaim_debug(GAIM_DEBUG_INFO, "Uploading icon: %s\n", iconfile); + if (!stat(iconfile, &st)) { + char *buf = g_malloc(st.st_size); + file = fopen(iconfile, "rb"); + if (file) { + int len = fread(buf, 1, st.st_size, file); + gaim_debug(GAIM_DEBUG_INFO, "oscar", + "Uploading icon to icon server\n"); + aim_icon_upload(od->sess, aim_getconn_type(od->sess, AIM_CONN_TYPE_ICON), buf, st.st_size); + fclose(file); + } else + gaim_debug(GAIM_DEBUG_ERROR, "oscar", + "Can't open buddy icon file!\n"); + g_free(buf); + } else + gaim_debug(GAIM_DEBUG_ERROR, "oscar", + "Can't stat buddy icon file!\n"); + } + od->set_icon = FALSE; + } + + if (!od->requesticon) { + gaim_debug(GAIM_DEBUG_MISC, "oscar", + "no more icons to request\n"); + return FALSE; + } + bi = g_hash_table_lookup(od->buddyinfo, (char *)od->requesticon->data); if (bi && (bi->iconcsumlen > 0)) { aim_icon_request(od->sess, od->requesticon->data, bi->iconcsum, bi->iconcsumlen); @@ -5424,6 +5455,60 @@ return ret; } + +static int gaim_buddyiconreq (aim_session_t *sess, aim_frame_t *fr, ...) { + GaimConnection *gc = sess->aux_data; + struct oscar_data *od = gc->proto_data; + + char *md5 = NULL; + fu16_t type; + fu8_t length, cached; + va_list ap; + va_start(ap, fr); + type = va_arg(ap, int); + switch (type) { + case 0x0001: + case 0x0000: + cached = va_arg(ap, int); + length = va_arg(ap, int); + md5 = va_arg(ap, char*); + break; + } + va_end(ap); + if (cached == 0x41) { + if (!aim_getconn_type(od->sess, AIM_CONN_TYPE_ICON) && !od->iconconnecting) { + od->iconconnecting = TRUE; + od->set_icon = TRUE; + aim_reqservice(od->sess, od->conn, AIM_CONN_TYPE_ICON); + } else { + const char *iconfile; + if ((iconfile = gaim_account_get_buddy_icon(gaim_connection_get_account(gc)))) { + FILE *file; + struct stat st; + + if (!stat(iconfile, &st)) { + char *buf = g_malloc(st.st_size); + file = fopen(iconfile, "rb"); + if (file) { + int len = fread(buf, 1, st.st_size, file); + gaim_debug(GAIM_DEBUG_INFO, "oscar", + "Uploading icon to icon server\n"); + aim_icon_upload(od->sess, aim_getconn_type(od->sess, AIM_CONN_TYPE_ICON), buf, st.st_size); + fclose(file); + } else + gaim_debug(GAIM_DEBUG_ERROR, "oscar", + "Can't open buddy icon file!\n"); + g_free(buf); + } else + gaim_debug(GAIM_DEBUG_ERROR, "oscar", + "Can't stat buddy icon file!\n"); + } + } + } else if (cached == 0x81) + aim_ssi_seticon(od->sess, md5, length); +} + + /* * We have just established a socket with the other dude, so set up some handlers. */ @@ -5929,6 +6014,37 @@ g_free(substituted); } +static void oscar_set_icon(GaimConnection *gc, const char *iconfile) +{ + struct oscar_data *od; + aim_session_t *sess; + od = gc->proto_data; + FILE *file; + struct stat st; + sess = od->sess; + if (!stat(iconfile, &st)) { + char *buf = g_malloc(st.st_size); + file = fopen(iconfile, "rb"); + if (file) { + int len = fread(buf, 1, st.st_size, file); + char md5[16]; + md5_state_t *state = g_malloc(sizeof(md5_state_t)); + md5_init(state); + md5_append(state, buf, len); + md5_finish(state, md5); + fclose(file); + g_free(state); + aim_ssi_seticon(sess, md5, 16); + } else + gaim_debug(GAIM_DEBUG_ERROR, "oscar", + "Can't open buddy icon file!\n"); + g_free(buf); + } else + gaim_debug(GAIM_DEBUG_ERROR, "oscar", + "Can't stat buddy icon file!\n"); +} + + static GList *oscar_actions(GaimConnection *gc) { struct oscar_data *od = gc->proto_data; @@ -5959,9 +6075,9 @@ /* AIM actions */ m = g_list_append(m, NULL); - pam = g_new0(struct proto_actions_menu, 1); + pam = g_new0(struct proto_actions_menu, 1); pam->label = _("Format Screenname"); - pam->callback = oscar_show_format_screenname; + pam->callback = oscar_format_screenname; pam->gc = gc; m = g_list_append(m, pam); @@ -6099,7 +6215,8 @@ #endif NULL, oscar_convo_closed, - NULL + NULL, + oscar_set_icon }; static GaimPluginInfo info = Index: service.c =================================================================== RCS file: /cvsroot/gaim/gaim/src/protocols/oscar/service.c,v retrieving revision 1.12 retrieving revision 1.13 diff -u -d -r1.12 -r1.13 --- service.c 12 Jun 2003 04:20:08 -0000 1.12 +++ service.c 12 Jun 2003 22:19:06 -0000 1.13 @@ -946,26 +946,38 @@ * Subtype 0x0021 - Receive our extended status * * This is used for MAC non-away "away" messages, and maybe ICQ extended status messages? + * It's also used to tell the client whether or not it needs to upload an SSI buddy icon... who engineers this stuff, anyway? */ static int aim_parse_extstatus(aim_session_t *sess, aim_module_t *mod, aim_frame_t *rx, aim_modsnac_t *snac, aim_bstream_t *bs) { - int ret = 0; + int ret = 0, i; aim_rxcallback_t userfunc; - char *msg = NULL; + char *msg = NULL, *md5 = NULL; fu16_t type; - fu8_t number, length; - - type = aimbs_get16(bs); /* 0x0002 */ - number = aimbs_get8(bs); /* 0x04 */ - length = aimbs_get8(bs); /* the first length */ - msg = aimbs_getstr(bs, aimbs_get16(bs)); /* the second length is just for the message */ - - if ((userfunc = aim_callhandler(sess, rx->conn, snac->family, snac->subtype))) - ret = userfunc(sess, rx, msg); - - free(msg); + fu8_t number, length, cached; - return ret; + type = aimbs_get16(bs); + printf("blah: %d\n", type); + if ((userfunc = aim_callhandler(sess, rx->conn, snac->family, snac->subtype))) { + switch (type) { + case 0x0001: + case 0x0000: /* not sure what the difference between 1 and 0 is */ + cached = aimbs_get8(bs); + length = aimbs_get8(bs); + md5 = aimbs_getraw(bs, length); + ret = userfunc(sess, rx, type, cached, length, md5); + free(md5); + break; + case 0x0002: + number = aimbs_get8(bs); /* 0x04 */ + length = aimbs_get8(bs); /* the first length */ + msg = aimbs_getstr(bs, aimbs_get16(bs)); /* the second length is just for the message */ + ret = userfunc(sess, rx, msg); + free(msg); + break; + } + return ret; + } } static int snachandler(aim_session_t *sess, aim_module_t *mod, aim_frame_t *rx, aim_modsnac_t *snac, aim_bstream_t *bs) Index: ssi.c =================================================================== RCS file: /cvsroot/gaim/gaim/src/protocols/oscar/ssi.c,v retrieving revision 1.43 retrieving revision 1.44 diff -u -d -r1.43 -r1.44 --- ssi.c 24 May 2003 13:47:14 -0000 1.43 +++ ssi.c 12 Jun 2003 22:19:07 -0000 1.44 @@ -1023,7 +1023,7 @@ * * @param sess The oscar session. * @param iconcsum The MD5 checksum of the icon you are using. - * @param iconcsumlen Length of the MD5 checksum given above. Should be 10 bytes. + * @param iconcsumlen Length of the MD5 checksum given above. Should be 0x10 bytes. * @return Return 0 if no errors, otherwise return the error number. */ faim_export int aim_ssi_seticon(aim_session_t *sess, fu8_t *iconsum, fu16_t iconsumlen) @@ -1035,30 +1035,31 @@ if (!sess || !iconsum || !iconsumlen) return -EINVAL; - /* Create the data for the TLV containing the icon checksum */ if (!(csumdata = (fu8_t *)malloc((iconsumlen+2)*sizeof(fu8_t)))) return -ENOMEM; csumdata[0] = 0x00; csumdata[1] = 0x10; memcpy(&csumdata[2], iconsum, iconsumlen); + + + /* Need to add the x00d5 TLV to the TLV chain */ + aim_addtlvtochain_raw(&data, 0x00d5, (iconsumlen+2) * sizeof(fu8_t), csumdata); - /* Need to add the x0131 TLV to the TLV chain */ + /* This TLV is added to cache the icon. */ aim_addtlvtochain_noval(&data, 0x0131); - /* Need to add the x00d5 TLV to the TLV chain */ - aim_addtlvtochain_raw(&data, 0x00d5, 0x0012, csumdata); if ((tmp = aim_ssi_itemlist_finditem(sess->ssi.local, NULL, "0", AIM_SSI_TYPE_ICONINFO))) { aim_freetlvchain(&tmp->data); tmp->data = data; } else { - tmp = aim_ssi_itemlist_add(&sess->ssi.local, "0", 0x0000, 0xFFFF, AIM_SSI_TYPE_ICONINFO, data); + tmp = aim_ssi_itemlist_add(&sess->ssi.local, "1", 0x0000, 0x51F4, AIM_SSI_TYPE_ICONINFO, data); aim_freetlvchain(&data); } /* Sync our local list with the server list */ aim_ssi_sync(sess); - + free(csumdata); return 0; } |