source: trunk/daemon/daemon.c @ 10947

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

(trunk daemon) #3394 "crash with last nightly build on Debian X86 #5189" -- fixed

  • Property svn:keywords set to Date Rev Author Id
File size: 18.9 KB
Line 
1/*
2 * This file Copyright (C) 2008-2010 Mnemosyne LLC
3 *
4 * This program is free software; you can redistribute it and/or modify
5 * it under the terms of the GNU General Public License version 2
6 * as published by the Free Software Foundation.
7 *
8 * http://www.gnu.org/licenses/old-licenses/gpl-2.0.html
9 *
10 * $Id: daemon.c 10947 2010-07-05 23:36:05Z charles $
11 */
12
13#include <errno.h>
14#include <stdio.h> /* printf */
15#include <stdlib.h> /* exit, atoi */
16#include <string.h> /* strcmp */
17
18#include <sys/types.h> /* umask*/
19#include <sys/stat.h> /* umask*/
20
21#include <fcntl.h> /* open */
22#include <signal.h>
23#ifdef HAVE_SYSLOG
24#include <syslog.h>
25#endif
26#include <unistd.h> /* daemon */
27
28#include <event.h>
29
30#include <libtransmission/transmission.h>
31#include <libtransmission/bencode.h>
32#include <libtransmission/tr-getopt.h>
33#include <libtransmission/utils.h>
34#include <libtransmission/version.h>
35
36#include "watch.h"
37
38#define MY_NAME "transmission-daemon"
39
40#define PREF_KEY_DIR_WATCH          "watch-dir"
41#define PREF_KEY_DIR_WATCH_ENABLED  "watch-dir-enabled"
42
43#define MEM_K 1024
44#define MEM_B_STR "B"
45#define MEM_K_STR "KiB"
46#define MEM_M_STR "MiB"
47#define MEM_G_STR "GiB"
48
49#define DISK_K 1000
50#define DISK_B_STR "B"
51#define DISK_K_STR "kB"
52#define DISK_M_STR "MB"
53#define DISK_G_STR "GB"
54
55#define SPEED_K 1000
56#define SPEED_B_STR "B/s"
57#define SPEED_K_STR "kB/s"
58#define SPEED_M_STR "MB/s"
59#define SPEED_G_STR "GB/s"
60
61static tr_bool paused = FALSE;
62static tr_bool closing = FALSE;
63static tr_session * mySession = NULL;
64static const char * pid_filename = NULL;
65
66/***
67****  Config File
68***/
69
70static const char *
71getUsage( void )
72{
73    return "Transmission " LONG_VERSION_STRING
74           "  http://www.transmissionbt.com/\n"
75           "A fast and easy BitTorrent client\n"
76           "\n"
77           MY_NAME " is a headless Transmission session\n"
78           "that can be controlled via transmission-remote\n"
79           "or the web interface.\n"
80           "\n"
81           "Usage: " MY_NAME " [options]";
82}
83
84static const struct tr_option options[] =
85{
86    { 'a', "allowed", "Allowed IP addresses.  (Default: " TR_DEFAULT_RPC_WHITELIST ")", "a", 1, "<list>" },
87    { 'b', "blocklist", "Enable peer blocklists", "b", 0, NULL },
88    { 'B', "no-blocklist", "Disable peer blocklists", "B", 0, NULL },
89    { 'c', "watch-dir", "Where to watch for new .torrent files", "c", 1, "<directory>" },
90    { 'C', "no-watch-dir", "Disable the watch-dir", "C", 0, NULL },
91    { 941, "incomplete-dir", "Where to store new torrents until they're complete", NULL, 1, "<directory>" },
92    { 942, "no-incomplete-dir", "Don't store incomplete torrents in a different location", NULL, 0, NULL },
93    { 'd', "dump-settings", "Dump the settings and exit", "d", 0, NULL },
94    { 'e', "logfile", "Dump the log messages to this filename", "e", 1, "<filename>" },
95    { 'f', "foreground", "Run in the foreground instead of daemonizing", "f", 0, NULL },
96    { 'g', "config-dir", "Where to look for configuration files", "g", 1, "<path>" },
97    { 'p', "port", "RPC port (Default: " TR_DEFAULT_RPC_PORT_STR ")", "p", 1, "<port>" },
98    { 't', "auth", "Require authentication", "t", 0, NULL },
99    { 'T', "no-auth", "Don't require authentication", "T", 0, NULL },
100    { 'u', "username", "Set username for authentication", "u", 1, "<username>" },
101    { 'v', "password", "Set password for authentication", "v", 1, "<password>" },
102    { 'V', "version", "Show version number and exit", "V", 0, NULL },
103    { 'w', "download-dir", "Where to save downloaded data", "w", 1, "<path>" },
104    { 800, "paused", "Pause all torrents on startup", NULL, 0, NULL },
105    { 'o', "dht", "Enable distributed hash tables (DHT)", "o", 0, NULL },
106    { 'O', "no-dht", "Disable distributed hash tables (DHT)", "O", 0, NULL },
107    { 'y', "lpd", "Enable local peer discovery (LPD)", "y", 0, NULL },
108    { 'Y', "no-lpd", "Disable local peer discovery (LPD)", "Y", 0, NULL },
109    { 'P', "peerport", "Port for incoming peers (Default: " TR_DEFAULT_PEER_PORT_STR ")", "P", 1, "<port>" },
110    { 'm', "portmap", "Enable portmapping via NAT-PMP or UPnP", "m", 0, NULL },
111    { 'M', "no-portmap", "Disable portmapping", "M", 0, NULL },
112    { 'L', "peerlimit-global", "Maximum overall number of peers (Default: " TR_DEFAULT_PEER_LIMIT_GLOBAL_STR ")", "L", 1, "<limit>" },
113    { 'l', "peerlimit-torrent", "Maximum number of peers per torrent (Default: " TR_DEFAULT_PEER_LIMIT_TORRENT_STR ")", "l", 1, "<limit>" },
114    { 910, "encryption-required",  "Encrypt all peer connections", "er", 0, NULL },
115    { 911, "encryption-preferred", "Prefer encrypted peer connections", "ep", 0, NULL },
116    { 912, "encryption-tolerated", "Prefer unencrypted peer connections", "et", 0, NULL },
117    { 'i', "bind-address-ipv4", "Where to listen for peer connections", "i", 1, "<ipv4 addr>" },
118    { 'I', "bind-address-ipv6", "Where to listen for peer connections", "I", 1, "<ipv6 addr>" },
119    { 'r', "rpc-bind-address", "Where to listen for RPC connections", "r", 1, "<ipv4 addr>" },
120    { 953, "global-seedratio", "All torrents, unless overridden by a per-torrent setting, should seed until a specific ratio", "gsr", 1, "ratio" },
121    { 954, "no-global-seedratio", "All torrents, unless overridden by a per-torrent setting, should seed regardless of ratio", "GSR", 0, NULL },
122    { 'x', "pid-file", "Enable PID file", "x", 1, "<pid-file>" },
123    { 0, NULL, NULL, NULL, 0, NULL }
124};
125
126static void
127showUsage( void )
128{
129    tr_getopt_usage( MY_NAME, getUsage( ), options );
130    exit( 0 );
131}
132
133static void
134gotsig( int sig )
135{
136    switch( sig )
137    {
138        case SIGHUP:
139        {
140            tr_benc settings;
141            const char * configDir = tr_sessionGetConfigDir( mySession );
142            tr_inf( "Reloading settings from \"%s\"", configDir );
143            tr_bencInitDict( &settings, 0 );
144            tr_bencDictAddBool( &settings, TR_PREFS_KEY_RPC_ENABLED, TRUE );
145            tr_sessionLoadSettings( &settings, configDir, MY_NAME );
146            tr_sessionSet( mySession, &settings );
147            tr_bencFree( &settings );
148            tr_sessionReloadBlocklists( mySession );
149            break;
150        }
151
152        default:
153            closing = TRUE;
154            break;
155    }
156}
157
158#if defined(WIN32)
159 #define USE_NO_DAEMON
160#elif !defined(HAVE_DAEMON) || defined(__UCLIBC__)
161 #define USE_TR_DAEMON
162#else
163 #define USE_OS_DAEMON
164#endif
165
166static int
167tr_daemon( int nochdir, int noclose )
168{
169#if defined(USE_OS_DAEMON)
170    return daemon( nochdir, noclose );
171#elif defined(USE_TR_DAEMON)
172    pid_t pid = fork( );
173    if( pid < 0 )
174        return -1;
175    else if( pid > 0 )
176        _exit( 0 );
177    else {
178        pid = setsid( );
179        if( pid < 0 )
180            return -1;
181
182        pid = fork( );
183        if( pid < 0 )
184            return -1;
185        else if( pid > 0 )
186            _exit( 0 );
187        else {
188
189            if( !nochdir )
190                if( chdir( "/" ) < 0 )
191                    return -1;
192
193            umask( (mode_t)0 );
194
195            if( !noclose ) {
196                /* send stdin, stdout, and stderr to /dev/null */
197                int i;
198                int fd = open( "/dev/null", O_RDWR, 0 );
199                if( fd < 0 )
200                    fprintf( stderr, "unable to open /dev/null: %s\n", tr_strerror(errno) );
201                for( i=0; i<3; ++i ) {
202                    if( close( i ) )
203                        return -1;
204                    dup2( fd, i );
205                }
206                close( fd );
207            }
208
209            return 0;
210        }
211    }
212#else /* USE_NO_DAEMON */
213    return 0;
214#endif
215}
216
217static const char*
218getConfigDir( int argc, const char ** argv )
219{
220    int c;
221    const char * configDir = NULL;
222    const char * optarg;
223    const int ind = tr_optind;
224
225    while(( c = tr_getopt( getUsage( ), argc, argv, options, &optarg ))) {
226        if( c == 'g' ) {
227            configDir = optarg;
228            break;
229        }
230    }
231
232    tr_optind = ind;
233
234    if( configDir == NULL )
235        configDir = tr_getDefaultConfigDir( MY_NAME );
236
237    return configDir;
238}
239
240static void
241onFileAdded( tr_session * session, const char * dir, const char * file )
242{
243    char * filename = tr_buildPath( dir, file, NULL );
244    tr_ctor * ctor = tr_ctorNew( session );
245    int err = tr_ctorSetMetainfoFromFile( ctor, filename );
246
247    if( !err )
248    {
249        tr_torrentNew( ctor, &err );
250
251        if( err == TR_PARSE_ERR )
252            tr_err( "Error parsing .torrent file \"%s\"", file );
253        else
254        {
255            tr_bool trash = FALSE;
256            int test = tr_ctorGetDeleteSource( ctor, &trash );
257
258            tr_inf( "Parsing .torrent file successful \"%s\"", file );
259
260            if( !test && trash )
261            {
262                tr_inf( "Deleting input .torrent file \"%s\"", file );
263                if( remove( filename ) )
264                    tr_err( "Error deleting .torrent file: %s", tr_strerror( errno ) );
265            }
266            else
267            {
268                char * new_filename = tr_strdup_printf( "%s.added", filename );
269                rename( filename, new_filename );
270                tr_free( new_filename );
271            }
272        }
273    }
274
275    tr_ctorFree( ctor );
276    tr_free( filename );
277}
278
279static void
280printMessage( FILE * logfile, int level, const char * name, const char * message, const char * file, int line )
281{
282    if( logfile != NULL )
283    {
284        char timestr[64];
285        tr_getLogTimeStr( timestr, sizeof( timestr ) );
286        if( name )
287            fprintf( logfile, "[%s] %s %s (%s:%d)\n", timestr, name, message, file, line );
288        else
289            fprintf( logfile, "[%s] %s (%s:%d)\n", timestr, message, file, line );
290    }
291#ifdef HAVE_SYSLOG
292    else /* daemon... write to syslog */
293    {
294        int priority;
295
296        /* figure out the syslog priority */
297        switch( level ) {
298            case TR_MSG_ERR: priority = LOG_ERR; break;
299            case TR_MSG_DBG: priority = LOG_DEBUG; break;
300            default: priority = LOG_INFO; break;
301        }
302
303        if( name )
304            syslog( priority, "%s %s (%s:%d)", name, message, file, line );
305        else
306            syslog( priority, "%s (%s:%d)", message, file, line );
307    }
308#endif
309}
310
311static void
312pumpLogMessages( FILE * logfile )
313{
314    const tr_msg_list * l;
315    tr_msg_list * list = tr_getQueuedMessages( );
316
317    for( l=list; l!=NULL; l=l->next )
318        printMessage( logfile, l->level, l->name, l->message, l->file, l->line );
319
320    tr_freeMessageList( list );
321}
322
323int
324main( int argc, char ** argv )
325{
326    int c;
327    const char * optarg;
328    tr_benc settings;
329    tr_bool boolVal;
330    tr_bool loaded;
331    tr_bool foreground = FALSE;
332    tr_bool dumpSettings = FALSE;
333    const char * configDir = NULL;
334    dtr_watchdir * watchdir = NULL;
335    FILE * logfile = NULL;
336    tr_bool pidfile_created = FALSE;
337
338    signal( SIGINT, gotsig );
339    signal( SIGTERM, gotsig );
340#ifndef WIN32
341    signal( SIGHUP, gotsig );
342#endif
343
344    /* load settings from defaults + config file */
345    tr_bencInitDict( &settings, 0 );
346    tr_bencDictAddBool( &settings, TR_PREFS_KEY_RPC_ENABLED, TRUE );
347    configDir = getConfigDir( argc, (const char**)argv );
348    loaded = tr_sessionLoadSettings( &settings, configDir, MY_NAME );
349
350    /* overwrite settings from the comamndline */
351    tr_optind = 1;
352    while(( c = tr_getopt( getUsage(), argc, (const char**)argv, options, &optarg ))) {
353        switch( c ) {
354            case 'a': tr_bencDictAddStr( &settings, TR_PREFS_KEY_RPC_WHITELIST, optarg );
355                      tr_bencDictAddBool( &settings, TR_PREFS_KEY_RPC_WHITELIST_ENABLED, TRUE );
356                      break;
357            case 'b': tr_bencDictAddBool( &settings, TR_PREFS_KEY_BLOCKLIST_ENABLED, TRUE );
358                      break;
359            case 'B': tr_bencDictAddBool( &settings, TR_PREFS_KEY_BLOCKLIST_ENABLED, FALSE );
360                      break;
361            case 'c': tr_bencDictAddStr( &settings, PREF_KEY_DIR_WATCH, optarg );
362                      tr_bencDictAddBool( &settings, PREF_KEY_DIR_WATCH_ENABLED, TRUE );
363                      break;
364            case 'C': tr_bencDictAddBool( &settings, PREF_KEY_DIR_WATCH_ENABLED, FALSE );
365                      break;
366            case 941: tr_bencDictAddStr( &settings, TR_PREFS_KEY_INCOMPLETE_DIR, optarg );
367                      tr_bencDictAddBool( &settings, TR_PREFS_KEY_INCOMPLETE_DIR_ENABLED, TRUE );
368                      break;
369            case 942: tr_bencDictAddBool( &settings, TR_PREFS_KEY_INCOMPLETE_DIR_ENABLED, FALSE );
370                      break;
371            case 'd': dumpSettings = TRUE;
372                      break;
373            case 'e': logfile = fopen( optarg, "a+" );
374                      if( logfile == NULL )
375                          fprintf( stderr, "Couldn't open \"%s\": %s\n", optarg, tr_strerror( errno ) );
376                      break;
377            case 'f': foreground = TRUE;
378                      break;
379            case 'g': /* handled above */
380                      break;
381            case 'V': /* version */
382                      fprintf(stderr, "Transmission %s\n", LONG_VERSION_STRING);
383                      exit( 0 );
384            case 'o': tr_bencDictAddBool( &settings, TR_PREFS_KEY_DHT_ENABLED, TRUE );
385                      break;
386            case 'O': tr_bencDictAddBool( &settings, TR_PREFS_KEY_DHT_ENABLED, FALSE );
387                      break;
388            case 'p': tr_bencDictAddInt( &settings, TR_PREFS_KEY_RPC_PORT, atoi( optarg ) );
389                      break;
390            case 't': tr_bencDictAddBool( &settings, TR_PREFS_KEY_RPC_AUTH_REQUIRED, TRUE );
391                      break;
392            case 'T': tr_bencDictAddBool( &settings, TR_PREFS_KEY_RPC_AUTH_REQUIRED, FALSE );
393                      break;
394            case 'u': tr_bencDictAddStr( &settings, TR_PREFS_KEY_RPC_USERNAME, optarg );
395                      break;
396            case 'v': tr_bencDictAddStr( &settings, TR_PREFS_KEY_RPC_PASSWORD, optarg );
397                      break;
398            case 'w': tr_bencDictAddStr( &settings, TR_PREFS_KEY_DOWNLOAD_DIR, optarg );
399                      break;
400            case 'P': tr_bencDictAddInt( &settings, TR_PREFS_KEY_PEER_PORT, atoi( optarg ) );
401                      break;
402            case 'm': tr_bencDictAddBool( &settings, TR_PREFS_KEY_PORT_FORWARDING, TRUE );
403                      break;
404            case 'M': tr_bencDictAddBool( &settings, TR_PREFS_KEY_PORT_FORWARDING, FALSE );
405                      break;
406            case 'L': tr_bencDictAddInt( &settings, TR_PREFS_KEY_PEER_LIMIT_GLOBAL, atoi( optarg ) );
407                      break;
408            case 'l': tr_bencDictAddInt( &settings, TR_PREFS_KEY_PEER_LIMIT_TORRENT, atoi( optarg ) );
409                      break;
410            case 800: paused = TRUE;
411                      break;
412            case 910: tr_bencDictAddInt( &settings, TR_PREFS_KEY_ENCRYPTION, TR_ENCRYPTION_REQUIRED );
413                      break;
414            case 911: tr_bencDictAddInt( &settings, TR_PREFS_KEY_ENCRYPTION, TR_ENCRYPTION_PREFERRED );
415                      break;
416            case 912: tr_bencDictAddInt( &settings, TR_PREFS_KEY_ENCRYPTION, TR_CLEAR_PREFERRED );
417                      break;
418            case 'i': tr_bencDictAddStr( &settings, TR_PREFS_KEY_BIND_ADDRESS_IPV4, optarg );
419                      break;
420            case 'I': tr_bencDictAddStr( &settings, TR_PREFS_KEY_BIND_ADDRESS_IPV6, optarg );
421                      break;
422            case 'r': tr_bencDictAddStr( &settings, TR_PREFS_KEY_RPC_BIND_ADDRESS, optarg );
423                      break;
424            case 953: tr_bencDictAddReal( &settings, TR_PREFS_KEY_RATIO, atof(optarg) );
425                      tr_bencDictAddBool( &settings, TR_PREFS_KEY_RATIO_ENABLED, TRUE );
426                      break;
427            case 954: tr_bencDictAddBool( &settings, TR_PREFS_KEY_RATIO_ENABLED, FALSE );
428                      break;
429            case 'x': pid_filename = optarg;
430                      break;
431            case 'y': tr_bencDictAddBool( &settings, TR_PREFS_KEY_LPD_ENABLED, TRUE );
432                      break;
433            case 'Y': tr_bencDictAddBool( &settings, TR_PREFS_KEY_LPD_ENABLED, FALSE );
434                      break;
435            default:  showUsage( );
436                      break;
437        }
438    }
439
440    if( foreground && !logfile )
441        logfile = stderr;
442
443    if( !loaded )
444    {
445        printMessage( logfile, TR_MSG_ERR, MY_NAME, "Error loading config file -- exiting.", __FILE__, __LINE__ );
446        return -1;
447    }
448
449    if( dumpSettings )
450    {
451        char * str = tr_bencToStr( &settings, TR_FMT_JSON, NULL );
452        fprintf( stderr, "%s", str );
453        tr_free( str );
454        return 0;
455    }
456
457    if( !foreground && tr_daemon( TRUE, FALSE ) < 0 )
458    {
459        char buf[256];
460        tr_snprintf( buf, sizeof( buf ), "Failed to daemonize: %s", tr_strerror( errno ) );
461        printMessage( logfile, TR_MSG_ERR, MY_NAME, buf, __FILE__, __LINE__ );
462        exit( 1 );
463    }
464
465    /* start the session */
466    tr_formatter_mem_init( MEM_K, MEM_B_STR, MEM_K_STR, MEM_M_STR, MEM_G_STR );
467    tr_formatter_size_init( DISK_K,DISK_B_STR, DISK_K_STR, DISK_M_STR, DISK_G_STR );
468    tr_formatter_speed_init( SPEED_K, SPEED_B_STR, SPEED_K_STR, SPEED_M_STR, SPEED_G_STR );
469    mySession = tr_sessionInit( "daemon", configDir, TRUE, &settings );
470    tr_ninf( NULL, "Using settings from \"%s\"", configDir );
471    tr_sessionSaveSettings( mySession, configDir, &settings );
472
473    if( pid_filename != NULL )
474    {
475        FILE * fp = fopen( pid_filename, "w+" );
476        if( fp != NULL )
477        {
478            fprintf( fp, "%d", (int)getpid() );
479            fclose( fp );
480            tr_inf( "Saved pidfile \"%s\"", pid_filename );
481            pidfile_created = TRUE;
482        }
483        else
484            tr_err( "Unable to save pidfile \"%s\": %s", pid_filename, strerror( errno ) );
485    }
486
487    if( tr_bencDictFindBool( &settings, TR_PREFS_KEY_RPC_AUTH_REQUIRED, &boolVal ) && boolVal )
488        tr_ninf( MY_NAME, "requiring authentication" );
489
490    /* maybe add a watchdir */
491    {
492        const char * dir;
493
494        if( tr_bencDictFindBool( &settings, PREF_KEY_DIR_WATCH_ENABLED, &boolVal )
495            && boolVal
496            && tr_bencDictFindStr( &settings, PREF_KEY_DIR_WATCH, &dir )
497            && dir
498            && *dir )
499        {
500            tr_inf( "Watching \"%s\" for new .torrent files", dir );
501            watchdir = dtr_watchdir_new( mySession, dir, onFileAdded );
502        }
503    }
504
505    /* load the torrents */
506    {
507        tr_torrent ** torrents;
508        tr_ctor * ctor = tr_ctorNew( mySession );
509        if( paused )
510            tr_ctorSetPaused( ctor, TR_FORCE, TRUE );
511        torrents = tr_sessionLoadTorrents( mySession, ctor, NULL );
512        tr_free( torrents );
513        tr_ctorFree( ctor );
514    }
515
516#ifdef HAVE_SYSLOG
517    if( !foreground )
518        openlog( MY_NAME, LOG_CONS|LOG_PID, LOG_DAEMON );
519#endif
520
521    while( !closing ) {
522        tr_wait_msec( 1000 ); /* sleep one second */
523        dtr_watchdir_update( watchdir );
524        pumpLogMessages( logfile );
525    }
526
527    /* shutdown */
528#if HAVE_SYSLOG
529    if( !foreground )
530    {
531        syslog( LOG_INFO, "%s", "Closing session" );
532        closelog( );
533    }
534#endif
535
536    printf( "Closing transmission session..." );
537    tr_sessionSaveSettings( mySession, configDir, &settings );
538    dtr_watchdir_free( watchdir );
539    tr_sessionClose( mySession );
540    printf( " done.\n" );
541
542    /* cleanup */
543    if( pidfile_created )
544        remove( pid_filename );
545    tr_bencFree( &settings );
546    return 0;
547}
Note: See TracBrowser for help on using the repository browser.