/* */ #include "DHTRoutingTable.h" #include #include "DHTNode.h" #include "DHTBucket.h" #include "DHTBucketTree.h" #include "DHTTaskQueue.h" #include "DHTTaskFactory.h" #include "DHTTask.h" #include "util.h" #include "LogFactory.h" #include "Logger.h" #include "fmt.h" namespace aria2 { DHTRoutingTable::DHTRoutingTable(const std::shared_ptr& localNode) : localNode_(localNode), root_(make_unique (std::make_shared(localNode_))), numBucket_(1), taskQueue_{nullptr}, taskFactory_{nullptr} {} DHTRoutingTable::~DHTRoutingTable() {} bool DHTRoutingTable::addNode(const std::shared_ptr& node) { return addNode(node, false); } bool DHTRoutingTable::addGoodNode(const std::shared_ptr& node) { return addNode(node, true); } bool DHTRoutingTable::addNode(const std::shared_ptr& node, bool good) { A2_LOG_DEBUG(fmt("Trying to add node:%s", node->toString().c_str())); if(*localNode_ == *node) { A2_LOG_DEBUG("Adding node with the same ID with localnode is not allowed."); return false; } auto treeNode = dht::findTreeNodeFor(root_.get(), node->getID()); while(1) { const std::shared_ptr& bucket = treeNode->getBucket(); if(bucket->addNode(node)) { A2_LOG_DEBUG("Added DHTNode."); return true; } else if(bucket->splitAllowed()) { A2_LOG_DEBUG(fmt("Splitting bucket. Range:%s-%s", util::toHex(bucket->getMinID(), DHT_ID_LENGTH).c_str(), util::toHex(bucket->getMaxID(), DHT_ID_LENGTH).c_str())); treeNode->split(); ++numBucket_; if(treeNode->getLeft()->isInRange(node->getID())) { treeNode = treeNode->getLeft(); } else { treeNode = treeNode->getRight(); } } else { if(good) { bucket->cacheNode(node); A2_LOG_DEBUG(fmt("Cached node=%s", node->toString().c_str())); } return false; } } return false; } void DHTRoutingTable::getClosestKNodes (std::vector >& nodes, const unsigned char* key) const { dht::findClosestKNodes(nodes, root_.get(), key); } int DHTRoutingTable::getNumBucket() const { return numBucket_; } void DHTRoutingTable::showBuckets() const {/* for(std::deque >::const_iterator itr = buckets_.begin(); itr != buckets_.end(); ++itr) { cerr << "prefix = " << (*itr)->getPrefixLength() << ", " << "nodes = " << (*itr)->countNode() << endl; } */ } std::shared_ptr DHTRoutingTable::getBucketFor(const unsigned char* nodeID) const { return dht::findBucketFor(root_.get(), nodeID); } std::shared_ptr DHTRoutingTable::getBucketFor(const std::shared_ptr& node) const { return getBucketFor(node->getID()); } std::shared_ptr DHTRoutingTable::getNode(const unsigned char* nodeID, const std::string& ipaddr, uint16_t port) const { std::shared_ptr bucket = getBucketFor(nodeID); return bucket->getNode(nodeID, ipaddr, port); } void DHTRoutingTable::dropNode(const std::shared_ptr& node) { getBucketFor(node)->dropNode(node); } /* void DHTRoutingTable::moveBucketHead(const std::shared_ptr& node) { getBucketFor(node)->moveToHead(node); } */ void DHTRoutingTable::moveBucketTail(const std::shared_ptr& node) { getBucketFor(node)->moveToTail(node); } void DHTRoutingTable::getBuckets (std::vector >& buckets) const { dht::enumerateBucket(buckets, root_.get()); } void DHTRoutingTable::setTaskQueue(DHTTaskQueue* taskQueue) { taskQueue_ = taskQueue; } void DHTRoutingTable::setTaskFactory(DHTTaskFactory* taskFactory) { taskFactory_ = taskFactory; } } // namespace aria2