--- /dev/null Thu Jan 01 00:00:00 1970 +0000 +++ b/src/protocols/msn/ft.c Thu Feb 06 10:13:18 2003 +0000 @@ -0,0 +1,465 @@ +/** + * @file msn.c The MSN protocol plugin + * + * gaim + * + * Copyright (C) 2003, Christian Hammond <chipx86@gnupdate.org> + * + * This program is free software; you can redistribute it and/or modify + * it under the terms of the GNU General Public License as published by + * the Free Software Foundation; either version 2 of the License, or + * (at your option) any later version. + * + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * + * You should have received a copy of the GNU General Public License + * along with this program; if not, write to the Free Software + * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA + * + */ +#include "msn.h" + +G_MODULE_IMPORT GSList *connections; + +static struct gaim_xfer * +find_xfer_by_cookie(struct gaim_connection *gc, unsigned long cookie) +{ + GSList *g; + struct msn_data *md = (struct msn_data *)gc->proto_data; + struct gaim_xfer *xfer = NULL; + struct msn_xfer_data *xfer_data; + + for (g = md->file_transfers; g != NULL; g = g->next) { + xfer = (struct gaim_xfer *)g->data; + xfer_data = (struct msn_xfer_data *)xfer->data; + + if (xfer_data->cookie == cookie) + break; + + xfer = NULL; + } + + return xfer; +} + +static void +msn_xfer_init(struct gaim_xfer *xfer) +{ + struct gaim_account *account; + struct msn_xfer_data *xfer_data; + struct msn_switchboard *ms; + char header[MSN_BUF_LEN]; + char buf[MSN_BUF_LEN]; + + account = gaim_xfer_get_account(xfer); + + ms = msn_find_switch(account->gc, xfer->who); + + xfer_data = (struct msn_xfer_data *)xfer->data; + + if (gaim_xfer_get_type(xfer) == GAIM_XFER_RECEIVE) { + /* + * NOTE: We actually have to wait for the next Invitation message + * before the transfer starts. We handle that in + * msn_xfer_start(). + */ + + g_snprintf(header, sizeof(header), + "MIME-Version: 1.0\r\n" + "Content-Type: text/x-msmsgsinvite; charset=UTF-8\r\n\r\n" + "Invitation-Command: ACCEPT\r\n" + "Invitation-Cookie: %lu\r\n" + "Launch-Application: FALSE\r\n" + "Request-Data: IP-Address:\r\n", + (unsigned long)xfer_data->cookie); + + g_snprintf(buf, sizeof(buf), "MSG %u N %d\r\n%s\r\n\r\n", + ++ms->trId, strlen(header) + strlen("\r\n\r\n"), + header); + + if (msn_write(ms->fd, buf, strlen(buf)) < 0) { + msn_kill_switch(ms); + gaim_xfer_destroy(xfer); + + return; + } + } +} + +static void +msn_xfer_start(struct gaim_xfer *xfer) +{ + struct msn_xfer_data *xfer_data; + + xfer_data = (struct msn_xfer_data *)xfer->data; + + xfer_data->transferring = TRUE; + + if (gaim_xfer_get_type(xfer) == GAIM_XFER_RECEIVE) { + char sendbuf[MSN_BUF_LEN]; + + /* Send the TFR string to request the start of a transfer. */ + g_snprintf(sendbuf, sizeof(sendbuf), "TFR\r\n"); + + if (msn_write(xfer->fd, sendbuf, strlen(sendbuf)) < 0) { + gaim_xfer_cancel(xfer); + } + } +} + +static void +msn_xfer_end(struct gaim_xfer *xfer) +{ + struct gaim_account *account; + struct msn_xfer_data *xfer_data; + struct msn_data *md; + + account = gaim_xfer_get_account(xfer); + xfer_data = (struct msn_xfer_data *)xfer->data; + md = (struct msn_data *)account->gc->proto_data; + + if (gaim_xfer_get_type(xfer) == GAIM_XFER_RECEIVE) { + char sendbuf[MSN_BUF_LEN]; + + g_snprintf(sendbuf, sizeof(sendbuf), "BYE 16777989\r\n"); + + msn_write(xfer->fd, sendbuf, strlen(sendbuf)); + + md->file_transfers = g_slist_remove(md->file_transfers, xfer); + + g_free(xfer_data); + xfer->data = NULL; + } +} + +static void +msn_xfer_cancel(struct gaim_xfer *xfer) +{ + struct gaim_account *account; + struct msn_xfer_data *xfer_data; + struct msn_data *md; + + account = gaim_xfer_get_account(xfer); + xfer_data = (struct msn_xfer_data *)xfer->data; + md = (struct msn_data *)account->gc->proto_data; + + if (gaim_xfer_get_type(xfer) == GAIM_XFER_RECEIVE) { + md->file_transfers = g_slist_remove(md->file_transfers, xfer); + + g_free(xfer_data); + xfer->data = NULL; + } +} + +static size_t +msn_xfer_read(char **buffer, struct gaim_xfer *xfer) +{ + unsigned char header[3]; + size_t len, size; + + if (read(xfer->fd, header, sizeof(header)) < 3) { + gaim_xfer_set_completed(xfer, TRUE); + return 0; + } + + if (header[0] != 0) { + debug_printf("MSNFTP: Invalid header[0]: %d. Aborting.\n", + header[0]); + return 0; + } + + size = header[1] | (header[2] << 8); + + *buffer = g_new0(char, size); + + for (len = 0; + len < size; + len += read(xfer->fd, *buffer + len, size - len)) + ; + + if (len == 0) + gaim_xfer_set_completed(xfer, TRUE); + + return len; +} + +static size_t +msn_xfer_write(const char *buffer, size_t size, struct gaim_xfer *xfer) +{ + return 0; +} + +static int +msn_process_msnftp(struct gaim_xfer *xfer, gint source, const char *buf) +{ + struct msn_xfer_data *xfer_data; + struct gaim_account *account; + char sendbuf[MSN_BUF_LEN]; + + xfer_data = (struct msn_xfer_data *)xfer->data; + account = gaim_xfer_get_account(xfer); + + if (!g_strncasecmp(buf, "VER MSNFTP", 10)) { + /* Send the USR string */ + g_snprintf(sendbuf, sizeof(sendbuf), "USR %s %lu\r\n", + account->gc->username, + (unsigned long)xfer_data->authcookie); + + if (msn_write(source, sendbuf, strlen(sendbuf)) < 0) { + gaim_xfer_cancel(xfer); /* ? */ + + return 0; + } + } + else if (!g_strncasecmp(buf, "FIL", 3)) { + gaim_input_remove(xfer_data->inpa); + xfer_data->inpa = 0; + + gaim_xfer_start(xfer, source, NULL, 0); + } +#if 0 + char *tmp = buf; + + /* + * This data is the size, but we already have + * the size, so who cares. + */ + GET_NEXT(tmp); + + /* Send the TFR string to request the start of a transfer. */ + g_snprintf(sendbuf, sizeof(sendbuf), "TFR\r\n"); + + + if (msn_write(source, sendbuf, strlen(sendbuf)) < 0) { + gaim_xfer_cancel(xfer); + + return 0; + } +#endif + + return 1; +} + +static void +msn_msnftp_cb(gpointer data, gint source, GaimInputCondition cond) +{ + struct gaim_xfer *xfer; + struct msn_xfer_data *xfer_data; + char buf[MSN_BUF_LEN]; + gboolean cont = TRUE; + size_t len; + + xfer = (struct gaim_xfer *)data; + xfer_data = (struct msn_xfer_data *)xfer->data; + + len = read(source, buf, sizeof(buf)); + + if (len <= 0) { + gaim_xfer_cancel(xfer); + return; + } + + xfer_data->rxqueue = g_realloc(xfer_data->rxqueue, + len + xfer_data->rxlen); + memcpy(xfer_data->rxqueue + xfer_data->rxlen, buf, len); + xfer_data->rxlen += len; + + while (cont) { + char *end = xfer_data->rxqueue; + char *cmd; + int cmdlen; + int i = 0; + + if (!xfer_data->rxlen) + return; + + for (i = 0; i < xfer_data->rxlen - 1; end++, i++) { + if (*end == '\r' && *(end + 1) == '\n') + break; + } + + if (i == xfer_data->rxlen - 1) + return; + + cmdlen = end - xfer_data->rxqueue + 2; + cmd = xfer_data->rxqueue; + + xfer_data->rxlen -= cmdlen; + + if (xfer_data->rxlen) + xfer_data->rxqueue = g_memdup(cmd + cmdlen, xfer_data->rxlen); + else { + xfer_data->rxqueue = NULL; + cmd = g_realloc(cmd, cmdlen + 1); + } + + cmd[cmdlen] = '\0'; + + g_strchomp(cmd); + + cont = msn_process_msnftp(xfer, source, cmd); + + g_free(cmd); + } +} + +static void +msn_msnftp_connect(gpointer data, gint source, GaimInputCondition cond) +{ + struct gaim_account *account; + struct gaim_xfer *xfer; + struct msn_xfer_data *xfer_data; + char buf[MSN_BUF_LEN]; + + xfer = (struct gaim_xfer *)data; + account = gaim_xfer_get_account(xfer); + xfer_data = (struct msn_xfer_data *)xfer->data; + + if (source == -1 || !g_slist_find(connections, account->gc)) { + debug_printf("MSNFTP: Error establishing connection\n"); + close(source); + + gaim_xfer_cancel(xfer); + + return; + } + + g_snprintf(buf, sizeof(buf), "VER MSNFTP\r\n"); + + if (msn_write(source, buf, strlen(buf)) < 0) { + gaim_xfer_cancel(xfer); + return; + } + + xfer_data->inpa = gaim_input_add(source, GAIM_INPUT_READ, + msn_msnftp_cb, xfer); +} + +void +msn_process_ft_msg(struct msn_switchboard *ms, char *msg) +{ + struct gaim_xfer *xfer; + struct msn_xfer_data *xfer_data; + struct msn_data *md = ms->gc->proto_data; + char *tmp = msg; + + if (strstr(msg, "Application-GUID: " MSN_FT_GUID) && + strstr(msg, "Invitation-Command: INVITE")) { + + /* + * First invitation message, requesting an ACCEPT or CANCEL from + * the recipient. Used in incoming file transfers. + */ + + char *filename; + char *cookie_s, *filesize_s; + + tmp = strstr(msg, "Invitation-Cookie"); + GET_NEXT(tmp); + cookie_s = tmp; + GET_NEXT(tmp); + GET_NEXT(tmp); + filename = tmp; + + /* Needed for filenames with spaces */ + tmp = strchr(tmp, '\r'); + *tmp = '\0'; + tmp += 2; + + GET_NEXT(tmp); + filesize_s = tmp; + GET_NEXT(tmp); + + /* Setup the MSN-specific file transfer data */ + xfer_data = g_new0(struct msn_xfer_data, 1); + xfer_data->cookie = atoi(cookie_s); + xfer_data->transferring = FALSE; + + /* Build the file transfer handle. */ + xfer = gaim_xfer_new(ms->gc->account, GAIM_XFER_RECEIVE, ms->msguser); + xfer->data = xfer_data; + + /* Set the info about the incoming file. */ + gaim_xfer_set_filename(xfer, filename); + gaim_xfer_set_size(xfer, atoi(filesize_s)); + + /* Setup our I/O op functions */ + gaim_xfer_set_init_fnc(xfer, msn_xfer_init); + gaim_xfer_set_start_fnc(xfer, msn_xfer_start); + gaim_xfer_set_end_fnc(xfer, msn_xfer_end); + gaim_xfer_set_cancel_fnc(xfer, msn_xfer_cancel); + gaim_xfer_set_read_fnc(xfer, msn_xfer_read); + gaim_xfer_set_write_fnc(xfer, msn_xfer_write); + + /* Keep track of this transfer for later. */ + md->file_transfers = g_slist_append(md->file_transfers, xfer); + + /* Now perform the request */ + gaim_xfer_request(xfer); + } + else if (strstr(msg, "Invitation-Command: ACCEPT")) { + + /* + * XXX I hope these checks don't return false positives, but they + * seem like they should work. The only issue is alternative + * protocols, *maybe*. + */ + + if (strstr(msg, "AuthCookie:")) { + + /* + * Second invitation request, sent after the recipient accepts + * the request. Used in incoming file transfers. + */ + char *cookie_s, *ip, *port_s, *authcookie_s; + char ip_s[16]; + + tmp = strstr(msg, "Invitation-Cookie"); + GET_NEXT(tmp); + cookie_s = tmp; + GET_NEXT(tmp); + GET_NEXT(tmp); + ip = tmp; + GET_NEXT(tmp); + GET_NEXT(tmp); + port_s = tmp; + GET_NEXT(tmp); + GET_NEXT(tmp); + authcookie_s = tmp; + GET_NEXT(tmp); + + xfer = find_xfer_by_cookie(ms->gc, atoi(cookie_s)); + + if (xfer == NULL) + { + debug_printf("MSNFTP : Cookie not found. " + "File transfer aborted.\n"); + return; + } + + xfer_data = (struct msn_xfer_data *)xfer->data; + xfer_data->authcookie = atol(authcookie_s); + + strncpy(ip_s, ip, sizeof(ip_s)); + + if (proxy_connect(ip_s, atoi(port_s), + msn_msnftp_connect, xfer) != 0) { + + gaim_xfer_cancel(xfer); + + return; + } + } + else + { + /* + * An accept message from the recipient. Used in outgoing + * file transfers. + */ + } + } +} +