/* */ #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 std::shared_ptr& localNode, const std::shared_ptr& remoteNode, const std::string& transactionID) : DHTResponseMessage{localNode, remoteNode, transactionID}, family_{family} { } void DHTFindNodeReplyMessage::doReceivedAction() { for (auto& node : closestKNodes_) { if (memcmp(node->getID(), getLocalNode()->getID(), DHT_ID_LENGTH) != 0) { getRoutingTable()->addNode(node); } } } std::unique_ptr DHTFindNodeReplyMessage::getResponse() { auto 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 (auto i = std::begin(closestKNodes_), eoi = std::end(closestKNodes_); i != eoi && k < DHTBucket::K; ++i) { memcpy(buffer + offset, (*i)->getID(), DHT_ID_LENGTH); unsigned char compact[COMPACT_LEN_IPV6]; int compactlen = bittorrent::packcompact(compact, (*i)->getIPAddress(), (*i)->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( std::vector> closestKNodes) { closestKNodes_ = std::move(closestKNodes); } std::string DHTFindNodeReplyMessage::toStringOptional() const { return fmt("nodes=%lu", static_cast(closestKNodes_.size())); } } // namespace aria2