|
|
//========= Copyright 1996-2005, Valve Corporation, All rights reserved. ============//
//
// Purpose:
//
// $NoKeywords: $
//
//=============================================================================//
//--------------------------------------------------------------------------------------------------------------
// downloadthread.cpp
//
// Implementation file for optional HTTP asset downloading thread
// Author: Matthew D. Campbell ([email protected]), 2004
//--------------------------------------------------------------------------------------------------------------
//--------------------------------------------------------------------------------------------------------------
// Includes
//--------------------------------------------------------------------------------------------------------------
#undef PROTECT_FILEIO_FUNCTIONS
#undef fopen
#if defined( WIN32 ) && !defined( _X360 )
#include "winlite.h"
#include <WinInet.h>
#endif
#include <assert.h>
#include <sys/stat.h>
#include <stdio.h>
#include "tier0/platform.h"
#include "tier0/dbg.h"
#include "tier0/threadtools.h"
#include "download_internal.h"
#include "tier1/strtools.h"
#if defined( _X360 )
#include "xbox/xbox_win32stubs.h"
#endif
// memdbgon must be the last include file in a .cpp file!!!
#include "tier0/memdbgon.h"
#ifdef _WIN32
//--------------------------------------------------------------------------------------------------------------
/**
* Formats a string to spit out via OutputDebugString (only in debug). OutputDebugString * is threadsafe, so this should be fine. * * Since I don't want to be playing with the developer cvar in the other thread, I'll use * the presence of _DEBUG as my developer flag. */ void Thread_DPrintf (char *fmt, ...) { #ifdef _DEBUG
va_list argptr; char msg[4096]; va_start( argptr, fmt ); Q_vsnprintf( msg, sizeof(msg), fmt, argptr ); va_end( argptr ); Plat_DebugString( msg ); #endif // _DEBUG
}
//--------------------------------------------------------------------------------------------------------------
/**
* Convenience function to name status states for debugging */ static const char *StateString( DWORD dwStatus ) { switch (dwStatus) { case INTERNET_STATUS_RESOLVING_NAME: return "INTERNET_STATUS_RESOLVING_NAME"; case INTERNET_STATUS_NAME_RESOLVED: return "INTERNET_STATUS_NAME_RESOLVED"; case INTERNET_STATUS_CONNECTING_TO_SERVER: return "INTERNET_STATUS_CONNECTING_TO_SERVER"; case INTERNET_STATUS_CONNECTED_TO_SERVER: return "INTERNET_STATUS_CONNECTED_TO_SERVER"; case INTERNET_STATUS_SENDING_REQUEST: return "INTERNET_STATUS_SENDING_REQUEST"; case INTERNET_STATUS_REQUEST_SENT: return "INTERNET_STATUS_REQUEST_SENT"; case INTERNET_STATUS_REQUEST_COMPLETE: return "INTERNET_STATUS_REQUEST_COMPLETE"; case INTERNET_STATUS_CLOSING_CONNECTION: return "INTERNET_STATUS_CLOSING_CONNECTION"; case INTERNET_STATUS_CONNECTION_CLOSED: return "INTERNET_STATUS_CONNECTION_CLOSED"; case INTERNET_STATUS_RECEIVING_RESPONSE: return "INTERNET_STATUS_RECEIVING_RESPONSE"; case INTERNET_STATUS_RESPONSE_RECEIVED: return "INTERNET_STATUS_RESPONSE_RECEIVED"; case INTERNET_STATUS_HANDLE_CLOSING: return "INTERNET_STATUS_HANDLE_CLOSING"; case INTERNET_STATUS_HANDLE_CREATED: return "INTERNET_STATUS_HANDLE_CREATED"; case INTERNET_STATUS_INTERMEDIATE_RESPONSE: return "INTERNET_STATUS_INTERMEDIATE_RESPONSE"; case INTERNET_STATUS_REDIRECT: return "INTERNET_STATUS_REDIRECT"; case INTERNET_STATUS_STATE_CHANGE: return "INTERNET_STATUS_STATE_CHANGE"; } return "Unknown"; }
//--------------------------------------------------------------------------------------------------------------
/**
* Callback function to update status information for a download (connecting to server, etc) */ void __stdcall DownloadStatusCallback( HINTERNET hOpenResource, DWORD dwContext, DWORD dwStatus, LPVOID pStatusInfo, DWORD dwStatusInfoLength ) { RequestContext *rc = (RequestContext*)pStatusInfo;
switch (dwStatus) { case INTERNET_STATUS_RESOLVING_NAME: case INTERNET_STATUS_NAME_RESOLVED: case INTERNET_STATUS_CONNECTING_TO_SERVER: case INTERNET_STATUS_CONNECTED_TO_SERVER: case INTERNET_STATUS_SENDING_REQUEST: case INTERNET_STATUS_REQUEST_SENT: case INTERNET_STATUS_REQUEST_COMPLETE: case INTERNET_STATUS_CLOSING_CONNECTION: case INTERNET_STATUS_CONNECTION_CLOSED: if ( rc ) { rc->fetchStatus = dwStatus; } else { //Thread_DPrintf( "** No RequestContext **\n" );
} //Thread_DPrintf( "DownloadStatusCallback %s\n", StateString(dwStatus) );
break; } }
//--------------------------------------------------------------------------------------------------------------
/**
* Reads data from a handle opened by InternetOpenUrl(). */ void ReadData( RequestContext& rc ) { const int BufferSize = 2048; unsigned char data[BufferSize]; DWORD dwSize = 0;
if ( !rc.nBytesTotal ) { rc.status = HTTP_ERROR; rc.error = HTTP_ERROR_ZERO_LENGTH_FILE; return; } rc.nBytesCurrent = rc.nBytesCached; rc.status = HTTP_FETCH;
while ( !rc.shouldStop ) { // InternetReadFile() will block until there is data, or the socket gets closed. This means the
// main thread could request an abort while we're blocked here. This is okay, because the main
// thread will not wait for this thread to finish, but will clean up the RequestContext at some
// later point when InternetReadFile() has returned and this thread has finished.
if ( !InternetReadFile( rc.hDataResource, (LPVOID)data, BufferSize, &dwSize ) ) { // if InternetReadFile() returns 0, there was a socket error (connection closed, etc)
rc.status = HTTP_ERROR; rc.error = HTTP_ERROR_CONNECTION_CLOSED; return; } if ( !dwSize ) { // if InternetReadFile() succeeded, but we read 0 bytes, we're at the end of the file.
// if the file doesn't exist, write it out
char path[_MAX_PATH]; Q_snprintf( path, sizeof(path), "%s\\%s", rc.basePath, rc.gamePath ); struct stat buf; int rt = stat(path, &buf); if ( rt == -1 ) { FILE *fp = fopen( path, "wb" ); if ( fp ) { fwrite( rc.data, rc.nBytesTotal, 1, fp ); fclose( fp ); } }
// Let the main thread know we finished reading data, and wait for it to let us exit.
rc.status = HTTP_DONE; return; } else { // We've read some data. Make sure we won't walk off the end of our buffer, then
// use memcpy() to save the data off.
DWORD safeSize = (DWORD)Min( rc.nBytesTotal - rc.nBytesCurrent, dwSize ); //Thread_DPrintf( "Read %d bytes @ offset %d\n", safeSize, rc.nBytesCurrent );
if ( safeSize != dwSize ) { //Thread_DPrintf( "Warning - read more data than expected!\n" );
} if ( rc.data && safeSize > 0 ) { memcpy( rc.data + rc.nBytesCurrent, data, safeSize ); } rc.nBytesCurrent += safeSize; } }
// If we get here, rc.shouldStop was set early (user hit cancel).
// Let the main thread know we aborted properly.
rc.status = HTTP_ABORTED; }
//--------------------------------------------------------------------------------------------------------------
const char *StatusString[] = { "HTTP_CONNECTING", "HTTP_FETCH", "HTTP_DONE", "HTTP_ABORTED", "HTTP_ERROR", };
//--------------------------------------------------------------------------------------------------------------
const char *ErrorString[] = { "HTTP_ERROR_NONE", "HTTP_ERROR_ZERO_LENGTH_FILE", "HTTP_ERROR_CONNECTION_CLOSED", "HTTP_ERROR_INVALID_URL", "HTTP_ERROR_INVALID_PROTOCOL", "HTTP_ERROR_CANT_BIND_SOCKET", "HTTP_ERROR_CANT_CONNECT", "HTTP_ERROR_NO_HEADERS", "HTTP_ERROR_FILE_NONEXISTENT", "HTTP_ERROR_MAX", };
//--------------------------------------------------------------------------------------------------------------
/**
* Closes all open handles, and waits until the main thread has given the OK * to quit. */ void CleanUpDownload( RequestContext& rc, HTTPStatus status, HTTPError error = HTTP_ERROR_NONE ) { if ( status != HTTP_DONE || error != HTTP_ERROR_NONE ) { //Thread_DPrintf( "CleanUpDownload() - http status is %s, error state is %s\n",
// StatusString[status], ErrorString[error] );
}
rc.status = status; rc.error = error;
// Close all HINTERNET handles we have open
if ( rc.hDataResource && !InternetCloseHandle(rc.hDataResource) ) { //Thread_DPrintf( "Failed to close data resource for %s%s\n", rc.baseURL, rc.gamePath );
} else if ( rc.hOpenResource && !InternetCloseHandle(rc.hOpenResource) ) { //Thread_DPrintf( "Failed to close open resource for %s%s\n", rc.baseURL, rc.gamePath );
} rc.hDataResource = NULL; rc.hOpenResource = NULL;
// wait until the main thread says we can go away (so it can look at rc.data).
while ( !rc.shouldStop ) { Sleep( 100 ); }
// Delete rc.data, which was allocated in this thread
if ( rc.data != NULL ) { delete[] rc.data; rc.data = NULL; }
// and tell the main thread we're exiting, so it can delete rc.cachedData and rc itself.
rc.threadDone = true; }
//--------------------------------------------------------------------------------------------------------------
static void DumpHeaders( RequestContext& rc ) { #ifdef _DEBUG
DWORD dwSize;
// First time we will find out the size of the headers.
HttpQueryInfo ( rc.hDataResource, HTTP_QUERY_RAW_HEADERS_CRLF, NULL, &dwSize, NULL ); char *lpBuffer = new char [dwSize + 2];
// Now we call HttpQueryInfo again to get the headers.
if (!HttpQueryInfo ( rc.hDataResource, HTTP_QUERY_RAW_HEADERS_CRLF, (LPVOID)lpBuffer, &dwSize, NULL)) { return; } *(lpBuffer + dwSize) = '\n'; *(lpBuffer + dwSize + 1) = '\0';
Thread_DPrintf( "------------------------------\n%s%s\n%s------------------------------\n", rc.baseURL, rc.gamePath, lpBuffer ); #endif
}
//--------------------------------------------------------------------------------------------------------------
/**
* Main download thread function - implements a (partial) synchronous HTTP download. */ uintp DownloadThread( void *voidPtr ) { RequestContext& rc = *(RequestContext *)voidPtr;
URL_COMPONENTS url; char urlBuf[6][BufferSize]; url.dwStructSize = sizeof(url);
url.dwSchemeLength = BufferSize; url.dwHostNameLength = BufferSize; url.dwUserNameLength = BufferSize; url.dwPasswordLength = BufferSize; url.dwUrlPathLength = BufferSize; url.dwExtraInfoLength = BufferSize;
url.lpszScheme = urlBuf[0]; url.lpszHostName = urlBuf[1]; url.lpszUserName = urlBuf[2]; url.lpszPassword = urlBuf[3]; url.lpszUrlPath = urlBuf[4]; url.lpszExtraInfo = urlBuf[5];
char fullURL[BufferSize*2]; DWORD fullURLLength = BufferSize*2; Q_snprintf( fullURL, fullURLLength, "%s%s", rc.baseURL, rc.gamePath ); /*
if ( !InternetCombineUrl( rc.baseURL, rc.gamePath, fullURL, &fullURLLength, 0 ) ) { CleanUpDownload( rc, HTTP_ERROR, HTTP_ERROR_INVALID_URL ); return rc.status; } */
if ( !InternetCrackUrl( fullURL, fullURLLength, 0, &url ) ) { CleanUpDownload( rc, HTTP_ERROR, HTTP_ERROR_INVALID_URL ); return rc.status; }
/// @TODO: Add FTP support (including restart of aborted transfers) -MDC 2004/01/08
// We should be able to handle FTP downloads as well, but I don't have a server to check against, so
// I'm gonna disallow it in case something bad would happen.
if ( url.nScheme != INTERNET_SCHEME_HTTP && url.nScheme != INTERNET_SCHEME_HTTPS ) { CleanUpDownload( rc, HTTP_ERROR, HTTP_ERROR_INVALID_PROTOCOL ); return rc.status; }
// Open a socket etc for the download.
// The first parameter, "Half-Life", is the User-Agent that gets sent with HTTP requests.
// INTERNET_OPEN_TYPE_PRECONFIG specifies using IE's proxy info from the registry for HTTP downloads.
rc.hOpenResource = InternetOpen( "Half-Life 2", INTERNET_OPEN_TYPE_PRECONFIG ,NULL, NULL, 0);
if ( !rc.hOpenResource ) { CleanUpDownload( rc, HTTP_ERROR, HTTP_ERROR_CANT_BIND_SOCKET ); return rc.status; }
InternetSetStatusCallback( rc.hOpenResource, (INTERNET_STATUS_CALLBACK)DownloadStatusCallback );
if ( rc.shouldStop ) { CleanUpDownload( rc, HTTP_ABORTED ); return rc.status; }
// Set up some flags
DWORD flags = 0; flags |= INTERNET_FLAG_RELOAD; // Get from server, not IE's cache
flags |= INTERNET_FLAG_NO_CACHE_WRITE; // Don't write to IE's cache, since we're doing our own caching of partial downloads
flags |= INTERNET_FLAG_KEEP_CONNECTION; // Use keep-alive semantics. Since each file downloaded is a separate connection
// from a separate thread, I don't think this does much. Can't hurt, though.
if ( url.nScheme == INTERNET_SCHEME_HTTPS ) { // The following flags allow us to use https:// URLs, but don't provide much in the way of authentication.
// In other words, this allows people with only access to https:// servers (?!?) to host files as transparently
// as possible.
flags |= INTERNET_FLAG_SECURE; // Use SSL, etc. Kinda need this for HTTPS URLs.
flags |= INTERNET_FLAG_IGNORE_CERT_CN_INVALID; // Don't check hostname on the SSL cert.
flags |= INTERNET_FLAG_IGNORE_CERT_DATE_INVALID; // Don't check for expired SSL certs.
}
// Request a partial if we have the data
char headers[BufferSize] = ""; DWORD headerLen = 0; char *headerPtr = NULL; if ( *rc.cachedTimestamp && rc.nBytesCached ) { if ( *rc.serverURL ) { Q_snprintf( headers, BufferSize, "If-Range: %s\nRange: bytes=%d-\nReferer: hl2://%s\n", rc.cachedTimestamp, rc.nBytesCached, rc.serverURL ); } else { Q_snprintf( headers, BufferSize, "If-Range: %s\nRange: bytes=%d-\n", rc.cachedTimestamp, rc.nBytesCached ); } headerPtr = headers; headerLen = (DWORD)-1L; // the DWORD cast is because we get a signed/unsigned mismatch even with an L on the -1.
//Thread_DPrintf( "Requesting partial download\n%s", headers );
} else if ( *rc.serverURL ) { Q_snprintf( headers, BufferSize, "Referer: hl2://%s\n", rc.serverURL ); headerPtr = headers; headerLen = (DWORD)-1L; // the DWORD cast is because we get a signed/unsigned mismatch even with an L on the -1.
//Thread_DPrintf( "Requesting full download\n%s", headers );
}
rc.hDataResource = InternetOpenUrl(rc.hOpenResource, fullURL, headerPtr, headerLen, flags,(DWORD)(&rc) );
// send the request off
if ( !rc.hDataResource ) { CleanUpDownload( rc, HTTP_ERROR, HTTP_ERROR_CANT_CONNECT ); return rc.status; }
if ( rc.shouldStop ) { CleanUpDownload( rc, HTTP_ABORTED ); return rc.status; }
//DumpHeaders( rc ); // debug
// check the status (are we gonna get anything?)
DWORD size = sizeof(DWORD); DWORD code; if ( !HttpQueryInfo( rc.hDataResource, HTTP_QUERY_STATUS_CODE | HTTP_QUERY_FLAG_NUMBER, &code, &size, NULL ) ) { CleanUpDownload( rc, HTTP_ERROR, HTTP_ERROR_NO_HEADERS ); return rc.status; }
// Only status codes we're looking for are HTTP_STATUS_OK (200) and HTTP_STATUS_PARTIAL_CONTENT (206)
if ( code != HTTP_STATUS_OK && code != HTTP_STATUS_PARTIAL_CONTENT ) { CleanUpDownload( rc, HTTP_ERROR, HTTP_ERROR_FILE_NONEXISTENT ); return rc.status; }
// get the timestamp, and save it off for future resumes, in case we abort this transfer later.
size = BufferSize; if ( !HttpQueryInfo( rc.hDataResource, HTTP_QUERY_LAST_MODIFIED, rc.cachedTimestamp, &size, NULL ) ) { rc.cachedTimestamp[0] = 0; } rc.cachedTimestamp[BufferSize-1] = 0;
// If we're not getting a partial download, don't use any cached data, even if we have some.
if ( code != HTTP_STATUS_PARTIAL_CONTENT ) { if ( rc.nBytesCached ) { //Thread_DPrintf( "Partial download refused - getting full version\n" );
} rc.nBytesCached = 0; // start from the beginning
}
// Check the resource size, and allocate a buffer
size = sizeof(code); if ( HttpQueryInfo( rc.hDataResource, HTTP_QUERY_CONTENT_LENGTH | HTTP_QUERY_FLAG_NUMBER, &code, &size, NULL ) ) { rc.nBytesTotal = code + rc.nBytesCached; if ( code > 0 ) { rc.data = new unsigned char[rc.nBytesTotal]; } } else { CleanUpDownload( rc, HTTP_ERROR, HTTP_ERROR_ZERO_LENGTH_FILE ); return rc.status; }
// copy cached data into buffer
if ( rc.cacheData && rc.nBytesCached ) { int len = Min( rc.nBytesCached, rc.nBytesTotal ); memcpy( rc.data, rc.cacheData, len ); }
if ( rc.shouldStop ) { CleanUpDownload( rc, HTTP_ABORTED ); return rc.status; }
// now download the actual data
ReadData( rc );
// and stick around until the main thread has gotten the data.
CleanUpDownload( rc, rc.status, rc.error ); return rc.status; }
#elif defined( _PS3 )
uintp DownloadThread( void *voidPtr ) { RequestContext& rc = *(RequestContext *)voidPtr;
Warning( "DownloadThread not implemented on PS3!\n" ); Assert( 0 );
return 0; } #elif defined( POSIX ) && !defined( DEDICATED )
#include "curl/curl.h"
// curl callback functions
static size_t curlWriteFn( void *ptr, size_t size, size_t nmemb, void *stream) { RequestContext *pRC = (RequestContext *) stream; if ( pRC->nBytesTotal && pRC->nBytesCurrent + ( size * nmemb ) <= pRC->nBytesTotal ) { Q_memcpy( pRC->data + pRC->nBytesCurrent, ptr, ( size * nmemb ) ); pRC->nBytesCurrent += size * nmemb; } return size * nmemb; }
int Q_StrTrim( char *pStr ) { char *pSource = pStr; char *pDest = pStr; // skip white space at the beginning
while ( *pSource != 0 && V_isspace( *pSource ) ) { pSource++; } // copy everything else
char *pLastWhiteBlock = NULL; char *pStart = pDest; while ( *pSource != 0 ) { *pDest = *pSource++; if ( V_isspace( *pDest ) ) { if ( pLastWhiteBlock == NULL ) pLastWhiteBlock = pDest; } else { pLastWhiteBlock = NULL; } pDest++; } *pDest = 0; // did we end in a whitespace block?
if ( pLastWhiteBlock != NULL ) { // yep; shorten the string
pDest = pLastWhiteBlock; *pLastWhiteBlock = 0; } return pDest - pStart; }
static size_t curlHeaderFn( void *ptr, size_t size, size_t nmemb, void *stream) { char *pszHeader = (char*)ptr; char *pszValue = NULL; RequestContext *pRC = (RequestContext *) stream; pszHeader[ ( size * nmemb - 1 ) ] = NULL; pszValue = Q_strstr( pszHeader, ":" ); if ( pszValue ) { // null terminate the header name, and point pszValue at it's value
*pszValue = NULL; pszValue++; Q_StrTrim( pszValue ); } if ( 0 == Q_stricmp( pszHeader, "Content-Length" ) ) { size_t len = atol( pszValue ); if ( pRC && len ) { pRC->nBytesTotal = len; pRC->data = (byte*)malloc( len ); } } return size * nmemb; }
// we're going to abuse this by using it for proxy pac fetching
// the cacheData field will hold the URL of the PAC, and the data
// field the contents of the pac
RequestContext g_pacRequestCtx;
// system specific headers for proxy configuration
#if defined(OSX)
#include <CoreFoundation/CoreFoundation.h>
#include <CoreServices/CoreServices.h>
#include <SystemConfiguration/SystemConfiguration.h>
#endif
void SetProxiesForURL( CURL *hMasterCURL, const char *pszURL ) { uint32 uProxyPort = 0; char rgchProxyHost[1024]; char *pszProxyExceptionList = NULL; rgchProxyHost[0] = '\0'; #if defined(OSX)
// create an urlref around the raw URL
CFURLRef url = CFURLCreateWithBytes( NULL, ( const UInt8 * ) pszURL, strlen( pszURL ), kCFStringEncodingASCII, NULL ); // copy the proxies dictionary
CFDictionaryRef proxyDict = SCDynamicStoreCopyProxies(NULL); // and ask the system what proxies it thinks I should consider for the given URL
CFArrayRef proxies = CFNetworkCopyProxiesForURL( url, proxyDict ); CFIndex iProxy; // walk through the returned set, looking for any types we (and lib curl) can handle
// the list is returned in "preference order", but we can only handle http, and pac urls
for( iProxy = 0; iProxy < CFArrayGetCount( proxies ); iProxy++ ) { CFDictionaryRef proxy = (CFDictionaryRef) CFArrayGetValueAtIndex( proxies, iProxy ); if ( proxy == NULL ) break; // what type of proxy is this one?
CFStringRef proxyType = (CFStringRef) CFDictionaryGetValue( proxy, kCFProxyTypeKey ); if ( CFEqual( proxyType, kCFProxyTypeNone ) ) { // no proxy should be used - we're done.
break; } else if ( CFEqual( proxyType, kCFProxyTypeHTTP ) ) { // manually configured HTTP proxy settings.
const void *val = NULL; // grab the proxy port
val = CFDictionaryGetValue( proxy, kCFProxyPortNumberKey ); if ( val == NULL || !CFNumberGetValue( (CFNumberRef) val, kCFNumberIntType, &uProxyPort ) ) // either we failed, or the port was invalid
continue; // no port specified - use the default http port
if ( uProxyPort == 0 ) uProxyPort = 80; int cbOffset = 0; // see if they've specified authentication (username/password)
val = CFDictionaryGetValue( proxy, kCFProxyUsernameKey ); if ( val != NULL && CFStringGetCString( (CFStringRef) val, rgchProxyHost + cbOffset, sizeof(rgchProxyHost) - cbOffset, kCFStringEncodingASCII ) && rgchProxyHost[cbOffset] != '\0' ) { // we've got "username" in rgchProxyHost
cbOffset = Q_strlen( rgchProxyHost ); val = CFDictionaryGetValue( proxy, kCFProxyPasswordKey ); if ( val != NULL && CFStringGetLength( (CFStringRef) val ) ) { // and there's a non-null password value - put a colon after username
rgchProxyHost[cbOffset++] = ':'; CFStringGetCString( (CFStringRef) val, rgchProxyHost + cbOffset, sizeof(rgchProxyHost) - cbOffset, kCFStringEncodingASCII ); // now we've got user:password in rgchProxyHost
cbOffset = Q_strlen( rgchProxyHost ); } // since we've got at least a username, we need an @
rgchProxyHost[cbOffset++] = '@'; } val = CFDictionaryGetValue( proxy, kCFProxyHostNameKey ); if ( val == NULL || !CFStringGetCString( (CFStringRef) val, rgchProxyHost + cbOffset, sizeof(rgchProxyHost) - cbOffset, kCFStringEncodingASCII ) || rgchProxyHost[cbOffset] == '\0' ) continue; break; } else if ( CFEqual( proxyType, kCFProxyTypeAutoConfigurationURL ) ) { // a proxy autoconfig URL has been provided
char rgchPacURL[1024]; // get the url (as an urlref) and turn it into a string
CFURLRef cfUrl = (CFURLRef) CFDictionaryGetValue( proxy, kCFProxyAutoConfigurationURLKey ); CFStringGetCString( (CFStringRef) CFStringCreateWithFormat( NULL, NULL, CFSTR("%@"), cfUrl ), rgchPacURL, sizeof( rgchPacURL ), kCFStringEncodingASCII ); CURLcode res = CURLE_OK; // see if we've not yet fetched this pac file
if ( !g_pacRequestCtx.cacheData || Q_strcmp( (const char *)g_pacRequestCtx.cacheData, rgchPacURL ) ) { if ( g_pacRequestCtx.cacheData ) { free( g_pacRequestCtx.cacheData ); g_pacRequestCtx.cacheData = NULL; } if ( g_pacRequestCtx.data ) { free( g_pacRequestCtx.data ); g_pacRequestCtx.data = NULL; } // grab the data, using the same request context structure (and callbacks) we use for real downloads
CURL *hCURL = curl_easy_init(); if ( !hCURL ) { AssertMsg( hCURL, "failed to initialize curl handle" ); break; } curl_easy_setopt( hCURL, CURLOPT_NOPROGRESS, 1 ); curl_easy_setopt( hCURL, CURLOPT_NOSIGNAL, 1 ); curl_easy_setopt( hCURL, CURLOPT_TIMEOUT, 30 ); curl_easy_setopt( hCURL, CURLOPT_FOLLOWLOCATION, 1 ); // follow 30x redirections from the web server
// and setup the callback fns
curl_easy_setopt( hCURL, CURLOPT_HEADERFUNCTION, &curlHeaderFn ); curl_easy_setopt( hCURL, CURLOPT_WRITEFUNCTION, &curlWriteFn ); // setup callback stream pointers
curl_easy_setopt( hCURL, CURLOPT_WRITEHEADER, &g_pacRequestCtx ); curl_easy_setopt( hCURL, CURLOPT_WRITEDATA, &g_pacRequestCtx ); curl_easy_setopt( hCURL, CURLOPT_URL, rgchPacURL ); res = curl_easy_perform( hCURL ); curl_easy_cleanup( hCURL ); } if ( res == CURLE_OK ) { // copy the URL into the "pac cache", if necessary
if ( !g_pacRequestCtx.cacheData ) { g_pacRequestCtx.cacheData = (unsigned char*) malloc( Q_strlen( rgchPacURL ) + 1 ); Q_memcpy( g_pacRequestCtx.cacheData, rgchPacURL, Q_strlen( rgchPacURL ) ); } if ( !g_pacRequestCtx.data ) // no data in the proxy.pac they have, so just ignore it
return; // wrap the data (the pac contents) into a cfstring
CFStringRef cfPacStr = CFStringCreateWithCString( kCFAllocatorDefault, (const char *)g_pacRequestCtx.data, kCFStringEncodingASCII ); // and ask the system, given this proxy pac, what (list of) proxies should I consider for this URL?
CFErrorRef err; CFArrayRef proxiesForUrl = CFNetworkCopyProxiesForAutoConfigurationScript( cfPacStr, cfUrl, &err ); if ( proxiesForUrl ) { // we're re-assigning the value that the loop is iterating over, the postincrement will fire after we do this,
// hence the -1 (rather than 0) assignment to iProxy
proxies = proxiesForUrl; iProxy = -1; } continue; } else { if ( g_pacRequestCtx.cacheData ) { free( g_pacRequestCtx.cacheData ); g_pacRequestCtx.cacheData = NULL; } } } else { Msg( "unsupported proxy type\n" ); break; } } #else
#warning "CHTTPDownloadThread doesn't know how to set proxy config"
#endif
if ( rgchProxyHost[0] == '\0' || uProxyPort <= 0 ) { if ( pszProxyExceptionList ) free( pszProxyExceptionList ); return; } curl_easy_setopt( hMasterCURL, CURLOPT_PROXY, rgchProxyHost ); curl_easy_setopt( hMasterCURL, CURLOPT_PROXYPORT, uProxyPort ); if ( pszProxyExceptionList ) { curl_easy_setopt( hMasterCURL, (CURLoption) (CURLOPTTYPE_OBJECTPOINT + 177) /*CURLOPT_NOPROXY*/ , pszProxyExceptionList ); free( pszProxyExceptionList ); } }
uintp DownloadThread( void *voidPtr ) { static bool bDoneInit = false; if ( !bDoneInit ) { bDoneInit = true; curl_global_init( CURL_GLOBAL_SSL ); } RequestContext& rc = *(RequestContext *)voidPtr; rc.status = HTTP_FETCH; CURL *hCURL = curl_easy_init(); if ( !hCURL ) { rc.error = HTTP_ERROR_INVALID_URL; rc.status = HTTP_ERROR; rc.threadDone = true; return rc.status; } curl_easy_setopt( hCURL, CURLOPT_NOPROGRESS, 1 ); curl_easy_setopt( hCURL, CURLOPT_NOSIGNAL, 1 ); curl_easy_setopt( hCURL, CURLOPT_TIMEOUT, 30 ); curl_easy_setopt( hCURL, CURLOPT_IPRESOLVE, CURL_IPRESOLVE_V4 ); // and now the callback fns
curl_easy_setopt( hCURL, CURLOPT_HEADERFUNCTION, &curlHeaderFn ); curl_easy_setopt( hCURL, CURLOPT_WRITEFUNCTION, &curlWriteFn ); uint32 cubURL = Q_strlen( rc.baseURL ) + Q_strlen( rc.gamePath ) + 2 /*one for the /, one for the null*/; char *pchURL = (char *) malloc( cubURL ); Q_snprintf( pchURL, cubURL, "%s%s", rc.baseURL, rc.gamePath ); // setup proxies
SetProxiesForURL( hCURL, pchURL ); // set the url
curl_easy_setopt( hCURL, CURLOPT_URL, pchURL ); // setup callback stream pointers
curl_easy_setopt( hCURL, CURLOPT_WRITEHEADER, &rc ); curl_easy_setopt( hCURL, CURLOPT_WRITEDATA, &rc ); curl_easy_setopt( hCURL, CURLOPT_FOLLOWLOCATION, 1 ); curl_easy_setopt( hCURL, CURLOPT_MAXREDIRS, 1 ); curl_easy_setopt( hCURL, CURLOPT_UNRESTRICTED_AUTH, 1 ); curl_easy_setopt( hCURL, CURLOPT_USERAGENT, "Half-Life 2" ); // g0g0g0
CURLcode res = curl_easy_perform( hCURL ); curl_easy_getinfo( hCURL , CURLINFO_RESPONSE_CODE , &rc.status ); free( pchURL ); if ( res == CURLE_OK ) { curl_easy_getinfo( hCURL , CURLINFO_RESPONSE_CODE , &rc.status ); if ( rc.status == 200 || rc.status == 206 ) { rc.status = HTTP_DONE; rc.error = HTTP_ERROR_NONE; } else { rc.status = HTTP_ERROR; rc.error = HTTP_ERROR_FILE_NONEXISTENT; } } else { rc.status = HTTP_ERROR; } // wait until the main thread says we can go away (so it can look at rc.data).
while ( !rc.shouldStop ) { ThreadSleep( 100 ); } // Delete rc.data, which was allocated in this thread
if ( rc.data != NULL ) { delete[] rc.data; rc.data = NULL; } curl_easy_cleanup( hCURL ); rc.threadDone = true; return rc.status; }
#else
uintp DownloadThread( void *voidPtr ) { Assert( !"Impl me" ); return 0; } #endif
|