/* */ #include "PeerReceiveHandshakeCommand.h" #include "PeerConnection.h" #include "DownloadEngine.h" #include "BtHandshakeMessage.h" #include "Util.h" #include "BtContext.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 "prefs.h" #include "Option.h" #include "RequestGroupMan.h" #include "StringFormat.h" #include "RequestGroup.h" namespace aria2 { PeerReceiveHandshakeCommand::PeerReceiveHandshakeCommand(int32_t cuid, const PeerHandle& peer, DownloadEngine* e, const SocketHandle& s, const SharedHandle& peerConnection): PeerAbstractCommand(cuid, peer, e, s), _peerConnection(peerConnection), _thresholdSpeed(e->option->getAsInt(PREF_BT_REQUEST_PEER_SPEED_LIMIT)) { if(_peerConnection.isNull()) { _peerConnection.reset(new PeerConnection(cuid, socket, e->option)); } unsigned int maxDownloadSpeed = e->option->getAsInt(PREF_MAX_DOWNLOAD_LIMIT); if(maxDownloadSpeed > 0) { _thresholdSpeed = std::min(maxDownloadSpeed, _thresholdSpeed); } } PeerReceiveHandshakeCommand::~PeerReceiveHandshakeCommand() {} bool PeerReceiveHandshakeCommand::exitBeforeExecute() { return e->isHaltRequested() || e->_requestGroupMan->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 = Util::toHex(&data[28], INFO_HASH_LENGTH); BtContextHandle btContext = BtRegistry::getBtContext(infoHash); if(btContext.isNull() || !BT_RUNTIME(btContext)->ready()) { throw DlAbortEx (StringFormat("Unknown info hash %s", infoHash.c_str()).str()); } TransferStat tstat = btContext->getOwnerRequestGroup()->calculateStat(); if((!PIECE_STORAGE(btContext)->downloadFinished() && tstat.getDownloadSpeed() < _thresholdSpeed) || BT_RUNTIME(btContext)->lessThanMaxPeers()) { if(PEER_STORAGE(btContext)->addPeer(peer)) { peer->usedBy(cuid); PeerInteractionCommand* command = new PeerInteractionCommand(cuid, btContext->getOwnerRequestGroup(), peer, e, btContext, socket, PeerInteractionCommand::RECEIVER_WAIT_HANDSHAKE, _peerConnection); e->commands.push_back(command); logger->debug(MSG_INCOMING_PEER_CONNECTION, cuid, peer->usedBy()); } } return true; } else { e->commands.push_back(this); return false; } } } // namespace aria2