aria2/src/main.cc

612 lines
20 KiB
C++
Raw Normal View History

2006-02-17 13:35:04 +00:00
/* <!-- copyright */
/*
* aria2 - a simple utility for downloading files faster
*
* Copyright (C) 2006 Tatsuhiro Tsujikawa
*
* This program is free software; you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation; either version 2 of the License, or
* (at your option) any later version.
*
* This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with this program; if not, write to the Free Software
* Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
*/
/* copyright --> */
#include "HttpInitiateConnectionCommand.h"
#include "ConsoleDownloadEngine.h"
2006-03-21 14:12:51 +00:00
#include "TorrentConsoleDownloadEngine.h"
2006-02-17 13:35:04 +00:00
#include "SegmentMan.h"
2006-03-21 14:12:51 +00:00
#include "TorrentMan.h"
2006-02-22 14:30:47 +00:00
#include "SplitSlowestSegmentSplitter.h"
2006-02-17 13:35:04 +00:00
#include "SimpleLogger.h"
#include "common.h"
#include "DefaultDiskWriter.h"
#include "Util.h"
#include "InitiateConnectionCommandFactory.h"
2006-02-21 14:00:58 +00:00
#include "prefs.h"
2006-03-21 14:12:51 +00:00
#include "TrackerInitCommand.h"
#include "PeerListenCommand.h"
#include "TorrentAutoSaveCommand.h"
#include "SleepCommand.h"
2006-02-17 13:35:04 +00:00
#include <vector>
#include <algorithm>
2006-03-21 14:12:51 +00:00
#include <time.h>
2006-02-17 13:35:04 +00:00
#include <signal.h>
#include <unistd.h>
#include <libgen.h>
#include <utility>
extern char* optarg;
extern int optind, opterr, optopt;
#include <getopt.h>
#ifdef HAVE_LIBSSL
// for SSL
# include <openssl/err.h>
# include <openssl/ssl.h>
#endif // HAVE_LIBSSL
2006-02-17 13:35:04 +00:00
using namespace std;
2006-03-21 14:12:51 +00:00
void printDownloadCompeleteMessage(string filename) {
printf(_("\nThe download was complete. <%s>\n"), filename.c_str());
}
void printDownloadAbortMessage() {
printf(_("\nThe download was not complete because of errors. Check the log.\n"));
}
2006-02-17 13:35:04 +00:00
void clearRequest(Request* req) {
delete(req);
}
DownloadEngine* e;
2006-03-21 14:12:51 +00:00
TorrentDownloadEngine* te;
2006-02-17 13:35:04 +00:00
void handler(int signal) {
cout << _("\nSIGINT signal received.") << endl;
2006-02-17 13:35:04 +00:00
e->segmentMan->save();
if(e->diskWriter != NULL) {
e->diskWriter->closeFile();
}
exit(0);
}
2006-03-21 14:12:51 +00:00
void torrentHandler(int signal) {
cout << _("\nSIGINT signal received.") << endl;
if(te->torrentMan->diskWriter != NULL) {
te->torrentMan->diskWriter->closeFile();
}
if(te->torrentMan->downloadComplete()) {
te->torrentMan->remove();
te->torrentMan->fixFilename();
printDownloadCompeleteMessage(te->torrentMan->getFilePath());
} else {
te->torrentMan->save();
}
exit(0);
}
2006-02-18 03:59:26 +00:00
void addCommand(int cuid, const char* url, string referer, vector<Request*> requests) {
2006-02-17 13:35:04 +00:00
Request* req = new Request();
2006-02-18 03:59:26 +00:00
req->setReferer(referer);
2006-02-17 13:35:04 +00:00
if(req->setUrl(url)) {
e->commands.push(InitiateConnectionCommandFactory::createInitiateConnectionCommand(cuid, req, e));
requests.push_back(req);
} else {
2006-03-05 06:32:01 +00:00
fprintf(stderr, _("Unrecognized URL or unsupported protocol: %s\n"), req->getUrl().c_str());
2006-02-17 13:35:04 +00:00
delete(req);
}
}
void showVersion() {
2006-03-05 06:32:01 +00:00
cout << PACKAGE << _(" version ") << PACKAGE_VERSION << endl;
2006-02-17 13:35:04 +00:00
cout << "Copyright (C) 2006 Tatsuhiro Tsujikawa" << endl;
cout << endl;
2006-03-05 06:32:01 +00:00
cout <<
_("This program is free software; you can redistribute it and/or modify\n"
"it under the terms of the GNU General Public License as published by\n"
"the Free Software Foundation; either version 2 of the License, or\n"
"(at your option) any later version.\n"
"\n"
"This program is distributed in the hope that it will be useful,\n"
"but WITHOUT ANY WARRANTY; without even the implied warranty of\n"
"MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the\n"
"GNU General Public License for more details.\n"
"\n"
"You should have received a copy of the GNU General Public License\n"
"along with this program; if not, write to the Free Software\n"
"Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA\n");
2006-02-17 13:35:04 +00:00
cout << endl;
2006-03-05 06:32:01 +00:00
printf(_("Contact Info: %s\n"), "Tasuhiro Tsujikawa <tujikawa at users dot sourceforge dot net>");
2006-02-17 13:35:04 +00:00
cout << endl;
}
void showUsage() {
2006-03-05 06:32:01 +00:00
printf(_("Usage: %s [options] URL ...\n"), PACKAGE_NAME);
2006-02-21 14:00:58 +00:00
cout << endl;
cout << _("Options:") << endl;
cout << _(" -d, --dir=DIR The directory to store downloaded file.") << endl;
cout << _(" -o, --out=FILE The file name for downloaded file.") << endl;
2006-03-05 06:32:01 +00:00
cout << _(" -l, --log=LOG The file path to store log. If '-' is specified,\n"
" log is written to stdout.") << endl;
cout << _(" -D, --daemon Run as daemon.") << endl;
2006-03-05 06:32:01 +00:00
cout << _(" -s, --split=N Download a file using N connections. N must be\n"
" between 1 and 5. This option affects all URLs.\n"
" Thus, aria2 connects to each URL with\n"
" N connections.") << endl;
cout << _(" --retry-wait=SEC Set amount of time in second between requests\n"
" for errors. Specify a value between 0 and 60.\n"
" Default: 5") << endl;
cout << _(" -t, --timeout=SEC Set timeout in second. Default: 60") << endl;
2006-03-05 06:32:01 +00:00
cout << _(" -m, --max-tries=N Set number of tries. 0 means unlimited.\n"
" Default: 5") << endl;
cout << _(" --min-segment-size=SIZE[K|M] Set minimum segment size. You can append\n"
" K or M(1K = 1024, 1M = 1024K). This\n"
" value must be greater than or equal to\n"
" 1024.") << endl;
cout << _(" --http-proxy=HOST:PORT Use HTTP proxy server. This affects to all\n"
" URLs.") << endl;
cout << _(" --http-user=USER Set HTTP user. This affects to all URLs.") << endl;
cout << _(" --http-passwd=PASSWD Set HTTP password. This affects to all URLs.") << endl;
cout << _(" --http-proxy-user=USER Set HTTP proxy user. This affects to all URLs") << endl;
cout << _(" --http-proxy-passwd=PASSWD Set HTTP proxy password. This affects to all URLs.") << endl;
2006-03-05 06:32:01 +00:00
cout << _(" --http-proxy-method=METHOD Set the method to use in proxy request.\n"
" METHOD is either 'get' or 'tunnel'.\n"
" Default: tunnel") << endl;
cout << _(" --http-auth-scheme=SCHEME Set HTTP authentication scheme. Currently, basic\n"
" is the only supported scheme. You MUST specify\n"
" this option in order to use HTTP authentication\n"
" as well as --http-user and --http-passwd.") << endl;
cout << _(" --referer=REFERER Set Referer. This affects to all URLs.") << endl;
2006-03-05 06:32:01 +00:00
cout << _(" --ftp-user=USER Set FTP user. This affects to all URLs.\n"
" Default: anonymous") << endl;
cout << _(" --ftp-passwd=PASSWD Set FTP password. This affects to all URLs.\n"
" Default: ARIA2USER@") << endl;
cout << _(" --ftp-type=TYPE Set FTP transfer type. TYPE is either 'binary'\n"
" or 'ascii'.\n"
" Default: binary") << endl;
cout << _(" -p, --ftp-pasv Use passive mode in FTP.") << endl;
2006-03-05 06:32:01 +00:00
cout << _(" --ftp-via-http-proxy=METHOD Use HTTP proxy in FTP. METHOD is either 'get' or\n"
" 'tunnel'.\n"
" Default: tunnel") << endl;
2006-03-21 16:07:22 +00:00
#ifdef HAVE_LIBSSL
2006-03-21 14:12:51 +00:00
cout << _(" --torrent-file=TORRENT_FILE The file path to .torrent file.") << endl;
cout << _(" --follow-torrent=true|false Setting this option to false prevents aria2 to\n"
" enter BitTorrent mode even if the filename of\n"
" downloaded file ends with .torrent.\n"
" Default: true") << endl;
2006-03-21 16:07:22 +00:00
#endif // HAVE_LIBSSL
cout << _(" -v, --version Print the version number and exit.") << endl;
cout << _(" -h, --help Print this message and exit.") << endl;
2006-02-21 14:00:58 +00:00
cout << endl;
2006-03-05 06:32:01 +00:00
cout << "URL:" << endl;
cout << _(" You can specify multiple URLs. All URLs must point to the same file\n"
" or downloading fails.") << endl;
2006-02-21 14:00:58 +00:00
cout << endl;
cout << _("Examples:") << endl;
cout << _(" Download a file by 1 connection:") << endl;
2006-02-17 13:35:04 +00:00
cout << " aria2c http://AAA.BBB.CCC/file.zip" << endl;
cout << _(" Download a file by 2 connections:") << endl;
2006-02-17 13:35:04 +00:00
cout << " aria2c -s 2 http://AAA.BBB.CCC/file.zip" << endl;
cout << _(" Download a file by 2 connections, each connects to a different server:") << endl;
2006-02-17 13:35:04 +00:00
cout << " aria2c http://AAA.BBB.CCC/file.zip http://DDD.EEE.FFF/GGG/file.zip" << endl;
cout << _(" You can mix up different protocols:") << endl;
2006-02-21 14:00:58 +00:00
cout << " aria2c http://AAA.BBB.CCC/file.zip ftp://DDD.EEE.FFF/GGG/file.zip" << endl;
2006-03-21 16:07:22 +00:00
#ifdef HAVE_LIBSSL
cout << _(" Download a torrent:") << endl;
2006-03-21 17:59:10 +00:00
cout << " aria2c -o test.torrent http://AAA.BBB.CCC/file.torrent" << endl;
2006-03-21 16:07:22 +00:00
cout << _(" Download a torrent using local .torrent file:") << endl;
cout << " aria2c --torrent-file test.torrent" << endl;
2006-02-21 14:00:58 +00:00
cout << endl;
2006-03-21 16:07:22 +00:00
#endif // HAVE_LIBSSL
2006-03-05 06:32:01 +00:00
printf(_("Reports bugs to %s"), "<tujikawa at users dot sourceforge dot net>");
cout << endl;
2006-02-17 13:35:04 +00:00
}
int main(int argc, char* argv[]) {
2006-03-05 06:32:01 +00:00
#ifdef ENABLE_NLS
setlocale (LC_CTYPE, "");
setlocale (LC_MESSAGES, "");
bindtextdomain (PACKAGE, LOCALEDIR);
textdomain (PACKAGE);
#endif // ENABLE_NLS
2006-02-17 13:35:04 +00:00
bool stdoutLog = false;
string logfile;
string dir;
string ufilename;
2006-02-22 12:16:10 +00:00
int split = 1;
2006-02-17 13:35:04 +00:00
bool daemonMode = false;
2006-02-18 03:59:26 +00:00
string referer;
2006-03-21 14:12:51 +00:00
string torrentFile;
2006-03-21 16:07:22 +00:00
#ifdef HAVE_LIBSSL
2006-03-21 14:12:51 +00:00
bool followTorrent = true;
2006-03-21 16:07:22 +00:00
#else
bool followTorrent = false;
#endif // HAVE_LIBSSL
2006-02-17 13:35:04 +00:00
int c;
Option* op = new Option();
2006-02-21 14:00:58 +00:00
op->put(PREF_RETRY_WAIT, "5");
op->put(PREF_TIMEOUT, "60");
2006-02-22 11:18:47 +00:00
op->put(PREF_MIN_SEGMENT_SIZE, "1048576");// 1M
op->put(PREF_MAX_TRIES, "5");
2006-02-22 15:40:04 +00:00
op->put(PREF_HTTP_PROXY_METHOD, V_TUNNEL);
2006-02-21 14:00:58 +00:00
op->put(PREF_FTP_USER, "anonymous");
op->put(PREF_FTP_PASSWD, "ARIA2USER@");
op->put(PREF_FTP_TYPE, V_BINARY);
op->put(PREF_FTP_VIA_HTTP_PROXY, V_TUNNEL);
2006-03-21 14:12:51 +00:00
op->put(PREF_AUTO_SAVE_INTERVAL, "60");
2006-02-21 12:27:17 +00:00
2006-02-17 13:35:04 +00:00
while(1) {
int optIndex = 0;
int lopt;
static struct option longOpts[] = {
{ "daemon", no_argument, NULL, 'D' },
{ "dir", required_argument, NULL, 'd' },
{ "out", required_argument, NULL, 'o' },
{ "log", required_argument, NULL, 'l' },
{ "split", required_argument, NULL, 's' },
2006-02-21 14:00:58 +00:00
{ "timeout", required_argument, NULL, 't' },
2006-03-05 06:32:01 +00:00
{ "max-tries", required_argument, NULL, 'm' },
2006-02-17 13:35:04 +00:00
{ "http-proxy", required_argument, &lopt, 1 },
{ "http-user", required_argument, &lopt, 2 },
{ "http-passwd", required_argument, &lopt, 3 },
{ "http-proxy-user", required_argument, &lopt, 4 },
{ "http-proxy-passwd", required_argument, &lopt, 5 },
{ "http-auth-scheme", required_argument, &lopt, 6 },
2006-02-18 03:59:26 +00:00
{ "referer", required_argument, &lopt, 7 },
{ "retry-wait", required_argument, &lopt, 8 },
2006-02-21 14:00:58 +00:00
{ "ftp-user", required_argument, &lopt, 9 },
{ "ftp-passwd", required_argument, &lopt, 10 },
{ "ftp-type", required_argument, &lopt, 11 },
{ "ftp-pasv", no_argument, NULL, 'p' },
{ "ftp-via-http-proxy", required_argument, &lopt, 12 },
2006-02-22 11:18:47 +00:00
{ "min-segment-size", required_argument, &lopt, 13 },
2006-02-22 15:40:04 +00:00
{ "http-proxy-method", required_argument, &lopt, 14 },
2006-03-21 16:07:22 +00:00
#ifdef HAVE_LIBSSL
2006-03-21 14:12:51 +00:00
{ "torrent-file", required_argument, &lopt, 15 },
{ "follow-torrent", required_argument, &lopt, 16 },
2006-03-21 16:07:22 +00:00
#endif // HAVE_LIBSSL
2006-02-17 13:35:04 +00:00
{ "version", no_argument, NULL, 'v' },
{ "help", no_argument, NULL, 'h' },
{ 0, 0, 0, 0 }
};
2006-02-22 11:18:47 +00:00
c = getopt_long(argc, argv, "Dd:o:l:s:pt:m:vh", longOpts, &optIndex);
2006-02-17 13:35:04 +00:00
if(c == -1) {
break;
}
switch(c) {
case 0:{
switch(lopt) {
case 1: {
pair<string, string> proxy;
Util::split(proxy, optarg, ':');
int port = (int)strtol(proxy.second.c_str(), NULL, 10);
if(proxy.first.empty() || proxy.second.empty() ||
!(0 < port && port <= 65535)) {
cerr << _("unrecognized proxy format") << endl;
2006-02-17 13:35:04 +00:00
showUsage();
exit(1);
}
2006-02-21 14:00:58 +00:00
op->put(PREF_HTTP_PROXY_HOST, proxy.first);
op->put(PREF_HTTP_PROXY_PORT, Util::itos(port));
op->put(PREF_HTTP_PROXY_ENABLED, V_TRUE);
2006-02-17 13:35:04 +00:00
break;
}
case 2:
2006-02-21 14:00:58 +00:00
op->put(PREF_HTTP_USER, optarg);
2006-02-17 13:35:04 +00:00
break;
case 3:
2006-02-21 14:00:58 +00:00
op->put(PREF_HTTP_PASSWD, optarg);
2006-02-17 13:35:04 +00:00
break;
case 4:
2006-02-21 14:00:58 +00:00
op->put(PREF_HTTP_PROXY_USER, optarg);
op->put(PREF_HTTP_PROXY_AUTH_ENABLED, V_TRUE);
2006-02-17 13:35:04 +00:00
break;
case 5:
2006-02-21 14:00:58 +00:00
op->put(PREF_HTTP_PROXY_PASSWD, optarg);
2006-02-17 13:35:04 +00:00
break;
case 6:
2006-02-21 14:00:58 +00:00
if(string(V_BASIC) == optarg) {
op->put(PREF_HTTP_AUTH_SCHEME, V_BASIC);
2006-02-17 13:35:04 +00:00
} else {
cerr << _("Currently, supported authentication scheme is basic.") << endl;
2006-02-17 13:35:04 +00:00
}
break;
2006-02-18 03:59:26 +00:00
case 7:
referer = optarg;
break;
case 8: {
int wait = (int)strtol(optarg, NULL, 10);
if(!(0 <= wait && wait <= 60)) {
cerr << _("retry-wait must be between 0 and 60.") << endl;
showUsage();
exit(1);
}
2006-02-21 14:00:58 +00:00
op->put(PREF_RETRY_WAIT, Util::itos(wait));
break;
}
2006-02-21 14:00:58 +00:00
case 9:
op->put(PREF_FTP_USER, optarg);
break;
case 10:
op->put(PREF_FTP_PASSWD, optarg);
break;
case 11:
if(string(optarg) == V_BINARY || string(optarg) == V_ASCII) {
op->put(PREF_FTP_TYPE, optarg);
} else {
cerr << _("ftp-type must be either 'binary' or 'ascii'.") << endl;
2006-02-21 14:00:58 +00:00
showUsage();
exit(1);
}
break;
case 12:
if(string(optarg) == V_GET || string(optarg) == V_TUNNEL) {
op->put(PREF_FTP_VIA_HTTP_PROXY, optarg);
} else {
cerr << _("ftp-via-http-proxy must be either 'get' or 'tunnel'.") << endl;
2006-02-21 14:00:58 +00:00
showUsage();
exit(1);
}
break;
2006-02-22 11:18:47 +00:00
case 13: {
string::size_type p = string(optarg).find_first_of("KM");
int mult = 1;
if(p != string::npos) {
if(optarg[p] == 'K') {
mult = 1024;
} else if(optarg[p] == 'M') {
mult = 1024*1024;
}
optarg[p] = '\0';
}
long long int size = strtoll(optarg, NULL, 10)*mult;
2006-03-05 06:32:01 +00:00
if(size < 1024) {
cerr << _("min-segment-size invalid") << endl;
2006-02-22 11:18:47 +00:00
showUsage();
exit(1);
}
op->put(PREF_MIN_SEGMENT_SIZE, Util::llitos(size));
break;
}
2006-02-22 15:40:04 +00:00
case 14:
if(string(optarg) == V_GET || string(optarg) == V_TUNNEL) {
op->put(PREF_HTTP_PROXY_METHOD, optarg);
} else {
cerr << _("http-proxy-method must be either 'get' or 'tunnel'.") << endl;
2006-02-22 15:40:04 +00:00
showUsage();
exit(1);
}
break;
2006-03-21 14:12:51 +00:00
case 15:
torrentFile = string(optarg);
break;
case 16:
2006-03-21 16:54:42 +00:00
if(string(optarg) == "true") {
2006-03-21 14:12:51 +00:00
followTorrent = true;
2006-03-21 16:54:42 +00:00
} else if(string(optarg) == "false") {
2006-03-21 14:12:51 +00:00
followTorrent = false;
} else {
cerr << _("follow-torrent must be either 'true' or 'false'.") << endl;
showUsage();
exit(1);
}
2006-02-17 13:35:04 +00:00
}
break;
}
case 'D':
daemonMode = true;
break;
case 'd':
dir = optarg;
break;
case 'o':
ufilename = optarg;
break;
case 'l':
if(strcmp("-", optarg) == 0) {
stdoutLog = true;
} else {
logfile = optarg;
}
break;
case 's':
split = (int)strtol(optarg, NULL, 10);
2006-02-21 14:00:58 +00:00
if(!(1 <= split && split <= 5)) {
cerr << _("split must be between 1 and 5.") << endl;
2006-02-17 13:35:04 +00:00
showUsage();
exit(1);
}
break;
2006-02-21 14:00:58 +00:00
case 't': {
int timeout = (int)strtol(optarg, NULL, 10);
if(1 <= timeout && timeout <= 600) {
op->put(PREF_TIMEOUT, Util::itos(timeout));
} else {
cerr << _("timeout must be between 1 and 600") << endl;
2006-02-21 14:00:58 +00:00
showUsage();
exit(1);
}
break;
}
2006-02-22 11:18:47 +00:00
case 'm': {
int retries = (int)strtol(optarg, NULL, 10);
if(retries < 0) {
2006-03-05 06:32:01 +00:00
cerr << _("max-tries invalid") << endl;
2006-02-22 11:18:47 +00:00
showUsage();
exit(1);
}
op->put(PREF_MAX_TRIES, Util::itos(retries));
break;
}
2006-02-21 14:00:58 +00:00
case 'p':
op->put(PREF_FTP_PASV_ENABLED, V_TRUE);
break;
2006-02-17 13:35:04 +00:00
case 'v':
showVersion();
exit(0);
case 'h':
showUsage();
exit(0);
default:
showUsage();
exit(1);
}
}
2006-03-21 14:12:51 +00:00
if(torrentFile.empty()) {
if(optind == argc) {
cerr << _("specify at least one URL") << endl;
showUsage();
exit(1);
}
2006-02-17 13:35:04 +00:00
}
if(daemonMode) {
if(daemon(1, 1) < 0) {
perror("daemon failed");
exit(1);
}
}
#ifdef HAVE_LIBSSL
// for SSL initialization
SSL_load_error_strings();
SSL_library_init();
#endif // HAVE_LIBSSL
2006-03-21 14:12:51 +00:00
srandom(time(NULL));
2006-02-17 13:35:04 +00:00
SimpleLogger* logger;
if(stdoutLog) {
logger = new SimpleLogger(stdout);
} else if(logfile.size()) {
logger = new SimpleLogger(logfile);
} else {
logger = new SimpleLogger("/dev/null");
}
2006-02-22 14:30:47 +00:00
SegmentSplitter* splitter = new SplitSlowestSegmentSplitter();
splitter->setMinSegmentSize(op->getAsLLInt(PREF_MIN_SEGMENT_SIZE));
2006-03-21 14:12:51 +00:00
struct sigaction sigactIgn;
sigactIgn.sa_handler = SIG_IGN;
sigactIgn.sa_flags = 0;
sigemptyset(&sigactIgn.sa_mask);
sigaction(SIGPIPE, &sigactIgn, NULL);
bool readyToTorrentMode = false;
string downloadedTorrentFile;
if(torrentFile.empty()) {
struct sigaction sigact;
sigact.sa_handler = handler;
sigact.sa_flags = 0;
sigemptyset(&sigact.sa_mask);
sigaction(SIGINT, &sigact, NULL);
splitter->logger = logger;
e = new ConsoleDownloadEngine();
e->logger = logger;
e->option = op;
e->diskWriter = new DefaultDiskWriter();
e->segmentMan = new SegmentMan();
e->segmentMan->dir = dir;
e->segmentMan->ufilename = ufilename;
e->segmentMan->logger = logger;
e->segmentMan->option = op;
e->segmentMan->splitter = splitter;
vector<Request*> requests;
for(int i = 1; optind+i-1 < argc; i++) {
for(int s = 1; s <= split; s++) {
2006-02-22 12:16:10 +00:00
addCommand(split*(i-1)+s, argv[optind+i-1], referer, requests);
2006-03-21 14:12:51 +00:00
}
2006-02-17 13:35:04 +00:00
}
2006-03-21 14:12:51 +00:00
e->run();
if(e->segmentMan->finished()) {
printDownloadCompeleteMessage(e->segmentMan->getFilePath());
if(Util::endsWith(e->segmentMan->getFilePath(), ".torrent")) {
downloadedTorrentFile = e->segmentMan->getFilePath();
readyToTorrentMode = true;
}
} else {
printDownloadAbortMessage();
}
for_each(requests.begin(), requests.end(), clearRequest);
requests.clear();
2006-02-17 13:35:04 +00:00
2006-03-21 14:12:51 +00:00
delete(e->segmentMan);
delete(e->diskWriter);
delete(e);
}
if(!torrentFile.empty() || followTorrent && readyToTorrentMode) {
try {
op->put(PREF_MAX_TRIES, "0");
struct sigaction sigact;
sigact.sa_handler = torrentHandler;
sigact.sa_flags = 0;
sigemptyset(&sigact.sa_mask);
sigaction(SIGINT, &sigact, NULL);
Request* req = new Request();
req->isTorrent = true;
req->setTrackerEvent(Request::STARTED);
te = new TorrentConsoleDownloadEngine();
te->logger = logger;
te->option = op;
te->diskWriter = new DefaultDiskWriter();
te->segmentMan = new SegmentMan();
te->segmentMan->logger = logger;
te->segmentMan->option = op;
te->segmentMan->splitter = splitter;
te->torrentMan = new TorrentMan();
te->torrentMan->setStoreDir(dir);
te->torrentMan->logger = logger;
te->torrentMan->setup(torrentFile.empty() ?
downloadedTorrentFile : torrentFile);
2006-03-21 14:12:51 +00:00
PeerListenCommand* listenCommand =
new PeerListenCommand(te->torrentMan->getNewCuid(), te);
int port = listenCommand->bindPort(6881, 6999);
if(port == -1) {
printf("an error occurred while binding port.\n");
exit(1);
}
te->torrentMan->setPort(port);
te->commands.push(listenCommand);
te->commands.push(new TrackerInitCommand(te->torrentMan->getNewCuid(),
req, te));
int autoSaveCommandCuid = te->torrentMan->getNewCuid();
te->commands.push(new SleepCommand(autoSaveCommandCuid, te,
new TorrentAutoSaveCommand(autoSaveCommandCuid, te, op->getAsInt(PREF_AUTO_SAVE_INTERVAL)),
op->getAsInt(PREF_AUTO_SAVE_INTERVAL)));
te->run();
if(te->torrentMan->downloadComplete()) {
printDownloadCompeleteMessage(te->torrentMan->getFilePath());
} else {
printDownloadAbortMessage();
}
delete(te->segmentMan);
delete(te->torrentMan);
delete(te);
} catch(Exception* ex) {
cerr << ex->getMsg() << endl;
delete ex;
exit(1);
}
}
2006-03-21 14:12:51 +00:00
2006-02-17 13:35:04 +00:00
delete(logger);
2006-03-21 14:12:51 +00:00
delete(op);
delete(splitter);
2006-02-17 13:35:04 +00:00
return 0;
}