/* */ #include "PeerReceiveHandshakeCommand.h" #include "PeerConnection.h" #include "DownloadEngine.h" #include "BtHandshakeMessage.h" #include "util.h" #include "DownloadContext.h" #include "DlAbortEx.h" #include "PeerInteractionCommand.h" #include "Peer.h" #include "BtRegistry.h" #include "PeerStorage.h" #include "PieceStorage.h" #include "BtRuntime.h" #include "BtConstants.h" #include "message.h" #include "Socket.h" #include "Logger.h" #include "LogFactory.h" #include "prefs.h" #include "Option.h" #include "RequestGroupMan.h" #include "fmt.h" #include "RequestGroup.h" namespace aria2 { PeerReceiveHandshakeCommand::PeerReceiveHandshakeCommand (cuid_t cuid, const SharedHandle& peer, DownloadEngine* e, const SocketHandle& s, const SharedHandle& peerConnection) : PeerAbstractCommand(cuid, peer, e, s), peerConnection_(peerConnection) { if(!peerConnection_) { peerConnection_.reset(new PeerConnection(cuid, getPeer(), getSocket())); } } PeerReceiveHandshakeCommand::~PeerReceiveHandshakeCommand() {} bool PeerReceiveHandshakeCommand::exitBeforeExecute() { return getDownloadEngine()->isHaltRequested() || getDownloadEngine()->getRequestGroupMan()->downloadFinished(); } bool PeerReceiveHandshakeCommand::executeInternal() { unsigned char data[BtHandshakeMessage::MESSAGE_LENGTH]; size_t dataLength = BtHandshakeMessage::MESSAGE_LENGTH; // ignore return value. The received data is kept in PeerConnection object // because of peek = true. peerConnection_->receiveHandshake(data, dataLength, true); // To handle tracker's NAT-checking feature if(dataLength >= 48) { // check info_hash std::string infoHash = std::string(&data[28], &data[28+INFO_HASH_LENGTH]); SharedHandle downloadContext = getDownloadEngine()->getBtRegistry()->getDownloadContext(infoHash); if(!downloadContext) { throw DL_ABORT_EX (fmt("Unknown info hash %s", util::toHex(infoHash).c_str())); } BtObject btObject = getDownloadEngine()->getBtRegistry()->get (downloadContext->getOwnerRequestGroup()->getGID()); SharedHandle btRuntime = btObject.btRuntime_; SharedHandle pieceStorage = btObject.pieceStorage_; SharedHandle peerStorage = btObject.peerStorage_; if(!btRuntime->ready()) { throw DL_ABORT_EX (fmt("Unknown info hash %s", util::toHex(infoHash).c_str())); } if(btRuntime->isHalt()) { A2_LOG_DEBUG("Info hash found but the download is over." " Dropping connection."); return true; } TransferStat tstat = downloadContext->getOwnerRequestGroup()->calculateStat(); const unsigned int maxDownloadLimit = downloadContext->getOwnerRequestGroup()->getMaxDownloadSpeedLimit(); unsigned int thresholdSpeed = downloadContext->getOwnerRequestGroup()-> getOption()->getAsInt(PREF_BT_REQUEST_PEER_SPEED_LIMIT); if(maxDownloadLimit > 0) { thresholdSpeed = std::min(maxDownloadLimit, thresholdSpeed); } if((!pieceStorage->downloadFinished() && tstat.getDownloadSpeed() < thresholdSpeed) || btRuntime->lessThanMaxPeers()) { if(peerStorage->addPeer(getPeer())) { getPeer()->usedBy(getCuid()); PeerInteractionCommand* command = new PeerInteractionCommand (getCuid(), downloadContext->getOwnerRequestGroup(), getPeer(), getDownloadEngine(), btRuntime, pieceStorage, peerStorage, getSocket(), PeerInteractionCommand::RECEIVER_WAIT_HANDSHAKE, peerConnection_); getDownloadEngine()->addCommand(command); A2_LOG_DEBUG(fmt(MSG_INCOMING_PEER_CONNECTION, getCuid(), util::itos(getPeer()->usedBy()).c_str())); } } return true; } else { getDownloadEngine()->addCommand(this); return false; } } } // namespace aria2