222 lines
4.9 KiB
C++
222 lines
4.9 KiB
C++
#include "Main.h"
|
|
|
|
#pragma warning(disable:4786)
|
|
|
|
#include "DownloadThread.h"
|
|
#include "api__ml_wire.h"
|
|
#include "api/service/waServiceFactory.h"
|
|
#include "../../..\Components\wac_network\wac_network_http_receiver_api.h"
|
|
#include "errors.h"
|
|
#include <strsafe.h>
|
|
|
|
extern int winampVersion;
|
|
|
|
#define USER_AGENT_SIZE (10 /*User-Agent*/ + 2 /*: */ + 6 /*Winamp*/ + 1 /*/*/ + 1 /*5*/ + 3/*.21*/ + 1 /*Null*/)
|
|
void SetUserAgent( api_httpreceiver *http )
|
|
{
|
|
char user_agent[ USER_AGENT_SIZE ] = { 0 };
|
|
int bigVer = ( ( winampVersion & 0x0000FF00 ) >> 12 );
|
|
int smallVer = ( ( winampVersion & 0x000000FF ) );
|
|
StringCchPrintfA( user_agent, USER_AGENT_SIZE, "User-Agent: Winamp/%01x.%02x", bigVer, smallVer );
|
|
http->addheader( user_agent );
|
|
}
|
|
|
|
#define HTTP_BUFFER_SIZE 32768
|
|
|
|
static int FeedXMLHTTP( api_httpreceiver *http, obj_xml *parser, bool *noData )
|
|
{
|
|
char downloadedData[ HTTP_BUFFER_SIZE ] = { 0 };
|
|
int xmlResult = API_XML_SUCCESS;
|
|
int downloadSize = http->get_bytes( downloadedData, HTTP_BUFFER_SIZE );
|
|
if ( downloadSize )
|
|
{
|
|
xmlResult = parser->xmlreader_feed( (void *)downloadedData, downloadSize );
|
|
*noData = false;
|
|
}
|
|
else
|
|
*noData = true;
|
|
|
|
return xmlResult;
|
|
}
|
|
|
|
|
|
DownloadThread::DownloadThread() : parser( 0 ), parserFactory( 0 )
|
|
{
|
|
parserFactory = plugin.service->service_getServiceByGuid( obj_xmlGUID );
|
|
if ( parserFactory )
|
|
parser = (obj_xml *)parserFactory->getInterface();
|
|
|
|
if ( parser )
|
|
{
|
|
parser->xmlreader_setCaseSensitive();
|
|
parser->xmlreader_registerCallback( L"*", &xmlDOM );
|
|
parser->xmlreader_open();
|
|
}
|
|
}
|
|
|
|
DownloadThread::~DownloadThread()
|
|
{
|
|
if ( parser )
|
|
{
|
|
parser->xmlreader_unregisterCallback( &xmlDOM );
|
|
parser->xmlreader_close();
|
|
}
|
|
|
|
if ( parserFactory && parser )
|
|
parserFactory->releaseInterface( parser );
|
|
|
|
parserFactory = 0;
|
|
parser = 0;
|
|
}
|
|
|
|
void URLToFileName( wchar_t *url )
|
|
{
|
|
while ( url && *url != 0 )
|
|
{
|
|
switch ( *url )
|
|
{
|
|
case ':':
|
|
case '/':
|
|
case '\\':
|
|
case '*':
|
|
case '?':
|
|
case '"':
|
|
case '<':
|
|
case '>':
|
|
case '|':
|
|
*url = '_';
|
|
}
|
|
url++;
|
|
}
|
|
}
|
|
|
|
#define FILE_BUFFER_SIZE 32768
|
|
void DownloadThread::DownloadFile( const wchar_t *fileName )
|
|
{
|
|
if ( !parser )
|
|
return; // no sense in continuing if there's no parser available
|
|
|
|
HANDLE file = CreateFile( fileName, GENERIC_READ, FILE_SHARE_READ, NULL, OPEN_EXISTING, FILE_FLAG_SEQUENTIAL_SCAN, NULL );
|
|
if ( file == INVALID_HANDLE_VALUE )
|
|
return;
|
|
|
|
while ( true )
|
|
{
|
|
char data[ FILE_BUFFER_SIZE ] = { 0 };
|
|
DWORD bytesRead = 0;
|
|
if ( ReadFile( file, data, FILE_BUFFER_SIZE, &bytesRead, NULL ) && bytesRead )
|
|
{
|
|
parser->xmlreader_feed( (void *)data, bytesRead );
|
|
}
|
|
else
|
|
break;
|
|
}
|
|
|
|
CloseHandle( file );
|
|
parser->xmlreader_feed( 0, 0 );
|
|
ReadNodes( fileName );
|
|
}
|
|
|
|
#ifdef _DEBUG
|
|
#include <iostream>
|
|
void ShowAllHeaders( api_httpreceiver *http )
|
|
{
|
|
std::cout << "--------------------" << std::endl;
|
|
const char *blah = http->getallheaders();
|
|
while ( blah && *blah )
|
|
{
|
|
std::cout << blah << std::endl;
|
|
blah += strlen( blah ) + 1;
|
|
}
|
|
}
|
|
#else
|
|
#define ShowAllHeaders(x)
|
|
#endif
|
|
|
|
int RunXMLDownload( api_httpreceiver *http, obj_xml *parser )
|
|
{
|
|
int ret;
|
|
bool noData;
|
|
do
|
|
{
|
|
Sleep( 50 );
|
|
ret = http->run();
|
|
if ( FeedXMLHTTP( http, parser, &noData ) != API_XML_SUCCESS )
|
|
return DOWNLOAD_ERROR_PARSING_XML;
|
|
} while ( ret == HTTPRECEIVER_RUN_OK );
|
|
|
|
// finish off the data
|
|
do
|
|
{
|
|
if ( FeedXMLHTTP( http, parser, &noData ) != API_XML_SUCCESS )
|
|
return DOWNLOAD_ERROR_PARSING_XML;
|
|
} while ( !noData );
|
|
|
|
parser->xmlreader_feed( 0, 0 );
|
|
if ( ret != HTTPRECEIVER_RUN_ERROR )
|
|
return DOWNLOAD_SUCCESS;
|
|
else
|
|
return DOWNLOAD_CONNECTIONRESET;
|
|
}
|
|
|
|
|
|
int DownloadThread::DownloadURL( const wchar_t *url )
|
|
{
|
|
if ( !parser )
|
|
return DOWNLOAD_NOPARSER; // no sense in continuing if there's no parser available
|
|
|
|
api_httpreceiver *http = 0;
|
|
waServiceFactory *sf = plugin.service->service_getServiceByGuid( httpreceiverGUID );
|
|
if ( sf )
|
|
http = (api_httpreceiver *)sf->getInterface();
|
|
|
|
if ( !http )
|
|
return DOWNLOAD_NOHTTP;
|
|
|
|
http->AllowCompression();
|
|
http->open( API_DNS_AUTODNS, HTTP_BUFFER_SIZE, mediaLibrary.GetProxy() );
|
|
|
|
SetUserAgent( http );
|
|
|
|
http->connect( AutoChar( url ) );
|
|
int ret;
|
|
|
|
do
|
|
{
|
|
Sleep( 50 );
|
|
ret = http->run();
|
|
if ( ret == -1 ) // connection failed
|
|
break;
|
|
|
|
// ---- check our reply code ----
|
|
int status = http->get_status();
|
|
switch ( status )
|
|
{
|
|
case HTTPRECEIVER_STATUS_CONNECTING:
|
|
case HTTPRECEIVER_STATUS_READING_HEADERS:
|
|
break;
|
|
|
|
case HTTPRECEIVER_STATUS_READING_CONTENT:
|
|
{
|
|
ShowAllHeaders( http ); // benski> don't cut, only enabled in debug mode
|
|
int downloadError;
|
|
downloadError = RunXMLDownload( http, parser );
|
|
if ( downloadError == DOWNLOAD_SUCCESS )
|
|
ReadNodes( url );
|
|
sf->releaseInterface( http );
|
|
return downloadError;
|
|
}
|
|
break;
|
|
case HTTPRECEIVER_STATUS_ERROR:
|
|
default:
|
|
sf->releaseInterface( http );
|
|
return DOWNLOAD_404;
|
|
}
|
|
} while ( ret == HTTPRECEIVER_RUN_OK );
|
|
|
|
const char *er = http->geterrorstr();
|
|
|
|
sf->releaseInterface( http );
|
|
|
|
return DOWNLOAD_404;
|
|
} |