Initial community commit
This commit is contained in:
parent
537bcbc862
commit
fc06254474
16440 changed files with 4239995 additions and 2 deletions
222
Src/Plugins/Library/ml_wire/DownloadThread.cpp
Normal file
222
Src/Plugins/Library/ml_wire/DownloadThread.cpp
Normal file
|
@ -0,0 +1,222 @@
|
|||
#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;
|
||||
}
|
Loading…
Add table
Add a link
Reference in a new issue