aria2/src/DefaultBtMessageReceiver.cc

159 lines
5.0 KiB
C++

/* <!-- 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 "DefaultBtMessageReceiver.h"
#include <cstring>
#include "BtHandshakeMessage.h"
#include "message.h"
#include "DownloadContext.h"
#include "Peer.h"
#include "PeerConnection.h"
#include "BtMessageDispatcher.h"
#include "BtMessageFactory.h"
#include "Logger.h"
#include "LogFactory.h"
#include "bittorrent_helper.h"
#include "BtPieceMessage.h"
#include "util.h"
#include "fmt.h"
#include "DlAbortEx.h"
namespace aria2 {
DefaultBtMessageReceiver::DefaultBtMessageReceiver()
: handshakeSent_{false},
downloadContext_{nullptr},
peerConnection_{nullptr},
dispatcher_{nullptr},
messageFactory_{nullptr}
{
}
std::unique_ptr<BtHandshakeMessage>
DefaultBtMessageReceiver::receiveHandshake(bool quickReply)
{
A2_LOG_DEBUG(
fmt("Receiving handshake bufferLength=%lu",
static_cast<unsigned long>(peerConnection_->getBufferLength())));
unsigned char data[BtHandshakeMessage::MESSAGE_LENGTH];
size_t dataLength = BtHandshakeMessage::MESSAGE_LENGTH;
if (handshakeSent_ || !quickReply ||
peerConnection_->getBufferLength() < 48) {
if (peerConnection_->receiveHandshake(data, dataLength)) {
auto msg = messageFactory_->createHandshakeMessage(data, dataLength);
msg->validate();
return msg;
}
}
else {
// Handle tracker's NAT-checking feature
handshakeSent_ = true;
// check info_hash
if (memcmp(bittorrent::getInfoHash(downloadContext_),
peerConnection_->getBuffer() + 28, INFO_HASH_LENGTH) == 0) {
sendHandshake();
}
else {
throw DL_ABORT_EX(
fmt("Bad Info Hash %s",
util::toHex(peerConnection_->getBuffer() + 28, INFO_HASH_LENGTH)
.c_str()));
}
if (peerConnection_->getBufferLength() ==
BtHandshakeMessage::MESSAGE_LENGTH &&
peerConnection_->receiveHandshake(data, dataLength)) {
auto msg = messageFactory_->createHandshakeMessage(data, dataLength);
msg->validate();
return msg;
}
}
return nullptr;
}
std::unique_ptr<BtHandshakeMessage>
DefaultBtMessageReceiver::receiveAndSendHandshake()
{
return receiveHandshake(true);
}
void DefaultBtMessageReceiver::sendHandshake()
{
dispatcher_->addMessageToQueue(messageFactory_->createHandshakeMessage(
bittorrent::getInfoHash(downloadContext_),
bittorrent::getStaticPeerId()));
dispatcher_->sendMessages();
}
std::unique_ptr<BtMessage> DefaultBtMessageReceiver::receiveMessage()
{
size_t dataLength = 0;
// Give 0 to PeerConnection::receiveMessage() to prevent memcpy.
if (!peerConnection_->receiveMessage(nullptr, dataLength)) {
return nullptr;
}
auto msg = messageFactory_->createBtMessage(
peerConnection_->getMsgPayloadBuffer(), dataLength);
msg->validate();
if (msg->getId() == BtPieceMessage::ID) {
auto piecemsg = static_cast<BtPieceMessage*>(msg.get());
piecemsg->setMsgPayload(peerConnection_->getMsgPayloadBuffer());
}
return msg;
}
void DefaultBtMessageReceiver::setDownloadContext(
DownloadContext* downloadContext)
{
downloadContext_ = downloadContext;
}
void DefaultBtMessageReceiver::setPeerConnection(PeerConnection* peerConnection)
{
peerConnection_ = peerConnection;
}
void DefaultBtMessageReceiver::setDispatcher(BtMessageDispatcher* dispatcher)
{
dispatcher_ = dispatcher;
}
void DefaultBtMessageReceiver::setBtMessageFactory(BtMessageFactory* factory)
{
messageFactory_ = factory;
}
} // namespace aria2