637 lines
17 KiB
C
637 lines
17 KiB
C
|
/*
|
||
|
* Copyright (c) 2005 Novell, Inc.
|
||
|
* All Rights Reserved.
|
||
|
*
|
||
|
* This program is free software; you can redistribute it and/or
|
||
|
* modify it under the terms of version 2 of the GNU General Public License as
|
||
|
* published by the Free Software Foundation.
|
||
|
*
|
||
|
* 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, contact Novell, Inc.
|
||
|
*
|
||
|
* To contact Novell about this file by physical or electronic mail,
|
||
|
* you may find current contact information at www.novell.com
|
||
|
*
|
||
|
* Author : Rohit Kumar
|
||
|
* Email ID : rokumar@novell.com
|
||
|
* Date : 14th July 2005
|
||
|
*/
|
||
|
|
||
|
|
||
|
#include <stdio.h>
|
||
|
#include <string.h>
|
||
|
#include <stdlib.h>
|
||
|
#include <fcntl.h>
|
||
|
#include <dirent.h>
|
||
|
#include <utime.h>
|
||
|
#include <errno.h>
|
||
|
#include <unistd.h>
|
||
|
#include <sys/stat.h>
|
||
|
#include <sys/types.h>
|
||
|
|
||
|
#include <rfb/rfb.h>
|
||
|
#include "rfbtightproto.h"
|
||
|
#include "filelistinfo.h"
|
||
|
#include "filetransfermsg.h"
|
||
|
#include "handlefiletransferrequest.h"
|
||
|
|
||
|
#define SZ_RFBBLOCKSIZE 8192
|
||
|
|
||
|
|
||
|
void
|
||
|
FreeFileTransferMsg(FileTransferMsg ftm)
|
||
|
{
|
||
|
|
||
|
if(ftm.data != NULL) {
|
||
|
free(ftm.data);
|
||
|
ftm.data = NULL;
|
||
|
}
|
||
|
|
||
|
ftm.length = 0;
|
||
|
|
||
|
}
|
||
|
|
||
|
|
||
|
/******************************************************************************
|
||
|
* Methods to handle file list request.
|
||
|
******************************************************************************/
|
||
|
|
||
|
int CreateFileListInfo(FileListInfoPtr pFileListInfo, char* path, int flag);
|
||
|
FileTransferMsg CreateFileListErrMsg(char flags);
|
||
|
FileTransferMsg CreateFileListMsg(FileListInfo fileListInfo, char flags);
|
||
|
|
||
|
|
||
|
/*
|
||
|
* This is the method called by HandleFileListRequest to get the file list
|
||
|
*/
|
||
|
|
||
|
FileTransferMsg
|
||
|
GetFileListResponseMsg(char* path, char flags)
|
||
|
{
|
||
|
FileTransferMsg fileListMsg;
|
||
|
FileListInfo fileListInfo;
|
||
|
int status = -1;
|
||
|
|
||
|
memset(&fileListMsg, 0, sizeof(FileTransferMsg));
|
||
|
memset(&fileListInfo, 0, sizeof(FileListInfo));
|
||
|
|
||
|
|
||
|
/* fileListInfo can have null data if the folder is Empty
|
||
|
or if some error condition has occured.
|
||
|
The return value is 'failure' only if some error condition has occured.
|
||
|
*/
|
||
|
status = CreateFileListInfo(&fileListInfo, path, !(flags & 0x10));
|
||
|
|
||
|
if(status == FAILURE) {
|
||
|
fileListMsg = CreateFileListErrMsg(flags);
|
||
|
}
|
||
|
else {
|
||
|
/* DisplayFileList(fileListInfo); For Debugging */
|
||
|
|
||
|
fileListMsg = CreateFileListMsg(fileListInfo, flags);
|
||
|
FreeFileListInfo(fileListInfo);
|
||
|
}
|
||
|
|
||
|
return fileListMsg;
|
||
|
}
|
||
|
|
||
|
#ifndef __GNUC__
|
||
|
#define __FUNCTION__ "unknown"
|
||
|
#endif
|
||
|
|
||
|
int
|
||
|
CreateFileListInfo(FileListInfoPtr pFileListInfo, char* path, int flag)
|
||
|
{
|
||
|
DIR* pDir = NULL;
|
||
|
struct dirent* pDirent = NULL;
|
||
|
|
||
|
if((path == NULL) || (strlen(path) == 0)) {
|
||
|
/* In this case we will send the list of entries in ftp root*/
|
||
|
sprintf(path, "%s%s", GetFtpRoot(), "/");
|
||
|
}
|
||
|
|
||
|
if((pDir = opendir(path)) == NULL) {
|
||
|
rfbLog("File [%s]: Method [%s]: not able to open the dir\n",
|
||
|
__FILE__, __FUNCTION__);
|
||
|
return FAILURE;
|
||
|
}
|
||
|
|
||
|
while((pDirent = readdir(pDir))) {
|
||
|
if(strcmp(pDirent->d_name, ".") && strcmp(pDirent->d_name, "..")) {
|
||
|
struct stat stat_buf;
|
||
|
/*
|
||
|
int fpLen = sizeof(char)*(strlen(pDirent->d_name)+strlen(path)+2);
|
||
|
*/
|
||
|
char fullpath[PATH_MAX];
|
||
|
|
||
|
memset(fullpath, 0, PATH_MAX);
|
||
|
|
||
|
strcpy(fullpath, path);
|
||
|
if(path[strlen(path)-1] != '/')
|
||
|
strcat(fullpath, "/");
|
||
|
strcat(fullpath, pDirent->d_name);
|
||
|
|
||
|
if(stat(fullpath, &stat_buf) < 0) {
|
||
|
rfbLog("File [%s]: Method [%s]: Reading stat for file %s failed\n",
|
||
|
__FILE__, __FUNCTION__, fullpath);
|
||
|
continue;
|
||
|
}
|
||
|
|
||
|
if(S_ISDIR(stat_buf.st_mode)) {
|
||
|
if(AddFileListItemInfo(pFileListInfo, pDirent->d_name, -1, 0) == 0) {
|
||
|
rfbLog("File [%s]: Method [%s]: Add directory %s in the"
|
||
|
" list failed\n", __FILE__, __FUNCTION__, fullpath);
|
||
|
continue;
|
||
|
}
|
||
|
}
|
||
|
else {
|
||
|
if(flag) {
|
||
|
if(AddFileListItemInfo(pFileListInfo, pDirent->d_name,
|
||
|
stat_buf.st_size,
|
||
|
stat_buf.st_mtime) == 0) {
|
||
|
rfbLog("File [%s]: Method [%s]: Add file %s in the "
|
||
|
"list failed\n", __FILE__, __FUNCTION__, fullpath);
|
||
|
continue;
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
if(closedir(pDir) < 0) {
|
||
|
rfbLog("File [%s]: Method [%s]: ERROR Couldn't close dir\n",
|
||
|
__FILE__, __FUNCTION__);
|
||
|
}
|
||
|
|
||
|
return SUCCESS;
|
||
|
}
|
||
|
|
||
|
|
||
|
FileTransferMsg
|
||
|
CreateFileListErrMsg(char flags)
|
||
|
{
|
||
|
FileTransferMsg fileListMsg;
|
||
|
rfbFileListDataMsg* pFLD = NULL;
|
||
|
char* data = NULL;
|
||
|
unsigned int length = 0;
|
||
|
|
||
|
memset(&fileListMsg, 0, sizeof(FileTransferMsg));
|
||
|
|
||
|
data = (char*) calloc(sizeof(rfbFileListDataMsg), sizeof(char));
|
||
|
if(data == NULL) {
|
||
|
return fileListMsg;
|
||
|
}
|
||
|
length = sizeof(rfbFileListDataMsg) * sizeof(char);
|
||
|
pFLD = (rfbFileListDataMsg*) data;
|
||
|
|
||
|
pFLD->type = rfbFileListData;
|
||
|
pFLD->numFiles = Swap16IfLE(0);
|
||
|
pFLD->dataSize = Swap16IfLE(0);
|
||
|
pFLD->compressedSize = Swap16IfLE(0);
|
||
|
pFLD->flags = flags | 0x80;
|
||
|
|
||
|
fileListMsg.data = data;
|
||
|
fileListMsg.length = length;
|
||
|
|
||
|
return fileListMsg;
|
||
|
}
|
||
|
|
||
|
|
||
|
FileTransferMsg
|
||
|
CreateFileListMsg(FileListInfo fileListInfo, char flags)
|
||
|
{
|
||
|
FileTransferMsg fileListMsg;
|
||
|
rfbFileListDataMsg* pFLD = NULL;
|
||
|
char *data = NULL, *pFileNames = NULL;
|
||
|
unsigned int length = 0, dsSize = 0, i = 0;
|
||
|
FileListItemSizePtr pFileListItemSize = NULL;
|
||
|
|
||
|
memset(&fileListMsg, 0, sizeof(FileTransferMsg));
|
||
|
dsSize = fileListInfo.numEntries * 8;
|
||
|
length = sz_rfbFileListDataMsg + dsSize +
|
||
|
GetSumOfFileNamesLength(fileListInfo) +
|
||
|
fileListInfo.numEntries;
|
||
|
|
||
|
data = (char*) calloc(length, sizeof(char));
|
||
|
if(data == NULL) {
|
||
|
return fileListMsg;
|
||
|
}
|
||
|
pFLD = (rfbFileListDataMsg*) data;
|
||
|
pFileListItemSize = (FileListItemSizePtr) &data[sz_rfbFileListDataMsg];
|
||
|
pFileNames = &data[sz_rfbFileListDataMsg + dsSize];
|
||
|
|
||
|
pFLD->type = rfbFileListData;
|
||
|
pFLD->flags = flags & 0xF0;
|
||
|
pFLD->numFiles = Swap16IfLE(fileListInfo.numEntries);
|
||
|
pFLD->dataSize = Swap16IfLE(GetSumOfFileNamesLength(fileListInfo) +
|
||
|
fileListInfo.numEntries);
|
||
|
pFLD->compressedSize = pFLD->dataSize;
|
||
|
|
||
|
for(i =0; i <fileListInfo.numEntries; i++) {
|
||
|
pFileListItemSize[i].size = Swap32IfLE(GetFileSizeAt(fileListInfo, i));
|
||
|
pFileListItemSize[i].data = Swap32IfLE(GetFileDataAt(fileListInfo, i));
|
||
|
strcpy(pFileNames, GetFileNameAt(fileListInfo, i));
|
||
|
|
||
|
if(i+1 < fileListInfo.numEntries)
|
||
|
pFileNames += strlen(pFileNames) + 1;
|
||
|
}
|
||
|
|
||
|
fileListMsg.data = data;
|
||
|
fileListMsg.length = length;
|
||
|
|
||
|
return fileListMsg;
|
||
|
}
|
||
|
|
||
|
|
||
|
/******************************************************************************
|
||
|
* Methods to handle File Download Request.
|
||
|
******************************************************************************/
|
||
|
|
||
|
FileTransferMsg CreateFileDownloadErrMsg(char* reason, unsigned int reasonLen);
|
||
|
FileTransferMsg CreateFileDownloadZeroSizeDataMsg(unsigned long mTime);
|
||
|
FileTransferMsg CreateFileDownloadBlockSizeDataMsg(unsigned short sizeFile, char *pFile);
|
||
|
|
||
|
FileTransferMsg
|
||
|
GetFileDownLoadErrMsg()
|
||
|
{
|
||
|
FileTransferMsg fileDownloadErrMsg;
|
||
|
|
||
|
char reason[] = "An internal error on the server caused download failure";
|
||
|
int reasonLen = strlen(reason);
|
||
|
|
||
|
memset(&fileDownloadErrMsg, 0, sizeof(FileTransferMsg));
|
||
|
|
||
|
fileDownloadErrMsg = CreateFileDownloadErrMsg(reason, reasonLen);
|
||
|
|
||
|
return fileDownloadErrMsg;
|
||
|
}
|
||
|
|
||
|
|
||
|
FileTransferMsg
|
||
|
GetFileDownloadReadDataErrMsg()
|
||
|
{
|
||
|
char reason[] = "Cannot open file, perhaps it is absent or is a directory";
|
||
|
int reasonLen = strlen(reason);
|
||
|
|
||
|
return CreateFileDownloadErrMsg(reason, reasonLen);
|
||
|
|
||
|
}
|
||
|
|
||
|
|
||
|
FileTransferMsg
|
||
|
GetFileDownloadLengthErrResponseMsg()
|
||
|
{
|
||
|
char reason [] = "Path length exceeds PATH_MAX (4096) bytes";
|
||
|
int reasonLen = strlen(reason);
|
||
|
|
||
|
return CreateFileDownloadErrMsg(reason, reasonLen);
|
||
|
}
|
||
|
|
||
|
|
||
|
FileTransferMsg
|
||
|
GetFileDownloadResponseMsgInBlocks(rfbClientPtr cl, rfbTightClientPtr rtcp)
|
||
|
{
|
||
|
/* const unsigned int sz_rfbBlockSize = SZ_RFBBLOCKSIZE; */
|
||
|
int numOfBytesRead = 0;
|
||
|
char pBuf[SZ_RFBBLOCKSIZE];
|
||
|
char* path = rtcp->rcft.rcfd.fName;
|
||
|
|
||
|
memset(pBuf, 0, SZ_RFBBLOCKSIZE);
|
||
|
|
||
|
if((rtcp->rcft.rcfd.downloadInProgress == FALSE) && (rtcp->rcft.rcfd.downloadFD == -1)) {
|
||
|
if((rtcp->rcft.rcfd.downloadFD = open(path, O_RDONLY)) == -1) {
|
||
|
rfbLog("File [%s]: Method [%s]: Error: Couldn't open file\n",
|
||
|
__FILE__, __FUNCTION__);
|
||
|
return GetFileDownloadReadDataErrMsg();
|
||
|
}
|
||
|
rtcp->rcft.rcfd.downloadInProgress = TRUE;
|
||
|
}
|
||
|
if((rtcp->rcft.rcfd.downloadInProgress == TRUE) && (rtcp->rcft.rcfd.downloadFD != -1)) {
|
||
|
if( (numOfBytesRead = read(rtcp->rcft.rcfd.downloadFD, pBuf, SZ_RFBBLOCKSIZE)) <= 0) {
|
||
|
close(rtcp->rcft.rcfd.downloadFD);
|
||
|
rtcp->rcft.rcfd.downloadFD = -1;
|
||
|
rtcp->rcft.rcfd.downloadInProgress = FALSE;
|
||
|
if(numOfBytesRead == 0) {
|
||
|
return CreateFileDownloadZeroSizeDataMsg(rtcp->rcft.rcfd.mTime);
|
||
|
}
|
||
|
return GetFileDownloadReadDataErrMsg();
|
||
|
}
|
||
|
return CreateFileDownloadBlockSizeDataMsg(numOfBytesRead, pBuf);
|
||
|
}
|
||
|
return GetFileDownLoadErrMsg();
|
||
|
}
|
||
|
|
||
|
|
||
|
FileTransferMsg
|
||
|
ChkFileDownloadErr(rfbClientPtr cl, rfbTightClientPtr rtcp)
|
||
|
{
|
||
|
FileTransferMsg fileDownloadMsg;
|
||
|
struct stat stat_buf;
|
||
|
int sz_rfbFileSize = 0;
|
||
|
char* path = rtcp->rcft.rcfd.fName;
|
||
|
|
||
|
memset(&fileDownloadMsg, 0, sizeof(FileTransferMsg));
|
||
|
|
||
|
if( (path == NULL) || (strlen(path) == 0) ||
|
||
|
(stat(path, &stat_buf) < 0) || (!(S_ISREG(stat_buf.st_mode))) ) {
|
||
|
|
||
|
char reason[] = "Cannot open file, perhaps it is absent or is not a regular file";
|
||
|
int reasonLen = strlen(reason);
|
||
|
|
||
|
rfbLog("File [%s]: Method [%s]: Reading stat for path %s failed\n",
|
||
|
__FILE__, __FUNCTION__, path);
|
||
|
|
||
|
fileDownloadMsg = CreateFileDownloadErrMsg(reason, reasonLen);
|
||
|
}
|
||
|
else {
|
||
|
rtcp->rcft.rcfd.mTime = stat_buf.st_mtime;
|
||
|
sz_rfbFileSize = stat_buf.st_size;
|
||
|
if(sz_rfbFileSize <= 0) {
|
||
|
fileDownloadMsg = CreateFileDownloadZeroSizeDataMsg(stat_buf.st_mtime);
|
||
|
}
|
||
|
|
||
|
}
|
||
|
return fileDownloadMsg;
|
||
|
}
|
||
|
|
||
|
|
||
|
FileTransferMsg
|
||
|
CreateFileDownloadErrMsg(char* reason, unsigned int reasonLen)
|
||
|
{
|
||
|
FileTransferMsg fileDownloadErrMsg;
|
||
|
int length = sz_rfbFileDownloadFailedMsg + reasonLen + 1;
|
||
|
rfbFileDownloadFailedMsg *pFDF = NULL;
|
||
|
char *pFollow = NULL;
|
||
|
|
||
|
char *pData = (char*) calloc(length, sizeof(char));
|
||
|
memset(&fileDownloadErrMsg, 0, sizeof(FileTransferMsg));
|
||
|
if(pData == NULL) {
|
||
|
rfbLog("File [%s]: Method [%s]: pData is NULL\n",
|
||
|
__FILE__, __FUNCTION__);
|
||
|
return fileDownloadErrMsg;
|
||
|
}
|
||
|
|
||
|
pFDF = (rfbFileDownloadFailedMsg *) pData;
|
||
|
pFollow = &pData[sz_rfbFileDownloadFailedMsg];
|
||
|
|
||
|
pFDF->type = rfbFileDownloadFailed;
|
||
|
pFDF->reasonLen = Swap16IfLE(reasonLen);
|
||
|
memcpy(pFollow, reason, reasonLen);
|
||
|
|
||
|
fileDownloadErrMsg.data = pData;
|
||
|
fileDownloadErrMsg.length = length;
|
||
|
|
||
|
return fileDownloadErrMsg;
|
||
|
}
|
||
|
|
||
|
|
||
|
FileTransferMsg
|
||
|
CreateFileDownloadZeroSizeDataMsg(unsigned long mTime)
|
||
|
{
|
||
|
FileTransferMsg fileDownloadZeroSizeDataMsg;
|
||
|
int length = sz_rfbFileDownloadDataMsg + sizeof(int);
|
||
|
rfbFileDownloadDataMsg *pFDD = NULL;
|
||
|
char *pFollow = NULL;
|
||
|
|
||
|
char *pData = (char*) calloc(length, sizeof(char));
|
||
|
memset(&fileDownloadZeroSizeDataMsg, 0, sizeof(FileTransferMsg));
|
||
|
if(pData == NULL) {
|
||
|
rfbLog("File [%s]: Method [%s]: pData is NULL\n",
|
||
|
__FILE__, __FUNCTION__);
|
||
|
return fileDownloadZeroSizeDataMsg;
|
||
|
}
|
||
|
|
||
|
pFDD = (rfbFileDownloadDataMsg *) pData;
|
||
|
pFollow = &pData[sz_rfbFileDownloadDataMsg];
|
||
|
|
||
|
pFDD->type = rfbFileDownloadData;
|
||
|
pFDD->compressLevel = 0;
|
||
|
pFDD->compressedSize = Swap16IfLE(0);
|
||
|
pFDD->realSize = Swap16IfLE(0);
|
||
|
|
||
|
memcpy(pFollow, &mTime, sizeof(unsigned long));
|
||
|
|
||
|
fileDownloadZeroSizeDataMsg.data = pData;
|
||
|
fileDownloadZeroSizeDataMsg.length = length;
|
||
|
|
||
|
return fileDownloadZeroSizeDataMsg;
|
||
|
|
||
|
}
|
||
|
|
||
|
|
||
|
FileTransferMsg
|
||
|
CreateFileDownloadBlockSizeDataMsg(unsigned short sizeFile, char *pFile)
|
||
|
{
|
||
|
FileTransferMsg fileDownloadBlockSizeDataMsg;
|
||
|
int length = sz_rfbFileDownloadDataMsg + sizeFile;
|
||
|
rfbFileDownloadDataMsg *pFDD = NULL;
|
||
|
char *pFollow = NULL;
|
||
|
|
||
|
char *pData = (char*) calloc(length, sizeof(char));
|
||
|
memset(&fileDownloadBlockSizeDataMsg, 0, sizeof(FileTransferMsg));
|
||
|
if(NULL == pData) {
|
||
|
rfbLog("File [%s]: Method [%s]: pData is NULL\n",
|
||
|
__FILE__, __FUNCTION__);
|
||
|
return fileDownloadBlockSizeDataMsg;
|
||
|
}
|
||
|
|
||
|
pFDD = (rfbFileDownloadDataMsg *) pData;
|
||
|
pFollow = &pData[sz_rfbFileDownloadDataMsg];
|
||
|
|
||
|
pFDD->type = rfbFileDownloadData;
|
||
|
pFDD->compressLevel = 0;
|
||
|
pFDD->compressedSize = Swap16IfLE(sizeFile);
|
||
|
pFDD->realSize = Swap16IfLE(sizeFile);
|
||
|
|
||
|
memcpy(pFollow, pFile, sizeFile);
|
||
|
|
||
|
fileDownloadBlockSizeDataMsg.data = pData;
|
||
|
fileDownloadBlockSizeDataMsg.length = length;
|
||
|
|
||
|
return fileDownloadBlockSizeDataMsg;
|
||
|
|
||
|
}
|
||
|
|
||
|
|
||
|
/******************************************************************************
|
||
|
* Methods to handle file upload request
|
||
|
******************************************************************************/
|
||
|
|
||
|
FileTransferMsg CreateFileUploadErrMsg(char* reason, unsigned int reasonLen);
|
||
|
|
||
|
FileTransferMsg
|
||
|
GetFileUploadLengthErrResponseMsg()
|
||
|
{
|
||
|
char reason [] = "Path length exceeds PATH_MAX (4096) bytes";
|
||
|
int reasonLen = strlen(reason);
|
||
|
|
||
|
return CreateFileUploadErrMsg(reason, reasonLen);
|
||
|
}
|
||
|
|
||
|
|
||
|
FileTransferMsg
|
||
|
ChkFileUploadErr(rfbClientPtr cl, rfbTightClientPtr rtcp)
|
||
|
{
|
||
|
FileTransferMsg fileUploadErrMsg;
|
||
|
|
||
|
memset(&fileUploadErrMsg, 0, sizeof(FileTransferMsg));
|
||
|
if( (rtcp->rcft.rcfu.fName == NULL) ||
|
||
|
(strlen(rtcp->rcft.rcfu.fName) == 0) ||
|
||
|
((rtcp->rcft.rcfu.uploadFD = creat(rtcp->rcft.rcfu.fName,
|
||
|
S_IRUSR | S_IWUSR | S_IRGRP | S_IWGRP | S_IROTH | S_IWOTH)) == -1)) {
|
||
|
|
||
|
char reason[] = "Could not create file";
|
||
|
int reasonLen = strlen(reason);
|
||
|
fileUploadErrMsg = CreateFileUploadErrMsg(reason, reasonLen);
|
||
|
}
|
||
|
else
|
||
|
rtcp->rcft.rcfu.uploadInProgress = TRUE;
|
||
|
|
||
|
return fileUploadErrMsg;
|
||
|
}
|
||
|
|
||
|
|
||
|
FileTransferMsg
|
||
|
GetFileUploadCompressedLevelErrMsg()
|
||
|
{
|
||
|
char reason[] = "Server does not support data compression on upload";
|
||
|
int reasonLen = strlen(reason);
|
||
|
|
||
|
return CreateFileUploadErrMsg(reason, reasonLen);
|
||
|
}
|
||
|
|
||
|
|
||
|
FileTransferMsg
|
||
|
ChkFileUploadWriteErr(rfbClientPtr cl, rfbTightClientPtr rtcp, char* pBuf)
|
||
|
{
|
||
|
FileTransferMsg ftm;
|
||
|
unsigned long numOfBytesWritten = 0;
|
||
|
|
||
|
memset(&ftm, 0, sizeof(FileTransferMsg));
|
||
|
|
||
|
numOfBytesWritten = write(rtcp->rcft.rcfu.uploadFD, pBuf, rtcp->rcft.rcfu.fSize);
|
||
|
|
||
|
if(numOfBytesWritten != rtcp->rcft.rcfu.fSize) {
|
||
|
char reason[] = "Error writing file data";
|
||
|
int reasonLen = strlen(reason);
|
||
|
ftm = CreateFileUploadErrMsg(reason, reasonLen);
|
||
|
CloseUndoneFileTransfer(cl, rtcp);
|
||
|
}
|
||
|
return ftm;
|
||
|
}
|
||
|
|
||
|
|
||
|
void
|
||
|
FileUpdateComplete(rfbClientPtr cl, rfbTightClientPtr rtcp)
|
||
|
{
|
||
|
/* Here we are settimg the modification and access time of the file */
|
||
|
/* Windows code stes mod/access/creation time of the file */
|
||
|
struct utimbuf utb;
|
||
|
|
||
|
utb.actime = utb.modtime = rtcp->rcft.rcfu.mTime;
|
||
|
if(utime(rtcp->rcft.rcfu.fName, &utb) == -1) {
|
||
|
rfbLog("File [%s]: Method [%s]: Setting the modification/access"
|
||
|
" time for the file <%s> failed\n", __FILE__,
|
||
|
__FUNCTION__, rtcp->rcft.rcfu.fName);
|
||
|
}
|
||
|
|
||
|
if(rtcp->rcft.rcfu.uploadFD != -1) {
|
||
|
close(rtcp->rcft.rcfu.uploadFD);
|
||
|
rtcp->rcft.rcfu.uploadFD = -1;
|
||
|
rtcp->rcft.rcfu.uploadInProgress = FALSE;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
|
||
|
FileTransferMsg
|
||
|
CreateFileUploadErrMsg(char* reason, unsigned int reasonLen)
|
||
|
{
|
||
|
FileTransferMsg fileUploadErrMsg;
|
||
|
int length = sz_rfbFileUploadCancelMsg + reasonLen;
|
||
|
rfbFileUploadCancelMsg *pFDF = NULL;
|
||
|
char *pFollow = NULL;
|
||
|
|
||
|
char *pData = (char*) calloc(length, sizeof(char));
|
||
|
memset(&fileUploadErrMsg, 0, sizeof(FileTransferMsg));
|
||
|
if(pData == NULL) {
|
||
|
rfbLog("File [%s]: Method [%s]: pData is NULL\n",
|
||
|
__FILE__, __FUNCTION__);
|
||
|
return fileUploadErrMsg;
|
||
|
}
|
||
|
|
||
|
pFDF = (rfbFileUploadCancelMsg *) pData;
|
||
|
pFollow = &pData[sz_rfbFileUploadCancelMsg];
|
||
|
|
||
|
pFDF->type = rfbFileUploadCancel;
|
||
|
pFDF->reasonLen = Swap16IfLE(reasonLen);
|
||
|
memcpy(pFollow, reason, reasonLen);
|
||
|
|
||
|
fileUploadErrMsg.data = pData;
|
||
|
fileUploadErrMsg.length = length;
|
||
|
|
||
|
return fileUploadErrMsg;
|
||
|
}
|
||
|
|
||
|
|
||
|
/******************************************************************************
|
||
|
* Method to cancel File Transfer operation.
|
||
|
******************************************************************************/
|
||
|
|
||
|
void
|
||
|
CloseUndoneFileTransfer(rfbClientPtr cl, rfbTightClientPtr rtcp)
|
||
|
{
|
||
|
/* TODO :: File Upload case is not handled currently */
|
||
|
/* TODO :: In case of concurrency we need to use Critical Section */
|
||
|
|
||
|
if(cl == NULL)
|
||
|
return;
|
||
|
|
||
|
|
||
|
if(rtcp->rcft.rcfu.uploadInProgress == TRUE) {
|
||
|
rtcp->rcft.rcfu.uploadInProgress = FALSE;
|
||
|
|
||
|
if(rtcp->rcft.rcfu.uploadFD != -1) {
|
||
|
close(rtcp->rcft.rcfu.uploadFD);
|
||
|
rtcp->rcft.rcfu.uploadFD = -1;
|
||
|
}
|
||
|
|
||
|
if(unlink(rtcp->rcft.rcfu.fName) == -1) {
|
||
|
rfbLog("File [%s]: Method [%s]: Delete operation on file <%s> failed\n",
|
||
|
__FILE__, __FUNCTION__, rtcp->rcft.rcfu.fName);
|
||
|
}
|
||
|
|
||
|
memset(rtcp->rcft.rcfu.fName, 0 , PATH_MAX);
|
||
|
}
|
||
|
|
||
|
if(rtcp->rcft.rcfd.downloadInProgress == TRUE) {
|
||
|
rtcp->rcft.rcfd.downloadInProgress = FALSE;
|
||
|
|
||
|
if(rtcp->rcft.rcfd.downloadFD != -1) {
|
||
|
close(rtcp->rcft.rcfd.downloadFD);
|
||
|
rtcp->rcft.rcfd.downloadFD = -1;
|
||
|
}
|
||
|
memset(rtcp->rcft.rcfd.fName, 0 , PATH_MAX);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
|
||
|
/******************************************************************************
|
||
|
* Method to handle create directory request.
|
||
|
******************************************************************************/
|
||
|
|
||
|
void
|
||
|
CreateDirectory(char* dirName)
|
||
|
{
|
||
|
if(dirName == NULL) return;
|
||
|
|
||
|
if(mkdir(dirName, 0700) == -1) {
|
||
|
rfbLog("File [%s]: Method [%s]: Create operation for directory <%s> failed\n",
|
||
|
__FILE__, __FUNCTION__, dirName);
|
||
|
}
|
||
|
}
|
||
|
|