import unittest, chronos, sequtils, chronicles, tables, stint, eth/[keys, rlp], eth/p2p/enode, eth/trie/db, eth/p2p/discoveryv5/[discovery_db, enr, node, types, routing_table, encoding], eth/p2p/discoveryv5/protocol as discv5_protocol, ./p2p_test_helper proc initDiscoveryNode*(privKey: PrivateKey, address: Address, bootstrapRecords: openarray[Record] = []): discv5_protocol.Protocol = var db = DiscoveryDB.init(newMemoryDB()) result = newProtocol(privKey, db, some(parseIpAddress("127.0.0.1")), address.tcpPort, address.udpPort, bootstrapRecords) result.open() proc nodeIdInNodes(id: NodeId, nodes: openarray[Node]): bool = for n in nodes: if id == n.id: return true # Creating a random packet with specific nodeid each time proc randomPacket(tag: PacketTag): seq[byte] = var authTag: AuthTag msg: array[44, byte] randomBytes2(authTag) randomBytes2(msg) result.add(tag) result.add(rlp.encode(authTag)) result.add(msg) proc generateNode(privKey = PrivateKey.random()[], port: int): Node = let port = Port(port) let enr = enr.Record.init(1, privKey, some(parseIpAddress("127.0.0.1")), port, port) result = newNode(enr) suite "Discovery v5 Tests": asyncTest "Random nodes": let bootNodeKey = PrivateKey.fromHex( "a2b50376a79b1a8c8a3296485572bdfbf54708bb46d3c25d73d2723aaaf6a617")[] bootNode = initDiscoveryNode(bootNodeKey, localAddress(20301)) let nodeKeys = [ PrivateKey.fromHex( "a2b50376a79b1a8c8a3296485572bdfbf54708bb46d3c25d73d2723aaaf6a618")[], PrivateKey.fromHex( "a2b50376a79b1a8c8a3296485572bdfbf54708bb46d3c25d73d2723aaaf6a619")[], PrivateKey.fromHex( "a2b50376a79b1a8c8a3296485572bdfbf54708bb46d3c25d73d2723aaaf6a620")[] ] var nodeAddrs = newSeqOfCap[Address](nodeKeys.len) for i in 0 ..< nodeKeys.len: nodeAddrs.add(localAddress(20302 + i)) var nodes = zip(nodeKeys, nodeAddrs).mapIt( initDiscoveryNode(it[0], it[1], @[bootNode.localNode.record])) nodes.add(bootNode) for node in nodes: let discovered = await node.lookupRandom() check discovered.len < nodes.len debug "Lookup from random id", node = node.localNode, discovered # Check for each node if the other nodes shows up in the routing table for i in nodes: for j in nodes: if j != i: check(nodeIdInNodes(i.localNode.id, j.randomNodes(nodes.len - 1))) for node in nodes: await node.closeWait() asyncTest "Lookup targets": const nodeCount = 17 let bootNode = initDiscoveryNode(PrivateKey.random()[], localAddress(20301)) bootNode.start() var nodes = newSeqOfCap[discv5_protocol.Protocol](nodeCount) nodes.add(bootNode) for i in 1 ..< nodeCount: nodes.add(initDiscoveryNode(PrivateKey.random()[], localAddress(20301 + i), @[bootNode.localNode.record])) nodes[i].start() for i in 0..