/* */ #include "PeerInteractionCommand.h" #include #include "DownloadEngine.h" #include "PeerInitiateConnectionCommand.h" #include "DefaultBtInteractive.h" #include "DlAbortEx.h" #include "message.h" #include "prefs.h" #include "Socket.h" #include "Option.h" #include "DownloadContext.h" #include "Peer.h" #include "BtMessage.h" #include "BtRuntime.h" #include "PeerStorage.h" #include "DefaultBtMessageDispatcher.h" #include "DefaultBtMessageReceiver.h" #include "DefaultBtRequestFactory.h" #include "DefaultBtMessageFactory.h" #include "DefaultBtInteractive.h" #include "PeerConnection.h" #include "ExtensionMessageFactory.h" #include "DHTRoutingTable.h" #include "DHTTaskQueue.h" #include "DHTTaskFactory.h" #include "DHTNode.h" #include "DHTRegistry.h" #include "DHTPeerAnnounceStorage.h" #include "DHTTokenTracker.h" #include "DHTMessageDispatcher.h" #include "DHTMessageReceiver.h" #include "DHTMessageFactory.h" #include "DHTMessageCallback.h" #include "PieceStorage.h" #include "RequestGroup.h" #include "DefaultExtensionMessageFactory.h" #include "RequestGroupMan.h" #include "ExtensionMessageRegistry.h" #include "bittorrent_helper.h" #include "UTMetadataRequestFactory.h" #include "UTMetadataRequestTracker.h" #include "BtRegistry.h" namespace aria2 { PeerInteractionCommand::PeerInteractionCommand (cuid_t cuid, RequestGroup* requestGroup, const SharedHandle& p, DownloadEngine* e, const SharedHandle& btRuntime, const SharedHandle& pieceStorage, const SharedHandle& peerStorage, const SocketHandle& s, Seq sequence, const PeerConnectionHandle& passedPeerConnection) :PeerAbstractCommand(cuid, p, e, s), requestGroup_(requestGroup), btRuntime_(btRuntime), pieceStorage_(pieceStorage), peerStorage_(peerStorage), sequence_(sequence) { // TODO move following bunch of processing to separate method, like init() if(sequence_ == INITIATOR_SEND_HANDSHAKE) { disableReadCheckSocket(); setWriteCheckSocket(getSocket()); setTimeout(getOption()->getAsInt(PREF_PEER_CONNECTION_TIMEOUT)); } int family; unsigned char compact[COMPACT_LEN_IPV6]; int compactlen = bittorrent::packcompact (compact, getPeer()->getIPAddress(), getPeer()->getPort()); if(compactlen == COMPACT_LEN_IPV6) { family = AF_INET6; } else { family = AF_INET; } SharedHandle torrentAttrs = bittorrent::getTorrentAttrs(requestGroup_->getDownloadContext()); bool metadataGetMode = torrentAttrs->metadata.empty(); SharedHandle exMsgRegistry (new ExtensionMessageRegistry()); SharedHandle utMetadataRequestFactory; SharedHandle utMetadataRequestTracker; if(metadataGetMode) { utMetadataRequestFactory.reset(new UTMetadataRequestFactory()); utMetadataRequestTracker.reset(new UTMetadataRequestTracker()); } SharedHandle extensionMessageFactory (new DefaultExtensionMessageFactory(getPeer(), exMsgRegistry)); extensionMessageFactory->setPeerStorage(peerStorage); extensionMessageFactory->setDownloadContext (requestGroup_->getDownloadContext()); extensionMessageFactory->setUTMetadataRequestTracker (utMetadataRequestTracker.get()); // PieceStorage will be set later. SharedHandle factory(new DefaultBtMessageFactory()); factory->setCuid(cuid); factory->setDownloadContext(requestGroup_->getDownloadContext()); factory->setPieceStorage(pieceStorage); factory->setPeerStorage(peerStorage); factory->setExtensionMessageFactory(extensionMessageFactory); factory->setPeer(getPeer()); if(family == AF_INET) { factory->setLocalNode(DHTRegistry::getData().localNode.get()); factory->setRoutingTable(DHTRegistry::getData().routingTable.get()); factory->setTaskQueue(DHTRegistry::getData().taskQueue.get()); factory->setTaskFactory(DHTRegistry::getData().taskFactory.get()); } else { factory->setLocalNode(DHTRegistry::getData6().localNode.get()); factory->setRoutingTable(DHTRegistry::getData6().routingTable.get()); factory->setTaskQueue(DHTRegistry::getData6().taskQueue.get()); factory->setTaskFactory(DHTRegistry::getData6().taskFactory.get()); } if(metadataGetMode) { factory->enableMetadataGetMode(); } PeerConnectionHandle peerConnection; if(!passedPeerConnection) { peerConnection.reset(new PeerConnection(cuid, getPeer(), getSocket())); } else { peerConnection = passedPeerConnection; if(sequence_ == RECEIVER_WAIT_HANDSHAKE && peerConnection->getBufferLength() > 0) { setStatus(Command::STATUS_ONESHOT_REALTIME); getDownloadEngine()->setNoWait(true); } } // If the number of pieces gets bigger, the length of Bitfield // message payload exceeds the initial buffer capacity of // PeerConnection, which is MAX_PAYLOAD_LEN. We expand buffer as // necessary so that PeerConnection can receive the Bitfield // message. size_t bitfieldPayloadSize = 1+(requestGroup_->getDownloadContext()->getNumPieces()+7)/8; peerConnection->reserveBuffer(bitfieldPayloadSize); SharedHandle dispatcher (new DefaultBtMessageDispatcher()); dispatcher->setCuid(cuid); dispatcher->setPeer(getPeer()); dispatcher->setDownloadContext(requestGroup_->getDownloadContext()); dispatcher->setPieceStorage(pieceStorage); dispatcher->setPeerStorage(peerStorage); dispatcher->setRequestTimeout(getOption()->getAsInt(PREF_BT_REQUEST_TIMEOUT)); dispatcher->setBtMessageFactory(factory.get()); dispatcher->setRequestGroupMan (getDownloadEngine()->getRequestGroupMan().get()); DefaultBtMessageReceiverHandle receiver(new DefaultBtMessageReceiver()); receiver->setDownloadContext(requestGroup_->getDownloadContext()); receiver->setPeerConnection(peerConnection.get()); receiver->setDispatcher(dispatcher.get()); receiver->setBtMessageFactory(factory.get()); SharedHandle reqFactory (new DefaultBtRequestFactory()); reqFactory->setPeer(getPeer()); reqFactory->setPieceStorage(pieceStorage); reqFactory->setBtMessageDispatcher(dispatcher.get()); reqFactory->setBtMessageFactory(factory.get()); reqFactory->setCuid(cuid); DefaultBtInteractiveHandle btInteractive (new DefaultBtInteractive(requestGroup_->getDownloadContext(), getPeer())); btInteractive->setBtRuntime(btRuntime_); btInteractive->setPieceStorage(pieceStorage_); btInteractive->setPeerStorage(peerStorage); // Note: Not a member variable. btInteractive->setCuid(cuid); btInteractive->setBtMessageReceiver(receiver); btInteractive->setDispatcher(dispatcher); btInteractive->setBtRequestFactory(reqFactory); btInteractive->setPeerConnection(peerConnection); btInteractive->setExtensionMessageFactory(extensionMessageFactory); btInteractive->setExtensionMessageRegistry(exMsgRegistry); btInteractive->setKeepAliveInterval (getOption()->getAsInt(PREF_BT_KEEP_ALIVE_INTERVAL)); btInteractive->setRequestGroupMan (getDownloadEngine()->getRequestGroupMan().get()); btInteractive->setBtMessageFactory(factory); if((metadataGetMode || !torrentAttrs->privateTorrent) && !getPeer()->isLocalPeer()) { if(getOption()->getAsBool(PREF_ENABLE_PEER_EXCHANGE)) { btInteractive->setUTPexEnabled(true); } if(family == AF_INET) { if(DHTRegistry::isInitialized()) { btInteractive->setDHTEnabled(true); factory->setDHTEnabled(true); btInteractive->setLocalNode(DHTRegistry::getData().localNode.get()); } } else { if(DHTRegistry::isInitialized6()) { btInteractive->setDHTEnabled(true); factory->setDHTEnabled(true); btInteractive->setLocalNode(DHTRegistry::getData6().localNode.get()); } } } btInteractive->setUTMetadataRequestFactory(utMetadataRequestFactory); btInteractive->setUTMetadataRequestTracker(utMetadataRequestTracker); btInteractive->setTcpPort(e->getBtRegistry()->getTcpPort()); if(metadataGetMode) { btInteractive->enableMetadataGetMode(); } btInteractive_ = btInteractive; // reverse depends factory->setBtMessageDispatcher(dispatcher.get()); factory->setBtRequestFactory(reqFactory.get()); factory->setPeerConnection(peerConnection.get()); extensionMessageFactory->setBtMessageDispatcher(dispatcher.get()); extensionMessageFactory->setBtMessageFactory(factory.get()); if(metadataGetMode) { utMetadataRequestFactory->setCuid(cuid); utMetadataRequestFactory->setDownloadContext (requestGroup_->getDownloadContext()); utMetadataRequestFactory->setBtMessageDispatcher(dispatcher.get()); utMetadataRequestFactory->setBtMessageFactory(factory.get()); utMetadataRequestFactory->setPeer(getPeer()); utMetadataRequestFactory->setUTMetadataRequestTracker (utMetadataRequestTracker.get()); } getPeer()->allocateSessionResource (requestGroup_->getDownloadContext()->getPieceLength(), requestGroup_->getDownloadContext()->getTotalLength()); getPeer()->setBtMessageDispatcher(dispatcher.get()); btRuntime_->increaseConnections(); requestGroup_->increaseNumCommand(); } PeerInteractionCommand::~PeerInteractionCommand() { if(getPeer()->getCompletedLength() > 0) { pieceStorage_->subtractPieceStats(getPeer()->getBitfield(), getPeer()->getBitfieldLength()); } getPeer()->releaseSessionResource(); requestGroup_->decreaseNumCommand(); btRuntime_->decreaseConnections(); } bool PeerInteractionCommand::executeInternal() { setNoCheck(false); bool done = false; while(!done) { switch(sequence_) { case INITIATOR_SEND_HANDSHAKE: if(!getSocket()->isWritable(0)) { done = true; break; } disableWriteCheckSocket(); setReadCheckSocket(getSocket()); //socket->setBlockingMode(); setTimeout(getOption()->getAsInt(PREF_BT_TIMEOUT)); btInteractive_->initiateHandshake(); sequence_ = INITIATOR_WAIT_HANDSHAKE; break; case INITIATOR_WAIT_HANDSHAKE: { if(btInteractive_->countPendingMessage() > 0) { btInteractive_->sendPendingMessage(); if(btInteractive_->countPendingMessage() > 0) { done = true; break; } } BtMessageHandle handshakeMessage = btInteractive_->receiveHandshake(); if(!handshakeMessage) { done = true; break; } btInteractive_->doPostHandshakeProcessing(); sequence_ = WIRED; break; } case RECEIVER_WAIT_HANDSHAKE: { BtMessageHandle handshakeMessage = btInteractive_->receiveAndSendHandshake(); if(!handshakeMessage) { done = true; break; } btInteractive_->doPostHandshakeProcessing(); sequence_ = WIRED; break; } case WIRED: // See the comment for writable check below. disableWriteCheckSocket(); btInteractive_->doInteractionProcessing(); if(btInteractive_->countReceivedMessageInIteration() > 0) { updateKeepAlive(); } if((getPeer()->amInterested() && !getPeer()->peerChoking()) || btInteractive_->countOutstandingRequest() || (getPeer()->peerInterested() && !getPeer()->amChoking())) { // Writable check to avoid slow seeding if(btInteractive_->isSendingMessageInProgress()) { setWriteCheckSocket(getSocket()); } if(getDownloadEngine()->getRequestGroupMan()-> doesOverallDownloadSpeedExceed() || requestGroup_->doesDownloadSpeedExceed()) { disableReadCheckSocket(); setNoCheck(true); } else { setReadCheckSocket(getSocket()); } } else { disableReadCheckSocket(); } done = true; break; } } if(btInteractive_->countPendingMessage() > 0) { setNoCheck(true); } getDownloadEngine()->addCommand(this); return false; } // TODO this method removed when PeerBalancerCommand is implemented bool PeerInteractionCommand::prepareForNextPeer(time_t wait) { if(peerStorage_->isPeerAvailable() && btRuntime_->lessThanEqMinPeers()) { SharedHandle peer = peerStorage_->getUnusedPeer(); peer->usedBy(getDownloadEngine()->newCUID()); PeerInitiateConnectionCommand* command = new PeerInitiateConnectionCommand (peer->usedBy(), requestGroup_, peer, getDownloadEngine(), btRuntime_); command->setPeerStorage(peerStorage_); command->setPieceStorage(pieceStorage_); getDownloadEngine()->addCommand(command); } return true; } void PeerInteractionCommand::onAbort() { btInteractive_->cancelAllPiece(); peerStorage_->returnPeer(getPeer()); } void PeerInteractionCommand::onFailure(const Exception& err) { requestGroup_->setLastErrorCode(err.getErrorCode()); requestGroup_->setHaltRequested(true); getDownloadEngine()->setRefreshInterval(0); } bool PeerInteractionCommand::exitBeforeExecute() { return btRuntime_->isHalt(); } const SharedHandle