273 lines
6.1 KiB
C++
273 lines
6.1 KiB
C++
/*++
|
|
|
|
Copyright (c) 1994 Microsoft Corporation
|
|
|
|
Module Name:
|
|
|
|
tcputil.cxx
|
|
|
|
Abstract:
|
|
|
|
Contains functions to connect to an FTP server
|
|
|
|
Contents:
|
|
FtpOpenServer
|
|
ResetSocket
|
|
|
|
Author:
|
|
|
|
Heath Hunnicutt (t-hheath) 21-Jun-1994
|
|
|
|
Environment:
|
|
|
|
Win32 user-level DLL
|
|
|
|
Revision History:
|
|
|
|
--*/
|
|
|
|
#include <wininetp.h>
|
|
#include "ftpapih.h"
|
|
|
|
|
|
// functions
|
|
|
|
|
|
DWORD
|
|
FtpOpenServer(
|
|
IN LPFTP_SESSION_INFO SessionInfo
|
|
)
|
|
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
Resolves a host name and makes a connection to the FTP server at that host.
|
|
If successful, the controlSocket field of the FTP_SESSION_INFO object will
|
|
contain an opened socket handle
|
|
|
|
Arguments:
|
|
|
|
SessionInfo - pointer to FTP_SESSION_INFO describing host to connect to
|
|
|
|
Return Value:
|
|
|
|
DWORD
|
|
Success - ERROR_SUCCESS
|
|
|
|
Failure - ERROR_INTERNET_NAME_NOT_RESOLVED
|
|
WSA error
|
|
|
|
--*/
|
|
|
|
{
|
|
DEBUG_ENTER((DBG_FTP,
|
|
Dword,
|
|
"FtpOpenServer",
|
|
"%#x",
|
|
SessionInfo
|
|
));
|
|
|
|
LPINTERNET_THREAD_INFO lpThreadInfo = InternetGetThreadInfo();
|
|
DWORD error;
|
|
BOOL fUseSocksProxy = FALSE;
|
|
INTERNET_CONNECT_HANDLE_OBJECT *pConnect;
|
|
AUTO_PROXY_ASYNC_MSG *pProxyInfoQuery = NULL;
|
|
|
|
if (lpThreadInfo == NULL) {
|
|
|
|
INET_ASSERT(FALSE);
|
|
|
|
error = ERROR_INTERNET_INTERNAL_ERROR;
|
|
goto quit;
|
|
}
|
|
|
|
DWORD asyncFlags;
|
|
|
|
asyncFlags = 0;
|
|
//asyncFlags = lpThreadInfo->IsAsyncWorkerThread ? SF_NON_BLOCKING : 0;
|
|
|
|
//
|
|
// attempt to resolve the host name, server port, and possibly service GUID
|
|
// to socket address(es)
|
|
//
|
|
|
|
SessionInfo->socketControl->SetPort(SessionInfo->Port);
|
|
|
|
|
|
//
|
|
// Using the object handle, check to see if we have a socks proxy.
|
|
// If so, use it to do our connections.
|
|
//
|
|
|
|
INTERNET_HANDLE_OBJECT * pInternet;
|
|
HINTERNET hConnectMapped;
|
|
|
|
INET_ASSERT(lpThreadInfo != NULL);
|
|
INET_ASSERT(lpThreadInfo->hObjectMapped != NULL);
|
|
INET_ASSERT(SessionInfo->Host != NULL);
|
|
|
|
//
|
|
// Get the Mapped Connect Handle Object
|
|
//
|
|
|
|
hConnectMapped = lpThreadInfo->hObjectMapped;
|
|
|
|
INET_ASSERT(hConnectMapped);
|
|
|
|
//
|
|
// Finally get the Internet Object, so we can query proxy information
|
|
// out of it.
|
|
//
|
|
|
|
pConnect = (INTERNET_CONNECT_HANDLE_OBJECT *) hConnectMapped;
|
|
|
|
|
|
pInternet = (INTERNET_HANDLE_OBJECT *)
|
|
((INTERNET_CONNECT_HANDLE_OBJECT *)hConnectMapped)->GetParent();
|
|
|
|
INET_ASSERT(pInternet);
|
|
|
|
{
|
|
CHAR szUrl[INTERNET_MAX_URL_LENGTH + sizeof("ftp:// /")];
|
|
|
|
PROXY_STATE * pProxyState = NULL;
|
|
|
|
INTERNET_SCHEME scheme = INTERNET_SCHEME_DEFAULT;
|
|
|
|
if (lstrlen(SessionInfo->Host)>INTERNET_MAX_URL_LENGTH)
|
|
{
|
|
error = ERROR_INSUFFICIENT_BUFFER;
|
|
goto quit;
|
|
}
|
|
wsprintf(szUrl, "ftp://%s/", SessionInfo->Host);
|
|
|
|
AUTO_PROXY_ASYNC_MSG proxyInfoQuery(
|
|
INTERNET_SCHEME_FTP,
|
|
szUrl,
|
|
lstrlen(szUrl),
|
|
SessionInfo->Host,
|
|
lstrlen(SessionInfo->Host),
|
|
(INTERNET_PORT) SessionInfo->Port
|
|
);
|
|
|
|
proxyInfoQuery.SetBlockUntilCompletetion(TRUE);
|
|
|
|
pProxyInfoQuery = &proxyInfoQuery;
|
|
|
|
error = pInternet->GetProxyInfo(
|
|
&pProxyInfoQuery
|
|
);
|
|
|
|
if ( error != ERROR_SUCCESS)
|
|
{
|
|
goto quit2;
|
|
}
|
|
|
|
|
|
if ( pProxyInfoQuery->IsUseProxy() &&
|
|
pProxyInfoQuery->GetProxyScheme() == INTERNET_SCHEME_SOCKS &&
|
|
pProxyInfoQuery->_lpszProxyHostName )
|
|
{
|
|
|
|
//
|
|
// If Socks is enabled, then turn it on.
|
|
//
|
|
|
|
error = SessionInfo->socketControl->EnableSocks(
|
|
pProxyInfoQuery->_lpszProxyHostName,
|
|
pProxyInfoQuery->_nProxyHostPort
|
|
);
|
|
|
|
if ( error != ERROR_SUCCESS)
|
|
{
|
|
goto quit2;
|
|
}
|
|
|
|
error = SessionInfo->socketData->EnableSocks(
|
|
pProxyInfoQuery->_lpszProxyHostName,
|
|
pProxyInfoQuery->_nProxyHostPort
|
|
);
|
|
|
|
|
|
if ( error != ERROR_SUCCESS)
|
|
{
|
|
goto quit2;
|
|
}
|
|
|
|
//
|
|
// Force Passive Mode, since Socks Firewalls may not
|
|
// support connections from the outside in.
|
|
//
|
|
|
|
SessionInfo->Flags |= FFTP_PASSIVE_MODE;
|
|
|
|
}
|
|
|
|
pConnect->SetServerInfo(SessionInfo->Host,
|
|
lstrlen(SessionInfo->Host));
|
|
|
|
|
|
|
|
//
|
|
// name was resolved ok, now let's try to connect to the server. Here
|
|
// we create the control socket
|
|
//
|
|
|
|
error = SessionInfo->socketControl->Connect(
|
|
GetTimeoutValue(INTERNET_OPTION_CONNECT_TIMEOUT),
|
|
GetTimeoutValue(INTERNET_OPTION_CONNECT_RETRIES),
|
|
SF_INDICATE | asyncFlags
|
|
);
|
|
|
|
quit2:
|
|
|
|
if ( pProxyInfoQuery && pProxyInfoQuery->IsAlloced() )
|
|
{
|
|
delete pProxyInfoQuery;
|
|
pProxyInfoQuery = NULL;
|
|
}
|
|
}
|
|
|
|
quit:
|
|
|
|
|
|
DEBUG_LEAVE(error);
|
|
|
|
return error;
|
|
}
|
|
|
|
|
|
BOOL
|
|
ResetSocket(
|
|
IN ICSocket * Socket
|
|
)
|
|
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
Sets linger time to zero on a socket, then closes the socket, causing a
|
|
"hard" close
|
|
|
|
Arguments:
|
|
|
|
Socket - The socket to reset the connection on
|
|
|
|
Return Value:
|
|
|
|
BOOL
|
|
Success - TRUE
|
|
Failure - FALSE
|
|
|
|
--*/
|
|
|
|
{
|
|
//
|
|
// ignore return code from linger - if error, socket is closed or aborted
|
|
//
|
|
|
|
Socket->SetLinger(TRUE, 0);
|
|
return (Socket->Close() == ERROR_SUCCESS) ? TRUE : FALSE;
|
|
}
|