2020-09-30 16:53:55 +02:00

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