2007-01-26 16:26:49 +00:00
/* This software was written by Dirk Engling <erdgeist@erdgeist.org>
2007-01-05 00:00:42 +00:00
It is considered beerware . Prost . Skol . Cheers or whatever .
Some of the stuff below is stolen from Fefes example libowfat httpd .
*/
2007-11-06 17:50:41 +00:00
/* System */
2006-12-06 18:36:14 +00:00
# include <string.h>
2006-12-05 12:56:56 +00:00
# include <sys/types.h>
# include <sys/stat.h>
2006-12-16 16:14:34 +00:00
# include <sys/socket.h>
2006-12-15 23:29:38 +00:00
# include <arpa/inet.h>
2006-12-05 12:56:56 +00:00
# include <unistd.h>
# include <stdlib.h>
# include <errno.h>
2006-12-08 22:37:44 +00:00
# include <signal.h>
# include <stdio.h>
2007-04-02 17:26:40 +00:00
# include <pwd.h>
2006-12-05 12:56:56 +00:00
2007-11-06 17:50:41 +00:00
/* Libowfat */
# include "socket.h"
# include "io.h"
# include "iob.h"
# include "buffer.h"
# include "array.h"
# include "byte.h"
# include "case.h"
# include "fmt.h"
# include "str.h"
# include "scan.h"
# include "ip4.h"
/* Opentracker */
2006-12-08 20:28:17 +00:00
# include "trackerlogic.h"
# include "scan_urlencoded_query.h"
2007-11-06 11:58:32 +00:00
# include "ot_stats.h"
# include "ot_sync.h"
2007-11-06 17:50:41 +00:00
# include "ot_udp.h"
2007-11-12 01:41:57 +00:00
# include "ot_fullscrape.h"
# include "ot_iovec.h"
2007-11-12 04:39:53 +00:00
# include "ot_accesslist.h"
2007-11-14 13:06:34 +00:00
# include "ot_mutex.h"
2007-11-19 21:10:53 +00:00
# include "ot_clean.h"
2006-12-08 20:28:17 +00:00
2007-01-26 16:26:49 +00:00
/* Globals */
2007-01-11 01:06:10 +00:00
static const size_t SUCCESS_HTTP_HEADER_LENGTH = 80 ;
static const size_t SUCCESS_HTTP_SIZE_OFF = 17 ;
2007-11-01 20:13:03 +00:00
static uint32_t g_adminip_addresses [ OT_ADMINIP_MAX ] ;
static unsigned int g_adminip_count = 0 ;
2007-11-19 21:10:53 +00:00
static time_t ot_last_clean_time ;
2007-11-06 17:50:41 +00:00
time_t ot_start_time ;
2007-10-13 17:40:37 +00:00
time_t g_now ;
2007-08-18 09:56:22 +00:00
2007-10-29 17:22:05 +00:00
# ifndef WANT_TRACKER_SYNC
# define add_peer_to_torrent(A,B,C) add_peer_to_torrent(A,B)
# endif
2007-10-16 18:23:36 +00:00
# ifndef NO_FULLSCRAPE_LOGGING
2007-10-15 18:01:38 +00:00
# define LOG_TO_STDERR( ... ) fprintf( stderr, __VA_ARGS__ )
2007-10-16 18:23:36 +00:00
# else
# define LOG_TO_STDERR( ... )
# endif
2007-10-15 18:01:38 +00:00
2007-01-11 01:06:10 +00:00
/* To always have space for error messages ;) */
2007-11-21 01:53:17 +00:00
static char static_inbuf [ 8192 ] ;
static char static_outbuf [ 8192 ] ;
2006-12-15 22:07:33 +00:00
2007-10-19 21:56:59 +00:00
# define OT_MAXMULTISCRAPE_COUNT 64
static ot_hash multiscrape_buf [ OT_MAXMULTISCRAPE_COUNT ] ;
2007-03-07 22:19:00 +00:00
static char * FLAG_TCP = " TCP " ;
static char * FLAG_UDP = " UDP " ;
static size_t ot_sockets_count = 0 ;
2007-03-05 21:14:50 +00:00
2007-01-24 20:13:30 +00:00
# ifdef _DEBUG_HTTPERROR
static char debug_request [ 8192 ] ;
2007-11-02 13:13:03 +00:00
# define _DEBUG_HTTPERROR_PARAM( param ) , param
# else
# define _DEBUG_HTTPERROR_PARAM( param )
2007-01-24 20:13:30 +00:00
# endif
2007-01-19 17:50:36 +00:00
2007-10-19 01:26:33 +00:00
typedef enum {
2007-11-14 13:06:34 +00:00
STRUCT_HTTP_FLAG_ARRAY_USED = 1 ,
STRUCT_HTTP_FLAG_IOB_USED = 2 ,
STRUCT_HTTP_FLAG_WAITINGFORTASK = 4
2007-10-19 01:26:33 +00:00
} STRUCT_HTTP_FLAG ;
2007-01-26 16:26:49 +00:00
struct http_data {
union {
2007-10-19 01:26:33 +00:00
array request ;
io_batch batch ;
2007-01-26 16:26:49 +00:00
} ;
2007-10-19 01:26:33 +00:00
unsigned char ip [ 4 ] ;
STRUCT_HTTP_FLAG flag ;
2007-01-26 16:26:49 +00:00
} ;
2007-11-01 20:13:03 +00:00
# define NOTBLESSED( h ) (!bsearch( &h->ip, g_adminip_addresses, g_adminip_count, 4, ot_ip_compare ))
static int ot_ip_compare ( const void * a , const void * b ) { return memcmp ( a , b , 4 ) ; }
2007-01-26 16:26:49 +00:00
/* Prototypes */
int main ( int argc , char * * argv ) ;
2007-01-29 02:02:03 +00:00
static void httperror ( const int64 s , const char * title , const char * message ) ;
2007-11-02 13:13:03 +00:00
static void httpresponse ( const int64 s , char * data _DEBUG_HTTPERROR_PARAM ( size_t l ) ) ;
2007-01-26 16:26:49 +00:00
2007-11-12 01:41:57 +00:00
static void sendiovecdata ( const int64 s , int iovec_entries , struct iovec * iovector ) ;
2007-01-29 02:02:03 +00:00
static void senddata ( const int64 s , char * buffer , const size_t size ) ;
2007-01-26 16:26:49 +00:00
2007-03-05 21:14:50 +00:00
static void server_mainloop ( ) ;
2007-01-26 16:26:49 +00:00
static void handle_timeouted ( void ) ;
static void handle_accept ( const int64 serversocket ) ;
static void handle_read ( const int64 clientsocket ) ;
static void handle_write ( const int64 clientsocket ) ;
2007-03-06 19:43:47 +00:00
2007-03-07 22:19:00 +00:00
static void ot_try_bind ( char ip [ 4 ] , uint16 port , int is_tcp ) ;
2007-01-26 16:26:49 +00:00
static void usage ( char * name ) ;
static void help ( char * name ) ;
static void carp ( const char * routine ) ;
static void panic ( const char * routine ) ;
2007-10-13 17:40:37 +00:00
static void signal_handler ( int s ) ;
2007-01-26 16:26:49 +00:00
2007-01-29 02:02:03 +00:00
# define HTTPERROR_400 return httperror( s, "400 Invalid Request", "This server only understands GET." )
# define HTTPERROR_400_PARAM return httperror( s, "400 Invalid Request", "Invalid parameter" )
# define HTTPERROR_400_COMPACT return httperror( s, "400 Invalid Request", "This server only delivers compact results." )
2007-03-27 16:09:03 +00:00
# define HTTPERROR_403_IP return httperror( s, "403 Access Denied", "Your ip address is not allowed to administrate this server." )
2007-01-29 02:02:03 +00:00
# define HTTPERROR_404 return httperror( s, "404 Not Found", "No such file or directory." )
# define HTTPERROR_500 return httperror( s, "500 Internal Server Error", "A server error has occured. Please retry later." )
2007-01-26 16:26:49 +00:00
/* End of prototypes */
static void carp ( const char * routine ) {
2007-01-24 22:23:18 +00:00
buffer_puts ( buffer_2 , routine ) ;
buffer_puts ( buffer_2 , " : " ) ;
buffer_puterror ( buffer_2 ) ;
buffer_putnlflush ( buffer_2 ) ;
2006-12-05 12:56:56 +00:00
}
2007-01-26 16:26:49 +00:00
static void panic ( const char * routine ) {
2007-01-24 22:23:18 +00:00
carp ( routine ) ;
exit ( 111 ) ;
2006-12-05 12:56:56 +00:00
}
2007-01-29 02:02:03 +00:00
static void httperror ( const int64 s , const char * title , const char * message ) {
size_t reply_size = sprintf ( static_outbuf , " HTTP/1.0 %s \r \n Content-Type: text/html \r \n Connection: close \r \n Content-Length: %zd \r \n \r \n <title>%s</title> \n " ,
2007-01-26 16:26:49 +00:00
title , strlen ( message ) + strlen ( title ) + 16 - 4 , title + 4 ) ;
# ifdef _DEBUG_HTTPERROR
fprintf ( stderr , " DEBUG: invalid request was: %s \n " , debug_request ) ;
# endif
2007-01-29 02:02:03 +00:00
senddata ( s , static_outbuf , reply_size ) ;
2007-01-26 16:26:49 +00:00
}
2007-11-12 01:41:57 +00:00
static void sendiovecdata ( const int64 s , int iovec_entries , struct iovec * iovector ) {
struct http_data * h = io_getcookie ( s ) ;
char * header ;
int i ;
size_t header_size , size = iovec_length ( & iovec_entries , & iovector ) ;
tai6464 t ;
2007-11-16 00:23:42 +00:00
/* No cookie? Bad socket. Leave. */
2007-11-12 01:41:57 +00:00
if ( ! h ) {
iovec_free ( & iovec_entries , & iovector ) ;
2007-11-16 00:23:42 +00:00
HTTPERROR_500 ;
2007-11-12 01:41:57 +00:00
}
2007-11-16 00:23:42 +00:00
/* If this socket collected request in a buffer,
free it now */
2007-11-12 01:41:57 +00:00
if ( h - > flag & STRUCT_HTTP_FLAG_ARRAY_USED ) {
h - > flag & = ~ STRUCT_HTTP_FLAG_ARRAY_USED ;
array_reset ( & h - > request ) ;
}
2007-11-16 00:23:42 +00:00
/* If we came here, wait for the answer is over */
h - > flag & = ~ STRUCT_HTTP_FLAG_WAITINGFORTASK ;
/* Our answers never are 0 bytes. Return an error. */
if ( ! iovec_entries | | ! iovector [ 0 ] . iov_len ) {
iovec_free ( & iovec_entries , & iovector ) ;
HTTPERROR_500 ;
}
/* Prepare space for http header */
2007-11-12 01:41:57 +00:00
header = malloc ( SUCCESS_HTTP_HEADER_LENGTH ) ;
if ( ! header ) {
iovec_free ( & iovec_entries , & iovector ) ;
HTTPERROR_500 ;
}
header_size = sprintf ( header , " HTTP/1.0 200 OK \r \n Content-Type: text/plain \r \n Content-Length: %zd \r \n \r \n " , size ) ;
iob_reset ( & h - > batch ) ;
iob_addbuf_free ( & h - > batch , header , header_size ) ;
2007-11-16 00:23:42 +00:00
2007-11-12 01:41:57 +00:00
/* Will move to ot_iovec.c */
for ( i = 0 ; i < iovec_entries ; + + i )
iob_addbuf_munmap ( & h - > batch , iovector [ i ] . iov_base , iovector [ i ] . iov_len ) ;
free ( iovector ) ;
h - > flag | = STRUCT_HTTP_FLAG_IOB_USED ;
/* writeable sockets timeout after twice the pool timeout
which defaults to 5 minutes ( e . g . after 10 minutes ) */
taia_now ( & t ) ; taia_addsec ( & t , & t , OT_CLIENT_TIMEOUT_SEND ) ;
io_timeout ( s , t ) ;
io_dontwantread ( s ) ;
io_wantwrite ( s ) ;
}
2007-01-29 02:02:03 +00:00
static void senddata ( const int64 s , char * buffer , size_t size ) {
struct http_data * h = io_getcookie ( s ) ;
2007-01-31 09:50:46 +00:00
ssize_t written_size ;
2007-01-08 00:34:37 +00:00
2007-01-24 22:23:18 +00:00
/* whoever sends data is not interested in its input-array */
2007-10-19 01:26:33 +00:00
if ( h & & ( h - > flag & STRUCT_HTTP_FLAG_ARRAY_USED ) ) {
h - > flag & = ~ STRUCT_HTTP_FLAG_ARRAY_USED ;
2007-01-24 22:23:18 +00:00
array_reset ( & h - > request ) ;
2007-10-19 01:26:33 +00:00
}
2007-01-24 22:23:18 +00:00
2007-01-09 06:30:37 +00:00
written_size = write ( s , buffer , size ) ;
2007-04-03 11:08:17 +00:00
if ( ( written_size < 0 ) | | ( ( size_t ) written_size = = size ) ) {
2007-01-24 22:23:18 +00:00
free ( h ) ; io_close ( s ) ;
2007-01-08 00:34:37 +00:00
} else {
2007-10-19 01:26:33 +00:00
char * outbuf ;
2007-01-20 01:50:28 +00:00
tai6464 t ;
2007-10-19 01:26:33 +00:00
if ( ! h ) return ;
if ( ! ( outbuf = malloc ( size - written_size ) ) ) {
2007-01-20 01:50:28 +00:00
free ( h ) ; io_close ( s ) ;
return ;
}
iob_reset ( & h - > batch ) ;
memmove ( outbuf , buffer + written_size , size - written_size ) ;
iob_addbuf_free ( & h - > batch , outbuf , size - written_size ) ;
2007-10-19 01:26:33 +00:00
h - > flag | = STRUCT_HTTP_FLAG_IOB_USED ;
2007-01-20 01:50:28 +00:00
2007-10-19 02:00:53 +00:00
/* writeable short data sockets just have a tcp timeout */
taia_uint ( & t , 0 ) ; io_timeout ( s , t ) ;
2007-01-20 11:13:30 +00:00
io_dontwantread ( s ) ;
io_wantwrite ( s ) ;
2007-01-08 00:34:37 +00:00
}
2006-12-05 12:56:56 +00:00
}
2007-11-02 13:13:03 +00:00
static void httpresponse ( const int64 s , char * data _DEBUG_HTTPERROR_PARAM ( size_t l ) ) {
2007-03-27 16:09:03 +00:00
struct http_data * h = io_getcookie ( s ) ;
2007-11-12 01:41:57 +00:00
char * c ;
2007-01-11 01:06:10 +00:00
ot_peer peer ;
ot_torrent * torrent ;
ot_hash * hash = NULL ;
2007-11-06 01:28:40 +00:00
int numwant , tmp , scanon , mode ;
2007-11-18 16:47:37 +00:00
ot_tasktype format = TASK_FULLSCRAPE ;
2007-01-11 01:06:10 +00:00
unsigned short port = htons ( 6881 ) ;
2007-01-31 09:50:46 +00:00
ssize_t len ;
2007-01-26 18:09:14 +00:00
size_t reply_size = 0 , reply_off ;
2007-01-11 01:06:10 +00:00
2007-01-24 20:13:30 +00:00
# ifdef _DEBUG_HTTPERROR
2007-10-21 03:37:26 +00:00
if ( l > = sizeof ( debug_request ) )
l = sizeof ( debug_request ) - 1 ;
memcpy ( debug_request , data , l ) ;
debug_request [ l ] = 0 ;
2007-01-24 20:13:30 +00:00
# endif
2007-01-29 02:02:03 +00:00
/* This one implicitely tests strlen < 5, too -- remember, it is \n terminated */
if ( byte_diff ( data , 5 , " GET / " ) ) HTTPERROR_400 ;
2006-12-08 20:50:06 +00:00
2007-01-29 02:02:03 +00:00
/* Query string MUST terminate with SP -- we know that theres at least a '\n' where this search terminates */
for ( c = data + 5 ; * c ! = ' ' & & * c ! = ' \t ' & & * c ! = ' \n ' & & * c ! = ' \r ' ; + + c ) ;
if ( * c ! = ' ' ) HTTPERROR_400 ;
2007-01-11 01:06:10 +00:00
2007-01-29 02:02:03 +00:00
/* Skip leading '/' */
for ( c = data + 4 ; * c = = ' / ' ; + + c ) ;
2007-01-11 01:06:10 +00:00
switch ( scan_urlencoded_query ( & c , data = c , SCAN_PATH ) ) {
2007-10-27 14:06:07 +00:00
# ifdef WANT_TRACKER_SYNC
2007-03-27 16:09:03 +00:00
/******************************
* S Y N C *
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2007-01-26 16:26:49 +00:00
case 4 : /* sync ? */
2007-10-21 05:36:10 +00:00
if ( * data = = ' a ' ) goto ANNOUNCE_WORKAROUND ;
2007-10-21 04:55:19 +00:00
if ( ! byte_diff ( data , 2 , " sc " ) ) goto SCRAPE_WORKAROUND ;
2007-01-26 16:26:49 +00:00
if ( byte_diff ( data , 4 , " sync " ) ) HTTPERROR_404 ;
2007-10-19 01:26:33 +00:00
if ( NOTBLESSED ( h ) ) HTTPERROR_403_IP ;
2007-03-27 16:09:03 +00:00
2007-10-15 18:01:38 +00:00
LOG_TO_STDERR ( " sync: %d.%d.%d.%d \n " , h - > ip [ 0 ] , h - > ip [ 1 ] , h - > ip [ 2 ] , h - > ip [ 3 ] ) ;
2007-03-27 16:09:03 +00:00
mode = SYNC_OUT ;
scanon = 1 ;
2007-03-27 12:07:29 +00:00
2007-03-27 16:09:03 +00:00
while ( scanon ) {
switch ( scan_urlencoded_query ( & c , data = c , SCAN_SEARCHPATH_PARAM ) ) {
case - 2 : scanon = 0 ; break ; /* TERMINATOR */
case - 1 : HTTPERROR_400_PARAM ; /* PARSE ERROR */
2007-10-18 23:33:07 +00:00
default : scan_urlencoded_skipvalue ( & c ) ; break ;
2007-03-27 16:09:03 +00:00
case 9 :
if ( byte_diff ( data , 9 , " changeset " ) ) {
2007-10-18 23:33:07 +00:00
scan_urlencoded_skipvalue ( & c ) ;
2007-03-27 16:09:03 +00:00
continue ;
}
/* ignore this, when we dont at least see "d4:syncdee" */
if ( ( len = scan_urlencoded_query ( & c , data = c , SCAN_SEARCHPATH_VALUE ) ) < 10 ) HTTPERROR_400_PARAM ;
if ( add_changeset_to_tracker ( ( ot_byte * ) data , len ) ) HTTPERROR_400_PARAM ;
mode = SYNC_IN ;
break ;
}
}
if ( mode = = SYNC_OUT ) {
2007-11-21 01:53:17 +00:00
/* Pass this task to the worker thread */
h - > flag | = STRUCT_HTTP_FLAG_WAITINGFORTASK ;
sync_deliver ( s ) ;
io_dontwantread ( s ) ;
return ;
2007-03-27 16:09:03 +00:00
}
/* Simple but proof for now */
2007-11-12 04:39:53 +00:00
memmove ( static_outbuf + SUCCESS_HTTP_HEADER_LENGTH , " OK " , 2 ) ;
2007-03-27 16:09:03 +00:00
reply_size = 2 ;
break ;
2007-10-27 14:06:07 +00:00
# endif
2007-03-27 16:09:03 +00:00
/******************************
* S T A T S *
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2007-01-24 22:23:18 +00:00
case 5 : /* stats ? */
2007-10-21 05:36:10 +00:00
if ( * data = = ' a ' ) goto ANNOUNCE_WORKAROUND ;
2007-10-21 04:55:19 +00:00
if ( ! byte_diff ( data , 2 , " sc " ) ) goto SCRAPE_WORKAROUND ;
2007-01-26 16:26:49 +00:00
if ( byte_diff ( data , 5 , " stats " ) ) HTTPERROR_404 ;
2007-01-17 11:51:55 +00:00
scanon = 1 ;
2007-11-18 16:47:37 +00:00
mode = TASK_STATS_PEERS ;
2007-01-17 11:51:55 +00:00
while ( scanon ) {
switch ( scan_urlencoded_query ( & c , data = c , SCAN_SEARCHPATH_PARAM ) ) {
2007-01-26 16:26:49 +00:00
case - 2 : scanon = 0 ; break ; /* TERMINATOR */
case - 1 : HTTPERROR_400_PARAM ; /* PARSE ERROR */
2007-10-18 23:33:07 +00:00
default : scan_urlencoded_skipvalue ( & c ) ; break ;
2007-01-17 11:51:55 +00:00
case 4 :
2007-01-24 22:23:18 +00:00
if ( byte_diff ( data , 4 , " mode " ) ) {
2007-10-18 23:33:07 +00:00
scan_urlencoded_skipvalue ( & c ) ;
2007-01-17 11:51:55 +00:00
continue ;
}
2007-01-31 09:50:46 +00:00
if ( scan_urlencoded_query ( & c , data = c , SCAN_SEARCHPATH_VALUE ) ! = 4 ) HTTPERROR_400_PARAM ;
2007-11-06 01:28:40 +00:00
if ( ! byte_diff ( data , 4 , " peer " ) )
2007-11-18 16:47:37 +00:00
mode = TASK_STATS_PEERS ;
2007-11-06 01:28:40 +00:00
else if ( ! byte_diff ( data , 4 , " conn " ) )
2007-11-18 16:47:37 +00:00
mode = TASK_STATS_CONNS ;
2007-01-17 11:51:55 +00:00
else if ( ! byte_diff ( data , 4 , " top5 " ) )
2007-11-18 16:47:37 +00:00
mode = TASK_STATS_TOP5 ;
2007-10-31 15:39:41 +00:00
else if ( ! byte_diff ( data , 4 , " fscr " ) )
2007-11-18 16:47:37 +00:00
mode = TASK_STATS_FULLSCRAPE ;
2007-03-15 23:14:14 +00:00
else if ( ! byte_diff ( data , 4 , " tcp4 " ) )
2007-11-18 16:47:37 +00:00
mode = TASK_STATS_TCP ;
2007-03-15 23:14:14 +00:00
else if ( ! byte_diff ( data , 4 , " udp4 " ) )
2007-11-18 16:47:37 +00:00
mode = TASK_STATS_UDP ;
2007-07-22 16:17:26 +00:00
else if ( ! byte_diff ( data , 4 , " s24s " ) )
2007-11-18 16:47:37 +00:00
mode = TASK_STATS_SLASH24S ;
else if ( ! byte_diff ( data , 4 , " tpbs " ) )
mode = TASK_STATS_TPB ;
2007-01-17 11:51:55 +00:00
else
2007-01-26 16:26:49 +00:00
HTTPERROR_400_PARAM ;
2007-11-18 16:47:37 +00:00
break ;
case 6 :
if ( byte_diff ( data , 6 , " format " ) ) {
scan_urlencoded_skipvalue ( & c ) ;
continue ;
}
if ( scan_urlencoded_query ( & c , data = c , SCAN_SEARCHPATH_VALUE ) ! = 3 ) HTTPERROR_400_PARAM ;
if ( ! byte_diff ( data , 3 , " bin " ) )
format = TASK_FULLSCRAPE_TPB_BINARY ;
else if ( ! byte_diff ( data , 3 , " ben " ) )
format = TASK_FULLSCRAPE ;
else if ( ! byte_diff ( data , 3 , " url " ) )
format = TASK_FULLSCRAPE_TPB_URLENCODED ;
else if ( ! byte_diff ( data , 3 , " txt " ) )
format = TASK_FULLSCRAPE_TPB_ASCII ;
else
HTTPERROR_400_PARAM ;
break ;
2007-01-17 11:51:55 +00:00
}
}
2007-11-18 16:47:37 +00:00
if ( mode = = TASK_STATS_TPB ) {
/* Pass this task to the worker thread */
h - > flag | = STRUCT_HTTP_FLAG_WAITINGFORTASK ;
fullscrape_deliver ( s , format ) ;
io_dontwantread ( s ) ;
return ;
}
2007-11-06 18:02:03 +00:00
// default format for now
if ( ! ( reply_size = return_stats_for_tracker ( static_outbuf + SUCCESS_HTTP_HEADER_LENGTH , mode , 0 ) ) ) HTTPERROR_500 ;
2007-01-16 02:59:39 +00:00
break ;
2007-03-27 16:09:03 +00:00
/******************************
* S C R A P E *
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2007-01-11 01:06:10 +00:00
case 6 : /* scrape ? */
2007-10-21 05:36:10 +00:00
if ( * data = = ' a ' ) goto ANNOUNCE_WORKAROUND ;
2007-01-26 16:26:49 +00:00
if ( byte_diff ( data , 6 , " scrape " ) ) HTTPERROR_404 ;
2007-01-11 01:06:10 +00:00
2007-10-19 22:13:59 +00:00
/* Full scrape... you might want to limit that */
if ( ! byte_diff ( data , 12 , " scrape HTTP/ " ) ) {
2007-10-19 22:20:42 +00:00
LOG_TO_STDERR ( " [%08d] scrp: %d.%d.%d.%d - FULL SCRAPE \n " , ( unsigned int ) ( g_now - ot_start_time ) , h - > ip [ 0 ] , h - > ip [ 1 ] , h - > ip [ 2 ] , h - > ip [ 3 ] ) ;
2007-10-19 22:36:28 +00:00
# ifdef _DEBUG_HTTPERROR
write ( 2 , debug_request , l ) ;
# endif
2007-11-16 00:23:42 +00:00
/* Pass this task to the worker thread */
h - > flag | = STRUCT_HTTP_FLAG_WAITINGFORTASK ;
2007-11-18 16:47:37 +00:00
fullscrape_deliver ( s , TASK_FULLSCRAPE ) ;
2007-11-16 00:23:42 +00:00
io_dontwantread ( s ) ;
return ;
2007-10-19 22:13:59 +00:00
}
2007-01-26 18:09:14 +00:00
SCRAPE_WORKAROUND :
2007-10-21 05:15:35 +00:00
/* This is to hack around stupid clients that send "scrape ?info_hash" */
2007-10-21 04:13:53 +00:00
if ( c [ - 1 ] ! = ' ? ' ) {
while ( ( * c ! = ' ? ' ) & & ( * c ! = ' \n ' ) ) + + c ;
if ( * c = = ' \n ' ) HTTPERROR_400_PARAM ;
2007-10-21 05:36:10 +00:00
+ + c ;
2007-10-21 04:13:53 +00:00
}
2007-01-26 18:09:14 +00:00
scanon = 1 ;
2007-11-06 01:28:40 +00:00
numwant = 0 ;
2007-01-11 01:06:10 +00:00
while ( scanon ) {
switch ( scan_urlencoded_query ( & c , data = c , SCAN_SEARCHPATH_PARAM ) ) {
2007-01-26 16:26:49 +00:00
case - 2 : scanon = 0 ; break ; /* TERMINATOR */
2007-10-21 05:15:35 +00:00
case - 1 :
2007-11-06 01:28:40 +00:00
if ( numwant )
2007-10-21 05:15:35 +00:00
goto UTORRENT1600_WORKAROUND ;
HTTPERROR_400_PARAM ; /* PARSE ERROR */
2007-10-18 23:33:07 +00:00
default : scan_urlencoded_skipvalue ( & c ) ; break ;
2007-01-11 01:06:10 +00:00
case 9 :
if ( byte_diff ( data , 9 , " info_hash " ) ) {
2007-10-18 23:33:07 +00:00
scan_urlencoded_skipvalue ( & c ) ;
2007-01-11 01:06:10 +00:00
continue ;
}
/* ignore this, when we have less than 20 bytes */
2007-10-23 00:01:10 +00:00
if ( scan_urlencoded_query ( & c , data = c , SCAN_SEARCHPATH_VALUE ) ! = ( ssize_t ) sizeof ( ot_hash ) ) {
# ifdef WANT_UTORRENT1600_WORKAROUND
if ( data [ 20 ] ! = ' ? ' )
# endif
HTTPERROR_400_PARAM ;
}
2007-11-06 01:28:40 +00:00
if ( numwant < OT_MAXMULTISCRAPE_COUNT )
memmove ( multiscrape_buf + numwant + + , data , sizeof ( ot_hash ) ) ;
2007-01-11 01:06:10 +00:00
break ;
2006-12-14 02:44:50 +00:00
}
2007-01-11 01:06:10 +00:00
}
2006-12-14 02:44:50 +00:00
2007-10-23 00:01:10 +00:00
UTORRENT1600_WORKAROUND :
2007-10-19 22:13:59 +00:00
/* No info_hash found? Inform user */
2007-11-06 01:28:40 +00:00
if ( ! numwant ) HTTPERROR_400_PARAM ;
2007-01-26 16:26:49 +00:00
/* Enough for http header + whole scrape string */
2007-11-06 01:28:40 +00:00
if ( ! ( reply_size = return_tcp_scrape_for_torrent ( multiscrape_buf , numwant , SUCCESS_HTTP_HEADER_LENGTH + static_outbuf ) ) ) HTTPERROR_500 ;
2007-11-06 17:50:41 +00:00
stats_issue_event ( EVENT_SCRAPE , 1 , reply_size ) ;
2007-01-11 01:06:10 +00:00
break ;
2007-03-27 16:09:03 +00:00
/******************************
* A N N O U N C E *
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2007-01-27 16:06:13 +00:00
case 8 :
2007-10-21 04:55:19 +00:00
if ( ! byte_diff ( data , 2 , " sc " ) ) goto SCRAPE_WORKAROUND ;
2007-10-21 05:36:10 +00:00
if ( * data ! = ' a ' ) HTTPERROR_404 ;
2007-01-11 01:06:10 +00:00
2007-01-26 18:09:14 +00:00
ANNOUNCE_WORKAROUND :
2007-10-19 23:15:13 +00:00
/* This is to hack around stupid clients that send "announce ?info_hash" */
2007-10-21 04:13:53 +00:00
if ( c [ - 1 ] ! = ' ? ' ) {
while ( ( * c ! = ' ? ' ) & & ( * c ! = ' \n ' ) ) + + c ;
if ( * c = = ' \n ' ) HTTPERROR_400_PARAM ;
2007-10-21 05:36:10 +00:00
+ + c ;
2007-10-21 04:13:53 +00:00
}
2007-10-19 23:15:13 +00:00
2007-01-31 09:50:46 +00:00
OT_SETIP ( & peer , ( ( struct http_data * ) io_getcookie ( s ) ) - > ip ) ;
2007-01-11 01:06:10 +00:00
OT_SETPORT ( & peer , & port ) ;
OT_FLAG ( & peer ) = 0 ;
numwant = 50 ;
scanon = 1 ;
while ( scanon ) {
switch ( scan_urlencoded_query ( & c , data = c , SCAN_SEARCHPATH_PARAM ) ) {
2007-01-26 16:26:49 +00:00
case - 2 : scanon = 0 ; break ; /* TERMINATOR */
case - 1 : HTTPERROR_400_PARAM ; /* PARSE ERROR */
2007-10-18 23:33:07 +00:00
default : scan_urlencoded_skipvalue ( & c ) ; break ;
2007-01-03 05:11:48 +00:00
# ifdef WANT_IP_FROM_QUERY_STRING
2007-01-11 01:06:10 +00:00
case 2 :
if ( ! byte_diff ( data , 2 , " ip " ) ) {
unsigned char ip [ 4 ] ;
2007-01-31 09:50:46 +00:00
len = scan_urlencoded_query ( & c , data = c , SCAN_SEARCHPATH_VALUE ) ;
2007-01-26 16:26:49 +00:00
if ( ( len < = 0 ) | | scan_fixed_ip ( data , len , ip ) ) HTTPERROR_400_PARAM ;
2007-01-24 22:23:18 +00:00
OT_SETIP ( & peer , ip ) ;
2007-01-11 01:06:10 +00:00
} else
2007-10-18 23:33:07 +00:00
scan_urlencoded_skipvalue ( & c ) ;
2007-01-11 01:06:10 +00:00
break ;
2007-01-03 05:11:48 +00:00
# endif
2007-01-11 01:06:10 +00:00
case 4 :
2007-01-31 09:50:46 +00:00
if ( ! byte_diff ( data , 4 , " port " ) ) {
len = scan_urlencoded_query ( & c , data = c , SCAN_SEARCHPATH_VALUE ) ;
2007-01-26 16:26:49 +00:00
if ( ( len < = 0 ) | | scan_fixed_int ( data , len , & tmp ) | | ( tmp > 0xffff ) ) HTTPERROR_400_PARAM ;
2007-01-24 22:23:18 +00:00
port = htons ( tmp ) ; OT_SETPORT ( & peer , & port ) ;
2007-01-31 09:50:46 +00:00
} else if ( ! byte_diff ( data , 4 , " left " ) ) {
if ( ( len = scan_urlencoded_query ( & c , data = c , SCAN_SEARCHPATH_VALUE ) ) < = 0 ) HTTPERROR_400_PARAM ;
2007-01-24 20:48:25 +00:00
if ( scan_fixed_int ( data , len , & tmp ) ) tmp = 0 ;
2007-01-11 01:06:10 +00:00
if ( ! tmp ) OT_FLAG ( & peer ) | = PEER_FLAG_SEEDING ;
} else
2007-10-18 23:33:07 +00:00
scan_urlencoded_skipvalue ( & c ) ;
2007-01-11 01:06:10 +00:00
break ;
case 5 :
2007-01-31 09:50:46 +00:00
if ( byte_diff ( data , 5 , " event " ) )
2007-10-18 23:33:07 +00:00
scan_urlencoded_skipvalue ( & c ) ;
2007-01-11 01:06:10 +00:00
else switch ( scan_urlencoded_query ( & c , data = c , SCAN_SEARCHPATH_VALUE ) ) {
case - 1 :
2007-01-26 16:26:49 +00:00
HTTPERROR_400_PARAM ;
2006-12-08 21:00:56 +00:00
case 7 :
2007-01-31 09:50:46 +00:00
if ( ! byte_diff ( data , 7 , " stopped " ) ) OT_FLAG ( & peer ) | = PEER_FLAG_STOPPED ;
2006-12-08 21:36:26 +00:00
break ;
case 9 :
2007-01-31 09:50:46 +00:00
if ( ! byte_diff ( data , 9 , " completed " ) ) OT_FLAG ( & peer ) | = PEER_FLAG_COMPLETED ;
2007-01-11 01:06:10 +00:00
default : /* Fall through intended */
2007-01-08 00:34:37 +00:00
break ;
2007-01-11 01:06:10 +00:00
}
break ;
case 7 :
if ( ! byte_diff ( data , 7 , " numwant " ) ) {
2007-01-31 09:50:46 +00:00
len = scan_urlencoded_query ( & c , data = c , SCAN_SEARCHPATH_VALUE ) ;
2007-01-26 16:26:49 +00:00
if ( ( len < = 0 ) | | scan_fixed_int ( data , len , & numwant ) ) HTTPERROR_400_PARAM ;
2007-10-23 00:39:21 +00:00
if ( numwant < 0 ) numwant = 50 ;
2007-01-11 01:06:10 +00:00
if ( numwant > 200 ) numwant = 200 ;
} else if ( ! byte_diff ( data , 7 , " compact " ) ) {
2007-01-31 09:50:46 +00:00
len = scan_urlencoded_query ( & c , data = c , SCAN_SEARCHPATH_VALUE ) ;
2007-01-26 16:26:49 +00:00
if ( ( len < = 0 ) | | scan_fixed_int ( data , len , & tmp ) ) HTTPERROR_400_PARAM ;
if ( ! tmp ) HTTPERROR_400_COMPACT ;
2007-01-11 01:06:10 +00:00
} else
2007-10-18 23:33:07 +00:00
scan_urlencoded_skipvalue ( & c ) ;
2007-01-11 01:06:10 +00:00
break ;
case 9 :
if ( byte_diff ( data , 9 , " info_hash " ) ) {
2007-10-18 23:33:07 +00:00
scan_urlencoded_skipvalue ( & c ) ;
2007-01-11 01:06:10 +00:00
continue ;
2006-12-08 20:07:26 +00:00
}
2007-01-11 01:06:10 +00:00
/* ignore this, when we have less than 20 bytes */
2007-01-26 16:26:49 +00:00
if ( scan_urlencoded_query ( & c , data = c , SCAN_SEARCHPATH_VALUE ) ! = 20 ) HTTPERROR_400_PARAM ;
2007-01-11 01:06:10 +00:00
hash = ( ot_hash * ) data ;
break ;
2006-12-08 20:50:06 +00:00
}
2007-01-11 01:06:10 +00:00
}
2006-12-08 21:36:26 +00:00
2007-01-31 09:58:32 +00:00
/* Scanned whole query string */
if ( ! hash ) {
reply_size = sprintf ( static_outbuf + SUCCESS_HTTP_HEADER_LENGTH , " d14:failure reason81:Your client forgot to send your torrent's info_hash. Please upgrade your client.e " ) ;
break ;
}
2007-09-10 02:43:11 +00:00
if ( OT_FLAG ( & peer ) & PEER_FLAG_STOPPED )
reply_size = remove_peer_from_torrent ( hash , & peer , SUCCESS_HTTP_HEADER_LENGTH + static_outbuf , 1 ) ;
else {
2007-03-27 16:09:03 +00:00
torrent = add_peer_to_torrent ( hash , & peer , 0 ) ;
2007-11-06 01:28:40 +00:00
if ( ! torrent | | ! ( reply_size = return_peers_for_torrent ( hash , numwant , SUCCESS_HTTP_HEADER_LENGTH + static_outbuf , 1 ) ) ) HTTPERROR_500 ;
2006-12-08 20:07:26 +00:00
}
2007-11-06 17:50:41 +00:00
stats_issue_event ( EVENT_ANNOUNCE , 1 , reply_size ) ;
2007-01-11 01:06:10 +00:00
break ;
2007-10-21 04:55:19 +00:00
default :
2007-10-23 00:30:46 +00:00
if ( ( * data = = ' a ' ) | | ( * data = = ' ? ' ) ) goto ANNOUNCE_WORKAROUND ;
2007-10-21 04:55:19 +00:00
if ( ! byte_diff ( data , 2 , " sc " ) ) goto SCRAPE_WORKAROUND ;
2007-01-26 16:26:49 +00:00
HTTPERROR_404 ;
2007-01-11 01:06:10 +00:00
}
2007-01-31 09:50:46 +00:00
if ( ! reply_size ) HTTPERROR_500 ;
2007-01-11 01:06:10 +00:00
2007-01-26 16:26:49 +00:00
/* This one is rather ugly, so I take you step by step through it.
2007-01-11 01:06:10 +00:00
2007-01-26 16:26:49 +00:00
1. In order to avoid having two buffers , one for header and one for content , we allow all above functions from trackerlogic to
write to a fixed location , leaving SUCCESS_HTTP_HEADER_LENGTH bytes in our static buffer , which is enough for the static string
2007-03-07 22:19:00 +00:00
plus dynamic space needed to expand our Content - Length value . We reserve SUCCESS_HTTP_SIZE_OFF for its expansion and calculate
2007-01-26 16:26:49 +00:00
the space NOT needed to expand in reply_off
*/
2007-01-29 02:02:03 +00:00
reply_off = SUCCESS_HTTP_SIZE_OFF - snprintf ( static_outbuf , 0 , " %zd " , reply_size ) ;
2007-01-11 01:06:10 +00:00
2007-01-26 16:26:49 +00:00
/* 2. Now we sprintf our header so that sprintf writes its terminating '\0' exactly one byte before content starts. Complete
packet size is increased by size of header plus one byte ' \n ' , we will copy over ' \0 ' in next step */
2007-01-29 02:02:03 +00:00
reply_size + = 1 + sprintf ( static_outbuf + reply_off , " HTTP/1.0 200 OK \r \n Content-Type: text/plain \r \n Content-Length: %zd \r \n \r " , reply_size ) ;
2007-01-11 01:06:10 +00:00
2007-01-26 16:26:49 +00:00
/* 3. Finally we join both blocks neatly */
2007-01-29 02:02:03 +00:00
static_outbuf [ SUCCESS_HTTP_HEADER_LENGTH - 1 ] = ' \n ' ;
2007-01-26 16:26:49 +00:00
2007-01-29 02:02:03 +00:00
senddata ( s , static_outbuf + reply_off , reply_size ) ;
2006-12-05 12:56:56 +00:00
}
2007-10-13 17:40:37 +00:00
static void signal_handler ( int s ) {
2006-12-08 22:53:32 +00:00
if ( s = = SIGINT ) {
signal ( SIGINT , SIG_IGN ) ;
2007-11-19 21:10:53 +00:00
2007-11-06 01:28:40 +00:00
trackerlogic_deinit ( ) ;
2006-12-08 22:53:32 +00:00
exit ( 0 ) ;
2007-10-13 17:40:37 +00:00
} else if ( s = = SIGALRM ) {
g_now = time ( NULL ) ;
alarm ( 5 ) ;
2006-12-08 22:53:32 +00:00
}
2006-12-08 22:37:44 +00:00
}
2007-01-26 16:26:49 +00:00
static void usage ( char * name ) {
2007-08-18 09:56:22 +00:00
fprintf ( stderr , " Usage: %s [-i ip] [-p port] [-P port] [-d dir] [-A ip] "
# ifdef WANT_BLACKLISTING
" [-b blacklistfile] "
# elif defined ( WANT_CLOSED_TRACKER )
" [-w whitelistfile] "
# endif
" \n " , name ) ;
2007-01-05 13:00:06 +00:00
}
2007-04-07 00:24:17 +00:00
# define HELPLINE(opt,desc) fprintf(stderr, "\t%-10s%s\n",opt,desc)
2007-01-26 16:26:49 +00:00
static void help ( char * name ) {
2007-01-05 13:00:06 +00:00
usage ( name ) ;
2007-04-07 00:24:17 +00:00
HELPLINE ( " -i ip " , " specify ip to bind to (default: *, you may specify more than one) " ) ;
HELPLINE ( " -p port " , " specify tcp port to bind to (default: 6969, you may specify more than one) " ) ;
HELPLINE ( " -P port " , " specify udp port to bind to (default: 6969, you may specify more than one) " ) ;
2007-07-22 00:40:10 +00:00
HELPLINE ( " -d dir " , " specify directory to try to chroot to (default: \" . \" ) " ) ;
2007-04-07 00:24:17 +00:00
HELPLINE ( " -A ip " , " bless an ip address as admin address (e.g. to allow syncs from this address) " ) ;
2007-07-22 00:40:10 +00:00
# ifdef WANT_BLACKLISTING
HELPLINE ( " -b file " , " specify blacklist file. " ) ;
2007-08-18 09:56:22 +00:00
# elif defined( WANT_CLOSED_TRACKER )
HELPLINE ( " -w file " , " specify whitelist file. " ) ;
2007-07-22 00:40:10 +00:00
# endif
2007-04-07 00:24:17 +00:00
fprintf ( stderr , " \n Example: ./opentracker -i 127.0.0.1 -p 6969 -P 6969 -i 10.1.1.23 -p 2710 -p 80 \n " ) ;
2007-01-05 12:25:44 +00:00
}
2007-04-07 00:24:17 +00:00
# undef HELPLINE
2007-01-05 12:25:44 +00:00
2007-11-14 13:06:34 +00:00
static void handle_dead ( const int64 socket ) {
struct http_data * h = io_getcookie ( socket ) ;
if ( h ) {
if ( h - > flag & STRUCT_HTTP_FLAG_IOB_USED )
iob_reset ( & h - > batch ) ;
if ( h - > flag & STRUCT_HTTP_FLAG_ARRAY_USED )
array_reset ( & h - > request ) ;
if ( h - > flag & STRUCT_HTTP_FLAG_WAITINGFORTASK )
mutex_workqueue_canceltask ( socket ) ;
free ( h ) ;
}
io_close ( socket ) ;
}
2007-01-26 16:26:49 +00:00
static void handle_read ( const int64 clientsocket ) {
2007-01-18 02:23:18 +00:00
struct http_data * h = io_getcookie ( clientsocket ) ;
2007-01-31 09:50:46 +00:00
ssize_t l ;
2007-01-18 02:23:18 +00:00
2007-11-14 13:06:34 +00:00
if ( ( l = io_tryread ( clientsocket , static_inbuf , sizeof static_inbuf ) ) < = 0 )
return handle_dead ( clientsocket ) ;
2007-01-18 02:23:18 +00:00
2007-01-24 20:13:30 +00:00
# ifdef _DEBUG_HTTPERROR
memcpy ( debug_request , " 500! \0 " , 5 ) ;
# endif
2007-01-29 02:02:03 +00:00
/* If we get the whole request in one packet, handle it without copying */
if ( ! array_start ( & h - > request ) ) {
2007-11-02 13:13:03 +00:00
if ( memchr ( static_inbuf , ' \n ' , l ) )
return httpresponse ( clientsocket , static_inbuf _DEBUG_HTTPERROR_PARAM ( l ) ) ;
2007-10-19 01:26:33 +00:00
h - > flag | = STRUCT_HTTP_FLAG_ARRAY_USED ;
2007-01-29 02:02:03 +00:00
return array_catb ( & h - > request , static_inbuf , l ) ;
}
2007-10-19 01:26:33 +00:00
h - > flag | = STRUCT_HTTP_FLAG_ARRAY_USED ;
2007-01-29 02:02:03 +00:00
array_catb ( & h - > request , static_inbuf , l ) ;
2007-01-24 22:23:18 +00:00
if ( array_failed ( & h - > request ) )
2007-03-27 12:07:29 +00:00
return httperror ( clientsocket , " 500 Server Error " , " Request too long. " ) ;
2007-10-19 01:26:33 +00:00
if ( ( array_bytes ( & h - > request ) > 8192 ) & & NOTBLESSED ( h ) )
return httperror ( clientsocket , " 500 request too long " , " You sent too much headers " ) ;
2007-03-27 12:07:29 +00:00
2007-11-02 13:13:03 +00:00
if ( memchr ( array_start ( & h - > request ) , ' \n ' , array_bytes ( & h - > request ) ) )
return httpresponse ( clientsocket , array_start ( & h - > request ) _DEBUG_HTTPERROR_PARAM ( array_bytes ( & h - > request ) ) ) ;
2007-01-18 02:23:18 +00:00
}
2007-01-26 16:26:49 +00:00
static void handle_write ( const int64 clientsocket ) {
2007-01-24 22:23:18 +00:00
struct http_data * h = io_getcookie ( clientsocket ) ;
2007-11-14 13:06:34 +00:00
if ( ! h | | ( iob_send ( clientsocket , & h - > batch ) < = 0 ) )
handle_dead ( clientsocket ) ;
2007-01-20 01:50:28 +00:00
}
2007-01-26 16:26:49 +00:00
static void handle_accept ( const int64 serversocket ) {
struct http_data * h ;
2007-01-18 02:23:18 +00:00
unsigned char ip [ 4 ] ;
uint16 port ;
2007-01-18 12:27:17 +00:00
tai6464 t ;
int64 i ;
while ( ( i = socket_accept4 ( serversocket , ( char * ) ip , & port ) ) ! = - 1 ) {
2007-11-20 02:21:53 +00:00
/* Put fd into a non-blocking mode */
io_nonblock ( i ) ;
2007-01-18 12:27:17 +00:00
if ( ! io_fd ( i ) | |
2007-01-25 14:16:26 +00:00
! ( h = ( struct http_data * ) malloc ( sizeof ( struct http_data ) ) ) ) {
2007-01-18 12:27:17 +00:00
io_close ( i ) ;
continue ;
}
2007-10-21 01:11:45 +00:00
io_setcookie ( i , h ) ;
2007-01-18 12:27:17 +00:00
io_wantread ( i ) ;
2007-03-16 23:37:04 +00:00
byte_zero ( h , sizeof ( struct http_data ) ) ;
memmove ( h - > ip , ip , sizeof ( ip ) ) ;
2007-03-28 23:24:30 +00:00
2007-11-06 17:50:41 +00:00
stats_issue_event ( EVENT_ACCEPT , 1 , 0 ) ;
2007-03-16 23:37:04 +00:00
2007-10-19 02:00:53 +00:00
/* That breaks taia encapsulation. But there is no way to take system
time this often in FreeBSD and libowfat does not allow to set unix time */
taia_uint ( & t , 0 ) ; /* Clear t */
tai_unix ( & ( t . sec ) , ( g_now + OT_CLIENT_TIMEOUT ) ) ;
2007-03-16 23:37:04 +00:00
io_timeout ( i , t ) ;
2007-01-18 12:27:17 +00:00
}
2007-03-16 23:37:04 +00:00
if ( errno = = EAGAIN )
2007-01-18 12:27:17 +00:00
io_eagain ( serversocket ) ;
}
2007-01-26 16:26:49 +00:00
static void handle_timeouted ( void ) {
2007-01-18 12:27:17 +00:00
int64 i ;
2007-11-14 13:06:34 +00:00
while ( ( i = io_timeouted ( ) ) ! = - 1 )
handle_dead ( i ) ;
2007-01-18 12:27:17 +00:00
}
2007-03-05 21:14:50 +00:00
static void server_mainloop ( ) {
2007-10-13 17:40:37 +00:00
time_t next_timeout_check = g_now + OT_CLIENT_TIMEOUT_CHECKINTERVAL ;
2007-11-14 13:06:34 +00:00
struct iovec * iovector ;
2007-11-16 00:23:42 +00:00
int iovec_entries ;
2007-01-18 02:23:18 +00:00
2007-01-24 22:23:18 +00:00
for ( ; ; ) {
2007-01-18 02:23:18 +00:00
int64 i ;
2007-01-18 02:40:18 +00:00
2007-10-13 17:58:20 +00:00
io_wait ( ) ;
2007-01-18 02:23:18 +00:00
2007-01-24 22:23:18 +00:00
while ( ( i = io_canread ( ) ) ! = - 1 ) {
2007-03-07 22:19:00 +00:00
const void * cookie = io_getcookie ( i ) ;
if ( cookie = = FLAG_TCP )
2007-01-18 12:27:17 +00:00
handle_accept ( i ) ;
2007-03-07 22:19:00 +00:00
else if ( cookie = = FLAG_UDP )
2007-03-05 21:14:50 +00:00
handle_udp4 ( i ) ;
2007-01-18 12:27:17 +00:00
else
handle_read ( i ) ;
}
2007-11-14 13:06:34 +00:00
while ( ( i = mutex_workqueue_popresult ( & iovec_entries , & iovector ) ) ! = - 1 )
2007-11-16 00:23:42 +00:00
sendiovecdata ( i , iovec_entries , iovector ) ;
2007-11-14 13:06:34 +00:00
2007-01-24 22:23:18 +00:00
while ( ( i = io_canwrite ( ) ) ! = - 1 )
2007-01-20 01:50:28 +00:00
handle_write ( i ) ;
2007-10-13 17:40:37 +00:00
if ( g_now > next_timeout_check ) {
2007-01-18 12:27:17 +00:00
handle_timeouted ( ) ;
2007-10-13 17:40:37 +00:00
next_timeout_check = g_now + OT_CLIENT_TIMEOUT_CHECKINTERVAL ;
2007-01-18 02:23:18 +00:00
}
2007-03-27 12:07:29 +00:00
/* See if we need to move our pools */
2007-11-21 01:53:17 +00:00
if ( NOW ! = ot_last_clean_time ) {
ot_last_clean_time = NOW ;
2007-11-19 21:10:53 +00:00
clean_all_torrents ( ) ;
}
2007-01-18 02:23:18 +00:00
}
}
2007-03-07 22:19:00 +00:00
static void ot_try_bind ( char ip [ 4 ] , uint16 port , int is_tcp ) {
int64 s = is_tcp ? socket_tcp4 ( ) : socket_udp4 ( ) ;
2007-03-05 21:14:50 +00:00
if ( socket_bind4_reuse ( s , ip , port ) = = - 1 )
panic ( " socket_bind4_reuse " ) ;
2007-03-07 22:19:00 +00:00
if ( is_tcp & & ( socket_listen ( s , SOMAXCONN ) = = - 1 ) )
2007-03-05 21:14:50 +00:00
panic ( " socket_listen " ) ;
if ( ! io_fd ( s ) )
panic ( " io_fd " ) ;
2007-03-07 22:19:00 +00:00
io_setcookie ( s , is_tcp ? FLAG_TCP : FLAG_UDP ) ;
2007-03-05 21:14:50 +00:00
io_wantread ( s ) ;
2007-03-07 22:19:00 +00:00
+ + ot_sockets_count ;
2007-03-05 21:14:50 +00:00
}
int main ( int argc , char * * argv ) {
2007-04-02 17:26:40 +00:00
struct passwd * pws = NULL ;
2007-03-05 21:14:50 +00:00
char serverip [ 4 ] = { 0 , 0 , 0 , 0 } ;
2007-01-08 00:57:35 +00:00
char * serverdir = " . " ;
2007-01-18 02:23:18 +00:00
int scanon = 1 ;
2007-11-12 04:39:53 +00:00
# ifdef WANT_ACCESS_CONTROL
char * accesslist_filename = NULL ;
# endif
2007-01-08 00:57:35 +00:00
2007-01-18 02:23:18 +00:00
while ( scanon ) {
2007-08-18 09:56:22 +00:00
switch ( getopt ( argc , argv , " :i:p:A:P:d: "
# ifdef WANT_BLACKLISTING
" b: "
# elif defined( WANT_CLOSED_TRACKER )
" w: "
# endif
" h " ) ) {
2007-01-18 02:23:18 +00:00
case - 1 : scanon = 0 ; break ;
2007-03-05 21:14:50 +00:00
case ' i ' : scan_ip4 ( optarg , serverip ) ; break ;
2007-07-22 00:40:10 +00:00
# ifdef WANT_BLACKLISTING
2007-08-18 09:56:22 +00:00
case ' b ' : accesslist_filename = optarg ; break ;
# elif defined( WANT_CLOSED_TRACKER )
case ' w ' : accesslist_filename = optarg ; break ;
2007-07-22 00:40:10 +00:00
# endif
2007-03-07 22:19:00 +00:00
case ' p ' : ot_try_bind ( serverip , ( uint16 ) atol ( optarg ) , 1 ) ; break ;
case ' P ' : ot_try_bind ( serverip , ( uint16 ) atol ( optarg ) , 0 ) ; break ;
2007-01-08 00:57:35 +00:00
case ' d ' : serverdir = optarg ; break ;
2007-11-01 20:13:03 +00:00
case ' A ' :
if ( g_adminip_count < OT_ADMINIP_MAX )
scan_ip4 ( optarg , ( char * ) ( g_adminip_addresses + g_adminip_count + + ) ) ;
break ;
2007-01-24 22:23:18 +00:00
case ' h ' : help ( argv [ 0 ] ) ; exit ( 0 ) ;
2007-01-08 00:57:35 +00:00
default :
2007-01-24 22:23:18 +00:00
case ' ? ' : usage ( argv [ 0 ] ) ; exit ( 1 ) ;
2007-01-05 12:25:44 +00:00
}
2007-01-08 00:57:35 +00:00
}
2006-12-08 20:07:26 +00:00
2007-11-01 20:13:03 +00:00
/* Sort our admin ips for quick lookup */
qsort ( g_adminip_addresses , g_adminip_count , 4 , ot_ip_compare ) ;
2007-07-20 10:20:02 +00:00
/* Bind to our default tcp/udp ports */
2007-03-16 23:37:04 +00:00
if ( ! ot_sockets_count ) {
2007-03-07 22:19:00 +00:00
ot_try_bind ( serverip , 6969 , 1 ) ;
2007-03-16 23:37:04 +00:00
ot_try_bind ( serverip , 6969 , 0 ) ;
}
2007-04-02 17:26:40 +00:00
2007-07-22 00:40:10 +00:00
/* Drop permissions */
pws = getpwnam ( " nobody " ) ;
2007-04-02 17:26:40 +00:00
if ( ! pws ) {
setegid ( ( gid_t ) - 2 ) ; setuid ( ( uid_t ) - 2 ) ;
setgid ( ( gid_t ) - 2 ) ; seteuid ( ( uid_t ) - 2 ) ;
} else {
setegid ( pws - > pw_gid ) ; setuid ( pws - > pw_uid ) ;
setgid ( pws - > pw_gid ) ; seteuid ( pws - > pw_uid ) ;
}
2007-07-20 10:20:02 +00:00
endpwent ( ) ;
2007-01-14 20:15:04 +00:00
2007-11-12 04:39:53 +00:00
accesslist_init ( accesslist_filename ) ;
2007-07-22 00:40:10 +00:00
2007-01-10 16:42:39 +00:00
signal ( SIGPIPE , SIG_IGN ) ;
2007-10-13 17:40:37 +00:00
signal ( SIGINT , signal_handler ) ;
signal ( SIGALRM , signal_handler ) ;
2007-07-22 00:40:10 +00:00
2007-11-06 01:28:40 +00:00
if ( trackerlogic_init ( serverdir ) = = - 1 )
2007-01-24 22:23:18 +00:00
panic ( " Logic not started " ) ;
2007-01-08 00:57:35 +00:00
2007-11-21 01:53:17 +00:00
g_now = ot_start_time = time ( NULL ) ;
ot_last_clean_time = NOW ;
2007-10-13 17:40:37 +00:00
alarm ( 5 ) ;
2007-01-14 20:15:04 +00:00
2007-03-05 21:14:50 +00:00
server_mainloop ( ) ;
2007-01-08 00:57:35 +00:00
return 0 ;
2006-12-05 12:56:56 +00:00
}