source: trunk/libtransmission/webseed.c @ 7078

Last change on this file since 7078 was 7078, checked in by charles, 12 years ago

(libT) #1442: removing a webseed torrent while it's running causes it to crash

  • Property svn:keywords set to Date Rev Author Id
File size: 9.0 KB
Line 
1/*
2 * This file Copyright (C) 2008 Charles Kerr <charles@rebelbase.com>
3 *
4 * This file is licensed by the GPL version 2.  Works owned by the
5 * Transmission project are granted a special exemption to clause 2(b)
6 * so that the bulk of its code can remain under the MIT license.
7 * This exemption does not extend to derived works not owned by
8 * the Transmission project.
9 *
10 * $Id: webseed.c 7078 2008-11-08 22:24:07Z charles $
11 */
12
13#include <assert.h>
14#include <string.h> /* strlen */
15
16#include <event.h>
17
18#include "transmission.h"
19#include "inout.h"
20#include "list.h"
21#include "ratecontrol.h"
22#include "torrent.h"
23#include "utils.h"
24#include "web.h"
25#include "webseed.h"
26
27struct tr_webseed
28{
29    unsigned int        busy : 1;
30    unsigned int        dead : 1;
31
32    uint8_t             hash[SHA_DIGEST_LENGTH];
33
34    char              * url;
35
36    tr_delivery_func  * callback;
37    void *              callback_userdata;
38
39    tr_piece_index_t    pieceIndex;
40    uint32_t            pieceOffset;
41    uint32_t            byteCount;
42
43    tr_ratecontrol    * rateDown;
44
45    tr_session        * session;
46
47    struct evbuffer   * content;
48};
49
50/***
51****
52***/
53
54static const tr_peer_event blankEvent = { 0, 0, 0, 0, 0.0f, 0 };
55
56static void
57publish( tr_webseed *    w,
58         tr_peer_event * e )
59{
60    if( w->callback )
61        w->callback( NULL, e, w->callback_userdata );
62}
63
64static void
65fireNeedReq( tr_webseed * w )
66{
67    tr_peer_event e = blankEvent;
68
69    e.eventType = TR_PEER_NEED_REQ;
70    publish( w, &e );
71}
72
73static void
74fireClientGotBlock( tr_webseed * w,
75                    uint32_t     pieceIndex,
76                    uint32_t     offset,
77                    uint32_t     length )
78{
79    tr_peer_event e = blankEvent;
80
81    e.eventType = TR_PEER_CLIENT_GOT_BLOCK;
82    e.pieceIndex = pieceIndex;
83    e.offset = offset;
84    e.length = length;
85    publish( w, &e );
86}
87
88static void
89fireClientGotData( tr_webseed * w,
90                   uint32_t     length )
91{
92    tr_peer_event e = blankEvent;
93
94    e.eventType = TR_PEER_CLIENT_GOT_DATA;
95    e.length = length;
96    publish( w, &e );
97}
98
99/***
100****
101***/
102
103static char*
104makeURL( tr_webseed *    w,
105         const tr_file * file )
106{
107    char *            ret;
108    struct evbuffer * out = evbuffer_new( );
109    const char *      url = w->url;
110    const size_t      url_len = strlen( url );
111
112    evbuffer_add( out, url, url_len );
113
114    /* if url ends with a '/', add the torrent name */
115    if( url[url_len - 1] == '/' )
116    {
117        const char * str = file->name;
118
119        /* this is like curl_escape() but doesn't munge the
120         * '/' directory separators in the path */
121        while( str && *str )
122        {
123            switch( *str )
124            {
125                case ',':
126                case '-':
127                case '.':
128                case '/':
129                case '0':
130                case '1':
131                case '2':
132                case '3':
133                case '4':
134                case '5':
135                case '6':
136                case '7':
137                case '8':
138                case '9':
139                case 'a':
140                case 'b':
141                case 'c':
142                case 'd':
143                case 'e':
144                case 'f':
145                case 'g':
146                case 'h':
147                case 'i':
148                case 'j':
149                case 'k':
150                case 'l':
151                case 'm':
152                case 'n':
153                case 'o':
154                case 'p':
155                case 'q':
156                case 'r':
157                case 's':
158                case 't':
159                case 'u':
160                case 'v':
161                case 'w':
162                case 'x':
163                case 'y':
164                case 'z':
165                case 'A':
166                case 'B':
167                case 'C':
168                case 'D':
169                case 'E':
170                case 'F':
171                case 'G':
172                case 'H':
173                case 'I':
174                case 'J':
175                case 'K':
176                case 'L':
177                case 'M':
178                case 'N':
179                case 'O':
180                case 'P':
181                case 'Q':
182                case 'R':
183                case 'S':
184                case 'T':
185                case 'U':
186                case 'V':
187                case 'W':
188                case 'X':
189                case 'Y':
190                case 'Z':
191                    evbuffer_add( out, str, 1 );
192                    break;
193
194                default:
195                    evbuffer_add_printf( out, "%%%02X", *str );
196                    break;
197            }
198            str++;
199        }
200    }
201
202    ret = tr_strndup( EVBUFFER_DATA( out ), EVBUFFER_LENGTH( out ) );
203    evbuffer_free( out );
204    return ret;
205}
206
207static void requestNextChunk( tr_webseed * w );
208
209static void
210webResponseFunc( tr_handle     * session,
211                 long            response_code,
212                 const void    * response,
213                 size_t          response_byte_count,
214                 void          * vw )
215{
216    tr_webseed * w = vw;
217    tr_torrent * tor = tr_torrentFindFromHash( session, w->hash );
218    const int    success = ( response_code == 206 );
219
220/*fprintf( stderr, "server responded with code %ld and %lu bytes\n",
221  response_code, (unsigned long)response_byte_count );*/
222    if( !success )
223    {
224        /* FIXME */
225    }
226    else if( tor != NULL )
227    {
228        evbuffer_add( w->content, response, response_byte_count );
229        if( !w->dead )
230        {
231            fireClientGotData( w, response_byte_count );
232            tr_rcTransferred( w->rateDown, response_byte_count );
233        }
234
235        if( EVBUFFER_LENGTH( w->content ) < w->byteCount )
236            requestNextChunk( w );
237        else {
238            tr_ioWrite( tor, w->pieceIndex, w->pieceOffset, w->byteCount, EVBUFFER_DATA(w->content) );
239            evbuffer_drain( w->content, EVBUFFER_LENGTH( w->content ) );
240            w->busy = 0;
241            if( w->dead )
242                tr_webseedFree( w );
243            else  {
244                fireClientGotBlock( w, w->pieceIndex, w->pieceOffset, w->byteCount );
245                fireNeedReq( w );
246            }
247        }
248    }
249}
250
251static void
252requestNextChunk( tr_webseed * w )
253{
254    tr_torrent * tor = tr_torrentFindFromHash( w->session, w->hash );
255
256    if( tor != NULL )
257    {
258        const tr_info * inf = tr_torrentInfo( tor );
259        const uint32_t have = EVBUFFER_LENGTH( w->content );
260        const uint32_t left = w->byteCount - have;
261        const uint32_t pieceOffset = w->pieceOffset + have;
262        tr_file_index_t fileIndex;
263        uint64_t fileOffset;
264        uint32_t thisPass;
265        char * url;
266        char * range;
267
268        tr_ioFindFileLocation( tor, w->pieceIndex, pieceOffset,
269                               &fileIndex, &fileOffset );
270        thisPass = MIN( left, inf->files[fileIndex].length - fileOffset );
271
272        url = makeURL( w, &inf->files[fileIndex] );
273/*fprintf( stderr, "url is [%s]\n", url );*/
274        range = tr_strdup_printf( "%"PRIu64"-%"PRIu64, fileOffset, fileOffset + thisPass - 1 );
275/*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 );*/
276        tr_webRun( w->session, url, range, webResponseFunc, w );
277        tr_free( range );
278        tr_free( url );
279    }
280}
281
282tr_addreq_t
283tr_webseedAddRequest( tr_webseed  * w,
284                      uint32_t      pieceIndex,
285                      uint32_t      pieceOffset,
286                      uint32_t      byteCount )
287{
288    int ret;
289
290    if( w->busy || w->dead )
291    {
292        ret = TR_ADDREQ_FULL;
293    }
294    else
295    {
296        w->busy = 1;
297        w->pieceIndex = pieceIndex;
298        w->pieceOffset = pieceOffset;
299        w->byteCount = byteCount;
300        evbuffer_drain( w->content, EVBUFFER_LENGTH( w->content ) );
301        requestNextChunk( w );
302        ret = TR_ADDREQ_OK;
303    }
304
305    return ret;
306}
307
308int
309tr_webseedIsActive( const tr_webseed * w )
310{
311    return w->busy != 0;
312}
313
314int
315tr_webseedGetSpeed( const tr_webseed * w,
316                    float *            setme_KiBs )
317{
318    const int isActive = tr_webseedIsActive( w );
319
320    *setme_KiBs = isActive ? tr_rcRate( w->rateDown ) : 0.0f;
321    return isActive;
322}
323
324/***
325****
326***/
327
328tr_webseed*
329tr_webseedNew( struct tr_torrent * torrent,
330               const char *        url,
331               tr_delivery_func    callback,
332               void *              callback_userdata )
333{
334    tr_webseed * w = tr_new0( tr_webseed, 1 );
335
336    memcpy( w->hash, torrent->info.hash, SHA_DIGEST_LENGTH );
337    w->session = torrent->session;
338    w->content = evbuffer_new( );
339    w->rateDown = tr_rcInit( );
340    w->url = tr_strdup( url );
341    w->callback = callback;
342    w->callback_userdata = callback_userdata;
343/*fprintf( stderr, "w->callback_userdata is %p\n", w->callback_userdata );*/
344    return w;
345}
346
347void
348tr_webseedFree( tr_webseed * w )
349{
350    if( w )
351    {
352        if( w->busy )
353        {
354            w->dead = 1;
355        }
356        else
357        {
358            evbuffer_free( w->content );
359            tr_rcClose( w->rateDown );
360            tr_free( w->url );
361            tr_free( w );
362        }
363    }
364}
Note: See TracBrowser for help on using the repository browser.