transmission/cli/transmissioncli.c

469 lines
14 KiB
C
Raw Normal View History

2006-07-16 19:39:23 +00:00
/******************************************************************************
* $Id$
*
* Copyright (c) 2005-2006 Transmission authors and contributors
*
* Permission is hereby granted, free of charge, to any person obtaining a
* copy of this software and associated documentation files (the "Software"),
* to deal in the Software without restriction, including without limitation
* the rights to use, copy, modify, merge, publish, distribute, sublicense,
* and/or sell copies of the Software, and to permit persons to whom the
* Software is furnished to do so, subject to the following conditions:
*
* The above copyright notice and this permission notice shall be included in
* all copies or substantial portions of the Software.
*
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
* FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
* AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
* LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING
* FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER
* DEALINGS IN THE SOFTWARE.
*****************************************************************************/
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <unistd.h>
#include <getopt.h>
#include <signal.h>
#include <libtransmission/transmission.h>
#include <libtransmission/makemeta.h>
2007-11-12 15:15:51 +00:00
#include <libtransmission/utils.h> /* tr_wait */
2006-07-16 19:39:23 +00:00
/* macro to shut up "unused parameter" warnings */
#ifdef __GNUC__
#define UNUSED __attribute__((unused))
#else
#define UNUSED
#endif
const char * USAGE =
"Usage: %s [-car[-m]] [-dfinpsuv] [-h] file.torrent [output-dir]\n\n"
"Options:\n"
" -c, --create-from <file> Create torrent from the specified source file.\n"
" -a, --announce <url> Used in conjunction with -c.\n"
" -r, --private Used in conjunction with -c.\n"
" -m, --comment <text> Adds an optional comment when creating a torrent.\n"
" -d, --download <int> Maximum download rate (-1 = no limit, default = -1)\n"
" -f, --finish <shell script> Command you wish to run on completion\n"
" -h, --help Print this help and exit\n"
" -i, --info Print metainfo and exit\n"
" -n --nat-traversal Attempt NAT traversal using NAT-PMP or UPnP IGD\n"
" -p, --port <int> Port we should listen on (default = %d)\n"
" -s, --scrape Print counts of seeders/leechers and exit\n"
" -u, --upload <int> Maximum upload rate (-1 = no limit, default = 20)\n"
" -v, --verbose <int> Verbose level (0 to 2, default = 0)\n"
" -y, --recheck Force a recheck of the torrent data\n";
2006-07-16 19:39:23 +00:00
static int showHelp = 0;
static int showInfo = 0;
static int showScrape = 0;
static int isPrivate = 0;
2006-07-16 19:39:23 +00:00
static int verboseLevel = 0;
static int bindPort = TR_DEFAULT_PORT;
static int uploadLimit = 20;
static int downloadLimit = -1;
static char * torrentPath = NULL;
static char * savePath = ".";
2006-09-25 18:37:45 +00:00
static int natTraversal = 0;
static int recheckData = 0;
2007-02-13 05:20:52 +00:00
static sig_atomic_t gotsig = 0;
static sig_atomic_t manualUpdate = 0;
static tr_torrent * tor;
2006-07-16 19:39:23 +00:00
static char * finishCall = NULL;
static char * announce = NULL;
static char * sourceFile = NULL;
static char * comment = NULL;
2006-07-16 19:39:23 +00:00
static int parseCommandLine ( int argc, char ** argv );
static void sigHandler ( int signal );
char * getStringRatio( float ratio )
{
static char string[20];
if( ratio == TR_RATIO_NA )
return "n/a";
snprintf( string, sizeof string, "%.3f", ratio );
return string;
}
#define LINEWIDTH 80
2007-09-28 15:12:21 +00:00
static void
torrentStateChanged( tr_torrent * torrent UNUSED,
cp_status_t status UNUSED,
void * user_data UNUSED )
{
system( finishCall );
}
2006-07-16 19:39:23 +00:00
int main( int argc, char ** argv )
{
2007-08-05 03:18:23 +00:00
int i, error;
tr_handle * h;
const tr_stat * s;
tr_handle_status * hstat;
tr_ctor * ctor;
2006-07-16 19:39:23 +00:00
printf( "Transmission %s - http://transmission.m0k.org/\n\n",
LONG_VERSION_STRING );
2006-07-16 19:39:23 +00:00
/* Get options */
if( parseCommandLine( argc, argv ) )
{
printf( USAGE, argv[0], TR_DEFAULT_PORT );
return EXIT_FAILURE;
2006-07-16 19:39:23 +00:00
}
if( showHelp )
{
printf( USAGE, argv[0], TR_DEFAULT_PORT );
return EXIT_SUCCESS;
2006-07-16 19:39:23 +00:00
}
if( verboseLevel < 0 )
{
verboseLevel = 0;
}
else if( verboseLevel > 9 )
{
verboseLevel = 9;
}
if( verboseLevel )
{
static char env[11];
2007-03-12 00:04:11 +00:00
snprintf( env, sizeof env, "TR_DEBUG=%d", verboseLevel );
2006-07-16 19:39:23 +00:00
putenv( env );
}
if( bindPort < 1 || bindPort > 65535 )
{
printf( "Invalid port '%d'\n", bindPort );
return EXIT_FAILURE;
2006-07-16 19:39:23 +00:00
}
/* Initialize libtransmission */
h = tr_init( "cli" );
2006-07-16 19:39:23 +00:00
if( sourceFile && *sourceFile ) /* creating a torrent */
{
int ret;
tr_metainfo_builder * builder = tr_metaInfoBuilderCreate( h, sourceFile );
tr_makeMetaInfo( builder, torrentPath, announce, comment, isPrivate );
while( !builder->isDone ) {
2007-11-12 15:15:51 +00:00
tr_wait( 1000 );
printf( "." );
}
ret = !builder->failed;
tr_metaInfoBuilderFree( builder );
return ret;
}
ctor = tr_ctorNew( h );
tr_ctorSetMetainfoFromFile( ctor, torrentPath );
tr_ctorSetPaused( ctor, TR_FORCE, 0 );
tr_ctorSetDestination( ctor, TR_FORCE, savePath );
tor = tr_torrentNew( h, ctor, &error );
tr_ctorFree( ctor );
if( tor == NULL )
2006-07-16 19:39:23 +00:00
{
printf( "Failed opening torrent file `%s'\n", torrentPath );
tr_close( h );
return EXIT_FAILURE;
2006-07-16 19:39:23 +00:00
}
if( showInfo )
{
const tr_info * info = tr_torrentInfo( tor );
2006-07-16 19:39:23 +00:00
s = tr_torrentStat( tor );
2006-07-16 19:39:23 +00:00
/* Print torrent info (quite <20> la btshowmetainfo) */
printf( "hash: " );
for( i = 0; i < SHA_DIGEST_LENGTH; i++ )
{
printf( "%02x", info->hash[i] );
}
printf( "\n" );
printf( "tracker: %s:%d\n",
s->tracker->address, s->tracker->port );
printf( "announce: %s\n", s->tracker->announce );
2006-07-16 19:39:23 +00:00
printf( "size: %"PRIu64" (%"PRIu64" * %d + %"PRIu64")\n",
info->totalSize, info->totalSize / info->pieceSize,
info->pieceSize, info->totalSize % info->pieceSize );
2006-12-02 05:39:27 +00:00
if( info->comment[0] )
2006-12-02 05:32:54 +00:00
{
printf( "comment: %s\n", info->comment );
}
2006-12-02 05:39:27 +00:00
if( info->creator[0] )
{
printf( "creator: %s\n", info->creator );
}
2007-10-15 20:58:39 +00:00
if( info->isPrivate )
{
printf( "private flag set\n" );
}
2006-07-16 19:39:23 +00:00
printf( "file(s):\n" );
for( i = 0; i < info->fileCount; i++ )
{
printf( " %s (%"PRIu64")\n", info->files[i].name,
info->files[i].length );
}
goto cleanup;
}
2007-11-12 15:15:51 +00:00
2006-07-16 19:39:23 +00:00
if( showScrape )
{
2007-11-12 15:15:51 +00:00
printf( "Scraping, Please wait...\n" );
const tr_stat * stats;
uint64_t start = tr_date();
do
2006-07-16 19:39:23 +00:00
{
2007-11-12 15:15:51 +00:00
stats = tr_torrentStat( tor );
if( stats == NULL || tr_date() - start > 20000 )
{
printf( "Scrape failed.\n" );
goto cleanup;
}
tr_wait( 2000 );
2006-07-16 19:39:23 +00:00
}
2007-11-12 15:15:51 +00:00
while( stats->completedFromTracker == -1 || stats->leechers == -1 || stats->seeders == -1 );
printf( "%d seeder(s), %d leecher(s), %d download(s).\n",
stats->seeders, stats->leechers, stats->completedFromTracker );
2006-07-16 19:39:23 +00:00
goto cleanup;
}
signal( SIGINT, sigHandler );
signal( SIGHUP, sigHandler );
2006-07-16 19:39:23 +00:00
tr_setBindPort( h, bindPort );
tr_setGlobalSpeedLimit ( h, TR_UP, uploadLimit );
tr_setUseGlobalSpeedLimit( h, TR_UP, uploadLimit > 0 );
tr_setGlobalSpeedLimit ( h, TR_DOWN, downloadLimit );
tr_setUseGlobalSpeedLimit( h, TR_DOWN, downloadLimit > 0 );
2006-09-25 18:37:45 +00:00
tr_natTraversalEnable( h, natTraversal );
2006-07-16 19:39:23 +00:00
2007-09-28 15:12:21 +00:00
tr_torrentSetStatusCallback( tor, torrentStateChanged, NULL );
2006-07-16 19:39:23 +00:00
tr_torrentStart( tor );
2007-01-14 12:00:21 +00:00
for( ;; )
2006-07-16 19:39:23 +00:00
{
char string[LINEWIDTH];
2006-07-16 19:39:23 +00:00
int chars = 0;
2007-11-12 15:15:51 +00:00
tr_wait( 1000 );
2006-07-16 19:39:23 +00:00
2007-02-13 05:20:52 +00:00
if( gotsig )
{
gotsig = 0;
tr_torrentStop( tor );
2007-07-18 17:31:00 +00:00
tr_natTraversalEnable( h, 0 );
2007-02-13 05:20:52 +00:00
}
if( manualUpdate )
{
manualUpdate = 0;
2007-10-19 23:21:32 +00:00
if ( !tr_torrentCanManualUpdate( tor ) )
fprintf( stderr, "\rReceived SIGHUP, but can't send a manual update now\n" );
else {
fprintf( stderr, "\rReceived SIGHUP: manual update scheduled\n" );
tr_manualUpdate( tor );
}
}
if( recheckData )
{
recheckData = 0;
tr_torrentRecheck( tor );
}
2007-02-13 05:20:52 +00:00
2006-07-16 19:39:23 +00:00
s = tr_torrentStat( tor );
if( s->status & TR_STATUS_CHECK_WAIT )
{
chars = snprintf( string, sizeof string,
2007-11-12 15:15:51 +00:00
"Waiting to verify local files..." );
}
2007-01-14 12:00:21 +00:00
else if( s->status & TR_STATUS_CHECK )
2006-07-16 19:39:23 +00:00
{
chars = snprintf( string, sizeof string,
2007-11-12 15:15:51 +00:00
"Verifying local files... %.2f%%, found %.2f%% valid", 100 * s->recheckProgress, 100.0 * s->percentDone );
2006-07-16 19:39:23 +00:00
}
else if( s->status & TR_STATUS_DOWNLOAD )
{
chars = snprintf( string, sizeof string,
2006-07-16 19:39:23 +00:00
"Progress: %.2f %%, %d peer%s, dl from %d (%.2f KB/s), "
"ul to %d (%.2f KB/s) [%s]", 100.0 * s->percentDone,
s->peersConnected, ( s->peersConnected == 1 ) ? "" : "s",
s->peersSendingToUs, s->rateDownload,
s->peersGettingFromUs, s->rateUpload,
getStringRatio(s->ratio) );
2006-07-16 19:39:23 +00:00
}
else if( s->status & TR_STATUS_SEED )
{
chars = snprintf( string, sizeof string,
"Seeding, uploading to %d of %d peer(s), %.2f KB/s [%s]",
s->peersGettingFromUs, s->peersConnected,
s->rateUpload, getStringRatio(s->ratio) );
2006-07-16 19:39:23 +00:00
}
else if( s->status & TR_STATUS_STOPPED )
2007-07-18 17:31:00 +00:00
{
break;
}
if( ( signed )sizeof string > chars )
2007-02-13 04:21:29 +00:00
{
memset( &string[chars], ' ', sizeof string - 1 - chars );
2007-02-13 04:21:29 +00:00
}
string[sizeof string - 1] = '\0';
2006-10-05 18:37:38 +00:00
fprintf( stderr, "\r%s", string );
2006-07-16 19:39:23 +00:00
2007-01-14 12:00:21 +00:00
if( s->error )
2006-07-16 19:39:23 +00:00
{
2007-01-14 12:00:21 +00:00
fprintf( stderr, "\n%s\n", s->errorString );
2006-07-16 19:39:23 +00:00
}
else if( verboseLevel > 0 )
{
fprintf( stderr, "\n" );
}
}
fprintf( stderr, "\n" );
2007-01-14 12:00:21 +00:00
/* Try for 5 seconds to delete any port mappings for nat traversal */
tr_natTraversalEnable( h, 0 );
2006-07-16 19:39:23 +00:00
for( i = 0; i < 10; i++ )
{
hstat = tr_handleStatus( h );
if( TR_NAT_TRAVERSAL_UNMAPPED == hstat->natTraversalStatus )
2006-07-16 19:39:23 +00:00
{
2007-01-14 12:00:21 +00:00
/* Port mappings were deleted */
2006-07-16 19:39:23 +00:00
break;
}
2007-11-12 15:15:51 +00:00
tr_wait( 500 );
2006-07-16 19:39:23 +00:00
}
cleanup:
2007-08-05 03:18:23 +00:00
tr_torrentClose( tor );
2006-07-16 19:39:23 +00:00
tr_close( h );
return EXIT_SUCCESS;
2006-07-16 19:39:23 +00:00
}
static int parseCommandLine( int argc, char ** argv )
{
for( ;; )
{
static struct option long_options[] =
{ { "help", no_argument, NULL, 'h' },
{ "info", no_argument, NULL, 'i' },
{ "scrape", no_argument, NULL, 's' },
{ "private", no_argument, NULL, 'r' },
{ "verbose", required_argument, NULL, 'v' },
{ "port", required_argument, NULL, 'p' },
{ "upload", required_argument, NULL, 'u' },
{ "download", required_argument, NULL, 'd' },
{ "finish", required_argument, NULL, 'f' },
{ "create", required_argument, NULL, 'c' },
{ "comment", required_argument, NULL, 'm' },
{ "announce", required_argument, NULL, 'a' },
{ "nat-traversal", no_argument, NULL, 'n' },
{ "recheck", no_argument, NULL, 'y' },
{ "output-dir", required_argument, NULL, 'o' },
2006-07-16 19:39:23 +00:00
{ 0, 0, 0, 0} };
int c, optind = 0;
c = getopt_long( argc, argv, "hisrv:p:u:d:f:c:m:a:n:o:y",
long_options, &optind );
2006-07-16 19:39:23 +00:00
if( c < 0 )
{
break;
}
switch( c )
{
case 'h':
showHelp = 1;
break;
case 'i':
showInfo = 1;
break;
case 's':
showScrape = 1;
break;
case 'r':
isPrivate = 1;
break;
2006-07-16 19:39:23 +00:00
case 'v':
verboseLevel = atoi( optarg );
break;
case 'p':
bindPort = atoi( optarg );
break;
case 'u':
uploadLimit = atoi( optarg );
break;
case 'd':
downloadLimit = atoi( optarg );
break;
case 'f':
finishCall = optarg;
break;
case 'c':
sourceFile = optarg;
break;
case 'm':
comment = optarg;
break;
case 'a':
announce = optarg;
break;
2006-09-25 18:37:45 +00:00
case 'n':
natTraversal = 1;
break;
case 'y':
recheckData = 1;
break;
case 'o':
savePath = optarg;
2006-07-16 19:39:23 +00:00
default:
return 1;
}
}
if( optind >= argc )
2006-07-16 19:39:23 +00:00
{
return !showHelp;
}
torrentPath = argv[optind];
return 0;
}
static void sigHandler( int signal )
{
switch( signal )
{
case SIGINT:
2007-02-13 05:20:52 +00:00
gotsig = 1;
2006-07-16 19:39:23 +00:00
break;
case SIGHUP:
manualUpdate = 1;
break;
2006-07-16 19:39:23 +00:00
default:
break;
}
}