/* 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 <http://www.gnu.org/licenses/>. */ /** @file net.cpp * @author Alex Leverington <nessence@gmail.com> * @date 2014 */ #include <boost/test/unit_test.hpp> #include <libdevcore/Worker.h> #include <libdevcore/Assertions.h> #include <libdevcrypto/Common.h> #include <libp2p/UDP.h> #include <libp2p/NodeTable.h> using namespace std; using namespace dev; using namespace dev::p2p; namespace ba = boost::asio; namespace bi = ba::ip; struct NetFixture { NetFixture() { dev::p2p::NodeIPEndpoint::test_allowLocal = true; } ~NetFixture() { dev::p2p::NodeIPEndpoint::test_allowLocal = false; } }; BOOST_FIXTURE_TEST_SUITE(net, NetFixture) /** * Only used for testing. Not useful beyond tests. */ class TestHost: public Worker { public: TestHost(): Worker("test",0), m_io() {}; virtual ~TestHost() { m_io.stop(); stopWorking(); } void start() { startWorking(); } void doWork() { m_io.run(); } void doneWorking() { m_io.reset(); m_io.poll(); m_io.reset(); } protected: ba::io_service m_io; }; struct TestNodeTable: public NodeTable { /// Constructor TestNodeTable(ba::io_service& _io, KeyPair _alias, bi::address const& _addr, uint16_t _port = 30300): NodeTable(_io, _alias, NodeIPEndpoint(_addr, _port, _port)) {} static std::vector<std::pair<KeyPair,unsigned>> createTestNodes(unsigned _count) { std::vector<std::pair<KeyPair,unsigned>> ret; asserts(_count < 1000); static uint16_t s_basePort = 30500; ret.clear(); for (unsigned i = 0; i < _count; i++) { KeyPair k = KeyPair::create(); ret.push_back(make_pair(k,s_basePort+i)); } return std::move(ret); } void pingTestNodes(std::vector<std::pair<KeyPair,unsigned>> const& _testNodes) { bi::address ourIp = bi::address::from_string("127.0.0.1"); for (auto& n: _testNodes) { ping(NodeIPEndpoint(ourIp, n.second, n.second)); this_thread::sleep_for(chrono::milliseconds(2)); } } void populateTestNodes(std::vector<std::pair<KeyPair,unsigned>> const& _testNodes, size_t _count = 0) { if (!_count) _count = _testNodes.size(); bi::address ourIp = bi::address::from_string("127.0.0.1"); for (auto& n: _testNodes) if (_count--) { // manually add node for test { Guard ln(x_nodes); shared_ptr<NodeEntry> node(new NodeEntry(m_node, n.first.pub(), NodeIPEndpoint(ourIp, n.second, n.second))); node->pending = false; m_nodes[node->id] = node; } noteActiveNode(n.first.pub(), bi::udp::endpoint(ourIp, n.second)); } else break; } void reset() { Guard l(x_state); for (auto& n: m_state) n.nodes.clear(); } }; /** * Only used for testing. Not useful beyond tests. */ struct TestNodeTableHost: public TestHost { TestNodeTableHost(unsigned _count = 8): m_alias(KeyPair::create()), nodeTable(new TestNodeTable(m_io, m_alias, bi::address::from_string("127.0.0.1"))), testNodes(TestNodeTable::createTestNodes(_count)) {}; ~TestNodeTableHost() { m_io.stop(); stopWorking(); } void setup() { for (auto n: testNodes) nodeTables.push_back(make_shared<TestNodeTable>(m_io,n.first, bi::address::from_string("127.0.0.1"),n.second)); } void pingAll() { for (auto& t: nodeTables) t->pingTestNodes(testNodes); } void populateAll(size_t _count = 0) { for (auto& t: nodeTables) t->populateTestNodes(testNodes, _count); } void populate(size_t _count = 0) { nodeTable->populateTestNodes(testNodes, _count); } KeyPair m_alias; shared_ptr<TestNodeTable> nodeTable; std::vector<std::pair<KeyPair,unsigned>> testNodes; // keypair and port std::vector<shared_ptr<TestNodeTable>> nodeTables; }; class TestUDPSocket: UDPSocketEvents, public TestHost { public: TestUDPSocket(): m_socket(new UDPSocket<TestUDPSocket, 1024>(m_io, *this, 30300)) {} void onDisconnected(UDPSocketFace*) {}; void onReceived(UDPSocketFace*, bi::udp::endpoint const&, bytesConstRef _packet) { if (_packet.toString() == "AAAA") success = true; } shared_ptr<UDPSocket<TestUDPSocket, 1024>> m_socket; bool success = false; }; BOOST_AUTO_TEST_CASE(requestTimeout) { using TimePoint = std::chrono::steady_clock::time_point; using RequestTimeout = std::pair<NodeId, TimePoint>; std::chrono::milliseconds timeout(300); std::list<RequestTimeout> timeouts; NodeId nodeA(sha3("a")); NodeId nodeB(sha3("b")); timeouts.push_back(make_pair(nodeA, chrono::steady_clock::now())); this_thread::sleep_for(std::chrono::milliseconds(100)); timeouts.push_back(make_pair(nodeB, chrono::steady_clock::now())); this_thread::sleep_for(std::chrono::milliseconds(210)); bool nodeAtriggered = false; bool nodeBtriggered = false; timeouts.remove_if([&](RequestTimeout const& t) { auto now = chrono::steady_clock::now(); auto diff = now - t.second; if (t.first == nodeA && diff < timeout) nodeAtriggered = true; if (t.first == nodeB && diff < timeout) nodeBtriggered = true; return (t.first == nodeA || t.first == nodeB); }); BOOST_REQUIRE(nodeAtriggered == false); BOOST_REQUIRE(nodeBtriggered == true); BOOST_REQUIRE(timeouts.size() == 0); } BOOST_AUTO_TEST_CASE(isIPAddressType) { string wildcard = "0.0.0.0"; BOOST_REQUIRE(bi::address::from_string(wildcard).is_unspecified()); string empty = ""; BOOST_REQUIRE_THROW(bi::address::from_string(empty).is_unspecified(), std::exception); string publicAddress192 = "192.169.0.0"; BOOST_REQUIRE(isPublicAddress(publicAddress192)); BOOST_REQUIRE(!isPrivateAddress(publicAddress192)); BOOST_REQUIRE(!isLocalHostAddress(publicAddress192)); string publicAddress172 = "172.32.0.0"; BOOST_REQUIRE(isPublicAddress(publicAddress172)); BOOST_REQUIRE(!isPrivateAddress(publicAddress172)); BOOST_REQUIRE(!isLocalHostAddress(publicAddress172)); string privateAddress192 = "192.168.1.0"; BOOST_REQUIRE(isPrivateAddress(privateAddress192)); BOOST_REQUIRE(!isPublicAddress(privateAddress192)); BOOST_REQUIRE(!isLocalHostAddress(privateAddress192)); string privateAddress172 = "172.16.0.0"; BOOST_REQUIRE(isPrivateAddress(privateAddress172)); BOOST_REQUIRE(!isPublicAddress(privateAddress172)); BOOST_REQUIRE(!isLocalHostAddress(privateAddress172)); string privateAddress10 = "10.0.0.0"; BOOST_REQUIRE(isPrivateAddress(privateAddress10)); BOOST_REQUIRE(!isPublicAddress(privateAddress10)); BOOST_REQUIRE(!isLocalHostAddress(privateAddress10)); } BOOST_AUTO_TEST_CASE(v2PingNodePacket) { // test old versino of pingNode packet w/new RLPStream s; s.appendList(3); s << "1.1.1.1" << 30303 << std::chrono::duration_cast<std::chrono::seconds>((std::chrono::system_clock::now() + chrono::seconds(60)).time_since_epoch()).count(); PingNode p((bi::udp::endpoint())); BOOST_REQUIRE_NO_THROW(p = PingNode::fromBytesConstRef(bi::udp::endpoint(), bytesConstRef(&s.out()))); BOOST_REQUIRE(p.version == 0); } BOOST_AUTO_TEST_CASE(neighboursPacketLength) { KeyPair k = KeyPair::create(); std::vector<std::pair<KeyPair,unsigned>> testNodes(TestNodeTable::createTestNodes(16)); bi::udp::endpoint to(boost::asio::ip::address::from_string("127.0.0.1"), 30000); // hash(32), signature(65), overhead: packetSz(3), type(1), nodeListSz(3), ts(5), static unsigned const nlimit = (1280 - 109) / 90; // neighbour: 2 + 65 + 3 + 3 + 17 for (unsigned offset = 0; offset < testNodes.size(); offset += nlimit) { Neighbours out(to); auto limit = nlimit ? std::min(testNodes.size(), (size_t)(offset + nlimit)) : testNodes.size(); for (auto i = offset; i < limit; i++) { Node n(testNodes[i].first.pub(), NodeIPEndpoint(boost::asio::ip::address::from_string("200.200.200.200"), testNodes[i].second, testNodes[i].second)); Neighbours::Neighbour neighbour(n); out.neighbours.push_back(neighbour); } out.sign(k.sec()); BOOST_REQUIRE_LE(out.data.size(), 1280); } } BOOST_AUTO_TEST_CASE(neighboursPacket) { KeyPair k = KeyPair::create(); std::vector<std::pair<KeyPair,unsigned>> testNodes(TestNodeTable::createTestNodes(16)); bi::udp::endpoint to(boost::asio::ip::address::from_string("127.0.0.1"), 30000); Neighbours out(to); for (auto n: testNodes) { Node node(n.first.pub(), NodeIPEndpoint(boost::asio::ip::address::from_string("200.200.200.200"), n.second, n.second)); Neighbours::Neighbour neighbour(node); out.neighbours.push_back(neighbour); } out.sign(k.sec()); bytesConstRef packet(out.data.data(), out.data.size()); bytesConstRef rlpBytes(packet.cropped(h256::size + Signature::size + 1)); Neighbours in = Neighbours::fromBytesConstRef(to, rlpBytes); int count = 0; for (auto n: in.neighbours) { BOOST_REQUIRE_EQUAL(testNodes[count].second, n.endpoint.udpPort); BOOST_REQUIRE_EQUAL(testNodes[count].first.pub(), n.node); BOOST_REQUIRE_EQUAL(sha3(testNodes[count].first.pub()), sha3(n.node)); count++; } } BOOST_AUTO_TEST_CASE(test_findnode_neighbours) { // Executing findNode should result in a list which is serialized // into Neighbours packet. Neighbours packet should then be deserialized // into the same list of nearest nodes. } BOOST_AUTO_TEST_CASE(kademlia) { // Not yet a 'real' test. TestNodeTableHost node(8); node.start(); node.nodeTable->discover(); // ideally, joining with empty node table logs warning we can check for node.setup(); node.populate(); clog << "NodeTable:\n" << *node.nodeTable.get() << endl; node.populateAll(); clog << "NodeTable:\n" << *node.nodeTable.get() << endl; auto nodes = node.nodeTable->nodes(); nodes.sort(); node.nodeTable->reset(); clog << "NodeTable:\n" << *node.nodeTable.get() << endl; node.populate(1); clog << "NodeTable:\n" << *node.nodeTable.get() << endl; node.nodeTable->discover(); this_thread::sleep_for(chrono::milliseconds(2000)); clog << "NodeTable:\n" << *node.nodeTable.get() << endl; BOOST_REQUIRE_EQUAL(node.nodeTable->count(), 8); auto netNodes = node.nodeTable->nodes(); netNodes.sort(); } BOOST_AUTO_TEST_CASE(udpOnce) { UDPDatagram d(bi::udp::endpoint(boost::asio::ip::address::from_string("127.0.0.1"), 30300), bytes({65,65,65,65})); TestUDPSocket a; a.m_socket->connect(); a.start(); a.m_socket->send(d); this_thread::sleep_for(chrono::seconds(1)); BOOST_REQUIRE_EQUAL(true, a.success); } BOOST_AUTO_TEST_SUITE_END() BOOST_AUTO_TEST_SUITE(netTypes) BOOST_AUTO_TEST_CASE(unspecifiedNode) { Node n = UnspecifiedNode; BOOST_REQUIRE(!n); Node node(Public(sha3("0")), NodeIPEndpoint(bi::address(), 0, 0)); BOOST_REQUIRE(node); BOOST_REQUIRE(n != node); Node nodeEq(Public(sha3("0")), NodeIPEndpoint(bi::address(), 0, 0)); BOOST_REQUIRE_EQUAL(node, nodeEq); } BOOST_AUTO_TEST_CASE(nodeTableReturnsUnspecifiedNode) { ba::io_service io; NodeTable t(io, KeyPair::create(), NodeIPEndpoint(bi::address::from_string("127.0.0.1"), 30303, 30303)); if (Node n = t.node(NodeId())) BOOST_REQUIRE(false); else BOOST_REQUIRE(n == UnspecifiedNode); } BOOST_AUTO_TEST_SUITE_END()