/* * This file Copyright (C) 2008-2010 Mnemosyne LLC * * This file is licensed by the GPL version 2. Works owned by the * Transmission project are granted a special exemption to clause 2(b) * so that the bulk of its code can remain under the MIT license. * This exemption does not extend to derived works not owned by * the Transmission project. * * $Id$ */ #include /* strlen */ #include #include "transmission.h" #include "inout.h" #include "ratecontrol.h" #include "torrent.h" #include "utils.h" #include "web.h" #include "webseed.h" struct tr_webseed { tr_bool busy; tr_bool dead; uint8_t hash[SHA_DIGEST_LENGTH]; char * url; tr_peer_callback * callback; void * callback_data; tr_piece_index_t pieceIndex; uint32_t pieceOffset; uint32_t byteCount; tr_ratecontrol rateDown; tr_session * session; struct evbuffer * content; }; /*** **** ***/ static const tr_peer_event blankEvent = { 0, 0, 0, 0, 0.0f, 0, 0, 0 }; static void publish( tr_webseed * w, tr_peer_event * e ) { if( w->callback != NULL ) w->callback( NULL, e, w->callback_data ); } static void fireNeedReq( tr_webseed * w ) { #if 0 tr_peer_event e = blankEvent; e.eventType = TR_PEER_NEED_REQ; publish( w, &e ); #endif } static void fireClientGotBlock( tr_webseed * w, uint32_t pieceIndex, uint32_t offset, uint32_t length ) { tr_peer_event e = blankEvent; e.eventType = TR_PEER_CLIENT_GOT_BLOCK; e.pieceIndex = pieceIndex; e.offset = offset; e.length = length; publish( w, &e ); } static void fireClientGotData( tr_webseed * w, uint32_t length ) { tr_peer_event e = blankEvent; e.eventType = TR_PEER_CLIENT_GOT_DATA; e.length = length; e.wasPieceData = TRUE; publish( w, &e ); } /*** **** ***/ static char* makeURL( tr_webseed * w, const tr_file * file ) { char * ret; struct evbuffer * out = evbuffer_new( ); const char * url = w->url; const size_t url_len = strlen( url ); evbuffer_add( out, url, url_len ); /* if url ends with a '/', add the torrent name */ if( url[url_len - 1] == '/' && file->name ) tr_http_escape( out, file->name, strlen(file->name), FALSE ); ret = tr_strndup( EVBUFFER_DATA( out ), EVBUFFER_LENGTH( out ) ); evbuffer_free( out ); return ret; } static void requestNextChunk( tr_webseed * w ); static void webResponseFunc( tr_session * session, long response_code, const void * response, size_t response_byte_count, void * vw ) { tr_webseed * w = vw; tr_torrent * tor = tr_torrentFindFromHash( session, w->hash ); const int success = ( response_code == 206 ); /*fprintf( stderr, "server responded with code %ld and %lu bytes\n", response_code, (unsigned long)response_byte_count );*/ if( !success ) { /* FIXME */ } else if( tor != NULL ) { evbuffer_add( w->content, response, response_byte_count ); if( !w->dead ) { fireClientGotData( w, response_byte_count ); tr_rcTransferred( &w->rateDown, response_byte_count ); } if( EVBUFFER_LENGTH( w->content ) < w->byteCount ) requestNextChunk( w ); else { tr_ioWrite( tor, w->pieceIndex, w->pieceOffset, w->byteCount, EVBUFFER_DATA(w->content) ); evbuffer_drain( w->content, EVBUFFER_LENGTH( w->content ) ); w->busy = 0; if( w->dead ) tr_webseedFree( w ); else { fireClientGotBlock( w, w->pieceIndex, w->pieceOffset, w->byteCount ); fireNeedReq( w ); } } } } static void requestNextChunk( tr_webseed * w ) { tr_torrent * tor = tr_torrentFindFromHash( w->session, w->hash ); if( tor != NULL ) { const tr_info * inf = tr_torrentInfo( tor ); const uint32_t have = EVBUFFER_LENGTH( w->content ); const uint32_t left = w->byteCount - have; const uint32_t pieceOffset = w->pieceOffset + have; tr_file_index_t fileIndex; uint64_t fileOffset; uint32_t thisPass; char * url; char * range; tr_ioFindFileLocation( tor, w->pieceIndex, pieceOffset, &fileIndex, &fileOffset ); thisPass = MIN( left, inf->files[fileIndex].length - fileOffset ); url = makeURL( w, &inf->files[fileIndex] ); /*fprintf( stderr, "url is [%s]\n", url );*/ range = tr_strdup_printf( "%"PRIu64"-%"PRIu64, fileOffset, fileOffset + thisPass - 1 ); /*fprintf( stderr, "range is [%s] ... we want %lu total, we have %lu, so %lu are left, and we're asking for %lu this time\n", range, (unsigned long)w->byteCount, (unsigned long)have, (unsigned long)left, (unsigned long)thisPass );*/ tr_webRun( w->session, url, range, webResponseFunc, w ); tr_free( range ); tr_free( url ); } } tr_addreq_t tr_webseedAddRequest( tr_webseed * w, uint32_t pieceIndex, uint32_t pieceOffset, uint32_t byteCount ) { int ret; if( w->busy || w->dead ) { ret = TR_ADDREQ_FULL; } else { w->busy = 1; w->pieceIndex = pieceIndex; w->pieceOffset = pieceOffset; w->byteCount = byteCount; evbuffer_drain( w->content, EVBUFFER_LENGTH( w->content ) ); requestNextChunk( w ); ret = TR_ADDREQ_OK; } return ret; } int tr_webseedIsActive( const tr_webseed * w ) { return w->busy != 0; } int tr_webseedGetSpeed_Bps( const tr_webseed * w, uint64_t now, int * setme_Bps ) { const int isActive = tr_webseedIsActive( w ); *setme_Bps = isActive ? tr_rcRate_Bps( &w->rateDown, now ) : 0; return isActive; } /*** **** ***/ tr_webseed* tr_webseedNew( struct tr_torrent * torrent, const char * url, tr_peer_callback * callback, void * callback_data ) { tr_webseed * w = tr_new0( tr_webseed, 1 ); memcpy( w->hash, torrent->info.hash, SHA_DIGEST_LENGTH ); w->session = torrent->session; w->content = evbuffer_new( ); w->url = tr_strdup( url ); w->callback = callback; w->callback_data = callback_data; tr_rcConstruct( &w->rateDown ); return w; } void tr_webseedFree( tr_webseed * w ) { if( w ) { if( w->busy ) { w->dead = 1; } else { evbuffer_free( w->content ); tr_rcDestruct( &w->rateDown ); tr_free( w->url ); tr_free( w ); } } }