/* */ #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 SharedHandle& localNode) : localNode_(localNode), root_(new DHTBucketTreeNode (SharedHandle(new DHTBucket(localNode_)))), numBucket_(1) {} DHTRoutingTable::~DHTRoutingTable() { delete root_; } bool DHTRoutingTable::addNode(const SharedHandle& node) { return addNode(node, false); } bool DHTRoutingTable::addGoodNode(const SharedHandle& node) { return addNode(node, true); } bool DHTRoutingTable::addNode(const SharedHandle& 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; } DHTBucketTreeNode* treeNode = dht::findTreeNodeFor(root_, node->getID()); while(1) { const SharedHandle& 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_, 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; } */ } SharedHandle DHTRoutingTable::getBucketFor(const unsigned char* nodeID) const { return dht::findBucketFor(root_, nodeID); } SharedHandle DHTRoutingTable::getBucketFor(const SharedHandle& node) const { return getBucketFor(node->getID()); } SharedHandle DHTRoutingTable::getNode(const unsigned char* nodeID, const std::string& ipaddr, uint16_t port) const { SharedHandle bucket = getBucketFor(nodeID); return bucket->getNode(nodeID, ipaddr, port); } void DHTRoutingTable::dropNode(const SharedHandle& node) { getBucketFor(node)->dropNode(node); } /* void DHTRoutingTable::moveBucketHead(const SharedHandle& node) { getBucketFor(node)->moveToHead(node); } */ void DHTRoutingTable::moveBucketTail(const SharedHandle& node) { getBucketFor(node)->moveToTail(node); } void DHTRoutingTable::getBuckets (std::vector >& buckets) const { dht::enumerateBucket(buckets, root_); } void DHTRoutingTable::setTaskQueue(const SharedHandle& taskQueue) { taskQueue_ = taskQueue; } void DHTRoutingTable::setTaskFactory(const SharedHandle& taskFactory) { taskFactory_ = taskFactory; } } // namespace aria2