2616 lines
62 KiB
C++
2616 lines
62 KiB
C++
/*++
|
|
|
|
Copyright (c) 1994 Microsoft Corporation
|
|
|
|
Module Name:
|
|
|
|
ftpapir.cxx
|
|
|
|
Abstract:
|
|
|
|
Contains the remote-side FTP API worker functions. In each case, the API
|
|
proper validates the arguments. The worker functions contained herein just
|
|
perform the requested operation with the supplied arguments.
|
|
|
|
These functions are the remote side of the RPC interface. If the DLL is
|
|
the abstract0 version (no RPC) then the A forms of the functions simply
|
|
call the w functions
|
|
|
|
Contents:
|
|
wFtpFindFirstFile
|
|
wFtpDeleteFile
|
|
wFtpRenameFile
|
|
wFtpOpenFile
|
|
wFtpCreateDirectory
|
|
wFtpRemoveDirectory
|
|
wFtpSetCurrentDirectory
|
|
wFtpGetCurrentDirectory
|
|
wFtpCommand
|
|
wFtpFindNextFile
|
|
wFtpFindClose
|
|
wFtpConnect
|
|
wFtpMakeConnection
|
|
wFtpDisconnect
|
|
wFtpReadFile
|
|
wFtpWriteFile
|
|
wFtpQueryDataAvailable
|
|
wFtpCloseFile
|
|
wFtpFindServerType
|
|
wFtpGetFileSize
|
|
|
|
Author:
|
|
|
|
Heath Hunnicutt [t-heathh] 13-Jul-1994
|
|
|
|
Environment:
|
|
|
|
Win32(s) user-level DLL
|
|
|
|
Revision History:
|
|
|
|
09-Mar-1995 rfirth
|
|
Created new file/worker functions from functions contained in
|
|
findfile.c, ftphelp.c
|
|
|
|
--*/
|
|
|
|
#include <wininetp.h>
|
|
#include "ftpapih.h"
|
|
|
|
|
|
// private macros
|
|
|
|
|
|
#define CASE_OF(constant) case constant: return # constant
|
|
|
|
|
|
// private debug functions
|
|
|
|
|
|
#if INET_DEBUG
|
|
|
|
PRIVATE
|
|
DEBUG_FUNCTION
|
|
LPSTR
|
|
InternetMapFtpServerType(
|
|
IN FTP_SERVER_TYPE ServerType
|
|
);
|
|
|
|
#else
|
|
|
|
#define InternetMapFtpServerType(x) (VOID)(x)
|
|
|
|
#endif // INET_DEBUG
|
|
|
|
|
|
// external functions
|
|
|
|
|
|
extern
|
|
DWORD
|
|
InbLocalEndCacheWrite(
|
|
IN HINTERNET hFtpFile,
|
|
IN LPSTR lpszFileExtension,
|
|
IN BOOL fNormal
|
|
);
|
|
|
|
|
|
// functions
|
|
|
|
|
|
|
|
DWORD
|
|
wFtpFindFirstFile(
|
|
IN HINTERNET hFtpSession,
|
|
IN LPCSTR lpszFilespec,
|
|
OUT LPWIN32_FIND_DATA lpFindFileData OPTIONAL,
|
|
OUT LPHINTERNET lphInternet
|
|
)
|
|
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
Download the remote site's directory listing and parse it into
|
|
WIN32_FIND_DATA structures that we can pass back to the app.
|
|
|
|
If the FTP session is currently involved in a data transfer, such as
|
|
a FtpOpenFile()....FtpCloseFile() series of calls, this function will
|
|
fail.
|
|
|
|
Arguments:
|
|
|
|
hFtpSession - Handle to an FTP session, as returned from FtpOpen()
|
|
|
|
lpszFilespec - Pointer to a string containing a file specification
|
|
to find. May be empty, but not NULL
|
|
|
|
lpFindFileData - Pointer to a buffer that will contain WIN32_FIND_DATA
|
|
information when this call succeeds.
|
|
If this parameter is not supplied, then any find data
|
|
will be returned via InternetFindNextFile()
|
|
|
|
lphInternet - place to return open find handle
|
|
|
|
Return Value:
|
|
|
|
DWORD
|
|
Success - ERROR_SUCCESS
|
|
*lphInternet contains new find handle
|
|
|
|
Failure - ERROR_INVALID_HANDLE
|
|
The session handle is not recognized
|
|
|
|
ERROR_FTP_TRANSFER_IN_PROGRESS
|
|
The data connection is already in use
|
|
|
|
ERROR_NO_MORE_FILES
|
|
The end of the directory listing has been reached
|
|
|
|
ERROR_INTERNET_EXTENDED_ERROR
|
|
Call InternetGetLastResponseInfo() for the text
|
|
|
|
ERROR_INTERNET_INTERNAL_ERROR
|
|
Something bad happened
|
|
--*/
|
|
|
|
{
|
|
DEBUG_ENTER((DBG_FTP,
|
|
Dword,
|
|
"wFtpFindFirstFile",
|
|
"%#x, %q, %#x, %#x",
|
|
hFtpSession,
|
|
lpszFilespec,
|
|
lpFindFileData,
|
|
lphInternet
|
|
));
|
|
|
|
LPINTERNET_THREAD_INFO lpThreadInfo = InternetGetThreadInfo();
|
|
LPSTR lpBuffer = NULL;
|
|
DWORD error;
|
|
|
|
if (lpThreadInfo == NULL) {
|
|
|
|
INET_ASSERT(FALSE);
|
|
|
|
error = ERROR_INTERNET_INTERNAL_ERROR;
|
|
goto quit;
|
|
}
|
|
|
|
LPFTP_SESSION_INFO lpSessionInfo;
|
|
|
|
if (!FindFtpSession(hFtpSession, &lpSessionInfo)) {
|
|
error = ERROR_INVALID_HANDLE;
|
|
goto quit;
|
|
}
|
|
|
|
|
|
// acquire the session lock while we check and optionally set the active
|
|
// find flag
|
|
|
|
|
|
AcquireFtpSessionLock(lpSessionInfo);
|
|
|
|
if (!(lpSessionInfo->Flags & FFTP_FIND_ACTIVE)) {
|
|
lpSessionInfo->Flags |= FFTP_FIND_ACTIVE;
|
|
error = ERROR_SUCCESS;
|
|
} else {
|
|
error = ERROR_FTP_TRANSFER_IN_PROGRESS;
|
|
}
|
|
|
|
ReleaseFtpSessionLock(lpSessionInfo);
|
|
|
|
|
|
// if we already have a directory listing on this connection, then we can
|
|
// not allow another one, until the current listing is cleared out by the
|
|
// app calling InternetCloseHandle()
|
|
|
|
|
|
if (error != ERROR_SUCCESS) {
|
|
goto deref_exit;
|
|
}
|
|
|
|
|
|
// the filespec may have a path component. We assume that any wild-cards
|
|
// will only be in the filename part. We use the path part in the directory
|
|
// request and the filename part when parsing the directory output
|
|
|
|
|
|
char pathBuf[INTERNET_MAX_PATH_LENGTH + 1];
|
|
LPSTR lpszPathPart;
|
|
LPSTR lpszFilePart;
|
|
BOOL isWild;
|
|
DWORD dwFilePartLength;
|
|
|
|
lpszFilePart = (LPSTR)lpszFilespec;
|
|
lpszPathPart = NULL;
|
|
dwFilePartLength = lstrlen(lpszFilePart);
|
|
|
|
if (*lpszFilePart != '\0') {
|
|
|
|
LPSTR pathSeparator;
|
|
|
|
pathSeparator = _memrchr(lpszFilePart, '\\', dwFilePartLength);
|
|
if (pathSeparator == NULL) {
|
|
pathSeparator = _memrchr(lpszFilePart, '/', dwFilePartLength);
|
|
}
|
|
if (pathSeparator != NULL) {
|
|
|
|
int len = (int) (pathSeparator - lpszFilePart) + 1;
|
|
|
|
if (len < sizeof(pathBuf)) {
|
|
memcpy(pathBuf, lpszFilePart, len);
|
|
pathBuf[len] = '\0';
|
|
lpszPathPart = pathBuf;
|
|
lpszFilePart = pathSeparator + 1;
|
|
|
|
DEBUG_PRINT(FTP,
|
|
INFO,
|
|
("lpszPathPart = %q, lpszFilePart = %q\n",
|
|
lpszPathPart,
|
|
lpszFilePart
|
|
));
|
|
|
|
}
|
|
}
|
|
|
|
|
|
// determine whether the caller is asking for a fuzzy file match, or
|
|
// (typically) the request is for the contents of a directory
|
|
|
|
|
|
isWild = IsFilespecWild(lpszFilePart);
|
|
} else {
|
|
|
|
|
|
// empty string - not asking for wildcard search
|
|
|
|
|
|
isWild = FALSE;
|
|
}
|
|
|
|
|
|
// and ask the FTP server for the directory listing
|
|
|
|
|
|
FTP_RESPONSE_CODE rcResponse;
|
|
|
|
error = Command(lpSessionInfo,
|
|
TRUE,
|
|
FTP_TRANSFER_TYPE_ASCII,
|
|
&rcResponse,
|
|
((lpszPathPart == NULL) && (isWild || (*lpszFilePart == '\0')))
|
|
? "LIST"
|
|
: "LIST %s",
|
|
(lpszPathPart == NULL)
|
|
? lpszFilePart
|
|
: isWild
|
|
? lpszPathPart
|
|
: lpszFilespec
|
|
);
|
|
|
|
|
|
// quit early if we failed to send the command, or the server didn't
|
|
// understand it
|
|
|
|
|
|
if (error != ERROR_SUCCESS) {
|
|
goto cleanup;
|
|
}
|
|
|
|
|
|
// presumably, the server has sent us a directory listing. Receive it
|
|
|
|
|
|
DWORD bufferLength;
|
|
DWORD bufferLeft;
|
|
DWORD bytesReceived;
|
|
BOOL eof;
|
|
|
|
bufferLength = 0;
|
|
bufferLeft = 0;
|
|
bytesReceived = 0;
|
|
|
|
error = lpSessionInfo->socketData->Receive((LPVOID *)&lpBuffer,
|
|
&bufferLength,
|
|
&bufferLeft,
|
|
&bytesReceived,
|
|
0,
|
|
SF_EXPAND
|
|
| SF_COMPRESS
|
|
| SF_RECEIVE_ALL
|
|
| SF_INDICATE,
|
|
&eof
|
|
);
|
|
|
|
|
|
// we are done with the data connection
|
|
|
|
|
|
lpSessionInfo->socketData->Close();
|
|
|
|
|
|
// quit now if we had an error while receiving
|
|
|
|
|
|
if (error != ERROR_SUCCESS) {
|
|
goto cleanup;
|
|
}
|
|
|
|
|
|
// if the previous response was preliminary then get the final response from
|
|
// the FTP server
|
|
|
|
|
|
if (rcResponse.Major != FTP_RESPONSE_COMPLETE) {
|
|
error = GetReply(lpSessionInfo, &rcResponse);
|
|
if (error != ERROR_SUCCESS) {
|
|
goto cleanup;
|
|
}
|
|
|
|
|
|
// check response for failure
|
|
|
|
|
|
if (rcResponse.Major != FTP_RESPONSE_COMPLETE) {
|
|
|
|
|
|
// <-- Return "command failed" error code
|
|
|
|
|
|
error = ERROR_INTERNET_EXTENDED_ERROR;
|
|
goto cleanup;
|
|
}
|
|
}
|
|
|
|
if (bytesReceived == 0) {
|
|
|
|
DEBUG_PRINT(WORKER,
|
|
ERROR,
|
|
("ReceiveData() returns 0 bytes\n"
|
|
));
|
|
|
|
error = ERROR_NO_MORE_FILES;
|
|
goto cleanup;
|
|
}
|
|
|
|
|
|
// trap bad servers which return a not-found message in the data stream. We
|
|
// only do this if we are not performing a wild-card search (because the
|
|
// wild-card match will fail to match anything if the target file or path
|
|
// cannot be found)
|
|
|
|
|
|
LPSTR lpszSearch;
|
|
DWORD dwSearch;
|
|
|
|
lpszSearch = (lpszPathPart == NULL) ? lpszFilePart : (LPSTR)lpszFilespec;
|
|
dwSearch = lstrlen(lpszSearch);
|
|
|
|
if (!isWild && (bytesReceived > dwSearch)) {
|
|
if (!_strnicmp(lpBuffer, lpszSearch, dwSearch)
|
|
&& (lpBuffer[dwSearch] == ':')) {
|
|
|
|
static char testChars[] = {'\r', '\n', '\0'};
|
|
LPSTR lpStartOfString = lpBuffer + dwSearch + 1;
|
|
LPSTR lpEndOfString;
|
|
|
|
for (int i = 0; i < ARRAY_ELEMENTS(testChars); ++i) {
|
|
lpEndOfString = strchr(lpStartOfString, testChars[i]);
|
|
if (lpEndOfString != NULL) {
|
|
break;
|
|
}
|
|
}
|
|
|
|
|
|
// we should have found at least one of the target characters
|
|
|
|
|
|
INET_ASSERT(lpEndOfString != NULL);
|
|
|
|
if (lpEndOfString != NULL) {
|
|
|
|
int lengthToTest = (int) (lpEndOfString - lpStartOfString);
|
|
|
|
|
|
// BUGBUG - internationalization?
|
|
|
|
|
|
if (strnistr(lpStartOfString, "not found", lengthToTest)
|
|
|| strnistr(lpStartOfString, "cannot find", lengthToTest)) {
|
|
error = ERROR_NO_MORE_FILES;
|
|
goto cleanup;
|
|
}
|
|
} else {
|
|
error = ERROR_INTERNET_INTERNAL_ERROR;
|
|
goto cleanup;
|
|
}
|
|
}
|
|
}
|
|
|
|
INET_ASSERT(lpBuffer != NULL);
|
|
INET_ASSERT((int)bytesReceived > 0);
|
|
|
|
error = ParseDirList(lpBuffer,
|
|
bytesReceived,
|
|
isWild ? (LPSTR)lpszFilePart : NULL,
|
|
&lpSessionInfo->FindFileList
|
|
);
|
|
|
|
|
|
// ParseDirList() may have failed
|
|
|
|
|
|
if (error != ERROR_SUCCESS) {
|
|
goto cleanup;
|
|
}
|
|
|
|
|
|
// if there's nothing in the list then no files matching the caller's
|
|
// specification were found
|
|
|
|
|
|
if (IsListEmpty(&lpSessionInfo->FindFileList)) {
|
|
error = ERROR_NO_MORE_FILES;
|
|
} else {
|
|
|
|
|
|
// if the caller supplied an output buffer then return the first entry
|
|
// and remove it from the list
|
|
|
|
|
|
if (ARGUMENT_PRESENT(lpFindFileData)) {
|
|
|
|
PLIST_ENTRY pEntry;
|
|
|
|
pEntry = RemoveHeadList(&lpSessionInfo->FindFileList);
|
|
CopyMemory(lpFindFileData,
|
|
(LPWIN32_FIND_DATA)(pEntry + 1),
|
|
sizeof(*lpFindFileData)
|
|
);
|
|
FREE_MEMORY(pEntry);
|
|
}
|
|
|
|
|
|
// FTP can only have one active operation per session, so we just return
|
|
// this session handle as the find handle
|
|
|
|
|
|
*lphInternet = hFtpSession;
|
|
error = ERROR_SUCCESS;
|
|
}
|
|
|
|
cleanup:
|
|
|
|
if (lpSessionInfo->socketData->IsValid()) {
|
|
lpSessionInfo->socketData->SetLinger(TRUE, 0);
|
|
lpSessionInfo->socketData->Close();
|
|
}
|
|
|
|
if (lpBuffer != NULL) {
|
|
(void)FREE_MEMORY((HLOCAL)lpBuffer);
|
|
}
|
|
|
|
|
|
// if we failed then reset the active find flag. We set it, so we know it
|
|
// is safe to reset without acquiring the session lock
|
|
|
|
|
|
if (error != ERROR_SUCCESS) {
|
|
lpSessionInfo->Flags &= ~FFTP_FIND_ACTIVE;
|
|
}
|
|
|
|
deref_exit:
|
|
|
|
DereferenceFtpSession(lpSessionInfo);
|
|
|
|
quit:
|
|
|
|
DEBUG_LEAVE(error);
|
|
|
|
return error;
|
|
}
|
|
|
|
|
|
DWORD
|
|
wFtpDeleteFile(
|
|
IN HINTERNET hFtpSession,
|
|
IN LPCSTR lpszFileName
|
|
)
|
|
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
Deletes a file at an FTP server
|
|
|
|
Arguments:
|
|
|
|
hFtpSession - identifies the FTP server
|
|
|
|
lpszFileName - name of file to delete
|
|
|
|
Return Value:
|
|
|
|
DWORD
|
|
Success - ERROR_SUCCESS
|
|
|
|
Failure -
|
|
|
|
--*/
|
|
|
|
{
|
|
DEBUG_ENTER((DBG_FTP,
|
|
Dword,
|
|
"wFtpDeleteFile",
|
|
"%#x, %q",
|
|
hFtpSession,
|
|
lpszFileName
|
|
));
|
|
|
|
LPFTP_SESSION_INFO lpSessionInfo;
|
|
DWORD error;
|
|
|
|
if (FindFtpSession(hFtpSession, &lpSessionInfo)) {
|
|
|
|
FTP_RESPONSE_CODE rcResponse;
|
|
|
|
error = Command(lpSessionInfo,
|
|
FALSE,
|
|
FTP_TRANSFER_TYPE_UNKNOWN,
|
|
&rcResponse,
|
|
"DELE %s",
|
|
lpszFileName
|
|
);
|
|
|
|
if ((error == ERROR_SUCCESS)
|
|
&& (rcResponse.Major != FTP_RESPONSE_COMPLETE)) {
|
|
error = ERROR_INTERNET_EXTENDED_ERROR;
|
|
}
|
|
DereferenceFtpSession(lpSessionInfo);
|
|
} else {
|
|
error = ERROR_INVALID_HANDLE;
|
|
}
|
|
|
|
DEBUG_LEAVE(error);
|
|
|
|
return error;
|
|
}
|
|
|
|
|
|
DWORD
|
|
wFtpRenameFile(
|
|
IN HINTERNET hFtpSession,
|
|
IN LPCSTR lpszExisting,
|
|
IN LPCSTR lpszNew
|
|
)
|
|
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
Renames a file at an FTP server
|
|
|
|
Arguments:
|
|
|
|
hFtpSession - identifies FTP server
|
|
|
|
lpszExisting - current file name
|
|
|
|
lpszNew - new file name
|
|
|
|
Return Value:
|
|
|
|
DWORD
|
|
Success - ERROR_SUCCESS
|
|
|
|
Failure -
|
|
|
|
--*/
|
|
|
|
{
|
|
DEBUG_ENTER((DBG_FTP,
|
|
Dword,
|
|
"wFtpRenameFile",
|
|
"%#x, %q, %q",
|
|
hFtpSession,
|
|
lpszExisting,
|
|
lpszNew
|
|
));
|
|
|
|
LPFTP_SESSION_INFO lpSessionInfo;
|
|
DWORD error;
|
|
|
|
if (FindFtpSession(hFtpSession, &lpSessionInfo)) {
|
|
|
|
FTP_RESPONSE_CODE rcResponse;
|
|
|
|
error = Command(lpSessionInfo,
|
|
FALSE,
|
|
FTP_TRANSFER_TYPE_UNKNOWN,
|
|
&rcResponse,
|
|
"RNFR %s",
|
|
lpszExisting
|
|
);
|
|
|
|
if ((error == ERROR_SUCCESS)
|
|
&& (rcResponse.Major != FTP_RESPONSE_CONTINUE)) {
|
|
error = ERROR_INTERNET_EXTENDED_ERROR;
|
|
}
|
|
if (error == ERROR_SUCCESS) {
|
|
error = Command(lpSessionInfo,
|
|
FALSE,
|
|
FTP_TRANSFER_TYPE_UNKNOWN,
|
|
&rcResponse,
|
|
"RNTO %s",
|
|
lpszNew
|
|
);
|
|
if ((error == ERROR_SUCCESS)
|
|
&& (rcResponse.Major != FTP_RESPONSE_COMPLETE)) {
|
|
error = ERROR_INTERNET_EXTENDED_ERROR;
|
|
}
|
|
}
|
|
DereferenceFtpSession(lpSessionInfo);
|
|
} else {
|
|
error = ERROR_INVALID_HANDLE;
|
|
}
|
|
|
|
DEBUG_LEAVE(error);
|
|
|
|
return error;
|
|
}
|
|
|
|
|
|
DWORD
|
|
wFtpOpenFile(
|
|
IN HINTERNET hFtpSession,
|
|
IN LPCSTR lpszFileName,
|
|
IN DWORD dwAccess,
|
|
IN DWORD dwFlags,
|
|
OUT LPHINTERNET lphInternet
|
|
)
|
|
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
Initiates the connection to read or write a file at the FTP server
|
|
|
|
Arguments:
|
|
|
|
hFtpSession - identifies FTP server
|
|
|
|
lpszFileName - name of file to open
|
|
|
|
dwAccess - access mode - GENERIC_READ or GENERIC_WRITE
|
|
|
|
dwFlags - flags controlling how to transfer the data
|
|
|
|
lphInternet - where to return the open file handle
|
|
|
|
Return Value:
|
|
|
|
DWORD
|
|
Success - ERROR_SUCCESS
|
|
|
|
Failure -
|
|
|
|
--*/
|
|
|
|
{
|
|
DEBUG_ENTER((DBG_FTP,
|
|
Dword,
|
|
"wFtpOpenFile",
|
|
"%#x, %q, %#x, %#x, %#x",
|
|
hFtpSession,
|
|
lpszFileName,
|
|
dwAccess,
|
|
dwFlags,
|
|
lphInternet
|
|
));
|
|
|
|
LPFTP_SESSION_INFO lpSessionInfo;
|
|
DWORD error;
|
|
|
|
if (FindFtpSession(hFtpSession, &lpSessionInfo)) {
|
|
|
|
|
|
// control session must be established
|
|
|
|
|
|
if (! lpSessionInfo->socketControl->IsValid()) {
|
|
error = ERROR_FTP_DROPPED;
|
|
} else if ((lpSessionInfo->socketData->IsValid())
|
|
|| (lpSessionInfo->Flags & FFTP_FILE_ACTIVE)) {
|
|
|
|
|
|
// there is a (file) transfer in progress if the socket is valid,
|
|
// or we are awaiting a call to InternetCloseHandle() before we can
|
|
// open another file (FFTP_FILE_ACTIVE is set. This stops another
|
|
// thread from closing our socket handle)
|
|
|
|
|
|
error = ERROR_FTP_TRANSFER_IN_PROGRESS;
|
|
} else {
|
|
|
|
FTP_RESPONSE_CODE rcResponse;
|
|
|
|
INET_ASSERT(!lpSessionInfo->socketData->IsValid());
|
|
|
|
|
|
// Clear the session's "known size bit" before we download the next file,
|
|
// this is to make sure we don't read an extranous size value off it.
|
|
|
|
|
|
lpSessionInfo->Flags &= ~(FFTP_KNOWN_FILE_SIZE);
|
|
|
|
|
|
// send the connection set-up commands, and issue either the send
|
|
// or the receive command
|
|
|
|
// Either "RETR filename" or "STOR filename"
|
|
|
|
|
|
error = NegotiateDataConnection(lpSessionInfo,
|
|
dwFlags,
|
|
&rcResponse,
|
|
(dwAccess & GENERIC_READ)
|
|
? "RETR %s"
|
|
: "STOR %s",
|
|
lpszFileName
|
|
);
|
|
|
|
if (error == ERROR_SUCCESS) {
|
|
|
|
|
|
// Check response for failure
|
|
|
|
|
|
if ((rcResponse.Major != FTP_RESPONSE_PRELIMINARY)
|
|
&& (rcResponse.Major != FTP_RESPONSE_COMPLETE)) {
|
|
|
|
ICSocket * socketData;
|
|
|
|
|
|
// BUGBUG - RLF - don't know if this is what's intended
|
|
// here, but the code just used to check
|
|
// socketData != INVALID_SOCKET. Since socketData
|
|
// was getting set to INVALID_SOCKET at the top
|
|
// of this routine, this branch would never be
|
|
// taken
|
|
|
|
|
|
socketData = lpSessionInfo->socketData;
|
|
if (socketData->IsValid()) {
|
|
ResetSocket(socketData);
|
|
}
|
|
error = ERROR_INTERNET_EXTENDED_ERROR;
|
|
} else {
|
|
|
|
lpSessionInfo->dwTransferAccess = dwAccess;
|
|
|
|
|
|
// Some FTP servers will send us back both the preliminary
|
|
// response and the complete response so quickly that we
|
|
// will never see the preliminary.
|
|
|
|
// In order for FtpCloseFile() to know that the completion
|
|
// response has been received, we store the response
|
|
// structure in the Session Info.
|
|
|
|
// The response structure only needs to be stored between
|
|
// API calls in this situation, it is not generally
|
|
// referred to.
|
|
|
|
|
|
SetSessionLastResponseCode(lpSessionInfo, &rcResponse);
|
|
|
|
|
|
// set the abort flag if the file was opened for read - this
|
|
// lets the server know it can clean up the session if we
|
|
// close early
|
|
|
|
|
|
if (dwAccess & GENERIC_READ) {
|
|
lpSessionInfo->Flags |= FFTP_ABORT_TRANSFER;
|
|
}
|
|
|
|
|
|
// FTP can only have one active operation per session, so
|
|
// we just return this session handle as the find handle
|
|
|
|
|
|
*lphInternet = hFtpSession;
|
|
|
|
|
|
// this session has an active file operation
|
|
|
|
|
|
lpSessionInfo->Flags |= FFTP_FILE_ACTIVE;
|
|
|
|
|
|
// N.B. error == ERROR_SUCCESS from above test after call
|
|
// to NegotiateDataConnection
|
|
|
|
|
|
INET_ASSERT(error == ERROR_SUCCESS);
|
|
}
|
|
}
|
|
}
|
|
DereferenceFtpSession(lpSessionInfo);
|
|
} else {
|
|
error = ERROR_INVALID_HANDLE;
|
|
}
|
|
|
|
DEBUG_LEAVE(error);
|
|
|
|
return error;
|
|
}
|
|
|
|
|
|
DWORD
|
|
wFtpCreateDirectory(
|
|
IN HINTERNET hFtpSession,
|
|
IN LPCSTR lpszDirectory
|
|
)
|
|
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
Creates a directory at the FTP server
|
|
|
|
Arguments:
|
|
|
|
hFtpSession - identifies the FTP server
|
|
|
|
lpszDirectory - directory to create
|
|
|
|
Return Value:
|
|
|
|
DWORD
|
|
Success - ERROR_SUCCESS
|
|
|
|
Failure -
|
|
|
|
--*/
|
|
|
|
{
|
|
DEBUG_ENTER((DBG_FTP,
|
|
Dword,
|
|
"wFtpCreateDirectory",
|
|
"%#x, %q",
|
|
hFtpSession,
|
|
lpszDirectory
|
|
));
|
|
|
|
LPFTP_SESSION_INFO lpSessionInfo;
|
|
DWORD error;
|
|
|
|
if (FindFtpSession(hFtpSession, &lpSessionInfo)) {
|
|
|
|
FTP_RESPONSE_CODE rcResponse;
|
|
|
|
error = Command(lpSessionInfo,
|
|
FALSE,
|
|
FTP_TRANSFER_TYPE_UNKNOWN,
|
|
&rcResponse,
|
|
"MKD %s",
|
|
lpszDirectory
|
|
);
|
|
if ((error == ERROR_SUCCESS)
|
|
&& (rcResponse.Major != FTP_RESPONSE_COMPLETE)) {
|
|
error = ERROR_INTERNET_EXTENDED_ERROR;
|
|
}
|
|
DereferenceFtpSession(lpSessionInfo);
|
|
} else {
|
|
error = ERROR_INVALID_HANDLE;
|
|
}
|
|
|
|
DEBUG_LEAVE(error);
|
|
|
|
return error;
|
|
}
|
|
|
|
|
|
DWORD
|
|
wFtpRemoveDirectory(
|
|
IN HINTERNET hFtpSession,
|
|
IN LPCSTR lpszDirectory
|
|
)
|
|
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
Removes the named directory at the FTP server
|
|
|
|
Arguments:
|
|
|
|
hFtpSession - identifies the FTP server
|
|
|
|
lpszDirectory - directory to remove
|
|
|
|
Return Value:
|
|
|
|
DWORD
|
|
Success - ERROR_SUCCESS
|
|
|
|
Failure -
|
|
|
|
--*/
|
|
|
|
{
|
|
DEBUG_ENTER((DBG_FTP,
|
|
Dword,
|
|
"wFtpRemoveDirectory",
|
|
"%#x, %q",
|
|
hFtpSession,
|
|
lpszDirectory
|
|
));
|
|
|
|
LPFTP_SESSION_INFO lpSessionInfo;
|
|
DWORD error;
|
|
|
|
if (FindFtpSession(hFtpSession, &lpSessionInfo)) {
|
|
|
|
FTP_RESPONSE_CODE rcResponse;
|
|
|
|
error = Command(lpSessionInfo,
|
|
FALSE,
|
|
FTP_TRANSFER_TYPE_UNKNOWN,
|
|
&rcResponse,
|
|
"RMD %s",
|
|
lpszDirectory
|
|
);
|
|
if ((error == ERROR_SUCCESS)
|
|
&& (rcResponse.Major != FTP_RESPONSE_COMPLETE)) {
|
|
error = ERROR_INTERNET_EXTENDED_ERROR;
|
|
}
|
|
DereferenceFtpSession(lpSessionInfo);
|
|
} else {
|
|
error = ERROR_INVALID_HANDLE;
|
|
}
|
|
|
|
DEBUG_LEAVE(error);
|
|
|
|
return error;
|
|
}
|
|
|
|
|
|
DWORD
|
|
wFtpSetCurrentDirectory(
|
|
IN HINTERNET hFtpSession,
|
|
IN LPCSTR lpszDirectory
|
|
)
|
|
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
Sets the current directory for this FTP server session
|
|
|
|
Arguments:
|
|
|
|
hFtpSession - identifies the FTP server/session
|
|
|
|
lpszDirectory - name of directory to set
|
|
|
|
Return Value:
|
|
|
|
DWORD
|
|
Success - ERROR_SUCCESS
|
|
|
|
Failure -
|
|
|
|
--*/
|
|
|
|
{
|
|
DEBUG_ENTER((DBG_FTP,
|
|
Dword,
|
|
"wFtpSetCurrentDirectory",
|
|
"%#x, %q",
|
|
hFtpSession,
|
|
lpszDirectory
|
|
));
|
|
|
|
LPFTP_SESSION_INFO lpSessionInfo;
|
|
DWORD error;
|
|
|
|
if (FindFtpSession(hFtpSession, &lpSessionInfo)) {
|
|
|
|
FTP_RESPONSE_CODE rcResponse;
|
|
|
|
error = Command(lpSessionInfo,
|
|
FALSE,
|
|
FTP_TRANSFER_TYPE_UNKNOWN,
|
|
&rcResponse,
|
|
"CWD %s",
|
|
lpszDirectory
|
|
);
|
|
if ((error == ERROR_SUCCESS)
|
|
&& (rcResponse.Major != FTP_RESPONSE_COMPLETE)) {
|
|
error = ERROR_INTERNET_EXTENDED_ERROR;
|
|
}
|
|
DereferenceFtpSession(lpSessionInfo);
|
|
} else {
|
|
error = ERROR_INVALID_HANDLE;
|
|
}
|
|
|
|
DEBUG_LEAVE(error);
|
|
|
|
return error;
|
|
}
|
|
|
|
|
|
DWORD
|
|
wFtpGetCurrentDirectory(
|
|
IN HINTERNET hFtpSession,
|
|
IN DWORD cchCurrentDirectory,
|
|
OUT LPSTR lpszCurrentDirectory,
|
|
OUT LPDWORD lpdwBytesReturned
|
|
)
|
|
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
Gets the current working directory at the FTP server for this session
|
|
|
|
Arguments:
|
|
|
|
hFtpSession - identifies FTP server
|
|
|
|
cchCurrentDirectory - number of characters in lpszCurrentDirectory
|
|
|
|
lpszCurrentDirectory - buffer where current directory string is written
|
|
|
|
lpdwBytesReturned - number of characters in output string NOT including
|
|
terminating NUL
|
|
|
|
Return Value:
|
|
|
|
DWORD
|
|
Success - ERROR_SUCCESS
|
|
|
|
Failure - ERROR_INVALID_HANDLE
|
|
|
|
ERROR_INSUFFICIENT_BUFFER
|
|
The buffer in lpszCurrentDirectory is not large enough to
|
|
hold the directory string. *lpdwBytesReturned will have
|
|
the required size
|
|
|
|
--*/
|
|
|
|
{
|
|
DEBUG_ENTER((DBG_FTP,
|
|
Dword,
|
|
"wFtpGetCurrentDirectory",
|
|
"%#x, %d, %#x, %#x",
|
|
hFtpSession,
|
|
cchCurrentDirectory,
|
|
lpszCurrentDirectory,
|
|
lpdwBytesReturned
|
|
));
|
|
|
|
LPFTP_SESSION_INFO lpSessionInfo;
|
|
DWORD cchCopied;
|
|
DWORD error;
|
|
|
|
if (FindFtpSession(hFtpSession, &lpSessionInfo)) {
|
|
|
|
FTP_RESPONSE_CODE rcResponse;
|
|
|
|
error = Command(lpSessionInfo,
|
|
FALSE,
|
|
FTP_TRANSFER_TYPE_UNKNOWN,
|
|
&rcResponse,
|
|
"PWD"
|
|
);
|
|
if ((error == ERROR_SUCCESS)
|
|
&& (rcResponse.Major != FTP_RESPONSE_COMPLETE)) {
|
|
error = ERROR_INTERNET_EXTENDED_ERROR;
|
|
}
|
|
if (error == ERROR_SUCCESS) {
|
|
|
|
LPSTR pchResponse;
|
|
|
|
|
|
// parse the returned directory name out of the response text
|
|
|
|
|
|
pchResponse = InternetLockErrorText();
|
|
if (pchResponse != NULL) {
|
|
pchResponse = strstr(pchResponse, "257 ");
|
|
if (pchResponse != NULL) {
|
|
pchResponse = strchr(pchResponse, '\"');
|
|
if (pchResponse != NULL) {
|
|
|
|
int idx;
|
|
|
|
++pchResponse;
|
|
for (idx = 0, cchCopied = 0; pchResponse[idx] != '\0'; idx++) {
|
|
if (pchResponse[idx] == '\"') {
|
|
if (pchResponse[idx + 1] == '\"') {
|
|
continue;
|
|
}
|
|
break;
|
|
}
|
|
if (cchCopied < cchCurrentDirectory) {
|
|
lpszCurrentDirectory[cchCopied] = pchResponse[idx];
|
|
}
|
|
cchCopied++;
|
|
}
|
|
if (cchCopied < cchCurrentDirectory) {
|
|
lpszCurrentDirectory[cchCopied] = '\0';
|
|
error = ERROR_SUCCESS;
|
|
} else {
|
|
error = ERROR_INSUFFICIENT_BUFFER;
|
|
++cchCopied;
|
|
}
|
|
} else {
|
|
error = ERROR_INTERNET_EXTENDED_ERROR;
|
|
}
|
|
}
|
|
//InternetUnlockErrorText();
|
|
}
|
|
}
|
|
DereferenceFtpSession(lpSessionInfo);
|
|
} else {
|
|
error = ERROR_INVALID_HANDLE;
|
|
}
|
|
|
|
if ((error == ERROR_SUCCESS) || (error == ERROR_INSUFFICIENT_BUFFER)) {
|
|
*lpdwBytesReturned = cchCopied;
|
|
}
|
|
|
|
DEBUG_LEAVE(error);
|
|
|
|
return error;
|
|
}
|
|
|
|
|
|
DWORD
|
|
wFtpCommand(
|
|
IN HINTERNET hFtpSession,
|
|
IN BOOL fExpectResponse,
|
|
IN DWORD dwFlags,
|
|
IN LPCSTR lpszCommand
|
|
)
|
|
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
Runs arbitrary command at an FTP server. Direct connect over Internet
|
|
|
|
Arguments:
|
|
|
|
hFtpSession - identifies the FTP server
|
|
|
|
fExpectResponse - TRUE if we expect a response from the server
|
|
|
|
dwFlags - type of response - ASCII text or BINARY data
|
|
|
|
lpszCommand - pointer to string describing command to run
|
|
|
|
Return Value:
|
|
|
|
DWORD
|
|
Success - ERROR_SUCCESS
|
|
|
|
Failure -
|
|
|
|
--*/
|
|
|
|
{
|
|
DEBUG_ENTER((DBG_FTP,
|
|
Dword,
|
|
"wFtpCommand",
|
|
"%#x, %#x, %#x, %q",
|
|
hFtpSession,
|
|
fExpectResponse,
|
|
dwFlags,
|
|
lpszCommand
|
|
));
|
|
|
|
LPFTP_SESSION_INFO lpSessionInfo;
|
|
DWORD error;
|
|
|
|
|
|
// Look up the given handle.
|
|
|
|
|
|
if (FindFtpSession(hFtpSession, &lpSessionInfo)) {
|
|
|
|
FTP_RESPONSE_CODE rcResponse;
|
|
|
|
|
|
// Issue the command.
|
|
|
|
|
|
error = Command(lpSessionInfo,
|
|
fExpectResponse,
|
|
dwFlags,
|
|
&rcResponse,
|
|
lpszCommand
|
|
);
|
|
if (fExpectResponse && (error == ERROR_SUCCESS)) {
|
|
|
|
INET_ASSERT(lpSessionInfo->socketData->IsValid());
|
|
|
|
lpSessionInfo->dwTransferAccess |= (GENERIC_READ|GENERIC_WRITE);
|
|
|
|
}
|
|
#if DBG
|
|
else {
|
|
|
|
INET_ASSERT(! lpSessionInfo->socketData->IsValid());
|
|
|
|
}
|
|
|
|
if (error == ERROR_SUCCESS) {
|
|
|
|
INET_ASSERT(lpSessionInfo->socketControl->IsValid());
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
DereferenceFtpSession(lpSessionInfo);
|
|
} else {
|
|
error = ERROR_INVALID_HANDLE;
|
|
}
|
|
|
|
DEBUG_LEAVE(error);
|
|
|
|
return error;
|
|
}
|
|
|
|
|
|
|
|
// Internet subordinate functions
|
|
|
|
|
|
DWORD
|
|
wFtpFindNextFile(
|
|
IN HINTERNET hFtpSession,
|
|
OUT LPWIN32_FIND_DATA lpFindFileData
|
|
)
|
|
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
Returns the next file found from a call to FtpFindFirstFile().
|
|
|
|
Arguments:
|
|
|
|
hFtpSession - Handle to an FTP session, as returned from FtpConnect()
|
|
|
|
lpFindFileData - Pointer to a buffer that will contain WIN32_FIND_DATA
|
|
information when this call succeeds.
|
|
|
|
Return Value:
|
|
|
|
DWORD
|
|
Success - ERROR_SUCCESS
|
|
|
|
Failure - ERROR_NO_MORE_FILES
|
|
The end of the file list has been reached.
|
|
|
|
ERROR_INVALID_HANDLE
|
|
Can't find session that knows about hFind
|
|
--*/
|
|
|
|
{
|
|
DEBUG_ENTER((DBG_FTP,
|
|
Dword,
|
|
"wFtpFindNextFile",
|
|
"%#x, %#x",
|
|
hFtpSession,
|
|
lpFindFileData
|
|
));
|
|
|
|
LPFTP_SESSION_INFO lpSessionInfo;
|
|
DWORD error;
|
|
|
|
if (FindFtpSession(hFtpSession, &lpSessionInfo)) {
|
|
|
|
|
|
// ISSUE this code is cut & paste from find first - they should both call a
|
|
// fn instead
|
|
|
|
|
|
if (!IsListEmpty(&lpSessionInfo->FindFileList)) {
|
|
|
|
PLIST_ENTRY pEntry;
|
|
|
|
|
|
// Enumerate the first entry and advance pointers
|
|
|
|
|
|
pEntry = RemoveHeadList(&lpSessionInfo->FindFileList);
|
|
|
|
INET_ASSERT(pEntry != NULL);
|
|
|
|
CopyMemory(lpFindFileData,
|
|
(LPWIN32_FIND_DATA)(pEntry + 1),
|
|
sizeof(WIN32_FIND_DATA)
|
|
);
|
|
FREE_MEMORY(pEntry);
|
|
error = ERROR_SUCCESS;
|
|
} else {
|
|
error = ERROR_NO_MORE_FILES;
|
|
}
|
|
DereferenceFtpSession(lpSessionInfo);
|
|
} else {
|
|
error = ERROR_INVALID_HANDLE;
|
|
}
|
|
|
|
DEBUG_LEAVE(error);
|
|
|
|
return error;
|
|
}
|
|
|
|
|
|
DWORD
|
|
wFtpFindClose(
|
|
IN HINTERNET hFtpSession
|
|
)
|
|
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
Frees the WIN32_FIND_DATA structures in the directory list for this session
|
|
|
|
Arguments:
|
|
|
|
hFtpSession - handle of an FTP session, created by InternetConnect
|
|
|
|
Return Value:
|
|
|
|
DWORD
|
|
Success - ERROR_SUCCESS
|
|
|
|
Failure - ERROR_INVALID_HANDLE
|
|
|
|
--*/
|
|
|
|
{
|
|
DEBUG_ENTER((DBG_FTP,
|
|
Dword,
|
|
"wFtpFindClose",
|
|
"%#x",
|
|
hFtpSession
|
|
));
|
|
|
|
LPFTP_SESSION_INFO lpSessionInfo;
|
|
DWORD error;
|
|
|
|
if (FindFtpSession(hFtpSession, &lpSessionInfo)) {
|
|
ClearFindList(&lpSessionInfo->FindFileList);
|
|
|
|
|
|
// this session no longer has an active directory listing
|
|
|
|
|
|
lpSessionInfo->Flags &= ~FFTP_FIND_ACTIVE;
|
|
DereferenceFtpSession(lpSessionInfo);
|
|
error = ERROR_SUCCESS;
|
|
} else {
|
|
error = ERROR_INVALID_HANDLE;
|
|
}
|
|
|
|
DEBUG_LEAVE(error);
|
|
|
|
return error;
|
|
}
|
|
|
|
|
|
DWORD
|
|
wFtpConnect(
|
|
IN LPCSTR lpszServerName,
|
|
IN INTERNET_PORT nServerPort,
|
|
IN LPCSTR lpszUsername,
|
|
IN LPCSTR lpszPassword,
|
|
IN DWORD dwService,
|
|
IN DWORD dwFlags,
|
|
OUT LPHINTERNET lphInternet
|
|
)
|
|
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
Creates a new FTP session object
|
|
|
|
Arguments:
|
|
|
|
lpszServerName - pointer to string identifying FTP server
|
|
|
|
nServerPort - port number to connect to
|
|
|
|
lpszUsername - pointer to string identifying user name to log on as
|
|
|
|
lpszPassword - pointer to string identifying password to use with user name
|
|
|
|
dwService - service type parameter (unused)
|
|
|
|
dwFlags - session flags. Currently only INTERNET_FLAG_PASSIVE
|
|
is defined
|
|
|
|
lphInternet - returned handle of created FTP session
|
|
|
|
Return Value:
|
|
|
|
DWORD
|
|
Success - ERROR_SUCCESS
|
|
|
|
Failure - ERROR_NOT_ENOUGH_MEMORY
|
|
Ran out of memory while creating the session object
|
|
|
|
ERROR_INTERNET_OUT_OF_HANDLES
|
|
Ran out of handles while creating the session object
|
|
|
|
ERROR_INTERNET_SHUTDOWN
|
|
The DLL is being unloaded
|
|
|
|
--*/
|
|
|
|
{
|
|
INET_ASSERT(lpszUsername != NULL);
|
|
INET_ASSERT(lpszPassword != NULL);
|
|
|
|
DEBUG_ENTER((DBG_FTP,
|
|
Dword,
|
|
"wFtpConnect",
|
|
"%q, %d, %q, %q, %d, %#x, %#x",
|
|
lpszServerName,
|
|
nServerPort,
|
|
lpszUsername,
|
|
lpszPassword,
|
|
dwService,
|
|
dwFlags,
|
|
lphInternet
|
|
));
|
|
|
|
DWORD error;
|
|
LPFTP_SESSION_INFO sessionInfo;
|
|
|
|
UNREFERENCED_PARAMETER(lpszUsername);
|
|
UNREFERENCED_PARAMETER(lpszPassword);
|
|
UNREFERENCED_PARAMETER(dwService);
|
|
|
|
|
|
// create a new FTP session object
|
|
|
|
|
|
error = CreateFtpSession((LPSTR)lpszServerName,
|
|
nServerPort,
|
|
|
|
|
|
// if INTERNET_FLAG_PASSIVE then create a passive
|
|
// session object
|
|
|
|
|
|
(dwFlags & INTERNET_FLAG_PASSIVE)
|
|
? FFTP_PASSIVE_MODE
|
|
: 0,
|
|
&sessionInfo
|
|
);
|
|
if (error == ERROR_SUCCESS) {
|
|
|
|
|
|
// return the FTP_SESSION_INFO handle
|
|
|
|
|
|
*lphInternet = sessionInfo->Handle;
|
|
}
|
|
|
|
DEBUG_LEAVE(error);
|
|
|
|
return error;
|
|
}
|
|
|
|
|
|
DWORD
|
|
wFtpMakeConnection(
|
|
IN HINTERNET hFtpSession,
|
|
IN LPCSTR lpszUsername,
|
|
IN LPCSTR lpszPassword
|
|
)
|
|
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
Connect with and log into an FTP server.
|
|
|
|
This function is cancellable
|
|
|
|
Arguments:
|
|
|
|
hFtpSession - handle of an FTP session, created by InternetConnect
|
|
|
|
pszUsername - pointer to string identifying user name to log on as
|
|
|
|
pszPassword - pointer to string identifying password to use with user name
|
|
|
|
Return Value:
|
|
|
|
DWORD
|
|
Success - ERROR_SUCCESS
|
|
|
|
Failure - ERROR_INTERNET_INCORRECT_USER_NAME
|
|
The server didn't like the user name
|
|
|
|
ERROR_INTERNET_INCORRECT_PASSWORD
|
|
The server didn't like the password
|
|
|
|
ERROR_INTERNET_LOGIN_FAILURE
|
|
The server rejected the login request
|
|
|
|
ERROR_FTP_DROPPED
|
|
The connection has been closed
|
|
|
|
ERROR_FTP_TRANSFER_IN_PROGRESS
|
|
There is already a transfer in progress on this connection
|
|
|
|
ERROR_INTERNET_NAME_NOT_RESOLVED
|
|
Couldn't resolve the server name
|
|
|
|
WSA error
|
|
Couldn't connect to the server, or problems while
|
|
communicating with it
|
|
|
|
--*/
|
|
|
|
{
|
|
DEBUG_ENTER((DBG_FTP,
|
|
Dword,
|
|
"wFtpMakeConnection",
|
|
"%#x, %q, %q",
|
|
hFtpSession,
|
|
lpszUsername,
|
|
lpszPassword
|
|
));
|
|
|
|
LPFTP_SESSION_INFO sessionInfo;
|
|
DWORD error;
|
|
|
|
if (FindFtpSession(hFtpSession, &sessionInfo)) {
|
|
|
|
|
|
// resolve the FTP server's host name and connect to the server
|
|
|
|
|
|
error = FtpOpenServer(sessionInfo);
|
|
if (error == ERROR_SUCCESS) {
|
|
|
|
FTP_RESPONSE_CODE rcResponse;
|
|
|
|
|
|
// set send and receive timeouts on the control channel socket.
|
|
// Ignore any errors
|
|
|
|
|
|
sessionInfo->socketControl->SetTimeout(
|
|
SEND_TIMEOUT,
|
|
GetTimeoutValue(INTERNET_OPTION_CONTROL_SEND_TIMEOUT)
|
|
);
|
|
|
|
sessionInfo->socketControl->SetTimeout(
|
|
RECEIVE_TIMEOUT,
|
|
GetTimeoutValue(INTERNET_OPTION_CONTROL_RECEIVE_TIMEOUT)
|
|
);
|
|
|
|
|
|
// check greeting and store in per-thread response text buffer
|
|
|
|
|
|
error = GetReply(sessionInfo, &rcResponse);
|
|
if (error == ERROR_SUCCESS) {
|
|
|
|
|
|
// check that the server sent us an affirmative response
|
|
|
|
|
|
if (rcResponse.Major == FTP_RESPONSE_COMPLETE) {
|
|
|
|
|
|
// send the user name
|
|
|
|
|
|
error = Command(sessionInfo,
|
|
FALSE,
|
|
FTP_TRANSFER_TYPE_UNKNOWN,
|
|
&rcResponse,
|
|
"USER %s",
|
|
lpszUsername
|
|
);
|
|
|
|
|
|
// BUGBUG - is it possible to get success from Command(),
|
|
// but an error from the server - e.g. 332, need
|
|
// account for login?
|
|
|
|
|
|
if (error == ERROR_SUCCESS) {
|
|
|
|
|
|
// send the password if required
|
|
|
|
|
|
if (rcResponse.Major == FTP_RESPONSE_CONTINUE) {
|
|
error = Command(sessionInfo,
|
|
FALSE,
|
|
FTP_TRANSFER_TYPE_UNKNOWN,
|
|
&rcResponse,
|
|
"PASS %s",
|
|
lpszPassword
|
|
);
|
|
|
|
|
|
// if we failed to send the password, or the password
|
|
// was rejected, or we are attempting to log on as
|
|
// "anonymous" and it turns out that the server does
|
|
// not allow anonymous logon, then return a password
|
|
// error. The caller can still check the response
|
|
// from the server
|
|
|
|
|
|
if (((error == ERROR_SUCCESS)
|
|
&& (rcResponse.Major != FTP_RESPONSE_COMPLETE))
|
|
|| (error == ERROR_INTERNET_EXTENDED_ERROR)) {
|
|
if (stricmp(lpszUsername, "anonymous") == 0) {
|
|
error = ERROR_INTERNET_LOGIN_FAILURE;
|
|
} else {
|
|
error = ERROR_INTERNET_INCORRECT_PASSWORD;
|
|
}
|
|
}
|
|
} else if (rcResponse.Major != FTP_RESPONSE_COMPLETE) {
|
|
error = ERROR_INTERNET_INCORRECT_USER_NAME;
|
|
}
|
|
|
|
|
|
// get the server type
|
|
|
|
|
|
//if (error == ERROR_SUCCESS) {
|
|
// error = wFtpFindServerType(hFtpSession);
|
|
//}
|
|
}
|
|
} else {
|
|
error = ERROR_INTERNET_LOGIN_FAILURE;
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
// success or fail: unlock the session object
|
|
|
|
|
|
DereferenceFtpSession(sessionInfo);
|
|
|
|
|
|
// if we failed to login then let wFtpDisconnect() clean up - it will
|
|
// also send a "QUIT" to the server (if we have a control connection)
|
|
// which will ensure a clean exit
|
|
|
|
|
|
if (error != ERROR_SUCCESS) {
|
|
|
|
|
|
// if we experience an error during disconnect, we will just ignore
|
|
// it and return the error generated during our failed login attempt
|
|
|
|
|
|
(void)wFtpDisconnect(hFtpSession, CF_EXPEDITED_CLOSE);
|
|
}
|
|
} else {
|
|
error = ERROR_INVALID_HANDLE;
|
|
}
|
|
|
|
DEBUG_LEAVE(error);
|
|
|
|
return error;
|
|
}
|
|
|
|
|
|
DWORD
|
|
wFtpDisconnect(
|
|
IN HINTERNET hFtpSession,
|
|
IN DWORD dwFlags
|
|
)
|
|
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
Closes the connection, issues the quit command, etc.,
|
|
|
|
Arguments:
|
|
|
|
hFtpSession - FTP session created by wFtpConnect
|
|
|
|
dwFlags - controlling operation. Can be:
|
|
|
|
CF_EXPEDITED_CLOSE - Don't send QUIT to the server, just
|
|
close the control connection
|
|
|
|
Return Value:
|
|
|
|
DWORD
|
|
Success - ERROR_SUCCESS
|
|
|
|
Failure - ERROR_INVALID_HANDLE
|
|
|
|
--*/
|
|
|
|
{
|
|
DEBUG_ENTER((DBG_FTP,
|
|
Dword,
|
|
"wFtpDisconnect",
|
|
"%#x, %#x",
|
|
hFtpSession,
|
|
dwFlags
|
|
));
|
|
|
|
LPFTP_SESSION_INFO lpSessionInfo;
|
|
DWORD error;
|
|
LPINTERNET_THREAD_INFO lpThreadInfo = InternetGetThreadInfo();
|
|
|
|
if (lpThreadInfo == NULL) {
|
|
|
|
INET_ASSERT(FALSE);
|
|
|
|
error = ERROR_INTERNET_INTERNAL_ERROR;
|
|
goto quit;
|
|
}
|
|
|
|
if (FindFtpSession(hFtpSession, &lpSessionInfo)) {
|
|
|
|
ICSocket * socketControl;
|
|
ICSocket * socketData;
|
|
|
|
socketControl = lpSessionInfo->socketControl;
|
|
socketData = lpSessionInfo->socketData;
|
|
|
|
|
|
// kill any active data transfer
|
|
|
|
|
|
if (socketData->IsValid()) {
|
|
|
|
|
|
// set the non-blocking state depending on whether we are called in
|
|
// an app thread context, or in the async scheduler thread context
|
|
|
|
|
|
//socketData->SetNonBlockingMode(lpThreadInfo->IsAsyncWorkerThread);
|
|
|
|
if (dwFlags & CF_EXPEDITED_CLOSE) {
|
|
error = socketData->Close();
|
|
} else {
|
|
error = wFtpCloseFile(hFtpSession);
|
|
if (error != ERROR_SUCCESS) {
|
|
|
|
DEBUG_PRINT(WORKER,
|
|
ERROR,
|
|
("wFtpCloseFile() returns %d\n",
|
|
error
|
|
));
|
|
|
|
}
|
|
}
|
|
}
|
|
|
|
INET_ASSERT(!lpSessionInfo->socketData->IsValid());
|
|
|
|
|
|
// perform graceful close to the server if we have a control connection
|
|
|
|
|
|
if (socketControl->IsValid()) {
|
|
|
|
|
|
// set the non-blocking state depending on whether we are called in
|
|
// an app thread context, or in the async scheduler thread context
|
|
|
|
|
|
//socketControl->SetNonBlockingMode(lpThreadInfo->IsAsyncWorkerThread);
|
|
|
|
if (!(dwFlags & CF_EXPEDITED_CLOSE)) {
|
|
|
|
FTP_RESPONSE_CODE rcResponse;
|
|
|
|
Command(lpSessionInfo,
|
|
FALSE,
|
|
FTP_TRANSFER_TYPE_UNKNOWN,
|
|
&rcResponse,
|
|
"QUIT"
|
|
);
|
|
}
|
|
lpSessionInfo->socketControl->Disconnect(SF_INDICATE);
|
|
}
|
|
|
|
|
|
// finally kill the FTP_SESSION_INFO structure
|
|
|
|
|
|
TerminateFtpSession(lpSessionInfo);
|
|
error = ERROR_SUCCESS;
|
|
} else {
|
|
error = ERROR_INVALID_HANDLE;
|
|
}
|
|
|
|
quit:
|
|
|
|
DEBUG_LEAVE(error);
|
|
|
|
return error;
|
|
}
|
|
|
|
|
|
DWORD
|
|
wFtpReadFile(
|
|
IN HINTERNET hFtpSession,
|
|
IN LPVOID lpBuffer,
|
|
IN DWORD dwNumberOfBytesToRead,
|
|
OUT LPDWORD lpdwNumberOfBytesRead
|
|
)
|
|
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
Reads data from the FTP server. We use the data channel
|
|
|
|
Arguments:
|
|
|
|
hFtpSession - handle identifying FTP session
|
|
|
|
lpBuffer - pointer to buffer for received data
|
|
|
|
dwNumberOfBytesToRead - size of lpBuffer in bytes
|
|
|
|
lpdwNumberOfBytesRead - returned number of bytes received
|
|
|
|
Return Value:
|
|
|
|
DWORD
|
|
Success - ERROR_SUCCESS
|
|
|
|
Failure - ERROR_INVALID_HANDLE
|
|
Couldn't find hFtpSession
|
|
|
|
ERROR_ACCESS_DENIED
|
|
This session doesn't have read access (?)
|
|
|
|
ERROR_FTP_DROPPED
|
|
The data channel has been closed
|
|
|
|
--*/
|
|
|
|
{
|
|
DEBUG_ENTER((DBG_FTP,
|
|
Dword,
|
|
"wFtpReadFile",
|
|
"%#x, %#x, %d, %#x",
|
|
hFtpSession,
|
|
lpBuffer,
|
|
dwNumberOfBytesToRead,
|
|
lpdwNumberOfBytesRead
|
|
));
|
|
|
|
LPINTERNET_THREAD_INFO lpThreadInfo = InternetGetThreadInfo();
|
|
DWORD error;
|
|
|
|
if (lpThreadInfo == NULL) {
|
|
|
|
INET_ASSERT(FALSE);
|
|
|
|
error = ERROR_INTERNET_INTERNAL_ERROR;
|
|
goto quit;
|
|
}
|
|
|
|
LPFTP_SESSION_INFO lpSessionInfo;
|
|
ICSocket * socketData;
|
|
BOOL eof;
|
|
DWORD bytesReceived;
|
|
|
|
|
|
// initialize variables in case we quit early (i.e. via goto)
|
|
|
|
|
|
bytesReceived = 0;
|
|
|
|
|
|
// find the FTP_SESSION_INFO and ensure it is set up to receive data
|
|
|
|
|
|
if (!FindFtpSession(hFtpSession, &lpSessionInfo)) {
|
|
error = ERROR_INVALID_HANDLE;
|
|
goto quit;
|
|
}
|
|
|
|
|
|
// if FFTP_EOF is set then we already reached the end the file
|
|
|
|
|
|
if (lpSessionInfo->Flags & FFTP_EOF) {
|
|
error = ERROR_SUCCESS;
|
|
goto unlock_and_quit;
|
|
}
|
|
|
|
|
|
// get the data socket. If it has become INVALID_SOCKET then the server
|
|
// closed the connection
|
|
|
|
|
|
socketData = lpSessionInfo->socketData;
|
|
if (!socketData->IsValid()) {
|
|
error = ERROR_FTP_DROPPED;
|
|
goto unlock_and_quit;
|
|
}
|
|
|
|
if (!(lpSessionInfo->dwTransferAccess & GENERIC_READ)) {
|
|
error = ERROR_ACCESS_DENIED;
|
|
goto unlock_and_quit;
|
|
}
|
|
|
|
|
|
// read until we fill the users buffer, get an error, or get to EOF
|
|
|
|
|
|
DWORD bufferRemaining;
|
|
|
|
bufferRemaining = dwNumberOfBytesToRead;
|
|
error = socketData->Receive(
|
|
&lpBuffer,
|
|
&dwNumberOfBytesToRead, // lpdwBufferLength
|
|
&bufferRemaining, // lpdwBufferRemaining
|
|
&bytesReceived, // lpdwBytesReceived
|
|
0, // dwExtraSpace
|
|
SF_RECEIVE_ALL
|
|
| SF_INDICATE,
|
|
&eof
|
|
);
|
|
if (error == ERROR_SUCCESS) {
|
|
|
|
|
|
// if we got to EOF then the server will have closed the data
|
|
// connection. We need to close the socket at our end. If this is
|
|
// a passive connection then we initiate session termination
|
|
|
|
|
|
if (eof) {
|
|
(void)socketData->Close();
|
|
|
|
INET_ASSERT(lpSessionInfo->socketData == socketData);
|
|
|
|
|
|
// reset the abort flag - we no longer have to send and ABOR command
|
|
// when we close the handle
|
|
|
|
|
|
lpSessionInfo->Flags &= ~FFTP_ABORT_TRANSFER;
|
|
|
|
|
|
// set EOF in the FTP_SESSION_INFO flags so we know next time
|
|
// we call this function that the session is not dropped, but
|
|
// that we already reached the end of the data
|
|
|
|
|
|
lpSessionInfo->Flags |= FFTP_EOF;
|
|
}
|
|
}
|
|
|
|
|
|
// BUGBUG - in error case we should probably close the socket, set
|
|
// INVALID_SOCKET in the FTP_SESSION_INFO, etc.
|
|
|
|
|
|
unlock_and_quit:
|
|
|
|
|
|
// update the output parameters if we succeeded
|
|
|
|
|
|
if (error == ERROR_SUCCESS) {
|
|
*lpdwNumberOfBytesRead = bytesReceived;
|
|
}
|
|
|
|
DereferenceFtpSession(lpSessionInfo);
|
|
|
|
quit:
|
|
|
|
DEBUG_LEAVE(error);
|
|
|
|
return error;
|
|
}
|
|
|
|
|
|
DWORD
|
|
wFtpWriteFile(
|
|
IN HINTERNET hFtpSession,
|
|
IN LPVOID lpBuffer,
|
|
IN DWORD dwNumberOfBytesToWrite,
|
|
OUT LPDWORD lpdwNumberOfBytesWritten
|
|
)
|
|
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
Writes data to the FTP server. We use the data channel
|
|
|
|
Arguments:
|
|
|
|
hFtpSession - handle identifying FTP session
|
|
|
|
lpBuffer - pointer to buffer containing data to write
|
|
|
|
dwNumberOfBytesToWrite - size of lpBuffer in bytes
|
|
|
|
lpdwNumberOfBytesWritten - returned number of bytes sent
|
|
|
|
Return Value:
|
|
|
|
DWORD
|
|
Success - ERROR_SUCCESS
|
|
|
|
Failure - ERROR_INVALID_HANDLE
|
|
Couldn't find hFtpSession
|
|
|
|
ERROR_ACCESS_DENIED
|
|
This session doesn't have write access (?)
|
|
|
|
ERROR_FTP_DROPPED
|
|
The data channel has been closed
|
|
|
|
--*/
|
|
|
|
{
|
|
DEBUG_ENTER((DBG_FTP,
|
|
Dword,
|
|
"wFtpWriteFile",
|
|
"%#x, %#x, %d, %#x",
|
|
hFtpSession,
|
|
lpBuffer,
|
|
dwNumberOfBytesToWrite,
|
|
lpdwNumberOfBytesWritten
|
|
));
|
|
|
|
LPINTERNET_THREAD_INFO lpThreadInfo = InternetGetThreadInfo();
|
|
DWORD error;
|
|
|
|
if (lpThreadInfo == NULL) {
|
|
|
|
INET_ASSERT(FALSE);
|
|
|
|
error = ERROR_INTERNET_INTERNAL_ERROR;
|
|
goto quit;
|
|
}
|
|
|
|
LPFTP_SESSION_INFO lpSessionInfo;
|
|
ICSocket * socketData;
|
|
int nSent;
|
|
|
|
|
|
// find the FTP_SESSION_INFO and ensure it is set up to send data
|
|
|
|
|
|
if (!FindFtpSession(hFtpSession, &lpSessionInfo)) {
|
|
error = ERROR_INVALID_HANDLE;
|
|
goto quit;
|
|
}
|
|
|
|
socketData = lpSessionInfo->socketData;
|
|
if (! socketData->IsValid()) {
|
|
error = ERROR_FTP_DROPPED;
|
|
goto unlock_and_quit;
|
|
}
|
|
|
|
if (!(lpSessionInfo->dwTransferAccess & GENERIC_WRITE)) {
|
|
error = ERROR_ACCESS_DENIED;
|
|
goto unlock_and_quit;
|
|
}
|
|
|
|
error = socketData->Send(lpBuffer, dwNumberOfBytesToWrite, SF_INDICATE);
|
|
if (error == ERROR_SUCCESS) {
|
|
*lpdwNumberOfBytesWritten = dwNumberOfBytesToWrite;
|
|
} else {
|
|
|
|
|
|
// we had a failure. We should check the control socket for any error
|
|
// info from the server
|
|
|
|
|
|
//FTP_RESPONSE_CODE response;
|
|
|
|
//(void)GetReply(lpSessionInfo, &response);
|
|
}
|
|
|
|
unlock_and_quit:
|
|
|
|
DereferenceFtpSession(lpSessionInfo);
|
|
|
|
quit:
|
|
|
|
DEBUG_LEAVE(error);
|
|
|
|
return error;
|
|
}
|
|
|
|
|
|
DWORD
|
|
wFtpQueryDataAvailable(
|
|
IN HINTERNET hFtpSession,
|
|
OUT LPDWORD lpdwNumberOfBytesAvailable
|
|
)
|
|
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
Determines amount of data available to be received on a data (file) socket
|
|
|
|
Arguments:
|
|
|
|
hFtpSession - identifies FTP session
|
|
|
|
lpdwNumberOfBytesAvailable - returned number of bytes available
|
|
|
|
Return Value:
|
|
|
|
DWORD
|
|
Success - ERROR_SUCCESS
|
|
|
|
Failure - ERROR_INVALID_HANDLE
|
|
|
|
--*/
|
|
|
|
{
|
|
DEBUG_ENTER((DBG_FTP,
|
|
Dword,
|
|
"wFtpQueryDataAvailable",
|
|
"%#x, %#x",
|
|
hFtpSession,
|
|
lpdwNumberOfBytesAvailable
|
|
));
|
|
|
|
LPINTERNET_THREAD_INFO lpThreadInfo = InternetGetThreadInfo();
|
|
DWORD error;
|
|
|
|
HINTERNET_HANDLE_TYPE handleType;
|
|
|
|
error = RGetHandleType(lpThreadInfo->hObjectMapped, &handleType);
|
|
|
|
if (error != ERROR_SUCCESS) {
|
|
return (error);
|
|
}
|
|
|
|
*lpdwNumberOfBytesAvailable = 0;
|
|
|
|
if (lpThreadInfo == NULL) {
|
|
|
|
INET_ASSERT(FALSE);
|
|
|
|
error = ERROR_INTERNET_INTERNAL_ERROR;
|
|
goto quit;
|
|
}
|
|
|
|
LPFTP_SESSION_INFO lpSessionInfo;
|
|
|
|
if (FindFtpSession(hFtpSession, &lpSessionInfo)) {
|
|
|
|
|
|
// if we are currently performing a directory list then return the size
|
|
// of a dir list entry
|
|
|
|
|
|
if (lpSessionInfo->Flags & FFTP_FIND_ACTIVE) {
|
|
*lpdwNumberOfBytesAvailable = !IsListEmpty(&lpSessionInfo->FindFileList)
|
|
? sizeof(WIN32_FIND_DATA) : 0;
|
|
} else {
|
|
|
|
|
|
// otherwise, if we are receiving data, find out how much
|
|
|
|
|
|
ICSocket * socketData;
|
|
|
|
socketData = lpSessionInfo->socketData;
|
|
if (socketData->IsValid()) {
|
|
error = socketData->DataAvailable(lpdwNumberOfBytesAvailable);
|
|
} else {
|
|
|
|
|
|
// there is no data connection
|
|
|
|
|
|
*lpdwNumberOfBytesAvailable = 0;
|
|
error = ERROR_SUCCESS;
|
|
}
|
|
}
|
|
DereferenceFtpSession(lpSessionInfo);
|
|
} else {
|
|
error = ERROR_INVALID_HANDLE;
|
|
}
|
|
|
|
quit:
|
|
|
|
if ((error == ERROR_SUCCESS) && (*lpdwNumberOfBytesAvailable == 0)) {
|
|
|
|
InbLocalEndCacheWrite(lpThreadInfo->hObjectMapped,
|
|
((handleType==TypeFtpFindHandleHtml)
|
|
?"htm":NULL),
|
|
TRUE);
|
|
}
|
|
|
|
DEBUG_LEAVE(error);
|
|
|
|
return error;
|
|
}
|
|
|
|
|
|
DWORD
|
|
wFtpCloseFile(
|
|
IN HINTERNET hFtpSession
|
|
)
|
|
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
Terminates the connection used for file transfer. The connection may already
|
|
be closed (by the server during a READ, or by the client during a WRITE) in
|
|
which case we just need to receive the confirmation (226) on the control
|
|
socket. If the connection is still open, or the abort flag is set for this
|
|
connection, then this is an abnormal termination, and we need to send an
|
|
ABORt command
|
|
|
|
Arguments:
|
|
|
|
hFtpSession - Identifies the session on which to terminate file transfer
|
|
|
|
Return Value:
|
|
|
|
DWORD
|
|
Success - ERROR_SUCCESS
|
|
|
|
Failure - ERROR_INVALID_HANDLE
|
|
Couldn't find the FTP_SESSION_INFO corresponding to
|
|
hFtpSession
|
|
|
|
--*/
|
|
|
|
{
|
|
DEBUG_ENTER((DBG_FTP,
|
|
Dword,
|
|
"wFtpCloseFile",
|
|
"%#x",
|
|
hFtpSession
|
|
));
|
|
|
|
LPFTP_SESSION_INFO lpSessionInfo;
|
|
DWORD error;
|
|
|
|
if (FindFtpSession(hFtpSession, &lpSessionInfo)) {
|
|
|
|
BOOL getResponse;
|
|
ICSocket * socketData;
|
|
FTP_RESPONSE_CODE rcResponse;
|
|
|
|
socketData = lpSessionInfo->socketData;
|
|
if (socketData->IsValid()) {
|
|
|
|
|
|
// if we are performing a read/write operation and the transfer
|
|
// isn't complete then abort the connection
|
|
|
|
|
|
if (lpSessionInfo->Flags & FFTP_ABORT_TRANSFER) {
|
|
AbortTransfer(lpSessionInfo);
|
|
ResetSocket(lpSessionInfo->socketData);
|
|
} else {
|
|
|
|
|
|
// in all other cases - completed READ, complete or incomplete
|
|
// WRITE - just close the socket
|
|
|
|
|
|
lpSessionInfo->socketData->Close();
|
|
}
|
|
} else if (lpSessionInfo->Flags & FFTP_ABORT_TRANSFER) {
|
|
|
|
|
|
// we have no data socket, but the abort transfer flag is set. We
|
|
// are probably closing a file we opened for read without having
|
|
// read any data. In this case we send an abort anyway
|
|
|
|
|
|
AbortTransfer(lpSessionInfo);
|
|
}
|
|
|
|
|
|
// get the server response - we expect either 226 to a good transfer,
|
|
// or 426 for an aborted transfer...
|
|
|
|
|
|
GetSessionLastResponseCode(lpSessionInfo, &rcResponse);
|
|
if (rcResponse.Major == FTP_RESPONSE_PRELIMINARY) {
|
|
error = GetReply(lpSessionInfo, &rcResponse);
|
|
if ((error == ERROR_SUCCESS)
|
|
&& (rcResponse.Major != FTP_RESPONSE_COMPLETE)) {
|
|
error = ERROR_INTERNET_EXTENDED_ERROR;
|
|
}
|
|
} else {
|
|
error = ERROR_SUCCESS;
|
|
}
|
|
|
|
|
|
// reset the ABORT, FILE_ACTIVE and EOF flags
|
|
|
|
|
|
lpSessionInfo->Flags &= ~(FFTP_ABORT_TRANSFER
|
|
| FFTP_EOF
|
|
| FFTP_FILE_ACTIVE
|
|
);
|
|
DereferenceFtpSession(lpSessionInfo);
|
|
} else {
|
|
error = ERROR_INVALID_HANDLE;
|
|
}
|
|
|
|
DEBUG_LEAVE(error);
|
|
|
|
return error;
|
|
}
|
|
|
|
|
|
DWORD
|
|
wFtpFindServerType(
|
|
IN HINTERNET hFtpSession
|
|
)
|
|
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
Determines the type of server we are talking to (NT or Unix)
|
|
|
|
Arguments:
|
|
|
|
hFtpSession - identifies FTP_SESSION_INFO. The structure ServerType field
|
|
will be updated with the discovered info
|
|
|
|
Return Value:
|
|
|
|
DWORD
|
|
Success - ERROR_SUCCESS
|
|
|
|
Failure - ERROR_INVALID_HANDLE
|
|
Couldn't find the FTP_SESSION_INFO corresponding to
|
|
hFtpSession
|
|
|
|
--*/
|
|
|
|
{
|
|
DEBUG_ENTER((DBG_FTP,
|
|
Dword,
|
|
"wFtpFindServerType",
|
|
"%#x",
|
|
hFtpSession
|
|
));
|
|
|
|
LPFTP_SESSION_INFO lpSessionInfo;
|
|
DWORD error;
|
|
|
|
if (FindFtpSession(hFtpSession, &lpSessionInfo)) {
|
|
|
|
FTP_RESPONSE_CODE rcResponse;
|
|
|
|
error = Command(lpSessionInfo,
|
|
FALSE,
|
|
FTP_TRANSFER_TYPE_UNKNOWN,
|
|
&rcResponse,
|
|
"SYST"
|
|
);
|
|
if (error == ERROR_SUCCESS) {
|
|
|
|
LPSTR lpszResponse = InternetLockErrorText();
|
|
|
|
if (lpszResponse != NULL) {
|
|
|
|
FTP_SERVER_TYPE serverType = FTP_SERVER_TYPE_UNKNOWN;
|
|
|
|
|
|
// "215 " must be first token in response text
|
|
|
|
|
|
lpszResponse = strstr(lpszResponse, "215 ");
|
|
if (lpszResponse != NULL) {
|
|
|
|
|
|
// check for existence of "Windows_NT" or "Unix" (case
|
|
// insensitive comparison)
|
|
|
|
|
|
|
|
// BUGBUG - find out from MuraliK/TerryK the values these
|
|
// ids can have
|
|
|
|
|
|
static struct {
|
|
LPCSTR lpszSystemName;
|
|
FTP_SERVER_TYPE ServerType;
|
|
} FtpServerTypes[] = {
|
|
"Windows_NT", FTP_SERVER_TYPE_NT,
|
|
"Unix", FTP_SERVER_TYPE_UNIX
|
|
};
|
|
|
|
DWORD textLength = strlen(lpszResponse);
|
|
|
|
for (int i = 0; i < ARRAY_ELEMENTS(FtpServerTypes); ++i) {
|
|
if (strnistr(lpszResponse,
|
|
(LPSTR)FtpServerTypes[i].lpszSystemName,
|
|
textLength
|
|
) != NULL) {
|
|
|
|
serverType = FtpServerTypes[i].ServerType;
|
|
|
|
DEBUG_PRINT(FTP,
|
|
INFO,
|
|
("serverType = %s (%d)\n",
|
|
InternetMapFtpServerType(serverType),
|
|
serverType
|
|
));
|
|
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
lpSessionInfo->ServerType = serverType;
|
|
//InternetUnlockErrorText();
|
|
}
|
|
}
|
|
DereferenceFtpSession(lpSessionInfo);
|
|
} else {
|
|
error = ERROR_INVALID_HANDLE;
|
|
}
|
|
|
|
DEBUG_LEAVE(error);
|
|
|
|
return error;
|
|
}
|
|
|
|
|
|
|
|
#if 0
|
|
|
|
|
|
// We don't use this today, because FtpGetFileSize does not support
|
|
// issuing backround commands through the FTP Control socket while
|
|
// the user is doing an FTP download (with FtpOpenFile)
|
|
|
|
|
|
|
|
DWORD
|
|
wFtpGetFileSize(
|
|
IN HINTERNET hMappedFtpSession,
|
|
IN LPFTP_SESSION_INFO lpSessionInfo,
|
|
OUT LPDWORD lpdwFileSizeLow,
|
|
OUT LPDWORD lpdwFileSizeHigh
|
|
)
|
|
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
Finds size of a file at server
|
|
|
|
Arguments:
|
|
|
|
hFtpSession - identifies mapped FTP handle obj
|
|
|
|
lpSessionInfo - LPFTP_SESSION_INFO structure ptr.
|
|
|
|
lpdwFileSizeLow - pointer to low dword of file size
|
|
|
|
lpdwFileSizeHigh - optional output pointer to high dword of file size
|
|
|
|
Return Value:
|
|
|
|
DWORD
|
|
Success - ERROR_SUCCESS
|
|
|
|
Failure - ERROR_INVALID_HANDLE
|
|
Couldn't find the FTP_SESSION_INFO corresponding to
|
|
hFtpSession
|
|
|
|
--*/
|
|
|
|
{
|
|
DEBUG_ENTER((DBG_FTP,
|
|
Dword,
|
|
"wFtpGetFileSize",
|
|
"%#x, %#x, %#x, %#x",
|
|
hMappedFtpSession,
|
|
lpSessionInfo,
|
|
lpdwFileSizeLow,
|
|
lpdwFileSizeHigh
|
|
));
|
|
|
|
|
|
DWORD error = ERROR_INTERNET_INTERNAL_ERROR;
|
|
FTP_FILE_HANDLE_OBJECT * pFileMapped = (FTP_FILE_HANDLE_OBJECT *) hMappedFtpSession;
|
|
|
|
*lpdwFileSizeLow = 0;
|
|
*lpdwFileSizeHigh = 0;
|
|
|
|
if (lpSessionInfo) {
|
|
|
|
FTP_RESPONSE_CODE rcResponse;
|
|
|
|
error = Command(lpSessionInfo,
|
|
FALSE,
|
|
FTP_TRANSFER_TYPE_UNKNOWN,
|
|
&rcResponse,
|
|
"SIZE %s",
|
|
pFileMapped->GetFileName()
|
|
);
|
|
if (error == ERROR_SUCCESS) {
|
|
|
|
LPSTR lpszResponse = InternetLockErrorText();
|
|
|
|
if (lpszResponse != NULL) {
|
|
|
|
FTP_SERVER_TYPE serverType = FTP_SERVER_TYPE_UNKNOWN;
|
|
|
|
|
|
// "213 " must be first token in response text of file size
|
|
|
|
|
|
lpszResponse = strstr(lpszResponse, "213 ");
|
|
if (lpszResponse != NULL) {
|
|
*lpdwFileSizeLow = atoi(lpszResponse);
|
|
error = ERROR_SUCCESS;
|
|
}
|
|
}
|
|
}
|
|
} else {
|
|
error = ERROR_INVALID_HANDLE;
|
|
}
|
|
|
|
DEBUG_LEAVE(error);
|
|
|
|
return error;
|
|
}
|
|
#endif
|
|
|
|
|
|
// private debug functions
|
|
|
|
|
|
#if INET_DEBUG
|
|
|
|
PRIVATE
|
|
DEBUG_FUNCTION
|
|
LPSTR
|
|
InternetMapFtpServerType(
|
|
IN FTP_SERVER_TYPE ServerType
|
|
)
|
|
{
|
|
switch (ServerType) {
|
|
CASE_OF(FTP_SERVER_TYPE_UNKNOWN);
|
|
CASE_OF(FTP_SERVER_TYPE_NT);
|
|
CASE_OF(FTP_SERVER_TYPE_UNIX);
|
|
}
|
|
return "?";
|
|
}
|
|
|
|
#endif // INET_DEBUG
|