| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247 | /* <!-- copyright *//* * aria2 - The high speed download utility * * 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., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA * * In addition, as a special exception, the copyright holders give * permission to link the code of portions of this program with the * OpenSSL library under certain conditions as described in each * individual source file, and distribute linked combinations * including the two. * You must obey the GNU General Public License in all respects * for all of the code used other than OpenSSL.  If you modify * file(s) with this exception, you may extend this exception to your * version of the file(s), but you are not obligated to do so.  If you * do not wish to do so, delete this exception statement from your * version.  If you delete this exception statement from all source * files in the program, then also delete it here. *//* copyright --> */#include "BtSetup.h"#include <cstring>#include "RequestGroup.h"#include "DownloadEngine.h"#include "Option.h"#include "BtRegistry.h"#include "PeerListenCommand.h"#include "TrackerWatcherCommand.h"#include "SeedCheckCommand.h"#include "PeerChokeCommand.h"#include "ActivePeerConnectionCommand.h"#include "PeerListenCommand.h"#include "UnionSeedCriteria.h"#include "TimeSeedCriteria.h"#include "ShareRatioSeedCriteria.h"#include "prefs.h"#include "LogFactory.h"#include "Logger.h"#include "util.h"#include "IntSequence.h"#include "DHTGetPeersCommand.h"#include "DHTPeerAnnounceStorage.h"#include "DHTSetup.h"#include "DHTRegistry.h"#include "BtProgressInfoFile.h"#include "BtAnnounce.h"#include "BtRuntime.h"#include "bittorrent_helper.h"#include "BtStopDownloadCommand.h"#include "LpdReceiveMessageCommand.h"#include "LpdDispatchMessageCommand.h"#include "LpdMessageReceiver.h"#include "LpdMessageDispatcher.h"#include "message.h"#include "SocketCore.h"namespace aria2 {BtSetup::BtSetup():_logger(LogFactory::getInstance()) {}void BtSetup::setup(std::vector<Command*>& commands,                    RequestGroup* requestGroup,                    DownloadEngine* e,                    const Option* option){  if(!requestGroup->getDownloadContext()->hasAttribute(bittorrent::BITTORRENT)){    return;  }  const BDE& torrentAttrs =    requestGroup->getDownloadContext()->getAttribute(bittorrent::BITTORRENT);  bool metadataGetMode = !torrentAttrs.containsKey(bittorrent::METADATA);  BtObject btObject = e->getBtRegistry()->get(requestGroup->getGID());  SharedHandle<PieceStorage> pieceStorage = btObject._pieceStorage;  SharedHandle<PeerStorage> peerStorage = btObject._peerStorage;  SharedHandle<BtRuntime> btRuntime = btObject._btRuntime;  SharedHandle<BtAnnounce> btAnnounce = btObject._btAnnounce;  // commands  {    TrackerWatcherCommand* c =      new TrackerWatcherCommand(e->newCUID(), requestGroup, e);    c->setPeerStorage(peerStorage);    c->setPieceStorage(pieceStorage);    c->setBtRuntime(btRuntime);    c->setBtAnnounce(btAnnounce);        commands.push_back(c);  }  if(!metadataGetMode) {    PeerChokeCommand* c =      new PeerChokeCommand(e->newCUID(), e);    c->setPeerStorage(peerStorage);    c->setBtRuntime(btRuntime);        commands.push_back(c);  }  {    ActivePeerConnectionCommand* c =      new ActivePeerConnectionCommand(e->newCUID(), requestGroup, e, 10);    c->setBtRuntime(btRuntime);    c->setPieceStorage(pieceStorage);    c->setPeerStorage(peerStorage);    c->setBtAnnounce(btAnnounce);                commands.push_back(c);  }  if((metadataGetMode || torrentAttrs[bittorrent::PRIVATE].i() == 0) &&     DHTSetup::initialized()) {    DHTGetPeersCommand* command =      new DHTGetPeersCommand(e->newCUID(), requestGroup, e);    command->setTaskQueue(DHTRegistry::_taskQueue);    command->setTaskFactory(DHTRegistry::_taskFactory);    command->setBtRuntime(btRuntime);    command->setPeerStorage(peerStorage);    commands.push_back(command);  }  if(!metadataGetMode) {    SharedHandle<UnionSeedCriteria> unionCri(new UnionSeedCriteria());    if(option->defined(PREF_SEED_TIME)) {      SharedHandle<SeedCriteria> cri        (new TimeSeedCriteria(option->getAsInt(PREF_SEED_TIME)*60));      unionCri->addSeedCriteria(cri);    }    {      double ratio = option->getAsDouble(PREF_SEED_RATIO);      if(ratio > 0.0) {        SharedHandle<ShareRatioSeedCriteria> cri          (new ShareRatioSeedCriteria(option->getAsDouble(PREF_SEED_RATIO),                                      requestGroup->getDownloadContext()));        cri->setPieceStorage(pieceStorage);        cri->setPeerStorage(peerStorage);        unionCri->addSeedCriteria(cri);      }    }    if(unionCri->getSeedCriterion().size() > 0) {      SeedCheckCommand* c =        new SeedCheckCommand(e->newCUID(), requestGroup, e, unionCri);      c->setPieceStorage(pieceStorage);      c->setBtRuntime(btRuntime);      commands.push_back(c);    }  }  if(PeerListenCommand::getNumInstance() == 0) {    PeerListenCommand* listenCommand = PeerListenCommand::getInstance(e);    IntSequence seq = util::parseIntRange(option->get(PREF_LISTEN_PORT));    uint16_t port;    if(listenCommand->bindPort(port, seq)) {      btRuntime->setListenPort(port);      // Add command to DownloadEngine directly.      e->commands.push_back(listenCommand);    } else {      delete listenCommand;      throw DL_ABORT_EX(_("Errors occurred while binding port.\n"));    }  } else {    PeerListenCommand* listenCommand = PeerListenCommand::getInstance(e);    btRuntime->setListenPort(listenCommand->getPort());  }  if(option->getAsBool(PREF_BT_ENABLE_LPD) &&     (metadataGetMode || torrentAttrs[bittorrent::PRIVATE].i() == 0)) {        if(LpdReceiveMessageCommand::getNumInstance() == 0) {      _logger->info("Initializing LpdMessageReceiver.");      SharedHandle<LpdMessageReceiver> receiver        (new LpdMessageReceiver(LPD_MULTICAST_ADDR, LPD_MULTICAST_PORT));      bool initialized = false;      const std::string& lpdInterface = option->get(PREF_BT_LPD_INTERFACE);      if(lpdInterface.empty()) {        if(receiver->init("")) {          initialized = true;        }      } else {        std::vector<std::pair<sockaddr_storage, socklen_t> > ifAddrs;        getInterfaceAddress(ifAddrs, lpdInterface, AF_INET, AI_NUMERICHOST);        for(std::vector<std::pair<sockaddr_storage, socklen_t> >::const_iterator              i = ifAddrs.begin(); i != ifAddrs.end(); ++i) {          sockaddr_in addr;          memcpy(&addr, &(*i).first, (*i).second);          if(receiver->init(inet_ntoa(addr.sin_addr))) {            initialized = true;            break;          }        }      }      if(initialized) {        _logger->info("LpdMessageReceiver initialized. multicastAddr=%s:%u,"                      " localAddr=%s",                      LPD_MULTICAST_ADDR, LPD_MULTICAST_PORT,                      receiver->getLocalAddress().c_str());        LpdReceiveMessageCommand* cmd =          LpdReceiveMessageCommand::getInstance(e, receiver);        e->commands.push_back(cmd);      } else {        _logger->info("LpdMessageReceiver not initialized.");      }    }    if(LpdReceiveMessageCommand::getNumInstance()) {      const unsigned char* infoHash =        bittorrent::getInfoHash(requestGroup->getDownloadContext());      SharedHandle<LpdMessageReceiver> receiver =        LpdReceiveMessageCommand::getInstance()->getLpdMessageReceiver();      _logger->info("Initializing LpdMessageDispatcher.");            SharedHandle<LpdMessageDispatcher> dispatcher        (new LpdMessageDispatcher         (std::string(&infoHash[0], &infoHash[INFO_HASH_LENGTH]),          btRuntime->getListenPort(),          LPD_MULTICAST_ADDR, LPD_MULTICAST_PORT));      if(dispatcher->init(receiver->getLocalAddress(), /*ttl*/1, /*loop*/0)) {        _logger->info("LpdMessageDispatcher initialized.");              LpdDispatchMessageCommand* cmd =          new LpdDispatchMessageCommand(e->newCUID(), dispatcher, e);        cmd->setBtRuntime(btRuntime);        e->commands.push_back(cmd);      } else {        _logger->info("LpdMessageDispatcher not initialized.");      }    }  }  time_t btStopTimeout = option->getAsInt(PREF_BT_STOP_TIMEOUT);  if(btStopTimeout > 0) {    BtStopDownloadCommand* stopDownloadCommand =      new BtStopDownloadCommand(e->newCUID(), requestGroup, e, btStopTimeout);    stopDownloadCommand->setBtRuntime(btRuntime);    stopDownloadCommand->setPieceStorage(pieceStorage);    commands.push_back(stopDownloadCommand);  }  btRuntime->setReady(true);}} // namespace aria2
 |