/*++ 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 #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(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(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(m_wDefaultDAUrl), ::wcslen(m_wDefaultDAUrl) * sizeof(WCHAR)) != ERROR_SUCCESS) { if (m_hKeyCU) { ::RegSetValueExW(m_hKeyCU, L"LoginServerUrl", 0, REG_SZ, reinterpret_cast(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(pwszDARealm), ::wcslen(pwszDARealm) * sizeof(WCHAR)) != ERROR_SUCCESS) { if (m_hKeyCU) { ::RegSetValueExW(m_hKeyCU, L"LoginServerRealm", 0, REG_SZ, reinterpret_cast(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(pwszDAReg), ::wcslen(pwszDAReg) * sizeof(WCHAR)) != ERROR_SUCCESS) { if (m_hKeyCU) { ::RegSetValueExW(m_hKeyCU, L"RegistrationUrl", 0, REG_SZ, reinterpret_cast(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_cast(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(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(m_wDefaultDAUrl), &dwUrlLen) != ERROR_SUCCESS) { if (m_hKeyLM && ::RegQueryValueExW(m_hKeyLM, L"LoginServerUrl", 0, &dwValueType, reinterpret_cast(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(m_wRegistrationUrl), &dwRegUrlLen); PP_ASSERT(!(dwError == ERROR_MORE_DATA)); */ m_hCredUI = ::LoadLibraryW(L"advapi32.dll"); if (m_hCredUI) { m_pfnReadDomainCred = reinterpret_cast(::GetProcAddress(m_hCredUI, "CredReadDomainCredentialsW")); if (m_pfnReadDomainCred == NULL) { DoTraceMessage(PP_LOG_WARNING, "failed to bind to CredReadDomainCredentialsW()"); } m_pfnCredFree = reinterpret_cast(::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(::GetProcAddress(m_hHttpStack, "InternetCloseHandle")); if (m_pfnCloseHandle == NULL) { DoTraceMessage(PP_LOG_ERROR, "function entry point InternetCloseHandle not found"); goto exit; } *ppfnInternetOpen = reinterpret_cast(::GetProcAddress(m_hHttpStack, "InternetOpenW")); if (*ppfnInternetOpen == NULL) { DoTraceMessage(PP_LOG_ERROR, "function entry point InternetOpenW not found"); goto exit; } m_pfnConnect = reinterpret_cast(::GetProcAddress(m_hHttpStack, "InternetConnectW")); if (m_pfnConnect == NULL) { DoTraceMessage(PP_LOG_ERROR, "function entry point InternetConnectW not found"); goto exit; } m_pfnOpenRequest = reinterpret_cast(::GetProcAddress(m_hHttpStack, "HttpOpenRequestW")); if (m_pfnOpenRequest == NULL) { DoTraceMessage(PP_LOG_ERROR, "function entry point HttpOpenRequestW not found"); goto exit; } m_pfnSendRequest = reinterpret_cast(::GetProcAddress(m_hHttpStack, "HttpSendRequestW")); if (m_pfnSendRequest == NULL) { DoTraceMessage(PP_LOG_ERROR, "function entry point HttpSendRequestW not found"); goto exit; } m_pfnQueryInfo = reinterpret_cast(::GetProcAddress(m_hHttpStack, "HttpQueryInfoW")); if (m_pfnQueryInfo == NULL) { DoTraceMessage(PP_LOG_ERROR, "function entry point HttpQueryInfoW not found"); goto exit; } m_pfnSetOption = reinterpret_cast(::GetProcAddress(m_hHttpStack, "InternetSetOptionW")); if (m_pfnSetOption == NULL) { DoTraceMessage(PP_LOG_ERROR, "function entry point InternetSetOptionW not found"); goto exit; } m_pfnOpenUrl = reinterpret_cast(::GetProcAddress(m_hHttpStack, "InternetOpenUrlW")); if (m_pfnOpenUrl == NULL) { DoTraceMessage(PP_LOG_ERROR, "function entry point InternetOpenUrlW not found"); goto exit; } m_pfnQueryOption = reinterpret_cast(::GetProcAddress(m_hHttpStack, "InternetQueryOptionW")); if (m_pfnQueryOption == NULL) { DoTraceMessage(PP_LOG_ERROR, "function entry point InternetQueryOptionW not found"); goto exit; } m_pfnCrack = reinterpret_cast(::GetProcAddress(m_hHttpStack, "InternetCrackUrlW")); if (m_pfnCrack == NULL) { DoTraceMessage(PP_LOG_ERROR, "function entry point InternetCrackUrlW not found"); goto exit; } m_pfnReadFile = reinterpret_cast(::GetProcAddress(m_hHttpStack, "InternetReadFile")); if (m_pfnReadFile == NULL) { DoTraceMessage(PP_LOG_ERROR, "function entry point InternetReadFile not found"); goto exit; } m_pfnStatusCallback = reinterpret_cast(::GetProcAddress(m_hHttpStack, "InternetSetStatusCallbackW")); if (m_pfnStatusCallback == NULL) { DoTraceMessage(PP_LOG_ERROR, "function entry point InternetSetStatusCallback not found"); goto exit; } m_pfnAddHeaders = reinterpret_cast(::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