/* */ #include "DHTFindNodeReplyMessage.h" #include #include "DHTNode.h" #include "DHTBucket.h" #include "DHTRoutingTable.h" #include "DHTMessageFactory.h" #include "DHTMessageDispatcher.h" #include "DHTMessageCallback.h" #include "bittorrent_helper.h" #include "util.h" namespace aria2 { const std::string DHTFindNodeReplyMessage::FIND_NODE("find_node"); const std::string DHTFindNodeReplyMessage::NODES("nodes"); const std::string DHTFindNodeReplyMessage::NODES6("nodes6"); DHTFindNodeReplyMessage::DHTFindNodeReplyMessage (int family, const SharedHandle& localNode, const SharedHandle& remoteNode, const std::string& transactionID): DHTResponseMessage(localNode, remoteNode, transactionID), family_(family) {} DHTFindNodeReplyMessage::~DHTFindNodeReplyMessage() {} void DHTFindNodeReplyMessage::doReceivedAction() { for(std::vector >::iterator i = closestKNodes_.begin(), eoi = closestKNodes_.end(); i != eoi; ++i) { if(memcmp((*i)->getID(), getLocalNode()->getID(), DHT_ID_LENGTH) != 0) { getRoutingTable()->addNode(*i); } } } SharedHandle DHTFindNodeReplyMessage::getResponse() { SharedHandle aDict = Dict::g(); aDict->put(DHTMessage::ID, String::g(getLocalNode()->getID(), DHT_ID_LENGTH)); unsigned char buffer[DHTBucket::K*38]; const int clen = bittorrent::getCompactLength(family_); const int unit = clen+20; assert(unit <= 38); size_t offset = 0; size_t k = 0; for(std::vector >::const_iterator i = closestKNodes_.begin(), eoi = closestKNodes_.end(); i != eoi && k < DHTBucket::K; ++i) { SharedHandle node = *i; memcpy(buffer+offset, node->getID(), DHT_ID_LENGTH); unsigned char compact[COMPACT_LEN_IPV6]; int compactlen = bittorrent::packcompact (compact, node->getIPAddress(), node->getPort()); if(compactlen == clen) { memcpy(buffer+20+offset, compact, compactlen); offset += unit; ++k; } } aDict->put(family_ == AF_INET?NODES:NODES6, String::g(buffer, offset)); return aDict; } const std::string& DHTFindNodeReplyMessage::getMessageType() const { return FIND_NODE; } void DHTFindNodeReplyMessage::accept(DHTMessageCallback* callback) { callback->visit(this); } void DHTFindNodeReplyMessage::setClosestKNodes (const std::vector >& closestKNodes) { closestKNodes_ = closestKNodes; } std::string DHTFindNodeReplyMessage::toStringOptional() const { return "nodes="+util::uitos(closestKNodes_.size()); } } // namespace aria2