Include local IP address to DHT GetPeers reply message

This is required when DHT is used in private network, where seeder is
initially not known to all nodes.  This functionality requires
--bt-external-ip is set since aria2 cannot figure out external IP
address to advertise in general.
This commit is contained in:
Tatsuhiro Tsujikawa 2016-05-27 23:26:46 +09:00
parent c57259f8e7
commit 3e00be26e8
7 changed files with 127 additions and 7 deletions

View file

@ -46,6 +46,10 @@
#include "DHTTokenTracker.h" #include "DHTTokenTracker.h"
#include "DHTGetPeersReplyMessage.h" #include "DHTGetPeersReplyMessage.h"
#include "util.h" #include "util.h"
#include "BtRegistry.h"
#include "DownloadContext.h"
#include "Option.h"
#include "SocketCore.h"
namespace aria2 { namespace aria2 {
@ -59,18 +63,61 @@ DHTGetPeersMessage::DHTGetPeersMessage(
const std::string& transactionID) const std::string& transactionID)
: DHTQueryMessage{localNode, remoteNode, transactionID}, : DHTQueryMessage{localNode, remoteNode, transactionID},
peerAnnounceStorage_{nullptr}, peerAnnounceStorage_{nullptr},
tokenTracker_{nullptr} tokenTracker_{nullptr},
btRegistry_{nullptr},
family_{AF_INET}
{ {
memcpy(infoHash_, infoHash, DHT_ID_LENGTH); memcpy(infoHash_, infoHash, DHT_ID_LENGTH);
} }
void DHTGetPeersMessage::addLocalPeer(std::vector<std::shared_ptr<Peer>>& peers)
{
if (!btRegistry_) {
return;
}
auto& dctx = btRegistry_->getDownloadContext(
std::string(infoHash_, infoHash_ + DHT_ID_LENGTH));
if (!dctx) {
return;
}
auto group = dctx->getOwnerRequestGroup();
auto& option = group->getOption();
auto& externalIP = option->get(PREF_BT_EXTERNAL_IP);
if (externalIP.empty()) {
return;
}
std::array<uint8_t, sizeof(struct in6_addr)> dst;
if (inetPton(family_, externalIP.c_str(), dst.data()) == -1) {
return;
}
auto tcpPort = btRegistry_->getTcpPort();
if (std::find_if(std::begin(peers), std::end(peers),
[&externalIP, tcpPort](const std::shared_ptr<Peer>& peer) {
return peer->getIPAddress() == externalIP &&
peer->getPort() == tcpPort;
}) != std::end(peers)) {
return;
}
peers.push_back(std::make_shared<Peer>(externalIP, tcpPort));
}
void DHTGetPeersMessage::doReceivedAction() void DHTGetPeersMessage::doReceivedAction()
{ {
std::string token = tokenTracker_->generateToken( std::string token = tokenTracker_->generateToken(
infoHash_, getRemoteNode()->getIPAddress(), getRemoteNode()->getPort()); infoHash_, getRemoteNode()->getIPAddress(), getRemoteNode()->getPort());
// Check to see localhost has the contents which has same infohash
std::vector<std::shared_ptr<Peer>> peers; std::vector<std::shared_ptr<Peer>> peers;
peerAnnounceStorage_->getPeers(peers, infoHash_); peerAnnounceStorage_->getPeers(peers, infoHash_);
// Check to see localhost has the contents which has same infohash
addLocalPeer(peers);
std::vector<std::shared_ptr<DHTNode>> nodes; std::vector<std::shared_ptr<DHTNode>> nodes;
getRoutingTable()->getClosestKNodes(nodes, infoHash_); getRoutingTable()->getClosestKNodes(nodes, infoHash_);
getMessageDispatcher()->addMessageToQueue( getMessageDispatcher()->addMessageToQueue(
@ -102,6 +149,13 @@ void DHTGetPeersMessage::setTokenTracker(DHTTokenTracker* tokenTracker)
tokenTracker_ = tokenTracker; tokenTracker_ = tokenTracker;
} }
void DHTGetPeersMessage::setBtRegistry(BtRegistry* btRegistry)
{
btRegistry_ = btRegistry;
}
void DHTGetPeersMessage::setFamily(int family) { family_ = family; }
std::string DHTGetPeersMessage::toStringOptional() const std::string DHTGetPeersMessage::toStringOptional() const
{ {
return "info_hash=" + util::toHex(infoHash_, INFO_HASH_LENGTH); return "info_hash=" + util::toHex(infoHash_, INFO_HASH_LENGTH);

View file

@ -36,6 +36,9 @@
#define D_DHT_GET_PEERS_MESSAGE_H #define D_DHT_GET_PEERS_MESSAGE_H
#include "DHTQueryMessage.h" #include "DHTQueryMessage.h"
#include <vector>
#include "DHTConstants.h" #include "DHTConstants.h"
#include "A2STR.h" #include "A2STR.h"
@ -43,6 +46,8 @@ namespace aria2 {
class DHTPeerAnnounceStorage; class DHTPeerAnnounceStorage;
class DHTTokenTracker; class DHTTokenTracker;
class BtRegistry;
class Peer;
class DHTGetPeersMessage : public DHTQueryMessage { class DHTGetPeersMessage : public DHTQueryMessage {
private: private:
@ -52,6 +57,12 @@ private:
DHTTokenTracker* tokenTracker_; DHTTokenTracker* tokenTracker_;
BtRegistry* btRegistry_;
int family_;
void addLocalPeer(std::vector<std::shared_ptr<Peer>>& peers);
protected: protected:
virtual std::string toStringOptional() const CXX11_OVERRIDE; virtual std::string toStringOptional() const CXX11_OVERRIDE;
@ -73,6 +84,10 @@ public:
void setTokenTracker(DHTTokenTracker* tokenTracker); void setTokenTracker(DHTTokenTracker* tokenTracker);
void setBtRegistry(BtRegistry* btRegistry);
void setFamily(int family);
static const std::string GET_PEERS; static const std::string GET_PEERS;
static const std::string INFO_HASH; static const std::string INFO_HASH;

View file

@ -70,7 +70,8 @@ DHTMessageFactoryImpl::DHTMessageFactoryImpl(int family)
dispatcher_{nullptr}, dispatcher_{nullptr},
routingTable_{nullptr}, routingTable_{nullptr},
peerAnnounceStorage_{nullptr}, peerAnnounceStorage_{nullptr},
tokenTracker_{nullptr} tokenTracker_{nullptr},
btRegistry_{nullptr}
{ {
} }
@ -409,6 +410,8 @@ DHTMessageFactoryImpl::createGetPeersMessage(
transactionID); transactionID);
m->setPeerAnnounceStorage(peerAnnounceStorage_); m->setPeerAnnounceStorage(peerAnnounceStorage_);
m->setTokenTracker(tokenTracker_); m->setTokenTracker(tokenTracker_);
m->setBtRegistry(btRegistry_);
m->setFamily(family_);
setCommonProperty(m.get()); setCommonProperty(m.get());
return m; return m;
} }
@ -529,4 +532,9 @@ void DHTMessageFactoryImpl::setLocalNode(
localNode_ = localNode; localNode_ = localNode;
} }
void DHTMessageFactoryImpl::setBtRegistry(BtRegistry* btRegistry)
{
btRegistry_ = btRegistry;
}
} // namespace aria2 } // namespace aria2

View file

@ -47,6 +47,7 @@ class DHTPeerAnnounceStorage;
class DHTTokenTracker; class DHTTokenTracker;
class DHTMessage; class DHTMessage;
class DHTAbstractMessage; class DHTAbstractMessage;
class BtRegistry;
class DHTMessageFactoryImpl : public DHTMessageFactory { class DHTMessageFactoryImpl : public DHTMessageFactory {
private: private:
@ -64,6 +65,8 @@ private:
DHTTokenTracker* tokenTracker_; DHTTokenTracker* tokenTracker_;
BtRegistry* btRegistry_;
// search node in routingTable. If it is not found, create new one. // search node in routingTable. If it is not found, create new one.
std::shared_ptr<DHTNode> getRemoteNode(const unsigned char* id, std::shared_ptr<DHTNode> getRemoteNode(const unsigned char* id,
const std::string& ipaddr, const std::string& ipaddr,
@ -154,6 +157,8 @@ public:
void setTokenTracker(DHTTokenTracker* tokenTracker); void setTokenTracker(DHTTokenTracker* tokenTracker);
void setLocalNode(const std::shared_ptr<DHTNode>& localNode); void setLocalNode(const std::shared_ptr<DHTNode>& localNode);
void setBtRegistry(BtRegistry* btRegistry);
}; };
} // namespace aria2 } // namespace aria2

View file

@ -180,6 +180,7 @@ DHTSetup::setup(DownloadEngine* e, int family)
factory->setPeerAnnounceStorage(peerAnnounceStorage.get()); factory->setPeerAnnounceStorage(peerAnnounceStorage.get());
factory->setTokenTracker(tokenTracker.get()); factory->setTokenTracker(tokenTracker.get());
factory->setLocalNode(localNode); factory->setLocalNode(localNode);
factory->setBtRegistry(e->getBtRegistry().get());
PrefPtr prefEntryPointHost = family == AF_INET ? PREF_DHT_ENTRY_POINT_HOST PrefPtr prefEntryPointHost = family == AF_INET ? PREF_DHT_ENTRY_POINT_HOST
: PREF_DHT_ENTRY_POINT_HOST6; : PREF_DHT_ENTRY_POINT_HOST6;

View file

@ -538,9 +538,13 @@
#define TEXT_EVENT_POLL \ #define TEXT_EVENT_POLL \
_(" --event-poll=POLL Specify the method for polling events.") _(" --event-poll=POLL Specify the method for polling events.")
#define TEXT_BT_EXTERNAL_IP \ #define TEXT_BT_EXTERNAL_IP \
_(" --bt-external-ip=IPADDRESS Specify the external IP address to report to a\n" \ _(" --bt-external-ip=IPADDRESS Specify the external IP address to use in\n" \
" BitTorrent tracker. Although this function is\n" \ " BitTorrent download and DHT. It may be sent to\n" \
" named 'external', it can accept any kind of IP\n" \ " BitTorrent tracker. For DHT, this option should\n" \
" be set to report that local node is downloading\n" \
" a particular torrent. This is critical to use\n" \
" DHT in a private network. Although this function\n" \
" is named 'external', it can accept any kind of IP\n" \
" addresses.") " addresses.")
#define TEXT_HTTP_AUTH_CHALLENGE \ #define TEXT_HTTP_AUTH_CHALLENGE \
_(" --http-auth-challenge[=true|false] Send HTTP authorization header only when it\n" \ _(" --http-auth-challenge[=true|false] Send HTTP authorization header only when it\n" \

View file

@ -12,6 +12,12 @@
#include "DHTPeerAnnounceStorage.h" #include "DHTPeerAnnounceStorage.h"
#include "DHTRoutingTable.h" #include "DHTRoutingTable.h"
#include "bencode2.h" #include "bencode2.h"
#include "GroupId.h"
#include "DownloadContext.h"
#include "Option.h"
#include "RequestGroup.h"
#include "BtRegistry.h"
#include "TorrentAttribute.h"
namespace aria2 { namespace aria2 {
@ -102,11 +108,32 @@ void DHTGetPeersMessageTest::testDoReceivedAction()
factory.setLocalNode(localNode_); factory.setLocalNode(localNode_);
DHTRoutingTable routingTable(localNode_); DHTRoutingTable routingTable(localNode_);
auto torrentAttrs = std::make_shared<TorrentAttribute>();
torrentAttrs->infoHash = std::string(infoHash, infoHash + DHT_ID_LENGTH);
auto dctx = std::make_shared<DownloadContext>();
dctx->setAttribute(CTX_ATTR_BT, torrentAttrs);
auto option = std::make_shared<Option>();
option->put(PREF_BT_EXTERNAL_IP, "192.168.0.1");
auto gid = GroupId::create();
RequestGroup group(gid, option);
dctx->setOwnerRequestGroup(&group);
BtRegistry btReg;
btReg.put(
gid->getNumericId(),
make_unique<BtObject>(dctx, nullptr, nullptr, nullptr, nullptr, nullptr));
btReg.setTcpPort(6890);
DHTGetPeersMessage msg(localNode_, remoteNode_, infoHash, transactionID); DHTGetPeersMessage msg(localNode_, remoteNode_, infoHash, transactionID);
msg.setRoutingTable(&routingTable); msg.setRoutingTable(&routingTable);
msg.setTokenTracker(&tokenTracker); msg.setTokenTracker(&tokenTracker);
msg.setMessageDispatcher(&dispatcher); msg.setMessageDispatcher(&dispatcher);
msg.setMessageFactory(&factory); msg.setMessageFactory(&factory);
msg.setBtRegistry(&btReg);
msg.setFamily(AF_INET);
{ {
// localhost has peer contact information for that infohash. // localhost has peer contact information for that infohash.
DHTPeerAnnounceStorage peerAnnounceStorage; DHTPeerAnnounceStorage peerAnnounceStorage;
@ -129,7 +156,7 @@ void DHTGetPeersMessageTest::testDoReceivedAction()
remoteNode_->getPort()), remoteNode_->getPort()),
m->getToken()); m->getToken());
CPPUNIT_ASSERT_EQUAL((size_t)0, m->getClosestKNodes().size()); CPPUNIT_ASSERT_EQUAL((size_t)0, m->getClosestKNodes().size());
CPPUNIT_ASSERT_EQUAL((size_t)2, m->getValues().size()); CPPUNIT_ASSERT_EQUAL((size_t)3, m->getValues().size());
{ {
auto peer = m->getValues()[0]; auto peer = m->getValues()[0];
CPPUNIT_ASSERT_EQUAL(std::string("192.168.0.100"), peer->getIPAddress()); CPPUNIT_ASSERT_EQUAL(std::string("192.168.0.100"), peer->getIPAddress());
@ -140,7 +167,13 @@ void DHTGetPeersMessageTest::testDoReceivedAction()
CPPUNIT_ASSERT_EQUAL(std::string("192.168.0.101"), peer->getIPAddress()); CPPUNIT_ASSERT_EQUAL(std::string("192.168.0.101"), peer->getIPAddress());
CPPUNIT_ASSERT_EQUAL((uint16_t)6889, peer->getPort()); CPPUNIT_ASSERT_EQUAL((uint16_t)6889, peer->getPort());
} }
{
auto peer = m->getValues()[2];
CPPUNIT_ASSERT_EQUAL(std::string("192.168.0.1"), peer->getIPAddress());
CPPUNIT_ASSERT_EQUAL((uint16_t)6890, peer->getPort());
}
} }
msg.setBtRegistry(nullptr);
dispatcher.messageQueue_.clear(); dispatcher.messageQueue_.clear();
{ {
// localhost doesn't have peer contact information for that infohash. // localhost doesn't have peer contact information for that infohash.