1549 lines
44 KiB
C++
1549 lines
44 KiB
C++
/*++
|
|
|
|
Copyright (c) 2000 Microsoft Corporation
|
|
|
|
Module Name:
|
|
|
|
InetSess.cpp
|
|
|
|
Abstract:
|
|
|
|
Implements the Passport Session that uses WinInet as the underlying transport.
|
|
|
|
Author:
|
|
|
|
Biao Wang (biaow) 01-Oct-2000
|
|
|
|
--*/
|
|
|
|
#include "PPdefs.h"
|
|
|
|
#ifdef HINTERNET
|
|
#undef HINTERNET
|
|
#endif
|
|
|
|
#ifdef INTERNET_PORT
|
|
#undef INTERNET_PORT
|
|
#endif
|
|
|
|
#include <WinInet.h>
|
|
#include "session.h"
|
|
|
|
// #include "inetsess.tmh"
|
|
|
|
//
|
|
// func pointer decl for WinInet
|
|
//
|
|
|
|
typedef HINTERNET
|
|
(WINAPI * PFN_INTERNET_OPEN)(
|
|
IN LPCWSTR lpwszAgent,
|
|
IN DWORD dwAccessType,
|
|
IN LPCWSTR lpwszProxy OPTIONAL,
|
|
IN LPCWSTR lpwszProxyBypass OPTIONAL,
|
|
IN DWORD dwFlags
|
|
);
|
|
|
|
typedef HINTERNET
|
|
(WINAPI * PFN_INTERNET_CONNECT)(
|
|
IN HINTERNET hInternet,
|
|
IN LPCWSTR lpwszServerName,
|
|
IN INTERNET_PORT nServerPort,
|
|
IN LPCWSTR lpwszUserName OPTIONAL,
|
|
IN LPCWSTR lpwszPassword OPTIONAL,
|
|
IN DWORD dwService,
|
|
IN DWORD dwFlags,
|
|
IN DWORD_PTR dwContext
|
|
);
|
|
|
|
typedef HINTERNET
|
|
(WINAPI * PFN_OPEN_REQUEST)(
|
|
IN HINTERNET hConnect,
|
|
IN LPCWSTR lpwszVerb,
|
|
IN LPCWSTR lpwszObjectName,
|
|
IN LPCWSTR lpwszVersion,
|
|
IN LPCWSTR lpwszReferrer OPTIONAL,
|
|
IN LPCWSTR FAR * lplpwszAcceptTypes OPTIONAL,
|
|
IN DWORD dwFlags,
|
|
IN DWORD_PTR dwContext
|
|
);
|
|
|
|
typedef BOOL
|
|
(WINAPI * PFN_SEND_REQUEST)(
|
|
IN HINTERNET hRequest,
|
|
IN LPCWSTR lpwszHeaders OPTIONAL,
|
|
IN DWORD dwHeadersLength,
|
|
IN LPVOID lpOptional OPTIONAL,
|
|
IN DWORD dwOptionalLength
|
|
);
|
|
|
|
typedef BOOL
|
|
(WINAPI * PFN_QUERY_INFO)(
|
|
IN HINTERNET hRequest,
|
|
IN DWORD dwInfoLevel,
|
|
IN LPVOID lpvBuffer,
|
|
IN LPDWORD lpdwBufferLength,
|
|
IN OUT LPDWORD lpdwIndex
|
|
);
|
|
|
|
typedef BOOL
|
|
(WINAPI* PFN_CLOSE_HANDLE)(
|
|
IN HINTERNET hInternet
|
|
);
|
|
|
|
typedef BOOL
|
|
(WINAPI* PFN_SET_OPTION)(
|
|
IN HINTERNET hInternet,
|
|
IN DWORD dwOption,
|
|
IN LPVOID lpBuffer,
|
|
IN DWORD dwBufferLength
|
|
);
|
|
|
|
typedef BOOL
|
|
(WINAPI* PFN_QUERY_OPTION)(
|
|
IN HINTERNET hInternet,
|
|
IN DWORD dwOption,
|
|
OUT LPVOID lpBuffer,
|
|
IN OUT LPDWORD lpdwBufferLength
|
|
);
|
|
|
|
typedef HINTERNET
|
|
(WINAPI* PFN_OPEN_URL)(
|
|
IN HINTERNET hInternet,
|
|
IN LPCWSTR lpwszUrl,
|
|
IN LPCWSTR lpwszHeaders,
|
|
IN DWORD dwHeadersLength,
|
|
IN DWORD dwFlags,
|
|
IN DWORD_PTR dwContext
|
|
);
|
|
|
|
typedef BOOL
|
|
(WINAPI* PFN_CRACK_URL)(
|
|
IN LPCWSTR lpszUrl,
|
|
IN DWORD dwUrlLength,
|
|
IN DWORD dwFlags,
|
|
IN OUT PVOID/*LPURL_COMPONENTSW*/ lpUrlComponents
|
|
);
|
|
|
|
typedef BOOL
|
|
(WINAPI* PFN_READ_FILE)(
|
|
IN HINTERNET hFile,
|
|
IN LPVOID lpBuffer,
|
|
IN DWORD dwNumberOfBytesToRead,
|
|
OUT LPDWORD lpdwNumberOfBytesRead
|
|
);
|
|
|
|
typedef INTERNET_STATUS_CALLBACK
|
|
(WINAPI* PFN_STATUS_CALLBACK)(
|
|
IN HINTERNET hInternet,
|
|
IN INTERNET_STATUS_CALLBACK lpfnInternetCallback
|
|
);
|
|
|
|
typedef BOOL
|
|
(WINAPI* PFN_ADD_HEADERS)(
|
|
IN HINTERNET hConnect,
|
|
IN LPCWSTR lpszHeaders,
|
|
IN DWORD dwHeadersLength,
|
|
IN DWORD dwModifiers
|
|
);
|
|
|
|
class WININET_SESSION : public SESSION
|
|
{
|
|
protected:
|
|
WININET_SESSION(void);
|
|
virtual ~WININET_SESSION(void);
|
|
|
|
virtual BOOL Open(PCWSTR pwszHttpStack, HINTERNET);
|
|
virtual void Close(void);
|
|
|
|
protected:
|
|
virtual HINTERNET Connect(
|
|
LPCWSTR lpwszServerName,
|
|
INTERNET_PORT
|
|
);
|
|
|
|
virtual HINTERNET OpenRequest(
|
|
HINTERNET hConnect,
|
|
LPCWSTR lpwszVerb,
|
|
LPCWSTR lpwszObjectName,
|
|
DWORD dwFlags,
|
|
DWORD_PTR dwContext = 0);
|
|
|
|
virtual BOOL SendRequest(
|
|
HINTERNET hRequest,
|
|
LPCWSTR lpwszHeaders,
|
|
DWORD dwHeadersLength,
|
|
DWORD_PTR dwContext = 0);
|
|
|
|
virtual BOOL QueryHeaders(
|
|
HINTERNET hRequest,
|
|
DWORD dwInfoLevel,
|
|
LPVOID lpvBuffer,
|
|
LPDWORD lpdwBufferLength,
|
|
LPDWORD lpdwIndex = NULL);
|
|
|
|
virtual BOOL CloseHandle(
|
|
IN HINTERNET hInternet);
|
|
|
|
virtual BOOL QueryOption(
|
|
HINTERNET hInternet,
|
|
DWORD dwOption,
|
|
LPVOID lpBuffer,
|
|
LPDWORD lpdwBufferLength);
|
|
|
|
virtual BOOL SetOption(
|
|
HINTERNET hInternet,
|
|
DWORD dwOption,
|
|
LPVOID lpBuffer,
|
|
DWORD dwBufferLength);
|
|
|
|
virtual HINTERNET OpenUrl(
|
|
LPCWSTR lpwszUrl,
|
|
LPCWSTR lpwszHeaders,
|
|
DWORD dwHeadersLength,
|
|
DWORD dwFlags);
|
|
|
|
virtual BOOL ReadFile(
|
|
HINTERNET hFile,
|
|
LPVOID lpBuffer,
|
|
DWORD dwNumberOfBytesToRead,
|
|
LPDWORD lpdwNumberOfBytesRead);
|
|
|
|
virtual BOOL CrackUrl(
|
|
LPCWSTR lpszUrl,
|
|
DWORD dwUrlLength,
|
|
DWORD dwFlags,
|
|
PVOID/*LPURL_COMPONENTSW*/ lpUrlComponents);
|
|
|
|
virtual PVOID SetStatusCallback(
|
|
HINTERNET hInternet,
|
|
PVOID lpfnCallback
|
|
);
|
|
|
|
virtual BOOL AddHeaders(
|
|
HINTERNET hConnect,
|
|
LPCWSTR lpszHeaders,
|
|
DWORD dwHeadersLength,
|
|
DWORD dwModifiers
|
|
);
|
|
|
|
|
|
#ifdef PP_DEMO
|
|
virtual BOOL ContactPartner(PCWSTR pwszPartnerUrl,
|
|
PCWSTR pwszVerb,
|
|
PCWSTR pwszHeaders,
|
|
PWSTR pwszData,
|
|
PDWORD pdwDataLength
|
|
);
|
|
#endif // PP_DEMO
|
|
|
|
BOOL InitHttpApi(PFN_INTERNET_OPEN*);
|
|
|
|
protected:
|
|
HINTERNET m_hInternet;
|
|
|
|
PFN_INTERNET_CONNECT m_pfnConnect;
|
|
PFN_OPEN_REQUEST m_pfnOpenRequest;
|
|
PFN_SEND_REQUEST m_pfnSendRequest;
|
|
PFN_QUERY_INFO m_pfnQueryInfo;
|
|
PFN_CLOSE_HANDLE m_pfnCloseHandle;
|
|
PFN_SET_OPTION m_pfnSetOption;
|
|
PFN_OPEN_URL m_pfnOpenUrl;
|
|
PFN_QUERY_OPTION m_pfnQueryOption;
|
|
PFN_CRACK_URL m_pfnCrack;
|
|
PFN_READ_FILE m_pfnReadFile;
|
|
PFN_STATUS_CALLBACK m_pfnStatusCallback;
|
|
PFN_ADD_HEADERS m_pfnAddHeaders;
|
|
|
|
friend class SESSION;
|
|
};
|
|
|
|
//
|
|
// Implementation for SESSION
|
|
//
|
|
|
|
SESSION* CreateWinHttpSession(void);
|
|
|
|
// -----------------------------------------------------------------------------
|
|
BOOL SESSION::CreateObject(PCWSTR pwszHttpStack, HINTERNET hSession, SESSION*& pSess)
|
|
{
|
|
PP_ASSERT(pwszHttpStack != NULL);
|
|
|
|
pSess = NULL;
|
|
|
|
if (!::_wcsicmp(pwszHttpStack, L"WinInet.dll") ||
|
|
!::_wcsicmp(pwszHttpStack, L"WinInet"))
|
|
{
|
|
pSess = new WININET_SESSION();
|
|
}
|
|
else
|
|
{
|
|
pSess = ::CreateWinHttpSession();
|
|
}
|
|
|
|
if (pSess)
|
|
{
|
|
return pSess->Open(pwszHttpStack, hSession);
|
|
}
|
|
else
|
|
{
|
|
DoTraceMessage(PP_LOG_ERROR, "CreateObject() failed; not enough memory");
|
|
return FALSE;
|
|
}
|
|
}
|
|
|
|
// -----------------------------------------------------------------------------
|
|
SESSION::SESSION(void)
|
|
{
|
|
m_hHttpStack = 0;
|
|
m_hCredUI = 0;
|
|
m_RefCount = 0;
|
|
|
|
m_pfnReadDomainCred = NULL;
|
|
m_pfnCredFree = NULL;
|
|
|
|
m_hKeyLM = NULL;
|
|
m_hKeyCU = NULL;
|
|
m_hKeyDAMap = NULL;
|
|
|
|
m_fLogout = FALSE;
|
|
|
|
m_LastNexusDownloadTime = 0xFFFFFFFF;
|
|
}
|
|
|
|
// -----------------------------------------------------------------------------
|
|
SESSION::~SESSION(void)
|
|
{
|
|
}
|
|
|
|
BOOL SESSION::GetDAInfoFromPPNexus(
|
|
IN PWSTR pwszRegUrl, // user supplied buffer ...
|
|
IN OUT PDWORD pdwRegUrlLen, // ... and length (will be updated to actual length
|
|
// on successful return)
|
|
IN PWSTR pwszRealm, // user supplied buffer ...
|
|
IN OUT PDWORD pdwRealmLen // ... and length (will be updated to actual length
|
|
// on successful return)
|
|
)
|
|
{
|
|
BOOL fRetVal = FALSE;
|
|
HINTERNET hRequest = NULL;
|
|
HINTERNET hConnect = NULL;
|
|
DWORD dwError;
|
|
|
|
#ifdef DBG
|
|
WCHAR wNexusHost[512] = L"nexus.msn.pp.test.microsoft.com";
|
|
#else
|
|
WCHAR wNexusHost[512] = L"nexus.passport.com";
|
|
#endif
|
|
|
|
DWORD dwHostLen = sizeof(wNexusHost); // note: size of the buffer, not # of UNICODE characters
|
|
|
|
#ifdef DBG
|
|
WCHAR wNexusObj[512] = L"sprdr/pprdr.asp";
|
|
#else
|
|
WCHAR wNexusObj[512] = L"rdr/pprdr.asp";
|
|
#endif
|
|
DWORD dwObjLen = sizeof(wNexusObj);
|
|
|
|
WCHAR wPassportUrls[1024];
|
|
DWORD dwUrlsLen = ARRAYSIZE(wPassportUrls);
|
|
DWORD dwValueType;
|
|
|
|
WCHAR Delimiters[] = L",";
|
|
PWSTR Token = NULL;
|
|
// we allow only one Nexus contact per session to avoid infinite loop due to Nexus misconfiguration
|
|
|
|
DWORD dwCurrentTime = ::GetTickCount();
|
|
|
|
if ((dwCurrentTime >= m_LastNexusDownloadTime) &&
|
|
(dwCurrentTime - m_LastNexusDownloadTime < 5*60*1000)) // 5 minutes
|
|
{
|
|
DoTraceMessage(PP_LOG_WARNING, "SESSION::GetDAInfoFromPPNexus() failed: Nexus info already downloaded");
|
|
goto exit;
|
|
}
|
|
|
|
// biaow-todo: when the Passport Team gives us THE final Nexus name, we'll then hard-code it here. And
|
|
// there will be no need to query registry here by then.
|
|
|
|
if (m_hKeyLM)
|
|
{
|
|
dwError = ::RegQueryValueExW(m_hKeyLM,
|
|
L"NexusHost",
|
|
0,
|
|
&dwValueType,
|
|
reinterpret_cast<LPBYTE>(wNexusHost),
|
|
&dwHostLen);
|
|
|
|
PP_ASSERT(!(dwError == ERROR_MORE_DATA));
|
|
// PP_ASSERT(dwValueType == REG_SZ); BVT Break!!!
|
|
|
|
dwError = ::RegQueryValueExW(m_hKeyLM,
|
|
L"NexusObj",
|
|
0,
|
|
&dwValueType,
|
|
reinterpret_cast<LPBYTE>(wNexusObj),
|
|
&dwObjLen);
|
|
|
|
PP_ASSERT(!(dwError == ERROR_MORE_DATA));
|
|
// PP_ASSERT(dwValueType == REG_SZ); BVT Break!!!
|
|
}
|
|
|
|
hConnect = Connect(wNexusHost,
|
|
#ifdef DISABLE_SSL
|
|
INTERNET_DEFAULT_HTTP_PORT
|
|
#else
|
|
INTERNET_DEFAULT_HTTPS_PORT
|
|
#endif
|
|
);
|
|
if (hConnect == NULL)
|
|
{
|
|
DWORD dwErrorCode = ::GetLastError();
|
|
DoTraceMessage(PP_LOG_ERROR,
|
|
"SESSION::GetDAInfoFromPPNexus(): failed to connect to %ws; Error = %d",
|
|
wNexusHost, dwErrorCode);
|
|
goto exit;
|
|
}
|
|
|
|
hRequest = OpenRequest(hConnect,
|
|
NULL,
|
|
wNexusObj,
|
|
#ifdef DISABLE_SSL
|
|
0
|
|
#else
|
|
INTERNET_FLAG_SECURE
|
|
#endif
|
|
);
|
|
|
|
if (hRequest == NULL)
|
|
|
|
{
|
|
DWORD dwErrorCode = ::GetLastError();
|
|
DoTraceMessage(PP_LOG_ERROR, "SESSION::GetDAInfoFromPPNexus() failed; OpenRequest() to %ws failed, Error Code = %d",
|
|
wNexusObj, dwErrorCode);
|
|
goto exit;
|
|
}
|
|
|
|
|
|
if (!SendRequest(hRequest, NULL, 0))
|
|
{
|
|
DWORD dwErrorCode = ::GetLastError();
|
|
|
|
#ifdef BAD_CERT_OK
|
|
if (dwErrorCode == ERROR_INTERNET_INVALID_CA)
|
|
{
|
|
DWORD dwSecFlags;
|
|
DWORD dwSecurityFlagsSize = sizeof(dwSecFlags);
|
|
|
|
if (!QueryOption(hRequest,
|
|
INTERNET_OPTION_SECURITY_FLAGS,
|
|
&dwSecFlags,
|
|
&dwSecurityFlagsSize))
|
|
{
|
|
dwSecFlags = 0;
|
|
}
|
|
else
|
|
{
|
|
dwSecFlags |= SECURITY_SET_MASK;
|
|
}
|
|
|
|
if (!SetOption(hRequest,
|
|
INTERNET_OPTION_SECURITY_FLAGS,
|
|
&dwSecFlags,
|
|
dwSecurityFlagsSize))
|
|
{
|
|
PP_ASSERT(TRUE); // shouldn't reach here
|
|
goto exit;
|
|
}
|
|
else
|
|
{
|
|
if (!SendRequest(hRequest, NULL, 0))
|
|
{
|
|
DWORD dwErrorCode = ::GetLastError();
|
|
DoTraceMessage(PP_LOG_ERROR, "SESSION::GetDAInfoFromPPNexus(): SendRequest() failed");
|
|
goto exit;
|
|
}
|
|
}
|
|
}
|
|
#else
|
|
DoTraceMessage(PP_LOG_ERROR, "SESSION::GetDAInfoFromPPNexus(): failed");
|
|
goto exit;
|
|
#endif // BAD_CERT_OK
|
|
}
|
|
|
|
if (QueryHeaders(hRequest,
|
|
HTTP_QUERY_PASSPORT_URLS,
|
|
wPassportUrls,
|
|
&dwUrlsLen) == FALSE)
|
|
{
|
|
DoTraceMessage(PP_LOG_ERROR, "SESSION::GetDAInfoFromPPNexus() failed; PassportUrls header not found");
|
|
goto exit;
|
|
}
|
|
|
|
Token = ::wcstok(wPassportUrls, Delimiters);
|
|
while (Token != NULL)
|
|
{
|
|
// skip leading white spaces
|
|
while (*Token == (L" ")[0]) { ++Token; }
|
|
if (Token == NULL)
|
|
{
|
|
DoTraceMessage(PP_LOG_WARNING, "SESSION::GetDAInfoFromPPNexus() : no text in between commas");
|
|
goto next_token;
|
|
}
|
|
|
|
// find DALocation
|
|
if (!::_wcsnicmp(Token, L"DALogin", ::wcslen(L"DALogin")))
|
|
{
|
|
PWSTR pwszDAUrl = ::wcsstr(Token, L"=");
|
|
if (pwszDAUrl == NULL)
|
|
{
|
|
DoTraceMessage(PP_LOG_WARNING, "SESSION::GetDAInfoFromPPNexus() : no = after DALocation");
|
|
goto exit;
|
|
}
|
|
|
|
pwszDAUrl++; // skip "="
|
|
|
|
while (*pwszDAUrl == (L" ")[0]) { ++pwszDAUrl; } // skip leading white spaces
|
|
|
|
::wcscpy(m_wDefaultDAUrl, L"https://");
|
|
::wcscat(m_wDefaultDAUrl, pwszDAUrl);
|
|
|
|
if (m_hKeyLM == NULL || ::RegSetValueExW(m_hKeyLM,
|
|
L"LoginServerUrl",
|
|
0,
|
|
REG_SZ,
|
|
reinterpret_cast<const LPBYTE>(m_wDefaultDAUrl),
|
|
::wcslen(m_wDefaultDAUrl) * sizeof(WCHAR)) != ERROR_SUCCESS)
|
|
{
|
|
if (m_hKeyCU)
|
|
{
|
|
::RegSetValueExW(m_hKeyCU,
|
|
L"LoginServerUrl",
|
|
0,
|
|
REG_SZ,
|
|
reinterpret_cast<const LPBYTE>(m_wDefaultDAUrl),
|
|
::wcslen(m_wDefaultDAUrl) * sizeof(WCHAR));
|
|
}
|
|
}
|
|
|
|
m_LastNexusDownloadTime = ::GetTickCount();
|
|
fRetVal = TRUE;
|
|
|
|
DoTraceMessage(PP_LOG_INFO, "DALocation URL %ws found", m_wDefaultDAUrl);
|
|
}
|
|
else if (!::_wcsnicmp(Token, L"DARealm", ::wcslen(L"DARealm")))
|
|
{
|
|
PWSTR pwszDARealm = ::wcsstr(Token, L"=");
|
|
if (pwszDARealm == NULL)
|
|
{
|
|
DoTraceMessage(PP_LOG_WARNING, "SESSION::GetDAInfoFromPPNexus() : no = after DARealm");
|
|
goto exit;
|
|
}
|
|
|
|
pwszDARealm++; // skip "="
|
|
|
|
while (*pwszDARealm == (L" ")[0]) { ++pwszDARealm; } // skip leading white spaces
|
|
|
|
if (m_hKeyLM == NULL || ::RegSetValueExW(m_hKeyLM,
|
|
L"LoginServerRealm",
|
|
0,
|
|
REG_SZ,
|
|
reinterpret_cast<const LPBYTE>(pwszDARealm),
|
|
::wcslen(pwszDARealm) * sizeof(WCHAR)) != ERROR_SUCCESS)
|
|
{
|
|
if (m_hKeyCU)
|
|
{
|
|
::RegSetValueExW(m_hKeyCU,
|
|
L"LoginServerRealm",
|
|
0,
|
|
REG_SZ,
|
|
reinterpret_cast<const LPBYTE>(pwszDARealm),
|
|
::wcslen(pwszDARealm) * sizeof(WCHAR));
|
|
}
|
|
}
|
|
|
|
if (pwszRealm)
|
|
{
|
|
if (*pdwRealmLen < ::wcslen(pwszDARealm) + 1)
|
|
{
|
|
*pdwRealmLen = ::wcslen(pwszDARealm) + 1;
|
|
fRetVal = FALSE;
|
|
goto exit;
|
|
}
|
|
else
|
|
{
|
|
::wcscpy(pwszRealm, pwszDARealm);
|
|
*pdwRealmLen = ::wcslen(pwszDARealm) + 1;
|
|
}
|
|
}
|
|
|
|
DoTraceMessage(PP_LOG_INFO, "DALocation URL %ws found", m_wDefaultDAUrl);
|
|
}
|
|
else if (!::_wcsnicmp(Token, L"DAReg", ::wcslen(L"DAReg")))
|
|
{
|
|
PWSTR pwszDAReg = ::wcsstr(Token, L"=");
|
|
if (pwszDAReg == NULL)
|
|
{
|
|
DoTraceMessage(PP_LOG_WARNING, "SESSION::GetDAInfoFromPPNexus() : no = after DAReg");
|
|
goto exit;
|
|
}
|
|
|
|
pwszDAReg++; // skip "="
|
|
|
|
while (*pwszDAReg == (L" ")[0]) { ++pwszDAReg; } // skip leading white spaces
|
|
|
|
if (m_hKeyLM == NULL || ::RegSetValueExW(m_hKeyLM,
|
|
L"RegistrationUrl",
|
|
0,
|
|
REG_SZ,
|
|
reinterpret_cast<const LPBYTE>(pwszDAReg),
|
|
::wcslen(pwszDAReg) * sizeof(WCHAR)) != ERROR_SUCCESS)
|
|
{
|
|
if (m_hKeyCU)
|
|
{
|
|
::RegSetValueExW(m_hKeyCU,
|
|
L"RegistrationUrl",
|
|
0,
|
|
REG_SZ,
|
|
reinterpret_cast<const LPBYTE>(pwszDAReg),
|
|
::wcslen(pwszDAReg) * sizeof(WCHAR));
|
|
}
|
|
}
|
|
|
|
if (pwszRegUrl)
|
|
{
|
|
if (*pdwRegUrlLen < ::wcslen(pwszDAReg) + 1)
|
|
{
|
|
*pdwRegUrlLen = ::wcslen(pwszDAReg) + 1;
|
|
fRetVal = FALSE;
|
|
goto exit;
|
|
}
|
|
else
|
|
{
|
|
::wcscpy(pwszRegUrl, pwszDAReg);
|
|
*pdwRegUrlLen = ::wcslen(pwszDAReg) + 1;
|
|
}
|
|
}
|
|
|
|
DoTraceMessage(PP_LOG_INFO, "DALocation URL %ws found", m_wDefaultDAUrl);
|
|
}
|
|
|
|
next_token:
|
|
|
|
Token = ::wcstok(NULL, Delimiters);
|
|
}
|
|
|
|
exit:
|
|
if (hRequest)
|
|
{
|
|
CloseHandle(hRequest);
|
|
}
|
|
if (hConnect)
|
|
{
|
|
CloseHandle(hConnect);
|
|
}
|
|
|
|
return fRetVal;
|
|
}
|
|
|
|
BOOL SESSION::UpdateDAInfo(
|
|
PCWSTR pwszSignIn,
|
|
PCWSTR pwszDAUrl
|
|
)
|
|
{
|
|
if (pwszSignIn)
|
|
{
|
|
LPCWSTR pwszDomain = ::wcsstr(pwszSignIn, L"@");
|
|
if (pwszDomain && m_hKeyDAMap)
|
|
{
|
|
DWORD dwError = ::RegSetValueExW(m_hKeyDAMap,
|
|
pwszDomain,
|
|
0,
|
|
REG_SZ,
|
|
reinterpret_cast<const LPBYTE>(const_cast<PWSTR>(pwszDAUrl)),
|
|
::wcslen(pwszDAUrl) * sizeof(WCHAR));
|
|
if (dwError == ERROR_SUCCESS)
|
|
{
|
|
return TRUE;
|
|
}
|
|
}
|
|
}
|
|
|
|
return FALSE;
|
|
}
|
|
|
|
BOOL SESSION::GetDAInfo(PCWSTR pwszSignIn,
|
|
LPWSTR pwszDAHostName,
|
|
DWORD HostNameLen,
|
|
LPWSTR pwszDAHostObj,
|
|
DWORD HostObjLen)
|
|
{
|
|
LPCWSTR pwszDAUrl = m_wDefaultDAUrl;
|
|
|
|
WCHAR wDomainDAUrl[1024];
|
|
DWORD dwDomainUrlLen = sizeof(wDomainDAUrl);
|
|
|
|
if (pwszSignIn)
|
|
{
|
|
LPCWSTR pwszDomain = ::wcsstr(pwszSignIn, L"@");
|
|
if (pwszDomain && m_hKeyDAMap)
|
|
{
|
|
DWORD dwValueType;
|
|
DWORD dwError = ::RegQueryValueExW(m_hKeyDAMap,
|
|
pwszDomain,
|
|
0,
|
|
&dwValueType,
|
|
reinterpret_cast<LPBYTE>(wDomainDAUrl),
|
|
&dwDomainUrlLen);
|
|
|
|
PP_ASSERT(!(dwError == ERROR_MORE_DATA));
|
|
// PP_ASSERT(dwValueType == REG_SZ);
|
|
|
|
if (dwError == ERROR_SUCCESS)
|
|
{
|
|
pwszDAUrl = wDomainDAUrl;
|
|
}
|
|
}
|
|
}
|
|
|
|
URL_COMPONENTSW UrlComps;
|
|
::memset(&UrlComps, 0, sizeof(UrlComps));
|
|
|
|
UrlComps.dwStructSize = sizeof(UrlComps);
|
|
|
|
UrlComps.lpszHostName = pwszDAHostName;
|
|
UrlComps.dwHostNameLength = HostNameLen;
|
|
|
|
UrlComps.lpszUrlPath = pwszDAHostObj;
|
|
UrlComps.dwUrlPathLength = HostObjLen;
|
|
|
|
if (CrackUrl(pwszDAUrl,
|
|
0,
|
|
0,
|
|
&UrlComps) == FALSE)
|
|
{
|
|
DoTraceMessage(PP_LOG_ERROR,
|
|
"WININET_SESSION::GetDAInfo() failed; can not crack the URL %ws",
|
|
pwszDAUrl);
|
|
return FALSE;
|
|
}
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
|
|
BOOL SESSION::Open(PCWSTR /*pwszHttpStack*/, HINTERNET)
|
|
{
|
|
BOOL fRetVal = FALSE;
|
|
DWORD dwError;
|
|
DWORD dwValueType;
|
|
DWORD dwUrlLen = sizeof(m_wDefaultDAUrl); // note: size of the buffer, not # of UNICODE characters
|
|
BOOL fDAInfoCached = FALSE; // assume NO DA info's cached locally
|
|
|
|
dwError = ::RegCreateKeyExW(HKEY_LOCAL_MACHINE,
|
|
L"SOFTWARE\\Microsoft\\Windows\\CurrentVersion\\Internet Settings\\Passport",
|
|
0,
|
|
NULL,
|
|
REG_OPTION_NON_VOLATILE,
|
|
KEY_READ | KEY_WRITE,
|
|
NULL,
|
|
&m_hKeyLM,
|
|
NULL);
|
|
|
|
if (dwError != ERROR_SUCCESS)
|
|
{
|
|
DoTraceMessage(PP_LOG_ERROR,
|
|
"SESSION::Open() failed; can not create/open the Passport key; Error = %d", dwError);
|
|
|
|
// we can't open the Passport key for read & write, let's try open it for read only
|
|
::RegCreateKeyExW(HKEY_LOCAL_MACHINE,
|
|
L"SOFTWARE\\Microsoft\\Windows\\CurrentVersion\\Internet Settings\\Passport",
|
|
0,
|
|
NULL,
|
|
REG_OPTION_NON_VOLATILE,
|
|
KEY_READ,
|
|
NULL,
|
|
&m_hKeyLM,
|
|
NULL);
|
|
|
|
// if we still can't open it for read, we are still fine since we can download the info from the
|
|
// Nexus server. *NOTE* m_hKeyLM could be NULL from this point on.
|
|
}
|
|
|
|
dwError = ::RegCreateKeyExW(HKEY_CURRENT_USER,
|
|
L"SOFTWARE\\Microsoft\\Windows\\CurrentVersion\\Internet Settings\\Passport",
|
|
0,
|
|
NULL,
|
|
REG_OPTION_NON_VOLATILE,
|
|
KEY_READ | KEY_WRITE,
|
|
NULL,
|
|
&m_hKeyCU,
|
|
NULL);
|
|
|
|
if (dwError != ERROR_SUCCESS)
|
|
{
|
|
DoTraceMessage(PP_LOG_ERROR,
|
|
"SESSION::Open() failed; can not create/open the Passport key; Error = %d", dwError);
|
|
|
|
::RegCreateKeyExW(HKEY_CURRENT_USER,
|
|
L"SOFTWARE\\Microsoft\\Windows\\CurrentVersion\\Internet Settings\\Passport",
|
|
0,
|
|
NULL,
|
|
REG_OPTION_NON_VOLATILE,
|
|
KEY_READ,
|
|
NULL,
|
|
&m_hKeyCU,
|
|
NULL);
|
|
}
|
|
|
|
if (m_hKeyCU)
|
|
{
|
|
dwError = ::RegCreateKeyExW(m_hKeyCU,
|
|
L"DAMap",
|
|
0,
|
|
NULL,
|
|
REG_OPTION_NON_VOLATILE,
|
|
KEY_READ | KEY_WRITE,
|
|
NULL,
|
|
&m_hKeyDAMap,
|
|
NULL);
|
|
|
|
if (dwError != ERROR_SUCCESS)
|
|
{
|
|
DoTraceMessage(PP_LOG_ERROR,
|
|
"SESSION::Open() failed; can not create/open the Passport key; Error = %d", dwError);
|
|
|
|
::RegCreateKeyExW(m_hKeyCU,
|
|
L"DAMap",
|
|
0,
|
|
NULL,
|
|
REG_OPTION_NON_VOLATILE,
|
|
KEY_READ,
|
|
NULL,
|
|
&m_hKeyDAMap,
|
|
NULL);
|
|
}
|
|
}
|
|
|
|
if (m_hKeyCU == NULL || ::RegQueryValueExW(m_hKeyCU,
|
|
L"LoginServerUrl",
|
|
0,
|
|
&dwValueType,
|
|
reinterpret_cast<LPBYTE>(m_wDefaultDAUrl),
|
|
&dwUrlLen) != ERROR_SUCCESS)
|
|
{
|
|
if (m_hKeyLM && ::RegQueryValueExW(m_hKeyLM,
|
|
L"LoginServerUrl",
|
|
0,
|
|
&dwValueType,
|
|
reinterpret_cast<LPBYTE>(m_wDefaultDAUrl),
|
|
&dwUrlLen) == ERROR_SUCCESS)
|
|
{
|
|
fDAInfoCached = TRUE;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
fDAInfoCached = TRUE;
|
|
}
|
|
|
|
PP_ASSERT(!(dwError == ERROR_MORE_DATA));
|
|
// PP_ASSERT(dwValueType == REG_SZ); BVT break!!!
|
|
|
|
if (!fDAInfoCached || (::wcslen(m_wDefaultDAUrl) == ::wcslen(L"")))
|
|
{
|
|
if (GetDAInfoFromPPNexus(NULL, 0, NULL, 0) == FALSE)
|
|
{
|
|
goto exit;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
/*
|
|
URL_COMPONENTSW UrlComps;
|
|
::memset(&UrlComps, 0, sizeof(UrlComps));
|
|
|
|
UrlComps.dwStructSize = sizeof(UrlComps) / sizeof(WCHAR);
|
|
|
|
UrlComps.lpszHostName = m_wDAHostName;
|
|
UrlComps.dwHostNameLength = ARRAYSIZE(m_wDAHostName);
|
|
|
|
UrlComps.lpszUrlPath = m_wDATargetObj;
|
|
UrlComps.dwUrlPathLength = ARRAYSIZE(m_wDATargetObj);
|
|
|
|
if (CrackUrl(m_wDefaultDAUrl,
|
|
0,
|
|
0,
|
|
&UrlComps) == FALSE)
|
|
{
|
|
DoTraceMessage(PP_LOG_ERROR,
|
|
"WININET_SESSION::Open() failed; can not crack the URL %ws",
|
|
m_wDefaultDAUrl);
|
|
goto exit;
|
|
}
|
|
*/
|
|
}
|
|
|
|
/*
|
|
DWORD dwRegUrlLen = sizeof(m_wRegistrationUrl);
|
|
dwError = ::RegQueryValueExW(m_hKeyLM,
|
|
L"RegistrationUrl",
|
|
0,
|
|
&dwValueType,
|
|
reinterpret_cast<LPBYTE>(m_wRegistrationUrl),
|
|
&dwRegUrlLen);
|
|
|
|
PP_ASSERT(!(dwError == ERROR_MORE_DATA));
|
|
*/
|
|
|
|
m_hCredUI = ::LoadLibraryW(L"advapi32.dll");
|
|
if (m_hCredUI)
|
|
{
|
|
m_pfnReadDomainCred =
|
|
reinterpret_cast<PFN_READ_DOMAIN_CRED_W>(::GetProcAddress(m_hCredUI, "CredReadDomainCredentialsW"));
|
|
if (m_pfnReadDomainCred == NULL)
|
|
{
|
|
DoTraceMessage(PP_LOG_WARNING, "failed to bind to CredReadDomainCredentialsW()");
|
|
}
|
|
|
|
m_pfnCredFree =
|
|
reinterpret_cast<PFN_CRED_FREE>(::GetProcAddress(m_hCredUI, "CredFree"));
|
|
if (m_pfnCredFree == NULL)
|
|
{
|
|
DoTraceMessage(PP_LOG_WARNING, "failed to bind to CredFree()");
|
|
}
|
|
}
|
|
|
|
fRetVal = TRUE;
|
|
|
|
exit:
|
|
|
|
return fRetVal;
|
|
}
|
|
|
|
void SESSION::Logout(void)
|
|
{
|
|
if (!m_fLogout)
|
|
{
|
|
m_fLogout = TRUE;
|
|
::GetSystemTime(&m_LogoutTimeStamp);
|
|
}
|
|
}
|
|
|
|
BOOL SESSION::IsLoggedOut(void) const
|
|
{
|
|
return m_fLogout;
|
|
}
|
|
|
|
void SESSION::ResetLogoutFlag(void)
|
|
{
|
|
m_fLogout = FALSE;
|
|
}
|
|
|
|
const SYSTEMTIME* SESSION::GetLogoutTimeStamp(void) const
|
|
{
|
|
return &m_LogoutTimeStamp;
|
|
}
|
|
|
|
void SESSION::Close(void)
|
|
{
|
|
if (m_hCredUI)
|
|
{
|
|
::FreeLibrary(m_hCredUI);
|
|
m_hCredUI = NULL;
|
|
}
|
|
|
|
if (m_hKeyDAMap)
|
|
{
|
|
::RegCloseKey(m_hKeyDAMap);
|
|
}
|
|
|
|
if (m_hKeyCU)
|
|
{
|
|
::RegCloseKey(m_hKeyCU);
|
|
}
|
|
|
|
if (m_hKeyLM)
|
|
{
|
|
::RegCloseKey(m_hKeyLM);
|
|
}
|
|
}
|
|
|
|
|
|
//
|
|
// Implementation for WININET_SESSION
|
|
//
|
|
|
|
// -----------------------------------------------------------------------------
|
|
WININET_SESSION::WININET_SESSION(void)
|
|
{
|
|
m_hInternet = NULL;
|
|
|
|
m_pfnConnect = NULL;
|
|
m_pfnOpenRequest = NULL;
|
|
m_pfnSendRequest = NULL;
|
|
m_pfnQueryInfo = NULL;
|
|
m_pfnCloseHandle = NULL;
|
|
m_pfnSetOption = NULL;
|
|
m_pfnOpenUrl = NULL;
|
|
m_pfnQueryOption = NULL;
|
|
m_pfnCrack = NULL;
|
|
m_pfnReadFile = NULL;
|
|
}
|
|
|
|
// -----------------------------------------------------------------------------
|
|
WININET_SESSION::~WININET_SESSION(void)
|
|
{
|
|
}
|
|
|
|
// -----------------------------------------------------------------------------
|
|
HINTERNET WININET_SESSION::Connect(
|
|
LPCWSTR lpwszServerName,
|
|
INTERNET_PORT nPort)
|
|
{
|
|
PP_ASSERT(m_pfnConnect != NULL);
|
|
|
|
return (*m_pfnConnect)(m_hInternet,
|
|
lpwszServerName,
|
|
nPort,
|
|
NULL,
|
|
NULL,
|
|
INTERNET_SERVICE_HTTP,
|
|
0,
|
|
0);
|
|
}
|
|
|
|
// -----------------------------------------------------------------------------
|
|
HINTERNET WININET_SESSION::OpenRequest(
|
|
HINTERNET hConnect,
|
|
LPCWSTR lpwszVerb,
|
|
LPCWSTR lpwszObjectName,
|
|
DWORD dwFlags,
|
|
DWORD_PTR dwContext)
|
|
{
|
|
PP_ASSERT(m_pfnOpenRequest != NULL);
|
|
|
|
return (*m_pfnOpenRequest)(hConnect,
|
|
lpwszVerb,
|
|
lpwszObjectName,
|
|
L"HTTP/1.1",
|
|
NULL,
|
|
NULL,
|
|
dwFlags,
|
|
dwContext);
|
|
}
|
|
|
|
// -----------------------------------------------------------------------------
|
|
BOOL WININET_SESSION::SendRequest(
|
|
HINTERNET hRequest,
|
|
LPCWSTR lpwszHeaders,
|
|
DWORD dwHeadersLength,
|
|
DWORD_PTR dwContext)
|
|
{
|
|
|
|
PP_ASSERT(m_pfnSendRequest != NULL);
|
|
|
|
return (*m_pfnSendRequest)(hRequest,
|
|
lpwszHeaders,
|
|
dwHeadersLength,
|
|
NULL,
|
|
0);
|
|
}
|
|
|
|
// -----------------------------------------------------------------------------
|
|
BOOL WININET_SESSION::QueryHeaders(
|
|
HINTERNET hRequest,
|
|
DWORD dwInfoLevel,
|
|
LPVOID lpvBuffer,
|
|
LPDWORD lpdwBufferLength,
|
|
LPDWORD lpdwIndex)
|
|
{
|
|
PP_ASSERT(m_pfnQueryInfo != NULL);
|
|
|
|
return (*m_pfnQueryInfo)(hRequest,
|
|
dwInfoLevel,
|
|
lpvBuffer,
|
|
lpdwBufferLength,
|
|
lpdwIndex);
|
|
}
|
|
|
|
// -----------------------------------------------------------------------------
|
|
BOOL WININET_SESSION::CloseHandle(
|
|
IN HINTERNET hInternet)
|
|
{
|
|
PP_ASSERT(m_pfnCloseHandle != NULL);
|
|
|
|
return (*m_pfnCloseHandle)(hInternet);
|
|
}
|
|
|
|
// -----------------------------------------------------------------------------
|
|
BOOL WININET_SESSION::QueryOption(
|
|
HINTERNET hInternet,
|
|
DWORD dwOption,
|
|
LPVOID lpBuffer,
|
|
LPDWORD lpdwBufferLength)
|
|
{
|
|
PP_ASSERT(m_pfnQueryOption != NULL);
|
|
|
|
return (*m_pfnQueryOption)(hInternet,
|
|
dwOption,
|
|
lpBuffer,
|
|
lpdwBufferLength
|
|
);
|
|
}
|
|
|
|
// -----------------------------------------------------------------------------
|
|
BOOL WININET_SESSION::SetOption(
|
|
HINTERNET hInternet,
|
|
DWORD dwOption,
|
|
LPVOID lpBuffer,
|
|
DWORD dwBufferLength)
|
|
{
|
|
PP_ASSERT(m_pfnSetOption != NULL);
|
|
|
|
return (*m_pfnSetOption)(hInternet,
|
|
dwOption,
|
|
lpBuffer,
|
|
dwBufferLength);
|
|
}
|
|
|
|
// -----------------------------------------------------------------------------
|
|
HINTERNET WININET_SESSION::OpenUrl(
|
|
LPCWSTR lpwszUrl,
|
|
LPCWSTR lpwszHeaders,
|
|
DWORD dwHeadersLength,
|
|
DWORD dwFlags)
|
|
{
|
|
PP_ASSERT(m_pfnOpenUrl != NULL);
|
|
|
|
return (*m_pfnOpenUrl)(m_hInternet,
|
|
lpwszUrl,
|
|
lpwszHeaders,
|
|
dwHeadersLength,
|
|
dwFlags,
|
|
0);
|
|
}
|
|
|
|
// -----------------------------------------------------------------------------
|
|
BOOL WININET_SESSION::ReadFile(
|
|
HINTERNET hFile,
|
|
LPVOID lpBuffer,
|
|
DWORD dwNumberOfBytesToRead,
|
|
LPDWORD lpdwNumberOfBytesRead)
|
|
{
|
|
PP_ASSERT(m_pfnReadFile != NULL);
|
|
|
|
return (*m_pfnReadFile)(
|
|
hFile,
|
|
lpBuffer,
|
|
dwNumberOfBytesToRead,
|
|
lpdwNumberOfBytesRead);
|
|
}
|
|
|
|
BOOL WININET_SESSION::CrackUrl(
|
|
LPCWSTR lpszUrl,
|
|
DWORD dwUrlLength,
|
|
DWORD dwFlags,
|
|
PVOID/*LPURL_COMPONENTSW*/ lpUrlComponents)
|
|
{
|
|
PP_ASSERT (m_pfnCrack != NULL);
|
|
|
|
return (*m_pfnCrack)(lpszUrl,
|
|
dwUrlLength,
|
|
dwFlags,
|
|
lpUrlComponents);
|
|
}
|
|
|
|
|
|
PVOID WININET_SESSION::SetStatusCallback(
|
|
HINTERNET hInternet,
|
|
PVOID lpfnCallback
|
|
)
|
|
{
|
|
PP_ASSERT (m_pfnStatusCallback != NULL);
|
|
|
|
return (*m_pfnStatusCallback)(hInternet,
|
|
(INTERNET_STATUS_CALLBACK)lpfnCallback);
|
|
|
|
}
|
|
|
|
BOOL WININET_SESSION::AddHeaders(
|
|
HINTERNET hConnect,
|
|
LPCWSTR lpszHeaders,
|
|
DWORD dwHeadersLength,
|
|
DWORD dwModifiers
|
|
)
|
|
{
|
|
PP_ASSERT(m_pfnAddHeaders != NULL);
|
|
|
|
return (*m_pfnAddHeaders)(hConnect,
|
|
lpszHeaders,
|
|
dwHeadersLength,
|
|
dwModifiers
|
|
);
|
|
}
|
|
|
|
|
|
|
|
BOOL WININET_SESSION::InitHttpApi(PFN_INTERNET_OPEN* ppfnInternetOpen)
|
|
{
|
|
BOOL fRet = FALSE;
|
|
|
|
m_pfnCloseHandle =
|
|
reinterpret_cast<PFN_CLOSE_HANDLE>(::GetProcAddress(m_hHttpStack, "InternetCloseHandle"));
|
|
if (m_pfnCloseHandle == NULL)
|
|
{
|
|
DoTraceMessage(PP_LOG_ERROR, "function entry point InternetCloseHandle not found");
|
|
goto exit;
|
|
}
|
|
|
|
*ppfnInternetOpen =
|
|
reinterpret_cast<PFN_INTERNET_OPEN>(::GetProcAddress(m_hHttpStack, "InternetOpenW"));
|
|
if (*ppfnInternetOpen == NULL)
|
|
{
|
|
DoTraceMessage(PP_LOG_ERROR, "function entry point InternetOpenW not found");
|
|
goto exit;
|
|
}
|
|
|
|
m_pfnConnect =
|
|
reinterpret_cast<PFN_INTERNET_CONNECT>(::GetProcAddress(m_hHttpStack, "InternetConnectW"));
|
|
if (m_pfnConnect == NULL)
|
|
{
|
|
DoTraceMessage(PP_LOG_ERROR, "function entry point InternetConnectW not found");
|
|
goto exit;
|
|
}
|
|
|
|
m_pfnOpenRequest =
|
|
reinterpret_cast<PFN_OPEN_REQUEST>(::GetProcAddress(m_hHttpStack, "HttpOpenRequestW"));
|
|
if (m_pfnOpenRequest == NULL)
|
|
{
|
|
DoTraceMessage(PP_LOG_ERROR, "function entry point HttpOpenRequestW not found");
|
|
goto exit;
|
|
}
|
|
|
|
m_pfnSendRequest =
|
|
reinterpret_cast<PFN_SEND_REQUEST>(::GetProcAddress(m_hHttpStack, "HttpSendRequestW"));
|
|
if (m_pfnSendRequest == NULL)
|
|
{
|
|
DoTraceMessage(PP_LOG_ERROR, "function entry point HttpSendRequestW not found");
|
|
goto exit;
|
|
}
|
|
|
|
m_pfnQueryInfo =
|
|
reinterpret_cast<PFN_QUERY_INFO>(::GetProcAddress(m_hHttpStack, "HttpQueryInfoW"));
|
|
if (m_pfnQueryInfo == NULL)
|
|
{
|
|
DoTraceMessage(PP_LOG_ERROR, "function entry point HttpQueryInfoW not found");
|
|
goto exit;
|
|
}
|
|
|
|
m_pfnSetOption =
|
|
reinterpret_cast<PFN_SET_OPTION>(::GetProcAddress(m_hHttpStack, "InternetSetOptionW"));
|
|
if (m_pfnSetOption == NULL)
|
|
{
|
|
DoTraceMessage(PP_LOG_ERROR, "function entry point InternetSetOptionW not found");
|
|
goto exit;
|
|
}
|
|
|
|
m_pfnOpenUrl =
|
|
reinterpret_cast<PFN_OPEN_URL>(::GetProcAddress(m_hHttpStack, "InternetOpenUrlW"));
|
|
if (m_pfnOpenUrl == NULL)
|
|
{
|
|
DoTraceMessage(PP_LOG_ERROR, "function entry point InternetOpenUrlW not found");
|
|
goto exit;
|
|
}
|
|
|
|
m_pfnQueryOption =
|
|
reinterpret_cast<PFN_QUERY_OPTION>(::GetProcAddress(m_hHttpStack, "InternetQueryOptionW"));
|
|
if (m_pfnQueryOption == NULL)
|
|
{
|
|
DoTraceMessage(PP_LOG_ERROR, "function entry point InternetQueryOptionW not found");
|
|
goto exit;
|
|
}
|
|
|
|
m_pfnCrack =
|
|
reinterpret_cast<PFN_CRACK_URL>(::GetProcAddress(m_hHttpStack, "InternetCrackUrlW"));
|
|
if (m_pfnCrack == NULL)
|
|
{
|
|
DoTraceMessage(PP_LOG_ERROR, "function entry point InternetCrackUrlW not found");
|
|
goto exit;
|
|
}
|
|
|
|
m_pfnReadFile =
|
|
reinterpret_cast<PFN_READ_FILE>(::GetProcAddress(m_hHttpStack, "InternetReadFile"));
|
|
if (m_pfnReadFile == NULL)
|
|
{
|
|
DoTraceMessage(PP_LOG_ERROR, "function entry point InternetReadFile not found");
|
|
goto exit;
|
|
}
|
|
|
|
m_pfnStatusCallback =
|
|
reinterpret_cast<PFN_STATUS_CALLBACK>(::GetProcAddress(m_hHttpStack, "InternetSetStatusCallbackW"));
|
|
if (m_pfnStatusCallback == NULL)
|
|
{
|
|
DoTraceMessage(PP_LOG_ERROR, "function entry point InternetSetStatusCallback not found");
|
|
goto exit;
|
|
}
|
|
|
|
m_pfnAddHeaders =
|
|
reinterpret_cast<PFN_ADD_HEADERS>(::GetProcAddress(m_hHttpStack, "HttpAddRequestHeadersW"));
|
|
if (m_pfnAddHeaders == NULL)
|
|
{
|
|
DoTraceMessage(PP_LOG_ERROR, "function entry point HttpAddRequestHeaders not found");
|
|
goto exit;
|
|
}
|
|
|
|
|
|
fRet = TRUE;
|
|
|
|
exit:
|
|
return fRet;
|
|
}
|
|
|
|
BOOL WININET_SESSION::Open(
|
|
PCWSTR pwszHttpStack,
|
|
HINTERNET hInternet
|
|
)
|
|
{
|
|
PP_ASSERT(pwszHttpStack != NULL);
|
|
|
|
DWORD dwErrorCode;
|
|
BOOL fRet = FALSE;
|
|
PFN_INTERNET_OPEN pfnInternetOpen = NULL;
|
|
|
|
PP_ASSERT(m_hHttpStack == 0);
|
|
m_hHttpStack = ::LoadLibraryW(pwszHttpStack);
|
|
if (m_hHttpStack == NULL)
|
|
{
|
|
dwErrorCode = ::GetLastError();
|
|
DoTraceMessage(PP_LOG_ERROR, "failed to load library %ws; error = %d", pwszHttpStack, dwErrorCode);
|
|
goto exit;
|
|
}
|
|
|
|
if (InitHttpApi(&pfnInternetOpen) == FALSE)
|
|
{
|
|
goto exit;
|
|
}
|
|
|
|
if (hInternet)
|
|
{
|
|
m_hInternet = hInternet;
|
|
m_fOwnedSession = FALSE;
|
|
}
|
|
else
|
|
{
|
|
m_hInternet = (*pfnInternetOpen)(
|
|
L"Microsoft.NET-Passport-Authentication-Service/1.4",
|
|
INTERNET_OPEN_TYPE_PRECONFIG,
|
|
NULL,
|
|
NULL,
|
|
0 /*INTERNET_FLAG_ASYNC*/ // biaow-todo: use async
|
|
);
|
|
m_fOwnedSession = TRUE;
|
|
}
|
|
if (m_hInternet == NULL)
|
|
{
|
|
dwErrorCode = ::GetLastError();
|
|
DoTraceMessage(PP_LOG_ERROR, "failed to open an HTTP session through %ws; error = %d",
|
|
pwszHttpStack, dwErrorCode);
|
|
goto exit;
|
|
}
|
|
|
|
if (SESSION::Open(pwszHttpStack, hInternet) == FALSE)
|
|
{
|
|
goto exit;
|
|
}
|
|
|
|
fRet = TRUE;
|
|
|
|
DoTraceMessage(PP_LOG_INFO, "WinInet Http Session opened");
|
|
|
|
exit:
|
|
if (!fRet)
|
|
{
|
|
if (m_hInternet && m_fOwnedSession)
|
|
{
|
|
(*m_pfnCloseHandle)(m_hInternet);
|
|
m_hInternet = NULL;
|
|
}
|
|
if (m_hHttpStack)
|
|
{
|
|
::FreeLibrary(m_hHttpStack);
|
|
m_hHttpStack = NULL;
|
|
}
|
|
|
|
DoTraceMessage(PP_LOG_ERROR, "WinInet Http Session failed");
|
|
}
|
|
|
|
return fRet;
|
|
}
|
|
|
|
void WININET_SESSION::Close(void)
|
|
{
|
|
PP_ASSERT(m_pfnCloseHandle);
|
|
if (m_hInternet && m_fOwnedSession)
|
|
{
|
|
(*m_pfnCloseHandle)(m_hInternet);
|
|
m_pfnCloseHandle = NULL;
|
|
}
|
|
|
|
if (m_hHttpStack)
|
|
{
|
|
::FreeLibrary(m_hHttpStack);
|
|
m_hHttpStack = NULL;
|
|
}
|
|
|
|
SESSION::Close();
|
|
|
|
DoTraceMessage(PP_LOG_INFO, "WinInet Http Session closed");
|
|
}
|
|
|
|
#ifdef PP_DEMO
|
|
BOOL WININET_SESSION::ContactPartner(PCWSTR pwszPartnerUrl,
|
|
PCWSTR pwszVerb,
|
|
PCWSTR pwszHeaders,
|
|
PWSTR pwszData,
|
|
PDWORD pdwDataLength
|
|
)
|
|
{
|
|
BOOL fRet = FALSE;
|
|
HINTERNET hConnect = NULL;
|
|
HINTERNET hRequest = NULL;
|
|
|
|
WCHAR ServerName[128];
|
|
WCHAR ObjectPath[1024];
|
|
|
|
URL_COMPONENTSW UrlComps;
|
|
::memset(&UrlComps, 0, sizeof(UrlComps));
|
|
|
|
UrlComps.dwStructSize = sizeof(UrlComps) / sizeof(WCHAR);
|
|
|
|
UrlComps.lpszHostName = ServerName;
|
|
UrlComps.dwHostNameLength = ARRAYSIZE(ServerName);
|
|
|
|
UrlComps.lpszUrlPath = ObjectPath;
|
|
UrlComps.dwUrlPathLength = ARRAYSIZE(ObjectPath);
|
|
|
|
PP_ASSERT(m_pfnCrack != NULL);
|
|
|
|
if ((*m_pfnCrack)(pwszPartnerUrl,
|
|
0,
|
|
0,
|
|
&UrlComps) == FALSE)
|
|
{
|
|
DoTraceMessage(PP_LOG_ERROR,
|
|
"WININET_SESSION::ContactPartner() failed; can not crack the URL %ws",
|
|
pwszPartnerUrl);
|
|
goto exit;
|
|
}
|
|
|
|
PP_ASSERT(m_pfnConnect != NULL);
|
|
|
|
hConnect = (*m_pfnConnect)(m_hInternet,
|
|
UrlComps.lpszHostName,
|
|
UrlComps.nPort,
|
|
NULL,
|
|
NULL,
|
|
INTERNET_SERVICE_HTTP,
|
|
0,
|
|
0);
|
|
if (hConnect == NULL)
|
|
{
|
|
DoTraceMessage(PP_LOG_ERROR,
|
|
"WININET_SESSION::ContactPartner() failed; can not open an HTTP sesstion to %ws:%d",
|
|
UrlComps.lpszHostName, UrlComps.nPort);
|
|
goto exit;
|
|
}
|
|
|
|
PP_ASSERT(m_pfnOpenRequest != NULL);
|
|
|
|
hRequest = (*m_pfnOpenRequest)(hConnect,
|
|
pwszVerb,
|
|
UrlComps.lpszUrlPath,
|
|
L"HTTP/1.1",
|
|
NULL,
|
|
NULL,
|
|
INTERNET_FLAG_NO_AUTO_REDIRECT | INTERNET_FLAG_NO_AUTH,
|
|
0);
|
|
if (hRequest == NULL)
|
|
{
|
|
DoTraceMessage(PP_LOG_ERROR,
|
|
"WININET_SESSION::ContactPartner() failed; can not open an HTTP request to %ws:(%ws)",
|
|
UrlComps.lpszUrlPath, pwszVerb);
|
|
goto exit;
|
|
}
|
|
|
|
PP_ASSERT(m_pfnSendRequest != NULL);
|
|
|
|
if ((*m_pfnSendRequest)(hRequest,
|
|
pwszHeaders,
|
|
0,
|
|
NULL,
|
|
0) == FALSE)
|
|
{
|
|
DoTraceMessage(PP_LOG_ERROR,
|
|
"WININET_SESSION::ContactPartner() failed; can not send an HTTP request");
|
|
|
|
goto exit;
|
|
}
|
|
|
|
if (pwszData == NULL)
|
|
{
|
|
fRet = TRUE;
|
|
goto exit;
|
|
}
|
|
|
|
DWORD dwStatus, dwStatusLen;
|
|
dwStatusLen = sizeof(dwStatus);
|
|
if (!QueryHeaders(hRequest,
|
|
HTTP_QUERY_FLAG_NUMBER | HTTP_QUERY_STATUS_CODE,
|
|
&dwStatus,
|
|
&dwStatusLen))
|
|
{
|
|
DoTraceMessage(PP_LOG_ERROR,
|
|
"WININET_SESSION::ContactPartner() failed; can not query status code");
|
|
goto exit;
|
|
}
|
|
|
|
if (dwStatus != HTTP_STATUS_REDIRECT)
|
|
{
|
|
DoTraceMessage(PP_LOG_ERROR,
|
|
"WININET_SESSION::ContactPartner() failed; expecting %d but get %d",
|
|
HTTP_STATUS_REDIRECT, dwStatus);
|
|
goto exit;
|
|
}
|
|
|
|
::wcscpy(pwszData, L"WWW-Authenticate: ");
|
|
if(!QueryHeaders(hRequest,
|
|
HTTP_QUERY_WWW_AUTHENTICATE,
|
|
(LPVOID)(pwszData + ::wcslen(L"WWW-Authenticate: ")),
|
|
pdwDataLength))
|
|
{
|
|
DoTraceMessage(PP_LOG_ERROR,
|
|
"WININET_SESSION::ContactPartner() failed; no auth headers found");
|
|
goto exit;
|
|
}
|
|
|
|
(*m_pfnCloseHandle)(hRequest);
|
|
hRequest = NULL;
|
|
(*m_pfnCloseHandle)(hConnect);
|
|
hConnect = NULL;
|
|
|
|
fRet = TRUE;
|
|
|
|
exit:
|
|
if (hRequest)
|
|
{
|
|
(*m_pfnCloseHandle)(hRequest);
|
|
hRequest = NULL;
|
|
}
|
|
|
|
if (hConnect)
|
|
{
|
|
(*m_pfnCloseHandle)(hConnect);
|
|
hConnect = NULL;
|
|
}
|
|
|
|
return fRet;
|
|
}
|
|
#endif // PP_DEMO
|