/* Tlen Protocol Plugin for Miranda NG Copyright (C) 2004-2007 Piotr Piastucki 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 "tlen.h" #include #include #include #include #include "tlen_list.h" #include "tlen_p2p_old.h" static void TlenFileReceiveParse(TLEN_FILE_TRANSFER *ft) { int i; char *p; TLEN_FILE_PACKET *rpacket, *packet; rpacket = NULL; if (ft->state == FT_CONNECTING) { rpacket = TlenP2PPacketReceive(ft->s); if (rpacket != NULL) { p = rpacket->packet; if (rpacket->type == TLEN_FILE_PACKET_FILE_LIST) { // list of files (length & name) ft->fileCount = (int)(*((DWORD*)p)); ft->files = (char **) mir_alloc(sizeof(char *) * ft->fileCount); ft->filesSize = (long *) mir_alloc(sizeof(long) * ft->fileCount); ft->currentFile = 0; ft->allFileTotalSize = 0; ft->allFileReceivedBytes = 0; p += sizeof(DWORD); for (i=0;ifileCount;i++) { ft->filesSize[i] = (long)(*((DWORD*)p)); ft->allFileTotalSize += ft->filesSize[i]; p += sizeof(DWORD); ft->files[i] = (char *)mir_alloc(256); memcpy(ft->files[i], p, 256); p += 256; } if ((packet=TlenP2PPacketCreate(3*sizeof(DWORD))) == NULL) { ft->state = FT_ERROR; } else { TlenP2PPacketSetType(packet, TLEN_FILE_PACKET_FILE_LIST_ACK); TlenP2PPacketSend(ft->s, packet); TlenP2PPacketFree(packet); ft->state = FT_INITIALIZING; TlenLog(ft->proto, "Change to FT_INITIALIZING"); } } TlenP2PPacketFree(rpacket); } else { ft->state = FT_ERROR; } } else if (ft->state == FT_INITIALIZING) { char *fullFileName; if ((packet=TlenP2PPacketCreate(3*sizeof(DWORD))) != NULL) { TlenP2PPacketSetType(packet, TLEN_FILE_PACKET_FILE_REQUEST); // file request TlenP2PPacketPackDword(packet, ft->currentFile); TlenP2PPacketPackDword(packet, 0); TlenP2PPacketPackDword(packet, 0); TlenP2PPacketSend(ft->s, packet); TlenP2PPacketFree(packet); fullFileName = (char *) mir_alloc(strlen(ft->szSavePath) + strlen(ft->files[ft->currentFile]) + 2); strcpy(fullFileName, ft->szSavePath); if (fullFileName[strlen(fullFileName)-1] != '\\') strcat(fullFileName, "\\"); strcat(fullFileName, ft->files[ft->currentFile]); ft->fileId = _open(fullFileName, _O_BINARY|_O_WRONLY|_O_CREAT|_O_TRUNC, _S_IREAD|_S_IWRITE); ft->fileReceivedBytes = 0; ft->fileTotalSize = ft->filesSize[ft->currentFile]; TlenLog(ft->proto, "Saving to [%s] [%d]", fullFileName, ft->filesSize[ft->currentFile]); mir_free(fullFileName); ft->state = FT_RECEIVING; TlenLog(ft->proto, "Change to FT_RECEIVING"); } else { ft->state = FT_ERROR; } } else if (ft->state == FT_RECEIVING) { PROTOFILETRANSFERSTATUS pfts; memset(&pfts, 0, sizeof(PROTOFILETRANSFERSTATUS)); pfts.cbSize = sizeof(PROTOFILETRANSFERSTATUS); pfts.hContact = ft->hContact; pfts.pszFiles = ft->files; pfts.totalFiles = ft->fileCount; pfts.currentFileNumber = ft->currentFile; pfts.totalBytes = ft->allFileTotalSize; pfts.szWorkingDir = NULL; pfts.szCurrentFile = ft->files[ft->currentFile]; pfts.currentFileSize = ft->filesSize[ft->currentFile]; pfts.currentFileTime = 0; TlenLog(ft->proto, "Receiving data..."); while (ft->state == FT_RECEIVING) { rpacket = TlenP2PPacketReceive(ft->s); if (rpacket != NULL) { p = rpacket->packet; if (rpacket->type == TLEN_FILE_PACKET_FILE_DATA) { // file data int writeSize; writeSize = rpacket->len - 2 * sizeof(DWORD) ; // skip file offset if (_write(ft->fileId, p + 2 * sizeof(DWORD), writeSize) != writeSize) { ft->state = FT_ERROR; } else { ft->fileReceivedBytes += writeSize; ft->allFileReceivedBytes += writeSize; pfts.totalProgress = ft->allFileReceivedBytes; pfts.currentFileProgress = ft->fileReceivedBytes; ProtoBroadcastAck(ft->proto->m_szModuleName, ft->hContact, ACKTYPE_FILE, ACKRESULT_DATA, ft, (LPARAM) &pfts); } } else if (rpacket->type == TLEN_FILE_PACKET_END_OF_FILE) { // end of file _close(ft->fileId); TlenLog(ft->proto, "Finishing this file..."); if (ft->currentFile >= ft->fileCount-1) { ft->state = FT_DONE; } else { ft->currentFile++; ft->state = FT_INITIALIZING; TlenLog(ft->proto, "File received, advancing to the next file..."); ProtoBroadcastAck(ft->proto->m_szModuleName, ft->hContact, ACKTYPE_FILE, ACKRESULT_NEXTFILE, ft, 0); } } TlenP2PPacketFree(rpacket); } else { ft->state = FT_ERROR; } } } } static void TlenFileReceivingConnection(HANDLE hConnection, DWORD dwRemoteIP, void * pExtra) { HANDLE slisten; TLEN_FILE_TRANSFER *ft; TlenProtocol *proto = (TlenProtocol *)pExtra; ft = TlenP2PEstablishIncomingConnection(proto, hConnection, LIST_FILE, TRUE); if (ft != NULL) { slisten = ft->s; ft->s = hConnection; TlenLog(ft->proto, "Set ft->s to %d (saving %d)", hConnection, slisten); TlenLog(ft->proto, "Entering send loop for this file connection... (ft->s is hConnection)"); while (ft->state != FT_DONE && ft->state != FT_ERROR) { TlenFileReceiveParse(ft); } if (ft->state == FT_DONE) ProtoBroadcastAck(ft->proto->m_szModuleName, ft->hContact, ACKTYPE_FILE, ACKRESULT_SUCCESS, ft, 0); else ProtoBroadcastAck(ft->proto->m_szModuleName, ft->hContact, ACKTYPE_FILE, ACKRESULT_FAILED, ft, 0); TlenLog(ft->proto, "Closing connection for this file transfer... (ft->s is now hBind)"); ft->s = slisten; TlenLog(ft->proto, "ft->s is restored to %d", ft->s); if (ft->s != hConnection) { Netlib_CloseHandle(hConnection); } if (ft->hFileEvent != NULL) SetEvent(ft->hFileEvent); } else { Netlib_CloseHandle(hConnection); } } static void __cdecl TlenFileReceiveThread(TLEN_FILE_TRANSFER *ft) { TlenLog(ft->proto, "Thread started: type=file_receive server='%s' port='%d'", ft->hostName, ft->wPort); ft->mode = FT_RECV; NETLIBOPENCONNECTION nloc = { sizeof(nloc) }; nloc.szHost = ft->hostName; nloc.wPort = ft->wPort; ProtoBroadcastAck(ft->proto->m_szModuleName, ft->hContact, ACKTYPE_FILE, ACKRESULT_CONNECTING, ft, 0); HANDLE s = (HANDLE) CallService(MS_NETLIB_OPENCONNECTION, (WPARAM) ft->proto->hNetlibUser, (LPARAM) &nloc); if (s != NULL) { ft->s = s; TlenLog(ft->proto, "Entering file receive loop"); TlenP2PEstablishOutgoingConnection(ft, TRUE); while (ft->state != FT_DONE && ft->state != FT_ERROR) { TlenFileReceiveParse(ft); } if (ft->s) { Netlib_CloseHandle(s); } ft->s = NULL; } else { ft->pfnNewConnectionV2 = TlenFileReceivingConnection; TlenLog(ft->proto, "Connection failed - receiving as server"); s = TlenP2PListen(ft); if (s != NULL) { HANDLE hEvent; char *nick; ft->s = s; hEvent = CreateEvent(NULL, FALSE, FALSE, NULL); ft->hFileEvent = hEvent; ft->currentFile = 0; ft->state = FT_CONNECTING; nick = TlenNickFromJID(ft->jid); TlenSend(ft->proto, "", nick, ft->iqId, ft->localName, ft->wLocalPort); mir_free(nick); TlenLog(ft->proto, "Waiting for the file to be received..."); WaitForSingleObject(hEvent, INFINITE); ft->hFileEvent = NULL; CloseHandle(hEvent); TlenLog(ft->proto, "Finish all files"); Netlib_CloseHandle(s); } else { ft->state = FT_ERROR; } } TlenListRemove(ft->proto, LIST_FILE, ft->iqId); if (ft->state == FT_DONE) ProtoBroadcastAck(ft->proto->m_szModuleName, ft->hContact, ACKTYPE_FILE, ACKRESULT_SUCCESS, ft, 0); else { char *nick; nick = TlenNickFromJID(ft->jid); TlenSend(ft->proto, "", nick, ft->iqId); mir_free(nick); ProtoBroadcastAck(ft->proto->m_szModuleName, ft->hContact, ACKTYPE_FILE, ACKRESULT_FAILED, ft, 0); } TlenLog(ft->proto, "Thread ended: type=file_receive server='%s'", ft->hostName); TlenP2PFreeFileTransfer(ft); } static void TlenFileSendParse(TLEN_FILE_TRANSFER *ft) { int i; char *p, *t; int currentFile, numRead; char *fileBuffer; TLEN_FILE_PACKET *rpacket, *packet; if (ft->state == FT_CONNECTING) { char filename[256]; // Must be 256 (0x100) if ((packet=TlenP2PPacketCreate(sizeof(DWORD)+(ft->fileCount*(sizeof(filename)+sizeof(DWORD))))) != NULL) { // Must pause a bit, sending these two packets back to back // will break the session because the receiver cannot take it :) SleepEx(1000, TRUE); TlenP2PPacketSetLen(packet, 0); // Reuse packet TlenP2PPacketSetType(packet, TLEN_FILE_PACKET_FILE_LIST); TlenP2PPacketPackDword(packet, (DWORD) ft->fileCount); for (i=0; ifileCount; i++) { // struct _stat statbuf; // _stat(ft->files[i], &statbuf); // TlenP2PPacketPackDword(packet, statbuf.st_size); TlenP2PPacketPackDword(packet, ft->filesSize[i]); memset(filename, 0, sizeof(filename)); if ((t=strrchr(ft->files[i], '\\')) != NULL) t++; else t = ft->files[i]; mir_snprintf(filename, sizeof(filename)-1, t); TlenP2PPacketPackBuffer(packet, filename, sizeof(filename)); } TlenP2PPacketSend(ft->s, packet); TlenP2PPacketFree(packet); ft->allFileReceivedBytes = 0; ft->state = FT_INITIALIZING; TlenLog(ft->proto, "Change to FT_INITIALIZING"); } else { ft->state = FT_ERROR; } } else if (ft->state == FT_INITIALIZING) { // FT_INITIALIZING rpacket = TlenP2PPacketReceive(ft->s); TlenLog(ft->proto, "FT_INITIALIZING: recv %d", rpacket); if (rpacket == NULL) { ft->state = FT_ERROR; return; } TlenLog(ft->proto, "FT_INITIALIZING: recv type %d", rpacket->type); p = rpacket->packet; // TYPE: TLEN_FILE_PACKET_FILE_LIST_ACK will be ignored // LEN: 0 if (rpacket->type == TLEN_FILE_PACKET_FILE_LIST_ACK) { } // Then the receiver will request each file // TYPE: TLEN_FILE_PACKET_REQUEST // LEN: // (DWORD) file number // (DWORD) 0 // (DWORD) 0 else if (rpacket->type == TLEN_FILE_PACKET_FILE_REQUEST) { PROTOFILETRANSFERSTATUS pfts; //struct _stat statbuf; currentFile = *((DWORD*)p); if (currentFile != ft->currentFile) { TlenLog(ft->proto, "Requested file (#%d) is invalid (must be %d)", currentFile, ft->currentFile); ft->state = FT_ERROR; } else { // _stat(ft->files[currentFile], &statbuf); // file size in statbuf.st_size TlenLog(ft->proto, "Sending [%s] [%d]", ft->files[currentFile], ft->filesSize[currentFile]); if ((ft->fileId=_open(ft->files[currentFile], _O_BINARY|_O_RDONLY)) < 0) { TlenLog(ft->proto, "File cannot be opened"); ft->state = FT_ERROR; } else { memset(&pfts, 0, sizeof(PROTOFILETRANSFERSTATUS)); pfts.cbSize = sizeof(PROTOFILETRANSFERSTATUS); pfts.hContact = ft->hContact; pfts.flags = PFTS_SENDING; pfts.pszFiles = ft->files; pfts.totalFiles = ft->fileCount; pfts.currentFileNumber = ft->currentFile; pfts.totalBytes = ft->allFileTotalSize; pfts.szWorkingDir = NULL; pfts.szCurrentFile = ft->files[ft->currentFile]; pfts.currentFileSize = ft->filesSize[ft->currentFile]; //statbuf.st_size; pfts.currentFileTime = 0; ft->fileReceivedBytes = 0; if ((packet = TlenP2PPacketCreate(2*sizeof(DWORD)+2048)) == NULL) { ft->state = FT_ERROR; } else { TlenP2PPacketSetType(packet, TLEN_FILE_PACKET_FILE_DATA); fileBuffer = (char *) mir_alloc(2048); TlenLog(ft->proto, "Sending file data..."); while ((numRead=_read(ft->fileId, fileBuffer, 2048)) > 0) { TlenP2PPacketSetLen(packet, 0); // Reuse packet TlenP2PPacketPackDword(packet, (DWORD) ft->fileReceivedBytes); TlenP2PPacketPackDword(packet, 0); TlenP2PPacketPackBuffer(packet, fileBuffer, numRead); if (TlenP2PPacketSend(ft->s, packet)) { ft->fileReceivedBytes += numRead; ft->allFileReceivedBytes += numRead; pfts.totalProgress = ft->allFileReceivedBytes; pfts.currentFileProgress = ft->fileReceivedBytes; ProtoBroadcastAck(ft->proto->m_szModuleName, ft->hContact, ACKTYPE_FILE, ACKRESULT_DATA, ft, (LPARAM) &pfts); } else { ft->state = FT_ERROR; break; } } mir_free(fileBuffer); _close(ft->fileId); if (ft->state != FT_ERROR) { if (ft->currentFile >= ft->fileCount-1) ft->state = FT_DONE; else { ft->currentFile++; ft->state = FT_INITIALIZING; TlenLog(ft->proto, "File sent, advancing to the next file..."); ProtoBroadcastAck(ft->proto->m_szModuleName, ft->hContact, ACKTYPE_FILE, ACKRESULT_NEXTFILE, ft, 0); } } TlenLog(ft->proto, "Finishing this file..."); TlenP2PPacketSetLen(packet, 0); // Reuse packet TlenP2PPacketSetType(packet, TLEN_FILE_PACKET_END_OF_FILE); TlenP2PPacketPackDword(packet, currentFile); TlenP2PPacketSend(ft->s, packet); TlenP2PPacketFree(packet); } } } TlenP2PPacketFree(rpacket); } else { ft->state = FT_ERROR; } } } static void TlenFileSendingConnection(HANDLE hConnection, DWORD dwRemoteIP, void * pExtra) { HANDLE slisten; TLEN_FILE_TRANSFER *ft; TlenProtocol *proto = (TlenProtocol *)pExtra; ft = TlenP2PEstablishIncomingConnection(proto, hConnection, LIST_FILE, TRUE); if (ft != NULL) { slisten = ft->s; ft->s = hConnection; TlenLog(ft->proto, "Set ft->s to %d (saving %d)", hConnection, slisten); TlenLog(ft->proto, "Entering send loop for this file connection... (ft->s is hConnection)"); while (ft->state != FT_DONE && ft->state != FT_ERROR) { TlenFileSendParse(ft); } if (ft->state == FT_DONE) ProtoBroadcastAck(ft->proto->m_szModuleName, ft->hContact, ACKTYPE_FILE, ACKRESULT_SUCCESS, ft, 0); else ProtoBroadcastAck(ft->proto->m_szModuleName, ft->hContact, ACKTYPE_FILE, ACKRESULT_FAILED, ft, 0); TlenLog(ft->proto, "Closing connection for this file transfer... (ft->s is now hBind)"); ft->s = slisten; TlenLog(ft->proto, "ft->s is restored to %d", ft->s); if (ft->s != hConnection) { Netlib_CloseHandle(hConnection); } if (ft->hFileEvent != NULL) SetEvent(ft->hFileEvent); } else { Netlib_CloseHandle(hConnection); } } int TlenFileCancelAll(TlenProtocol *proto) { TLEN_LIST_ITEM *item; HANDLE hEvent; int i = 0; while ((i=TlenListFindNext(proto, LIST_FILE, 0)) >=0 ) { if ((item=TlenListGetItemPtrFromIndex(proto, i)) != NULL) { TLEN_FILE_TRANSFER *ft = item->ft; TlenListRemoveByIndex(proto, i); if (ft != NULL) { if (ft->s) { //ProtoBroadcastAck(m_szModuleName, ft->hContact, ACKTYPE_FILE, ACKRESULT_FAILED, ft, 0); TlenLog(ft->proto, "Closing ft->s = %d", ft->s); ft->state = FT_ERROR; Netlib_CloseHandle(ft->s); ft->s = NULL; if (ft->hFileEvent != NULL) { hEvent = ft->hFileEvent; ft->hFileEvent = NULL; SetEvent(hEvent); } } else { TlenLog(ft->proto, "freeing ft struct"); TlenP2PFreeFileTransfer(ft); } } } } return 0; } static void __cdecl TlenFileSendingThread(TLEN_FILE_TRANSFER *ft) { HANDLE s = NULL; HANDLE hEvent; char *nick; TlenLog(ft->proto, "Thread started: type=tlen_file_send"); ft->mode = FT_SEND; ft->pfnNewConnectionV2 = TlenFileSendingConnection; s = TlenP2PListen(ft); if (s != NULL) { ProtoBroadcastAck(ft->proto->m_szModuleName, ft->hContact, ACKTYPE_FILE, ACKRESULT_CONNECTING, ft, 0); ft->s = s; //TlenLog("ft->s = %d", s); //TlenLog("fileCount = %d", ft->fileCount); hEvent = CreateEvent(NULL, FALSE, FALSE, NULL); ft->hFileEvent = hEvent; ft->currentFile = 0; ft->state = FT_CONNECTING; nick = TlenNickFromJID(ft->jid); TlenSend(ft->proto, "", nick, ft->iqId, ft->localName, ft->wLocalPort); mir_free(nick); TlenLog(ft->proto, "Waiting for the file to be sent..."); WaitForSingleObject(hEvent, INFINITE); ft->hFileEvent = NULL; CloseHandle(hEvent); TlenLog(ft->proto, "Finish all files"); Netlib_CloseHandle(s); ft->s = NULL; TlenLog(ft->proto, "ft->s is NULL"); if (ft->state == FT_SWITCH) { TlenLog(ft->proto, "Sending as client..."); ft->state = FT_CONNECTING; NETLIBOPENCONNECTION nloc = { sizeof(nloc) }; nloc.szHost = ft->hostName; nloc.wPort = ft->wPort; HANDLE s = (HANDLE) CallService(MS_NETLIB_OPENCONNECTION, (WPARAM) ft->proto->hNetlibUser, (LPARAM) &nloc); if (s != NULL) { ProtoBroadcastAck(ft->proto->m_szModuleName, ft->hContact, ACKTYPE_FILE, ACKRESULT_CONNECTING, ft, 0); ft->s = s; TlenP2PEstablishOutgoingConnection(ft, TRUE); TlenLog(ft->proto, "Entering send loop for this file connection..."); while (ft->state != FT_DONE && ft->state != FT_ERROR) { TlenFileSendParse(ft); } if (ft->state == FT_DONE) ProtoBroadcastAck(ft->proto->m_szModuleName, ft->hContact, ACKTYPE_FILE, ACKRESULT_SUCCESS, ft, 0); else ProtoBroadcastAck(ft->proto->m_szModuleName, ft->hContact, ACKTYPE_FILE, ACKRESULT_FAILED, ft, 0); TlenLog(ft->proto, "Closing connection for this file transfer... "); Netlib_CloseHandle(s); } else { ft->state = FT_ERROR; } } } else { TlenLog(ft->proto, "Cannot allocate port to bind for file server thread, thread ended."); ft->state = FT_ERROR; } TlenListRemove(ft->proto, LIST_FILE, ft->iqId); switch (ft->state) { case FT_DONE: TlenLog(ft->proto, "Finish successfully"); ProtoBroadcastAck(ft->proto->m_szModuleName, ft->hContact, ACKTYPE_FILE, ACKRESULT_SUCCESS, ft, 0); break; case FT_DENIED: ProtoBroadcastAck(ft->proto->m_szModuleName, ft->hContact, ACKTYPE_FILE, ACKRESULT_DENIED, ft, 0); break; default: // FT_ERROR: nick = TlenNickFromJID(ft->jid); TlenSend(ft->proto, "", nick, ft->iqId); mir_free(nick); TlenLog(ft->proto, "Finish with errors"); ProtoBroadcastAck(ft->proto->m_szModuleName, ft->hContact, ACKTYPE_FILE, ACKRESULT_FAILED, ft, 0); break; } TlenLog(ft->proto, "Thread ended: type=file_send"); TlenP2PFreeFileTransfer(ft); } TLEN_FILE_TRANSFER *TlenFileCreateFT(TlenProtocol *proto, const char *jid) { TLEN_FILE_TRANSFER *ft; ft = (TLEN_FILE_TRANSFER *) mir_alloc(sizeof(TLEN_FILE_TRANSFER)); memset(ft, 0, sizeof(TLEN_FILE_TRANSFER)); ft->proto = proto; ft->jid = mir_strdup(jid); return ft; } /* * File transfer */ void TlenProcessF(XmlNode *node, ThreadData *info) { TLEN_FILE_TRANSFER *ft; char *from, *p, *e; char jid[128], szFilename[MAX_PATH]; int numFiles; TLEN_LIST_ITEM *item; // if (!node->name || strcmp(node->name, "f")) return; if (info == NULL) return; if ((from=TlenXmlGetAttrValue(node, "f")) != NULL) { if (strchr(from, '@') == NULL) { mir_snprintf(jid, sizeof(jid), "%s@%s", from, info->server); } else { mir_snprintf(jid, sizeof(jid), "%s", from); } if ((e=TlenXmlGetAttrValue(node, "e")) != NULL) { if (!strcmp(e, "1")) { // FILE_RECV : e='1' : File transfer request ft = TlenFileCreateFT(info->proto, jid); ft->hContact = TlenHContactFromJID(info->proto, jid); if ((p=TlenXmlGetAttrValue(node, "i")) != NULL) ft->iqId = mir_strdup(p); szFilename[0] = '\0'; if ((p=TlenXmlGetAttrValue(node, "c")) != NULL) { numFiles = atoi(p); if (numFiles == 1) { if ((p=TlenXmlGetAttrValue(node, "n")) != NULL) { p = TlenTextDecode(p); strncpy(szFilename, p, sizeof(szFilename)); mir_free(p); } else { strcpy(szFilename, Translate("1 File")); } } else if (numFiles > 1) { mir_snprintf(szFilename, sizeof(szFilename), Translate("%d Files"), numFiles); } } if (szFilename[0] != '\0' && ft->iqId != NULL) { TCHAR* filenameT = mir_utf8decodeT((char*)szFilename); PROTORECVFILET pre = {0}; pre.flags = PREF_TCHAR; pre.fileCount = 1; pre.timestamp = time(NULL); pre.tszDescription = filenameT; pre.ptszFiles = &filenameT; pre.lParam = (LPARAM)ft; TlenLog(ft->proto, "sending chainrecv"); ProtoChainRecvFile(ft->hContact, &pre); mir_free(filenameT); } else { // malformed request, reject if (ft->iqId) TlenSend(ft->proto, "", ft->iqId, from); else TlenSend(ft->proto, "", from); TlenP2PFreeFileTransfer(ft); } } else if (!strcmp(e, "3")) { // FILE_RECV : e='3' : invalid transfer error if ((p=TlenXmlGetAttrValue(node, "i")) != NULL) { if ((item=TlenListGetItemPtr(info->proto, LIST_FILE, p)) != NULL) { if (item->ft != NULL) { ProtoBroadcastAck(info->proto->m_szModuleName, item->ft->hContact, ACKTYPE_FILE, ACKRESULT_FAILED, item->ft, 0); info->proto->FileCancel(NULL, item->ft); } TlenListRemove(info->proto, LIST_FILE, p); } } } else if (!strcmp(e, "4")) { // FILE_SEND : e='4' : File sending request was denied by the remote client if ((p=TlenXmlGetAttrValue(node, "i")) != NULL) { if ((item=TlenListGetItemPtr(info->proto, LIST_FILE, p)) != NULL) { if (!strcmp(item->ft->jid, jid)) { ProtoBroadcastAck(info->proto->m_szModuleName, item->ft->hContact, ACKTYPE_FILE, ACKRESULT_DENIED, item->ft, 0); TlenListRemove(info->proto, LIST_FILE, p); } } } } else if (!strcmp(e, "5")) { // FILE_SEND : e='5' : File sending request was accepted if ((p=TlenXmlGetAttrValue(node, "i")) != NULL) { if ((item=TlenListGetItemPtr(info->proto, LIST_FILE, p)) != NULL) { if (!strcmp(item->ft->jid, jid)) forkthread((void (__cdecl *)(void*))TlenFileSendingThread, 0, item->ft); } } } else if (!strcmp(e, "6")) { // FILE_RECV : e='6' : IP and port information to connect to get file if ((p=TlenXmlGetAttrValue(node, "i")) != NULL) { if ((item=TlenListGetItemPtr(info->proto, LIST_FILE, p)) != NULL) { if ((p=TlenXmlGetAttrValue(node, "a")) != NULL) { item->ft->hostName = mir_strdup(p); if ((p=TlenXmlGetAttrValue(node, "p")) != NULL) { item->ft->wPort = atoi(p); forkthread((void (__cdecl *)(void*))TlenFileReceiveThread, 0, item->ft); } } } } } else if (!strcmp(e, "7")) { // FILE_RECV : e='7' : IP and port information to connect to send file // in case the conection to the given server was not successful if ((p=TlenXmlGetAttrValue(node, "i")) != NULL) { if ((item=TlenListGetItemPtr(info->proto, LIST_FILE, p)) != NULL) { if ((p=TlenXmlGetAttrValue(node, "a")) != NULL) { if (item->ft->hostName != NULL) mir_free(item->ft->hostName); item->ft->hostName = mir_strdup(p); if ((p=TlenXmlGetAttrValue(node, "p")) != NULL) { item->ft->wPort = atoi(p); item->ft->state = FT_SWITCH; SetEvent(item->ft->hFileEvent); } } } } } else if (!strcmp(e, "8")) { // FILE_RECV : e='8' : transfer error if ((p=TlenXmlGetAttrValue(node, "i")) != NULL) { if ((item=TlenListGetItemPtr(info->proto, LIST_FILE, p)) != NULL) { item->ft->state = FT_ERROR; if (item->ft->hFileEvent != NULL) { SetEvent(item->ft->hFileEvent); } else { ProtoBroadcastAck(info->proto->m_szModuleName, item->ft->hContact, ACKTYPE_FILE, ACKRESULT_FAILED, item->ft, 0); } } } } } } }