/* * This file Copyright (C) 2008 Charles Kerr * * 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 #include #include /* printf */ #include /* exit, atoi */ #include /* strcmp */ #include /* open */ #include #include /* daemon */ #include #include #include #include #include #define MY_NAME "transmission-daemon" static int closing = FALSE; static tr_session * mySession = NULL; /*** **** Config File ***/ static const char * getUsage( void ) { return "Transmission " LONG_VERSION_STRING " http://www.transmissionbt.com/\n" "A fast and easy BitTorrent client\n" "\n" MY_NAME " is a headless Transmission session\n" "that can be controlled via transmission-remote or Clutch.\n" "\n" "Usage: " MY_NAME " [options]"; } static const struct tr_option options[] = { { 'a', "allowed", "Allowed IP addresses. (Default: " TR_DEFAULT_RPC_WHITELIST ")", "a", 1, "" }, { 'b', "blocklist", "Enable peer blocklists", "b", 0, NULL }, { 'B', "no-blocklist", "Disable peer blocklists", "B", 0, NULL }, { 'd', "dump-settings", "Dump the settings and exit", "d", 0, NULL }, { 'f', "foreground", "Run in the foreground instead of daemonizing", "f", 0, NULL }, { 'g', "config-dir", "Where to look for configuration files", "g", 1, "" }, { 'p', "port", "RPC port (Default: " TR_DEFAULT_RPC_PORT_STR ")", "p", 1, "" }, { 't', "auth", "Require authentication", "t", 0, NULL }, { 'T', "no-auth", "Don't require authentication", "T", 0, NULL }, { 'u', "username", "Set username for authentication", "u", 1, "" }, { 'v', "password", "Set password for authentication", "v", 1, "" }, { 'w', "download-dir", "Where to save downloaded data", "w", 1, "" }, { 0, NULL, NULL, NULL, 0, NULL } }; static void showUsage( void ) { tr_getopt_usage( MY_NAME, getUsage( ), options ); exit( 0 ); } static void gotsig( int sig UNUSED ) { closing = TRUE; } #if !defined( WIN32 ) #if !defined( HAVE_DAEMON ) static int daemon( int nochdir, int noclose ) { switch( fork( ) ) { case 0: break; case - 1: tr_nerr( MY_NAME, "Error daemonizing (fork)! %d - %s", errno, strerror( errno ) ); return -1; default: _exit( 0 ); } if( setsid( ) < 0 ) { tr_nerr( MY_NAME, "Error daemonizing (setsid)! %d - %s", errno, strerror( errno ) ); return -1; } switch( fork( ) ) { case 0: break; case - 1: tr_nerr( MY_NAME, "Error daemonizing (fork2)! %d - %s", errno, strerror( errno ) ); return -1; default: _exit( 0 ); } if( !nochdir && 0 > chdir( "/" ) ) { tr_nerr( MY_NAME, "Error daemonizing (chdir)! %d - %s", errno, strerror( errno ) ); return -1; } if( !noclose ) { int fd; if( ( ( fd = open( "/dev/null", O_RDONLY ) ) ) != 0 ) { dup2( fd, 0 ); close( fd ); } if( ( ( fd = open( "/dev/null", O_WRONLY ) ) ) != 1 ) { dup2( fd, 1 ); close( fd ); } if( ( ( fd = open( "/dev/null", O_WRONLY ) ) ) != 2 ) { dup2( fd, 2 ); close( fd ); } } return 0; } #endif #endif static const char* getConfigDir( int argc, const char ** argv ) { int c; const char * configDir = NULL; const char * optarg; const int ind = tr_optind; while(( c = tr_getopt( getUsage( ), argc, argv, options, &optarg ))) if( c == 'g' ) configDir = optarg; tr_optind = ind; if( configDir == NULL ) configDir = tr_getDefaultConfigDir( MY_NAME ); return configDir; } int main( int argc, char ** argv ) { int c; int64_t i; const char * optarg; tr_benc settings; tr_bool foreground = FALSE; tr_bool dumpSettings = FALSE; const char * configDir = NULL; signal( SIGINT, gotsig ); signal( SIGTERM, gotsig ); #ifndef WIN32 signal( SIGQUIT, gotsig ); signal( SIGPIPE, SIG_IGN ); signal( SIGHUP, SIG_IGN ); #endif /* load settings from defaults + config file */ tr_bencInitDict( &settings, 0 ); configDir = getConfigDir( argc, (const char**)argv ); tr_sessionLoadSettings( &settings, configDir, MY_NAME ); /* overwrite settings from the comamndline */ tr_optind = 1; while(( c = tr_getopt( getUsage(), argc, (const char**)argv, options, &optarg ))) { switch( c ) { case 'a': tr_bencDictAddStr( &settings, TR_PREFS_KEY_RPC_WHITELIST, optarg ); tr_bencDictAddInt( &settings, TR_PREFS_KEY_RPC_WHITELIST_ENABLED, 1 ); break; case 'b': tr_bencDictAddInt( &settings, TR_PREFS_KEY_BLOCKLIST_ENABLED, 1 ); break; case 'B': tr_bencDictAddInt( &settings, TR_PREFS_KEY_BLOCKLIST_ENABLED, 0 ); break; case 'd': dumpSettings = TRUE; break; case 'f': foreground = TRUE; break; case 'g': /* handled above */ break; case 'p': tr_bencDictAddInt( &settings, TR_PREFS_KEY_RPC_PORT, atoi( optarg ) ); break; case 't': tr_bencDictAddInt( &settings, TR_PREFS_KEY_RPC_AUTH_REQUIRED, 0 ); break; case 'T': tr_bencDictAddInt( &settings, TR_PREFS_KEY_RPC_AUTH_REQUIRED, 1 ); break; case 'u': tr_bencDictAddStr( &settings, TR_PREFS_KEY_RPC_USERNAME, optarg ); break; case 'v': tr_bencDictAddStr( &settings, TR_PREFS_KEY_RPC_PASSWORD, optarg ); break; case 'w': tr_bencDictAddStr( &settings, TR_PREFS_KEY_DOWNLOAD_DIR, optarg ); break; default: showUsage( ); break; } } if( dumpSettings ) { char * str = tr_bencSaveAsJSON( &settings, NULL ); fprintf( stderr, "%s", str ); tr_free( str ); return 0; } #ifndef WIN32 if( !foreground ) { if( 0 > daemon( 1, 0 ) ) { fprintf( stderr, "failed to daemonize: %s\n", strerror( errno ) ); exit( 1 ); } } #endif /* start the session */ mySession = tr_sessionInit( "daemon", configDir, FALSE, &settings ); if( tr_bencDictFindInt( &settings, TR_PREFS_KEY_RPC_AUTH_REQUIRED, &i ) && i!=0 ) tr_ninf( MY_NAME, "requiring authentication" ); /* load the torrents */ { tr_ctor * ctor = tr_ctorNew( mySession ); tr_torrent ** torrents = tr_sessionLoadTorrents( mySession, ctor, NULL ); tr_free( torrents ); tr_ctorFree( ctor ); } while( !closing ) tr_wait( 1000 ); /* sleep one second */ /* shutdown */ printf( "Closing transmission session..." ); tr_sessionSaveSettings( mySession, configDir, &settings ); tr_sessionClose( mySession ); printf( " done.\n" ); /* cleanup */ tr_bencFree( &settings ); return 0; }