diff --git a/libp2p/NodeTable.cpp b/libp2p/NodeTable.cpp index f04068852..90a7ac459 100644 --- a/libp2p/NodeTable.cpp +++ b/libp2p/NodeTable.cpp @@ -1,16 +1,16 @@ /* This file is part of cpp-ethereum. - + cpp-ethereum is free software: you can redistribute it and/or modify it under the terms of the GNU General Public License as published by the Free Software Foundation, either version 3 of the License, or (at your option) any later version. - + cpp-ethereum is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for more details. - + You should have received a copy of the GNU General Public License along with cpp-ethereum. If not, see . */ @@ -54,17 +54,17 @@ NodeTable::NodeTable(ba::io_service& _io, KeyPair const& _alias, NodeIPEndpoint m_state[i].distance = i; m_state[i].modified = chrono::steady_clock::now() - chrono::seconds(1); } - + m_socketPointer->connect(); doRefreshBuckets(boost::system::error_code()); } - + NodeTable::~NodeTable() { // Cancel scheduled tasks to ensure. m_evictionCheckTimer.cancel(); m_bucketRefreshTimer.cancel(); - + // Disconnect socket so that deallocation is safe. m_socketPointer->disconnect(); } @@ -85,10 +85,10 @@ shared_ptr NodeTable::addNode(Node const& _node, NodeRelation _relati noteActiveNode(_node.id, _node.endpoint); return ret; } - + if (!_node.endpoint) return move(shared_ptr()); - + // ping address to recover nodeid if nodeid is empty if (!_node.id) { @@ -100,13 +100,13 @@ shared_ptr NodeTable::addNode(Node const& _node, NodeRelation _relati ping(_node.endpoint); return move(shared_ptr()); } - + { Guard ln(x_nodes); if (m_nodes.count(_node.id)) return m_nodes[_node.id]; } - + shared_ptr ret(new NodeEntry(m_node, _node.id, _node.endpoint)); m_nodes[_node.id] = ret; clog(NodeTableConnect) << "addNode pending for" << _node.endpoint; @@ -167,7 +167,7 @@ void NodeTable::discover(NodeId _node, unsigned _round, shared_ptrisOpen() || _round == s_maxSteps) return; - + if (_round == s_maxSteps) { clog(NodeTableEvent) << "Terminating discover after " << _round << " rounds."; @@ -176,7 +176,7 @@ void NodeTable::discover(NodeId _node, unsigned _round, shared_ptr>()); - + auto nearest = nearestNodeEntries(_node); list> tried; for (unsigned i = 0; i < nearest.size() && tried.size() < s_alpha; i++) @@ -189,19 +189,19 @@ void NodeTable::discover(NodeId _node, unsigned _round, shared_ptrid, chrono::steady_clock::now())); m_socketPointer->send(p); } - + if (tried.empty()) { clog(NodeTableEvent) << "Terminating discover after " << _round << " rounds."; return; } - + while (!tried.empty()) { _tried->insert(tried.front()); tried.pop_front(); } - + auto self(shared_from_this()); m_evictionCheckTimer.expires_from_now(boost::posix_time::milliseconds(c_reqTimeout.count() * 2)); m_evictionCheckTimer.async_wait([this, self, _node, _round, _tried](boost::system::error_code const& _ec) @@ -218,10 +218,10 @@ vector> NodeTable::nearestNodeEntries(NodeId _target) static unsigned lastBin = s_bins - 1; unsigned head = distance(m_node.id, _target); unsigned tail = head == 0 ? lastBin : (head - 1) % s_bins; - - unordered_multimap> found; + + map>> found; unsigned count = 0; - + // if d is 0, then we roll look forward, if last, we reverse, else, spread from d if (head > 1 && tail != lastBin) while (head != tail && head < s_bins && count < s_bucketSize) @@ -231,17 +231,17 @@ vector> NodeTable::nearestNodeEntries(NodeId _target) if (auto p = n.lock()) { if (count < s_bucketSize) - found.insert(make_pair(distance(_target, p->id), p)); + found[distance(_target, p->id)].push_back(p); else break; } - + if (count < s_bucketSize && tail) for (auto n: m_state[tail].nodes) if (auto p = n.lock()) { if (count < s_bucketSize) - found.insert(make_pair(distance(_target, p->id), p)); + found[distance(_target, p->id)].push_back(p); else break; } @@ -258,7 +258,7 @@ vector> NodeTable::nearestNodeEntries(NodeId _target) if (auto p = n.lock()) { if (count < s_bucketSize) - found.insert(make_pair(distance(_target, p->id), p)); + found[distance(_target, p->id)].push_back(p); else break; } @@ -272,17 +272,18 @@ vector> NodeTable::nearestNodeEntries(NodeId _target) if (auto p = n.lock()) { if (count < s_bucketSize) - found.insert(make_pair(distance(_target, p->id), p)); + found[distance(_target, p->id)].push_back(p); else break; } tail--; } - + vector> ret; - for (auto n: found) - if (ret.size() < s_bucketSize && !!n.second->endpoint && n.second->endpoint.isAllowed()) - ret.push_back(n.second); + for (auto& nodes: found) + for (auto n: nodes.second) + if (ret.size() < s_bucketSize && !!n->endpoint && n->endpoint.isAllowed()) + ret.push_back(n); return move(ret); } @@ -303,7 +304,7 @@ void NodeTable::evict(shared_ptr _leastSeen, shared_ptr _n { if (!m_socketPointer->isOpen()) return; - + { Guard l(x_evictions); m_evictions.push_back(EvictionTimeout(make_pair(_leastSeen->id,chrono::steady_clock::now()), _new->id)); @@ -324,7 +325,7 @@ void NodeTable::noteActiveNode(Public const& _pubk, bi::udp::endpoint const& _en clog(NodeTableConnect) << "Noting active node:" << _pubk << _endpoint.address().to_string() << ":" << _endpoint.port(); node->endpoint.address = _endpoint.address(); node->endpoint.udpPort = _endpoint.port(); - + shared_ptr contested; { Guard l(x_state); @@ -336,7 +337,7 @@ void NodeTable::noteActiveNode(Public const& _pubk, bi::udp::endpoint const& _en removed = true; return removed; }); - + if (s.nodes.size() >= s_bucketSize) { // It's only contested iff nodeentry exists @@ -346,7 +347,7 @@ void NodeTable::noteActiveNode(Public const& _pubk, bi::udp::endpoint const& _en s.nodes.pop_front(); s.nodes.push_back(node); s.touch(); - + if (!removed && m_nodeEventHandler) m_nodeEventHandler->appendEvent(node->id, NodeEntryAdded); } @@ -355,12 +356,12 @@ void NodeTable::noteActiveNode(Public const& _pubk, bi::udp::endpoint const& _en { s.nodes.push_back(node); s.touch(); - + if (!removed && m_nodeEventHandler) m_nodeEventHandler->appendEvent(node->id, NodeEntryAdded); } } - + if (contested) evict(contested, node); } @@ -374,7 +375,7 @@ void NodeTable::dropNode(shared_ptr _n) NodeBucket& s = bucket_UNSAFE(_n.get()); s.nodes.remove_if([&_n](weak_ptr n) { return n.lock() == _n; }); } - + // notify host clog(NodeTableUpdate) << "p2p.nodes.drop " << _n->id; if (m_nodeEventHandler) @@ -394,7 +395,7 @@ void NodeTable::onReceived(UDPSocketFace*, bi::udp::endpoint const& _from, bytes clog(NodeTableTriviaSummary) << "Invalid message size from " << _from.address().to_string() << ":" << _from.port(); return; } - + bytesConstRef hashedBytes(_packet.cropped(h256::size, _packet.size() - h256::size)); h256 hashSigned(sha3(hashedBytes)); if (!_packet.cropped(0, h256::size).contentsEqual(hashSigned.asBytes())) @@ -402,11 +403,11 @@ void NodeTable::onReceived(UDPSocketFace*, bi::udp::endpoint const& _from, bytes clog(NodeTableTriviaSummary) << "Invalid message hash from " << _from.address().to_string() << ":" << _from.port(); return; } - + bytesConstRef signedBytes(hashedBytes.cropped(Signature::size, hashedBytes.size() - Signature::size)); // todo: verify sig via known-nodeid and MDC - + bytesConstRef sigBytes(_packet.cropped(h256::size, Signature::size)); Public nodeid(dev::recover(*(Signature const*)sigBytes.data(), sha3(signedBytes))); if (!nodeid) @@ -414,7 +415,7 @@ void NodeTable::onReceived(UDPSocketFace*, bi::udp::endpoint const& _from, bytes clog(NodeTableTriviaSummary) << "Invalid message signature from " << _from.address().to_string() << ":" << _from.port(); return; } - + unsigned packetType = signedBytes[0]; bytesConstRef rlpBytes(_packet.cropped(h256::size + Signature::size + 1)); RLP rlp(rlpBytes); @@ -424,7 +425,7 @@ void NodeTable::onReceived(UDPSocketFace*, bi::udp::endpoint const& _from, bytes case Pong::type: { Pong in = Pong::fromBytesConstRef(_from, rlpBytes); - + // whenever a pong is received, check if it's in m_evictions Guard le(x_evictions); bool evictionEntry = false; @@ -434,13 +435,13 @@ void NodeTable::onReceived(UDPSocketFace*, bi::udp::endpoint const& _from, bytes evictionEntry = true; if (auto n = nodeEntry(it->second)) dropNode(n); - + if (auto n = nodeEntry(it->first.first)) n->pending = false; - + it = m_evictions.erase(it); } - + // if not, check if it's known/pending or a pubk discovery ping if (!evictionEntry) { @@ -458,16 +459,16 @@ void NodeTable::onReceived(UDPSocketFace*, bi::udp::endpoint const& _from, bytes else return; // unsolicited pong; don't note node as active } - + // update our endpoint address and UDP port if ((!m_node.endpoint || !m_node.endpoint.isAllowed()) && isPublicAddress(in.destination.address)) m_node.endpoint.address = in.destination.address; m_node.endpoint.udpPort = in.destination.udpPort; - + clog(NodeTableConnect) << "PONG from " << nodeid << _from; break; } - + case Neighbours::type: { bool expected = false; @@ -480,13 +481,13 @@ void NodeTable::onReceived(UDPSocketFace*, bi::udp::endpoint const& _from, bytes return true; return false; }); - + if (!expected) { clog(NetConnect) << "Dropping unsolicited neighbours packet from " << _from.address(); break; } - + Neighbours in = Neighbours::fromBytesConstRef(_from, rlpBytes); for (auto n: in.neighbours) addNode(Node(n.node, n.endpoint)); @@ -530,13 +531,13 @@ void NodeTable::onReceived(UDPSocketFace*, bi::udp::endpoint const& _from, bytes else return; } - + if (RLPXDatagramFace::secondsSinceEpoch() > in.ts) { clog(NodeTableTriviaSummary) << "Received expired PingNode from " << _from.address().to_string() << ":" << _from.port(); return; } - + in.source.address = _from.address(); in.source.udpPort = _from.port(); addNode(Node(nodeid, in.source)); @@ -546,7 +547,7 @@ void NodeTable::onReceived(UDPSocketFace*, bi::udp::endpoint const& _from, bytes m_socketPointer->send(p); break; } - + default: clog(NodeTableWarn) << "Invalid message, " << hex << packetType << ", received from " << _from.address().to_string() << ":" << dec << _from.port(); return; @@ -571,7 +572,7 @@ void NodeTable::doCheckEvictions(boost::system::error_code const& _ec) { if (_ec) return; - + bool evictionsRemain = false; list> drop; { @@ -583,11 +584,11 @@ void NodeTable::doCheckEvictions(boost::system::error_code const& _ec) drop.push_back(m_nodes[e.second]); evictionsRemain = m_evictions.size() - drop.size() > 0; } - + drop.unique(); for (auto n: drop) dropNode(n); - + if (evictionsRemain) doCheckEvictions(boost::system::error_code()); });