mirror of
https://github.com/WinampDesktop/winamp.git
synced 2025-01-28 19:48:23 +00:00
873 lines
23 KiB
C++
873 lines
23 KiB
C++
|
#include <string.h>
|
||
|
#include <strsafe.h>
|
||
|
|
||
|
#include <iostream>
|
||
|
#include <cstdio>
|
||
|
|
||
|
#include <QtCore/qglobal.h>
|
||
|
|
||
|
#include <QWebEngineProfile>
|
||
|
#include <QtWebEngineWidgets/QtWebEngineWidgets>
|
||
|
#include <QWebEnginePage>
|
||
|
#include <QWebEngineSettings>
|
||
|
|
||
|
#include "api__wac_downloadManager.h"
|
||
|
|
||
|
#include "wac_downloadManager.h"
|
||
|
#include "wac_download_http_receiver_api.h"
|
||
|
|
||
|
#include "..\wac_network\wac_network_http_receiver_api.h"
|
||
|
|
||
|
#include "api/service/waservicefactory.h"
|
||
|
|
||
|
#include "../nu/threadname.h"
|
||
|
#include "../nu/AutoChar.h"
|
||
|
#include "../nu/threadpool/timerhandle.hpp"
|
||
|
|
||
|
#include "..\WAT\WAT.h"
|
||
|
|
||
|
#include "..\Winamp\buildType.h"
|
||
|
|
||
|
static const GUID internetConfigGroupGUID =
|
||
|
{
|
||
|
0xc0a565dc, 0xcfe, 0x405a, { 0xa2, 0x7c, 0x46, 0x8b, 0xc, 0x8a, 0x3a, 0x5c }
|
||
|
};
|
||
|
|
||
|
#define DOWNLOAD_TIMEOUT_MS 60000 // 60 second timeout
|
||
|
#define DOWNLOAD_SLEEP_MS 50
|
||
|
#define DOWNLOAD_BUFFER_SIZE 1310720 // gives a maximum download rate of 25 mb/sec per file
|
||
|
|
||
|
|
||
|
/**********************************************************************************
|
||
|
**********************************************************************************/
|
||
|
|
||
|
|
||
|
/**********************************************************************************
|
||
|
* PUBLIC *
|
||
|
**********************************************************************************/
|
||
|
wa::Components::WAC_DownloadData::WAC_DownloadData( api_wac_download_manager_http_receiver *p_http, const char *p_url, int p_flags, ifc_downloadManagerCallback *p_callback )
|
||
|
{
|
||
|
_http = p_http;
|
||
|
|
||
|
strcpy_s( this->_url, 1024, p_url );
|
||
|
|
||
|
_flags = p_flags;
|
||
|
_callback = p_callback;
|
||
|
|
||
|
if ( _callback )
|
||
|
_callback->AddRef();
|
||
|
|
||
|
_hFile = INVALID_HANDLE_VALUE;
|
||
|
_filepath[ 0 ] = 0;
|
||
|
_fileext = 0;
|
||
|
|
||
|
int download_method = ( api_downloadManager::DOWNLOADEX_MASK_DOWNLOADMETHOD & _flags );
|
||
|
switch ( download_method )
|
||
|
{
|
||
|
case api_downloadManager::DOWNLOADEX_TEMPFILE:
|
||
|
{
|
||
|
wchar_t temppath[ MAX_PATH - 14 ] = { 0 }; // MAX_PATH-14 'cause MSDN said so
|
||
|
GetTempPathW( MAX_PATH - 14, temppath );
|
||
|
GetTempFileNameW( temppath, L"wdl", 0, _filepath );
|
||
|
_hFile = CreateFileW( _filepath, GENERIC_WRITE, FILE_SHARE_READ | FILE_SHARE_WRITE, 0, CREATE_ALWAYS, 0, 0 );
|
||
|
}
|
||
|
break;
|
||
|
case api_downloadManager::DOWNLOADEX_CALLBACK:
|
||
|
if ( _callback )
|
||
|
_callback->GetLocation( _filepath, MAX_PATH );
|
||
|
break;
|
||
|
}
|
||
|
|
||
|
|
||
|
|
||
|
_source[ 0 ] = 0;
|
||
|
_title[ 0 ] = 0;
|
||
|
|
||
|
if ( _flags & api_downloadManager::DOWNLOADEX_CALLBACK )
|
||
|
{
|
||
|
if ( _callback )
|
||
|
{
|
||
|
_callback->GetSource( _source, 1024 );
|
||
|
_callback->GetTitle( _title, 1024 );
|
||
|
}
|
||
|
}
|
||
|
|
||
|
_connectionStart = _lastDownloadTick = GetTickCount();
|
||
|
_last_status = HTTP_RECEIVER_STATUS_ERROR;
|
||
|
_pending = ( _flags & api_downloadManager::DOWNLOADEX_PENDING ) > 0;
|
||
|
}
|
||
|
|
||
|
wa::Components::WAC_DownloadData::~WAC_DownloadData()
|
||
|
{
|
||
|
ServiceRelease( _http, httpreceiverGUID2 );
|
||
|
|
||
|
_http = NULL;
|
||
|
|
||
|
if ( _fileext )
|
||
|
delete _fileext;
|
||
|
|
||
|
int download_method = ( api_downloadManager::DOWNLOADEX_MASK_DOWNLOADMETHOD & _flags );
|
||
|
if ( download_method == api_downloadManager::DOWNLOADEX_TEMPFILE && _filepath[ 0 ] )
|
||
|
DeleteFileW( _filepath );
|
||
|
|
||
|
if ( _callback )
|
||
|
_callback->Release();
|
||
|
|
||
|
_callback = NULL;
|
||
|
}
|
||
|
|
||
|
|
||
|
void wa::Components::WAC_DownloadData::Retain()
|
||
|
{
|
||
|
this->_refCount.fetch_add( 1 );
|
||
|
}
|
||
|
|
||
|
void wa::Components::WAC_DownloadData::Release()
|
||
|
{
|
||
|
if ( this->_refCount.fetch_sub( 1 ) == 0 )
|
||
|
delete this;
|
||
|
}
|
||
|
|
||
|
|
||
|
void wa::Components::WAC_DownloadData::Close( ifc_downloadManagerCallback **callbackCopy )
|
||
|
{
|
||
|
if ( _hFile != INVALID_HANDLE_VALUE )
|
||
|
CloseHandle( _hFile );
|
||
|
|
||
|
_hFile = INVALID_HANDLE_VALUE;
|
||
|
|
||
|
if ( callbackCopy != NULL )
|
||
|
{
|
||
|
*callbackCopy = _callback;
|
||
|
if ( _callback != NULL )
|
||
|
_callback->AddRef();
|
||
|
}
|
||
|
else if ( _callback != NULL )
|
||
|
{
|
||
|
_callback->Release();
|
||
|
_callback = NULL;
|
||
|
}
|
||
|
|
||
|
// don't want to close http here, because someone might still want to get the headers out of it
|
||
|
}
|
||
|
|
||
|
bool wa::Components::WAC_DownloadData::getExtention()
|
||
|
{
|
||
|
if ( _fileext && *_fileext )
|
||
|
return _fileext;
|
||
|
|
||
|
char l_header_name_content_type[] = "Content-Type";
|
||
|
|
||
|
char *l_content_type = _http->getheader( l_header_name_content_type );
|
||
|
if ( l_content_type && *l_content_type )
|
||
|
{
|
||
|
if ( _CONTENT_TYPES_EXTENSIONS.count( l_content_type ) == 1 )
|
||
|
_fileext = _strdup( _CONTENT_TYPES_EXTENSIONS.find( l_content_type )->second.c_str() );
|
||
|
}
|
||
|
|
||
|
return _fileext;
|
||
|
}
|
||
|
|
||
|
|
||
|
|
||
|
|
||
|
/**********************************************************************************
|
||
|
* PRIVATE *
|
||
|
**********************************************************************************/
|
||
|
|
||
|
|
||
|
|
||
|
/**********************************************************************************
|
||
|
**********************************************************************************/
|
||
|
|
||
|
/**********************************************************************************
|
||
|
* PUBLIC *
|
||
|
**********************************************************************************/
|
||
|
wa::Components::WAC_DownloadManager::WAC_DownloadManager( QObject *parent ) : QNetworkAccessManager( parent )
|
||
|
{
|
||
|
this->setObjectName( "DownloadManagerService" );
|
||
|
|
||
|
this->init();
|
||
|
}
|
||
|
|
||
|
wa::Components::WAC_DownloadManager::~WAC_DownloadManager()
|
||
|
{
|
||
|
disconnect( _connection_authentication_required );
|
||
|
}
|
||
|
|
||
|
|
||
|
DownloadToken wa::Components::WAC_DownloadManager::Download( const char *p_url, ifc_downloadManagerCallback *p_callback )
|
||
|
{
|
||
|
return DownloadEx( p_url, p_callback, api_downloadManager::DOWNLOADEX_TEMPFILE );
|
||
|
}
|
||
|
|
||
|
DownloadToken wa::Components::WAC_DownloadManager::DownloadEx( const char *p_url, ifc_downloadManagerCallback *p_callback, int p_flags )
|
||
|
{
|
||
|
|
||
|
|
||
|
|
||
|
return DownloadToken();
|
||
|
}
|
||
|
|
||
|
|
||
|
|
||
|
|
||
|
|
||
|
/**********************************************************************************
|
||
|
* PRIVATE *
|
||
|
**********************************************************************************/
|
||
|
void wa::Components::WAC_DownloadManager::init()
|
||
|
{
|
||
|
QString l_winamp_user_agent = QString( "%1 Winamp/%2" ).arg( QWebEngineProfile::defaultProfile()->httpUserAgent(), STR_WINAMP_PRODUCTVER ).replace( ",", "." );
|
||
|
|
||
|
QWebEngineProfile::defaultProfile()->setHttpUserAgent( l_winamp_user_agent );
|
||
|
|
||
|
|
||
|
_connection_authentication_required = connect( this, &QNetworkAccessManager::authenticationRequired, this, &wa::Components::WAC_DownloadManager::on_s_authentication_required );
|
||
|
|
||
|
|
||
|
}
|
||
|
|
||
|
|
||
|
QNetworkReply *wa::Components::WAC_DownloadManager::createRequest( Operation p_operation, const QNetworkRequest &p_request, QIODevice *p_outgoing_data )
|
||
|
{
|
||
|
return QNetworkAccessManager::createRequest( p_operation, p_request, p_outgoing_data );
|
||
|
}
|
||
|
|
||
|
|
||
|
|
||
|
/**********************************************************************************
|
||
|
* PRIVATE SLOTS *
|
||
|
**********************************************************************************/
|
||
|
void wa::Components::WAC_DownloadManager::on_s_authentication_required( QNetworkReply *p_reply, QAuthenticator *p_authenticator )
|
||
|
{
|
||
|
Q_UNUSED( p_reply );
|
||
|
Q_UNUSED( p_authenticator );
|
||
|
}
|
||
|
|
||
|
|
||
|
|
||
|
|
||
|
|
||
|
/**********************************************************************************
|
||
|
**********************************************************************************/
|
||
|
DownloadData::DownloadData( api_httpreceiver *p_http, const char *p_url, int p_flags, ifc_downloadManagerCallback *p_callback )
|
||
|
{
|
||
|
flags = p_flags;
|
||
|
http = p_http;
|
||
|
callback = p_callback;
|
||
|
|
||
|
if ( callback )
|
||
|
callback->AddRef();
|
||
|
|
||
|
hFile = INVALID_HANDLE_VALUE;
|
||
|
filepath[ 0 ] = 0;
|
||
|
fileext = 0;
|
||
|
|
||
|
int download_method = ( api_downloadManager::DOWNLOADEX_MASK_DOWNLOADMETHOD & flags );
|
||
|
switch ( download_method )
|
||
|
{
|
||
|
case api_downloadManager::DOWNLOADEX_TEMPFILE:
|
||
|
{
|
||
|
wchar_t temppath[ MAX_PATH - 14 ] = { 0 }; // MAX_PATH-14 'cause MSDN said so
|
||
|
GetTempPathW( MAX_PATH - 14, temppath );
|
||
|
GetTempFileNameW( temppath, L"wdl", 0, filepath );
|
||
|
hFile = CreateFileW( filepath, GENERIC_WRITE, FILE_SHARE_READ | FILE_SHARE_WRITE, 0, CREATE_ALWAYS, 0, 0 );
|
||
|
}
|
||
|
break;
|
||
|
case api_downloadManager::DOWNLOADEX_CALLBACK:
|
||
|
if ( callback )
|
||
|
callback->GetLocation( filepath, MAX_PATH );
|
||
|
break;
|
||
|
}
|
||
|
|
||
|
strcpy_s( this->url, 1024, p_url );
|
||
|
source[ 0 ] = 0;
|
||
|
title[ 0 ] = 0;
|
||
|
if ( flags & api_downloadManager::DOWNLOADEX_CALLBACK )
|
||
|
{
|
||
|
if ( callback )
|
||
|
{
|
||
|
callback->GetSource( source, 1024 );
|
||
|
callback->GetTitle( title, 1024 );
|
||
|
}
|
||
|
}
|
||
|
|
||
|
connectionStart = lastDownloadTick = GetTickCount();
|
||
|
last_status = HTTPRECEIVER_STATUS_ERROR;
|
||
|
pending = ( flags & api_downloadManager::DOWNLOADEX_PENDING ) > 0;
|
||
|
}
|
||
|
|
||
|
DownloadData::~DownloadData()
|
||
|
{
|
||
|
ServiceRelease( http, httpreceiverGUID );
|
||
|
|
||
|
http = NULL;
|
||
|
|
||
|
if ( fileext )
|
||
|
delete fileext;
|
||
|
|
||
|
int download_method = ( api_downloadManager::DOWNLOADEX_MASK_DOWNLOADMETHOD & flags );
|
||
|
if ( download_method == api_downloadManager::DOWNLOADEX_TEMPFILE && filepath[ 0 ] )
|
||
|
DeleteFileW( filepath );
|
||
|
|
||
|
if ( callback )
|
||
|
callback->Release();
|
||
|
|
||
|
callback = NULL;
|
||
|
}
|
||
|
|
||
|
|
||
|
void DownloadData::Retain()
|
||
|
{
|
||
|
this->_refCount.fetch_add( 1 );
|
||
|
}
|
||
|
|
||
|
void DownloadData::Release()
|
||
|
{
|
||
|
if ( this->_refCount.fetch_sub( 1 ) == 0 )
|
||
|
delete this;
|
||
|
}
|
||
|
|
||
|
|
||
|
void DownloadData::Close( ifc_downloadManagerCallback **callbackCopy )
|
||
|
{
|
||
|
if ( hFile != INVALID_HANDLE_VALUE )
|
||
|
CloseHandle( hFile );
|
||
|
|
||
|
hFile = INVALID_HANDLE_VALUE;
|
||
|
|
||
|
if ( callbackCopy != NULL )
|
||
|
{
|
||
|
*callbackCopy = callback;
|
||
|
if ( callback != NULL )
|
||
|
callback->AddRef();
|
||
|
}
|
||
|
else if ( callback != NULL )
|
||
|
{
|
||
|
callback->Release();
|
||
|
callback = NULL;
|
||
|
}
|
||
|
|
||
|
// don't want to close http here, because someone might still want to get the headers out of it
|
||
|
}
|
||
|
|
||
|
bool DownloadData::getExtention()
|
||
|
{
|
||
|
if ( fileext && *fileext )
|
||
|
return fileext;
|
||
|
|
||
|
char l_header_name_content_type[] = "Content-Type";
|
||
|
char *l_content_type = http->getheader( l_header_name_content_type );
|
||
|
if ( l_content_type && *l_content_type )
|
||
|
{
|
||
|
if ( _CONTENT_TYPES_EXTENSIONS.count( l_content_type ) == 1 )
|
||
|
fileext = _strdup( _CONTENT_TYPES_EXTENSIONS.find( l_content_type )->second.c_str() );
|
||
|
}
|
||
|
|
||
|
return fileext;
|
||
|
}
|
||
|
|
||
|
|
||
|
|
||
|
/**********************************************************************************
|
||
|
**********************************************************************************/
|
||
|
|
||
|
/**********************************************************************************
|
||
|
* PUBLIC *
|
||
|
**********************************************************************************/
|
||
|
DownloadManager::DownloadManager()
|
||
|
{
|
||
|
download_thread = NULL;
|
||
|
killswitch = CreateEvent( NULL, TRUE, FALSE, NULL );
|
||
|
|
||
|
InitializeCriticalSection( &downloadsCS );
|
||
|
}
|
||
|
|
||
|
|
||
|
void DownloadManager::Kill()
|
||
|
{
|
||
|
SetEvent( killswitch );
|
||
|
if ( download_thread )
|
||
|
{
|
||
|
WaitForSingleObject( download_thread, 3000 );
|
||
|
CloseHandle( download_thread );
|
||
|
}
|
||
|
|
||
|
DeleteCriticalSection( &downloadsCS );
|
||
|
CloseHandle( killswitch );
|
||
|
}
|
||
|
|
||
|
|
||
|
static void SetUserAgent( api_httpreceiver *p_http )
|
||
|
{
|
||
|
char agent[ 256 ] = { 0 };
|
||
|
StringCchPrintfA( agent, 256, "User-Agent: %S/%S", WASABI_API_APP->main_getAppName(), WASABI_API_APP->main_getVersionNumString() );
|
||
|
p_http->addheader( agent );
|
||
|
|
||
|
//QString l_winamp_user_agent = QString( "User-Agent: %1 Winamp/%2" ).arg( QWebEngineProfile::defaultProfile()->httpUserAgent(), STR_WINAMP_PRODUCTVER ).replace( ",", "." );
|
||
|
|
||
|
//http->addheader( l_winamp_user_agent.toStdString().c_str() );
|
||
|
}
|
||
|
|
||
|
|
||
|
DownloadToken DownloadManager::Download( const char *url, ifc_downloadManagerCallback *callback )
|
||
|
{
|
||
|
return DownloadEx( url, callback, api_downloadManager::DOWNLOADEX_TEMPFILE );
|
||
|
}
|
||
|
|
||
|
DownloadToken DownloadManager::DownloadEx( const char *url, ifc_downloadManagerCallback *callback, int flags )
|
||
|
{
|
||
|
if ( InitDownloadThread() )
|
||
|
{
|
||
|
api_httpreceiver *http = NULL;
|
||
|
|
||
|
ServiceBuild( http, httpreceiverGUID );
|
||
|
|
||
|
if ( http )
|
||
|
{
|
||
|
DownloadData *downloadData = new DownloadData( http, url, flags, callback );
|
||
|
|
||
|
int use_proxy = 1;
|
||
|
bool proxy80 = AGAVE_API_CONFIG->GetBool( internetConfigGroupGUID, L"proxy80", false );
|
||
|
|
||
|
if ( proxy80 && strstr( url, ":" ) && ( !strstr( url, ":80/" ) && strstr( url, ":80" ) != ( url + strlen( url ) - 3 ) ) )
|
||
|
use_proxy = 0;
|
||
|
|
||
|
const wchar_t *proxy = use_proxy ? AGAVE_API_CONFIG->GetString( internetConfigGroupGUID, L"proxy", 0 ) : 0;
|
||
|
|
||
|
http->open( API_DNS_AUTODNS, DOWNLOAD_BUFFER_SIZE, ( proxy && proxy[ 0 ] ) ? (const char *)AutoChar( proxy ) : NULL );
|
||
|
|
||
|
SetUserAgent( http );
|
||
|
|
||
|
if ( callback )
|
||
|
callback->OnInit( downloadData );
|
||
|
|
||
|
if ( downloadData->flags & api_downloadManager::DOWNLOADEX_UI )
|
||
|
{
|
||
|
for ( ifc_downloadManagerCallback *l_status : status_callbacks )
|
||
|
l_status->OnInit( downloadData );
|
||
|
}
|
||
|
|
||
|
//only call http->connect when it is not pending download request
|
||
|
if ( !( flags & DOWNLOADEX_PENDING ) )
|
||
|
http->connect( url, 1 );
|
||
|
|
||
|
//http->run(); // let's get this party started
|
||
|
EnterCriticalSection( &downloadsCS );
|
||
|
downloads.push_back( downloadData );
|
||
|
LeaveCriticalSection( &downloadsCS );
|
||
|
|
||
|
return downloadData;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
return 0;
|
||
|
}
|
||
|
|
||
|
|
||
|
void DownloadManager::ResumePendingDownload( DownloadToken p_token )
|
||
|
{
|
||
|
if ( !p_token )
|
||
|
return;
|
||
|
|
||
|
DownloadData *data = (DownloadData *)p_token;
|
||
|
if ( data->pending )
|
||
|
{
|
||
|
data->pending = false;
|
||
|
data->connectionStart = data->lastDownloadTick = GetTickCount();
|
||
|
|
||
|
data->http->connect( data->url );
|
||
|
}
|
||
|
}
|
||
|
|
||
|
void DownloadManager::CancelDownload( DownloadToken p_token )
|
||
|
{
|
||
|
if ( !p_token )
|
||
|
return;
|
||
|
|
||
|
DownloadData *data = (DownloadData *)p_token;
|
||
|
EnterCriticalSection( &downloadsCS );
|
||
|
|
||
|
if ( downloads.end() != std::find( downloads.begin(), downloads.end(), data ) )
|
||
|
{
|
||
|
ifc_downloadManagerCallback *callback;
|
||
|
data->Close( &callback );
|
||
|
|
||
|
//downloads.eraseObject(p_data);
|
||
|
auto it = std::find( downloads.begin(), downloads.end(), data );
|
||
|
if ( it != downloads.end() )
|
||
|
{
|
||
|
downloads.erase( it );
|
||
|
}
|
||
|
|
||
|
LeaveCriticalSection( &downloadsCS );
|
||
|
|
||
|
if ( callback )
|
||
|
{
|
||
|
callback->OnCancel( p_token );
|
||
|
if ( data->flags & api_downloadManager::DOWNLOADEX_UI )
|
||
|
{
|
||
|
for ( ifc_downloadManagerCallback *l_status : status_callbacks )
|
||
|
l_status->OnCancel( p_token );
|
||
|
}
|
||
|
|
||
|
callback->Release();
|
||
|
}
|
||
|
|
||
|
data->Release();
|
||
|
}
|
||
|
else
|
||
|
LeaveCriticalSection( &downloadsCS );
|
||
|
}
|
||
|
|
||
|
void DownloadManager::RetainDownload( DownloadToken p_token )
|
||
|
{
|
||
|
if ( !p_token )
|
||
|
return;
|
||
|
|
||
|
DownloadData *data = (DownloadData *)p_token;
|
||
|
if ( data )
|
||
|
data->Retain();
|
||
|
}
|
||
|
|
||
|
void DownloadManager::ReleaseDownload( DownloadToken p_token )
|
||
|
{
|
||
|
if ( !p_token )
|
||
|
return;
|
||
|
|
||
|
DownloadData *data = (DownloadData *)p_token;
|
||
|
if ( data )
|
||
|
data->Release();
|
||
|
}
|
||
|
|
||
|
|
||
|
void DownloadManager::RegisterStatusCallback( ifc_downloadManagerCallback *callback )
|
||
|
{
|
||
|
EnterCriticalSection( &downloadsCS );
|
||
|
status_callbacks.push_back( callback );
|
||
|
LeaveCriticalSection( &downloadsCS );
|
||
|
}
|
||
|
|
||
|
void DownloadManager::UnregisterStatusCallback( ifc_downloadManagerCallback *callback )
|
||
|
{
|
||
|
EnterCriticalSection( &downloadsCS );
|
||
|
|
||
|
auto it = std::find( status_callbacks.begin(), status_callbacks.end(), callback );
|
||
|
if ( it != status_callbacks.end() )
|
||
|
status_callbacks.erase( it );
|
||
|
|
||
|
LeaveCriticalSection( &downloadsCS );
|
||
|
}
|
||
|
|
||
|
|
||
|
bool DownloadManager::IsPending( DownloadToken token )
|
||
|
{
|
||
|
DownloadData *data = (DownloadData *)token;
|
||
|
if ( data )
|
||
|
return data->pending;
|
||
|
else
|
||
|
return FALSE;
|
||
|
}
|
||
|
|
||
|
|
||
|
/**********************************************************************************
|
||
|
* PRIVATE *
|
||
|
**********************************************************************************/
|
||
|
bool DownloadManager::DownloadThreadTick()
|
||
|
{
|
||
|
unsigned int i = 0;
|
||
|
char *downloadBuffer = (char *)malloc( DOWNLOAD_BUFFER_SIZE );
|
||
|
|
||
|
while ( WaitForSingleObject( killswitch, 0 ) != WAIT_OBJECT_0 )
|
||
|
{
|
||
|
EnterCriticalSection( &downloadsCS );
|
||
|
if ( downloads.empty() )
|
||
|
{
|
||
|
// TODO: might be nice to dynamically increase the sleep time if this happens
|
||
|
// (maybe to INFINITE and have Download() wake us?)
|
||
|
LeaveCriticalSection( &downloadsCS );
|
||
|
|
||
|
free( downloadBuffer );
|
||
|
|
||
|
return true;
|
||
|
}
|
||
|
|
||
|
if ( i >= downloads.size() )
|
||
|
{
|
||
|
LeaveCriticalSection( &downloadsCS );
|
||
|
free( downloadBuffer );
|
||
|
|
||
|
return true;
|
||
|
}
|
||
|
|
||
|
DownloadData *thisDownload = downloads[ i ];
|
||
|
if ( thisDownload->pending )
|
||
|
{
|
||
|
LeaveCriticalSection( &downloadsCS );
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
thisDownload->Retain();
|
||
|
LeaveCriticalSection( &downloadsCS );
|
||
|
|
||
|
|
||
|
INT tick = Tick( thisDownload, downloadBuffer, DOWNLOAD_BUFFER_SIZE );
|
||
|
switch ( tick )
|
||
|
{
|
||
|
case TICK_NODATA:
|
||
|
// do nothing
|
||
|
break;
|
||
|
|
||
|
case TICK_CONNECTING:
|
||
|
break;
|
||
|
|
||
|
case TICK_CONNECTED:
|
||
|
if ( thisDownload->callback )
|
||
|
thisDownload->callback->OnConnect( thisDownload );
|
||
|
if ( thisDownload->flags & api_downloadManager::DOWNLOADEX_UI )
|
||
|
{
|
||
|
for ( ifc_downloadManagerCallback *l_status : status_callbacks )
|
||
|
l_status->OnConnect( thisDownload );
|
||
|
}
|
||
|
break;
|
||
|
|
||
|
case TICK_SUCCESS:
|
||
|
if ( thisDownload->callback )
|
||
|
thisDownload->callback->OnTick( thisDownload );
|
||
|
if ( thisDownload->flags & api_downloadManager::DOWNLOADEX_UI )
|
||
|
{
|
||
|
for ( ifc_downloadManagerCallback *l_status : status_callbacks )
|
||
|
l_status->OnTick( thisDownload );
|
||
|
}
|
||
|
|
||
|
// TODO: send out update l_callback
|
||
|
break;
|
||
|
|
||
|
case TICK_FINISHED:
|
||
|
case TICK_FAILURE:
|
||
|
case TICK_TIMEOUT:
|
||
|
case TICK_CANT_CONNECT:
|
||
|
case TICK_WRITE_ERROR:
|
||
|
FinishDownload( thisDownload, tick );
|
||
|
break;
|
||
|
}
|
||
|
thisDownload->Release();
|
||
|
}
|
||
|
|
||
|
i++;
|
||
|
}
|
||
|
|
||
|
free( downloadBuffer );
|
||
|
|
||
|
return false; // we only get here when killswitch is set
|
||
|
}
|
||
|
|
||
|
int DownloadManager::DownloadTickThreadPoolFunc( HANDLE handle, void *user_data, intptr_t id )
|
||
|
{
|
||
|
DownloadManager *dlmgr = (DownloadManager *)user_data;
|
||
|
if ( dlmgr->DownloadThreadTick() )
|
||
|
{
|
||
|
TimerHandle t( handle );
|
||
|
t.Wait( DOWNLOAD_SLEEP_MS );
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
WASABI_API_THREADPOOL->RemoveHandle( 0, handle );
|
||
|
SetEvent( dlmgr->download_thread );
|
||
|
}
|
||
|
|
||
|
return 0;
|
||
|
}
|
||
|
|
||
|
bool DownloadManager::InitDownloadThread()
|
||
|
{
|
||
|
if ( download_thread == NULL )
|
||
|
{
|
||
|
download_thread = CreateEvent( 0, FALSE, FALSE, 0 );
|
||
|
TimerHandle t;
|
||
|
WASABI_API_THREADPOOL->AddHandle( 0, t, DownloadTickThreadPoolFunc, this, 0, api_threadpool::FLAG_LONG_EXECUTION );
|
||
|
t.Wait( DOWNLOAD_SLEEP_MS );
|
||
|
}
|
||
|
|
||
|
return ( download_thread != NULL );
|
||
|
}
|
||
|
|
||
|
void DownloadManager::FinishDownload( DownloadData *p_data, int code )
|
||
|
{
|
||
|
if ( p_data == NULL )
|
||
|
return;
|
||
|
|
||
|
ifc_downloadManagerCallback *l_callback = NULL;
|
||
|
|
||
|
EnterCriticalSection( &downloadsCS );
|
||
|
p_data->Close( &l_callback );
|
||
|
LeaveCriticalSection( &downloadsCS );
|
||
|
|
||
|
if ( l_callback != NULL )
|
||
|
{
|
||
|
if ( code == TICK_FINISHED )
|
||
|
{
|
||
|
l_callback->OnFinish( p_data );
|
||
|
if ( p_data->flags & api_downloadManager::DOWNLOADEX_UI )
|
||
|
{
|
||
|
for ( ifc_downloadManagerCallback *l_data : status_callbacks )
|
||
|
l_data->OnFinish( p_data );
|
||
|
}
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
l_callback->OnError( p_data, code );
|
||
|
if ( p_data->flags & api_downloadManager::DOWNLOADEX_UI )
|
||
|
{
|
||
|
for ( ifc_downloadManagerCallback *l_data : status_callbacks )
|
||
|
l_data->OnError( p_data, code );
|
||
|
}
|
||
|
}
|
||
|
|
||
|
l_callback->Release();
|
||
|
}
|
||
|
|
||
|
EnterCriticalSection( &downloadsCS );
|
||
|
|
||
|
auto it = std::find( downloads.begin(), downloads.end(), p_data );
|
||
|
if ( it != downloads.end() )
|
||
|
downloads.erase( it );
|
||
|
|
||
|
LeaveCriticalSection( &downloadsCS );
|
||
|
|
||
|
p_data->Release();
|
||
|
}
|
||
|
|
||
|
int DownloadManager::Tick( DownloadData *thisDownload, void *buffer, int bufferSize )
|
||
|
{
|
||
|
if ( !thisDownload )
|
||
|
return api_downloadManager::TICK_FAILURE;
|
||
|
|
||
|
int state = thisDownload->http->run();
|
||
|
if ( state == HTTPRECEIVER_RUN_ERROR || thisDownload == NULL )
|
||
|
return api_downloadManager::TICK_FAILURE;
|
||
|
|
||
|
if ( !thisDownload->fileext )
|
||
|
thisDownload->getExtention();
|
||
|
|
||
|
|
||
|
int downloaded = thisDownload->http->get_bytes( buffer, bufferSize );
|
||
|
if ( downloaded )
|
||
|
{
|
||
|
switch ( thisDownload->flags & DOWNLOADEX_MASK_DOWNLOADMETHOD )
|
||
|
{
|
||
|
case api_downloadManager::DOWNLOADEX_BUFFER:
|
||
|
{
|
||
|
thisDownload->buffer.reserve( thisDownload->http->content_length() );
|
||
|
thisDownload->buffer.add( buffer, downloaded );
|
||
|
}
|
||
|
break;
|
||
|
case api_downloadManager::DOWNLOADEX_TEMPFILE:
|
||
|
{
|
||
|
DWORD written = 0;
|
||
|
WriteFile( thisDownload->hFile, buffer, downloaded, &written, NULL );
|
||
|
if ( written != downloaded )
|
||
|
return api_downloadManager::TICK_WRITE_ERROR;
|
||
|
}
|
||
|
break;
|
||
|
case api_downloadManager::DOWNLOADEX_CALLBACK:
|
||
|
{
|
||
|
if ( thisDownload->flags & api_downloadManager::DOWNLOADEX_UI )
|
||
|
{
|
||
|
for ( ifc_downloadManagerCallback *l_status : status_callbacks )
|
||
|
l_status->OnData( thisDownload, buffer, downloaded );
|
||
|
}
|
||
|
|
||
|
if ( thisDownload->callback )
|
||
|
thisDownload->callback->OnData( thisDownload, buffer, downloaded );
|
||
|
}
|
||
|
}
|
||
|
|
||
|
thisDownload->lastDownloadTick = GetTickCount();
|
||
|
thisDownload->bytesDownloaded += downloaded;
|
||
|
|
||
|
return api_downloadManager::TICK_SUCCESS;
|
||
|
}
|
||
|
else // nothing in the buffer
|
||
|
{
|
||
|
if ( state == HTTPRECEIVER_RUN_CONNECTION_CLOSED ) // see if the connection is closed
|
||
|
{
|
||
|
return api_downloadManager::TICK_FINISHED; // yay we're done
|
||
|
}
|
||
|
|
||
|
if ( GetTickCount() - thisDownload->lastDownloadTick > DOWNLOAD_TIMEOUT_MS ) // check for timeout
|
||
|
return api_downloadManager::TICK_TIMEOUT;
|
||
|
|
||
|
switch ( thisDownload->http->get_status() )
|
||
|
{
|
||
|
case HTTPRECEIVER_STATUS_CONNECTING:
|
||
|
if ( thisDownload->last_status != HTTPRECEIVER_STATUS_CONNECTING )
|
||
|
{
|
||
|
thisDownload->last_status = HTTPRECEIVER_STATUS_CONNECTING;
|
||
|
return api_downloadManager::TICK_CONNECTING;
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
return api_downloadManager::TICK_NODATA;
|
||
|
}
|
||
|
case HTTPRECEIVER_STATUS_READING_HEADERS:
|
||
|
if ( thisDownload->last_status != HTTPRECEIVER_STATUS_READING_HEADERS )
|
||
|
{
|
||
|
thisDownload->last_status = HTTPRECEIVER_STATUS_READING_HEADERS;
|
||
|
return api_downloadManager::TICK_CONNECTED;
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
return api_downloadManager::TICK_NODATA;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
if ( !thisDownload->replyCode )
|
||
|
thisDownload->replyCode = thisDownload->http->getreplycode();
|
||
|
|
||
|
switch ( thisDownload->replyCode )
|
||
|
{
|
||
|
case 0:
|
||
|
case 100:
|
||
|
case 200:
|
||
|
case 201:
|
||
|
case 202:
|
||
|
case 203:
|
||
|
case 204:
|
||
|
case 205:
|
||
|
case 206:
|
||
|
return api_downloadManager::TICK_NODATA;
|
||
|
default:
|
||
|
return api_downloadManager::TICK_CANT_CONNECT;
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
|
||
|
|
||
|
|
||
|
|
||
|
#define CBCLASS DownloadManager
|
||
|
START_DISPATCH;
|
||
|
CB( API_DOWNLOADMANAGER_DOWNLOAD, Download )
|
||
|
CB( API_DOWNLOADMANAGER_DOWNLOADEX, DownloadEx )
|
||
|
CB( API_DOWNLOADMANAGER_GETRECEIVER, GetReceiver )
|
||
|
CB( API_DOWNLOADMANAGER_GETLOCATION, GetLocation )
|
||
|
VCB( API_DOWNLOADMANAGER_SETLOCATION, SetLocation )
|
||
|
CB( API_DOWNLOADMANAGER_GETEXTENTION, GetExtention )
|
||
|
CB( API_DOWNLOADMANAGER_GETURL, GetUrl )
|
||
|
CB( API_DOWNLOADMANAGER_GETBYTESDOWNLOADED, GetBytesDownloaded );
|
||
|
CB( API_DOWNLOADMANAGER_GETBUFFER, GetBuffer );
|
||
|
VCB( API_DOWNLOADMANAGER_RESUMEPENDINGDOWNLOAD, ResumePendingDownload );
|
||
|
VCB( API_DOWNLOADMANAGER_CANCELDOWNLOAD, CancelDownload );
|
||
|
VCB( API_DOWNLOADMANAGER_RETAINDOWNLOAD, RetainDownload );
|
||
|
VCB( API_DOWNLOADMANAGER_RELEASEDOWNLOAD, ReleaseDownload );
|
||
|
VCB( API_DOWNLOADMANAGER_REGISTERSTATUSCALLBACK, RegisterStatusCallback );
|
||
|
VCB( API_DOWNLOADMANAGER_UNREGISTERSTATUSCALLBACK, UnregisterStatusCallback );
|
||
|
CB( API_DOWNLOADMANAGER_GETSOURCE, GetSource );
|
||
|
CB( API_DOWNLOADMANAGER_GETTITLE, GetTitle );
|
||
|
CB( API_DOWNLOADMANAGER_ISPENDING, IsPending );
|
||
|
END_DISPATCH;
|
||
|
#undef CBCLASS
|