{.used.} import std/[sequtils, algorithm], stew/results, stew/shims/net, chronos, chronicles, testutils/unittests, libp2p/crypto/crypto as libp2p_keys, eth/keys as eth_keys import ../../../waku/[waku_core/topics, waku_enr, waku_discv5, common/enr], ../testlib/[wakucore, testasync, assertions, futures], ../waku_enr/utils, ./utils import eth/p2p/discoveryv5/enr as ethEnr procSuite "Waku Discovery v5": let rng = eth_keys.newRng() pk1 = eth_keys.PrivateKey.random(rng[]) pk2 = eth_keys.PrivateKey.random(rng[]) enrNoCapabilities = initRecord(1, pk1, {"rs": @[0.byte, 0.byte, 1.byte, 0.byte, 0.byte]}).value() enrRelay = initRecord( 1, pk2, {"waku2": @[1.byte], "rs": @[0.byte, 1.byte, 1.byte, 0.byte, 1.byte]} ).value() enrNoShardingInfo = initRecord(1, pk1, {"waku2": @[1.byte]}).value() suite "shardingPredicate": var recordCluster21 {.threadvar.}: Record recordCluster22Indices1 {.threadvar.}: Record recordCluster22Indices2 {.threadvar.}: Record asyncSetup: recordCluster21 = block: let enrSeqNum = 1u64 enrPrivKey = generatesecp256k1key() let clusterId: uint16 = 21 shardIds: seq[uint16] = @[1u16, 2u16, 5u16, 7u16, 9u16, 11u16] let shardsTopics = RelayShards.init(clusterId, shardIds).expect("Valid shardIds") var builder = EnrBuilder.init(enrPrivKey, seqNum = enrSeqNum) require builder.withWakuRelaySharding(shardsTopics).isOk() builder.withWakuCapabilities(Relay) let recordRes = builder.build() require recordRes.isOk() recordRes.tryGet() recordCluster22Indices1 = block: let enrSeqNum = 1u64 enrPrivKey = generatesecp256k1key() let clusterId: uint16 = 22 shardIds: seq[uint16] = @[2u16, 4u16, 5u16, 8u16, 10u16, 12u16] let shardsTopics = RelayShards.init(clusterId, shardIds).expect("Valid shardIds") var builder = EnrBuilder.init(enrPrivKey, seqNum = enrSeqNum) require builder.withWakuRelaySharding(shardsTopics).isOk() builder.withWakuCapabilities(Relay) let recordRes = builder.build() require recordRes.isOk() recordRes.tryGet() recordCluster22Indices2 = block: let enrSeqNum = 1u64 enrPrivKey = generatesecp256k1key() let clusterId: uint16 = 22 shardIds: seq[uint16] = @[1u16, 3u16, 6u16, 7u16, 9u16, 11u16] let shardsTopics = RelayShards.init(clusterId, shardIds).expect("Valid shardIds") var builder = EnrBuilder.init(enrPrivKey, seqNum = enrSeqNum) require builder.withWakuRelaySharding(shardsTopics).isOk() builder.withWakuCapabilities(Relay) let recordRes = builder.build() require recordRes.isOk() recordRes.tryGet() asyncTest "filter peer per contained shard": # When let predicateCluster21Op = shardingPredicate(recordCluster21) require predicateCluster21Op.isSome() let predicateCluster21 = predicateCluster21Op.get() let predicateCluster22Op = shardingPredicate(recordCluster22Indices1) require predicateCluster22Op.isSome() let predicateCluster22 = predicateCluster22Op.get() # Then check: predicateCluster21(recordCluster21) == true predicateCluster21(recordCluster22Indices1) == false predicateCluster21(recordCluster22Indices2) == false predicateCluster22(recordCluster21) == false predicateCluster22(recordCluster22Indices1) == true predicateCluster22(recordCluster22Indices2) == false asyncTest "filter peer per bootnode": let predicateNoCapabilities = shardingPredicate(enrNoCapabilities, @[enrNoCapabilities]).get() predicateNoCapabilitiesWithBoth = shardingPredicate(enrNoCapabilities, @[enrNoCapabilities, enrRelay]).get() check: predicateNoCapabilities(enrNoCapabilities) == true predicateNoCapabilities(enrRelay) == false predicateNoCapabilitiesWithBoth(enrNoCapabilities) == true predicateNoCapabilitiesWithBoth(enrRelay) == true let predicateRelay = shardingPredicate(enrRelay, @[enrRelay]).get() predicateRelayWithBoth = shardingPredicate(enrRelay, @[enrRelay, enrNoCapabilities]).get() check: predicateRelay(enrNoCapabilities) == false predicateRelay(enrRelay) == true predicateRelayWithBoth(enrNoCapabilities) == true predicateRelayWithBoth(enrRelay) == true asyncTest "does not conform to typed record": let record = ethEnr.Record(raw: @[]) predicateRecord = shardingPredicate(record, @[]) check: predicateRecord.isNone() asyncTest "no relay sharding info": let predicateNoShardingInfo = shardingPredicate(enrNoShardingInfo, @[enrNoShardingInfo]) check: predicateNoShardingInfo.isNone() suite "findRandomPeers": proc buildNode( tcpPort: uint16, udpPort: uint16, bindIp: string = "0.0.0.0", extIp: string = "127.0.0.1", indices: seq[uint64] = @[], recordFlags: Option[CapabilitiesBitfield] = none(CapabilitiesBitfield), bootstrapRecords: seq[waku_enr.Record] = @[], ): (WakuDiscoveryV5, Record) = let privKey = generateSecp256k1Key() record = newTestEnrRecord( privKey = privKey, extIp = extIp, tcpPort = tcpPort, udpPort = udpPort, indices = indices, flags = recordFlags, ) node = newTestDiscv5( privKey = privKey, bindIp = bindIp, tcpPort = tcpPort, udpPort = udpPort, record = record, bootstrapRecords = bootstrapRecords, ) (node, record) let filterForStore: WakuDiscv5Predicate = proc(record: waku_enr.Record): bool = let typedRecord = record.toTyped() if typedRecord.isErr(): return false let capabilities = typedRecord.value.waku2 if capabilities.isNone(): return false return capabilities.get().supportsCapability(Capabilities.Store) asyncTest "find random peers without predicate": # Given 3 nodes let (node1, record1) = buildNode(tcpPort = 61500u16, udpPort = 9000u16) (node2, record2) = buildNode(tcpPort = 61502u16, udpPort = 9002u16) (node3, record3) = buildNode( tcpPort = 61504u16, udpPort = 9004u16, bootstrapRecords = @[record1, record2], ) let res1 = await node1.start() assertResultOk res1 let res2 = await node2.start() assertResultOk res2 let res3 = await node3.start() assertResultOk res3 await sleepAsync(FUTURE_TIMEOUT) ## When we find random peers let res = await node3.findRandomPeers() var tcpPortList = res.mapIt(it.toTypedRecord().value().tcp.get()) tcpPortList.sort() ## Then check: res.len == 2 tcpPortList == @[61500, 61502] ## Cleanup await allFutures(node1.stop(), node2.stop(), node3.stop()) asyncTest "find random peers with parameter predicate": # Given 4 nodes let (node3, record3) = buildNode( tcpPort = 61504u16, udpPort = 9004u16, recordFlags = some(CapabilitiesBitfield.init(Capabilities.Relay, Capabilities.Filter)), ) (node4, record4) = buildNode( tcpPort = 61506u16, udpPort = 9006u16, recordFlags = some(CapabilitiesBitfield.init(Capabilities.Relay, Capabilities.Store)), ) (node2, record2) = buildNode( tcpPort = 61502u16, udpPort = 9002u16, recordFlags = some(CapabilitiesBitfield.init(Capabilities.Relay, Capabilities.Store)), bootstrapRecords = @[record3, record4], ) (node1, record1) = buildNode( tcpPort = 61500u16, udpPort = 9000u16, recordFlags = some(CapabilitiesBitfield.init(Capabilities.Relay)), bootstrapRecords = @[record2], ) # Start nodes' discoveryV5 protocols let res1 = await node1.start() assertResultOk res1 let res2 = await node2.start() assertResultOk res2 let res3 = await node3.start() assertResultOk res3 let res4 = await node4.start() assertResultOk res4 await sleepAsync(FUTURE_TIMEOUT) ## When let peers = await node1.findRandomPeers(some(filterForStore)) ## Then check: peers.len >= 1 peers.allIt(it.supportsCapability(Capabilities.Store)) # Cleanup await allFutures(node1.stop(), node2.stop(), node3.stop(), node4.stop()) asyncTest "find random peers with instance predicate": ## Setup # Records let (node3, record3) = buildNode( tcpPort = 61504u16, udpPort = 9004u16, recordFlags = some(CapabilitiesBitfield.init(Capabilities.Relay, Capabilities.Filter)), ) (node4, record4) = buildNode( tcpPort = 61506u16, udpPort = 9006u16, recordFlags = some(CapabilitiesBitfield.init(Capabilities.Relay, Capabilities.Store)), ) (node2, record2) = buildNode( tcpPort = 61502u16, udpPort = 9002u16, recordFlags = some(CapabilitiesBitfield.init(Capabilities.Relay, Capabilities.Store)), bootstrapRecords = @[record3, record4], ) let (node1, record1) = buildNode( tcpPort = 61500u16, udpPort = 9000u16, recordFlags = some(CapabilitiesBitfield.init(Capabilities.Relay)), indices = @[0u64, 0u64, 1u64, 0u64, 0u64], bootstrapRecords = @[record2], ) # Start nodes' discoveryV5 protocols let res1 = await node1.start() assertResultOk res1 let res2 = await node2.start() assertResultOk res2 let res3 = await node3.start() assertResultOk res3 let res4 = await node4.start() assertResultOk res4 await sleepAsync(FUTURE_TIMEOUT) ## When let peers = await node1.findRandomPeers() ## Then check: peers.len >= 1 peers.allIt(it.supportsCapability(Capabilities.Store)) # Cleanup await allFutures(node1.stop(), node2.stop(), node3.stop(), node4.stop()) suite "addBoostrapNode": let validEnr = "enr:-I-4QG3mX250ArniAs2DLpW-QHOLKSD5x_Ibp8AYcQZbz1HhHFJtl2dNDGcha" & "U5ugLbDKRgtTDZH8NsxXlTXDpYAgzgBgmlkgnY0gnJzjwAVBgABAAIABQAHAAkAC4" & "lzZWNwMjU2azGhA4_KwN0NRRmmfQ-B9B2h2PZjoJvBnaIOi6sR_b2UTQBBhXdha3U" & "yAQ" asyncTest "address is valid": # Given an empty list of enrs var enrs: seq[Record] = @[] # When adding a valid enr addBootstrapNode(validEnr, enrs) var r: Record echo r.fromURI(validEnr) echo r # Then the enr is added to the list check: enrs.len == 1 enrs[0].toBase64() == validEnr[4..^1] asyncTest "address is empty": # Given an empty list of enrs var enrs: seq[Record] = @[] # When adding an empty enr addBootstrapNode("", enrs) # Then the enr is not added to the list check: enrs.len == 0 asyncTest "address is valid but starts with #": # Given an empty list of enrs var enrs: seq[Record] = @[] # When adding any enr that starts with # let enr = "#" & validEnr addBootstrapNode(enr, enrs) # Then the enr is not added to the list check: enrs.len == 0 asyncTest "address is not valid": # Given an empty list of enrs var enrs: seq[Record] = @[] # When adding an invalid enr let enr = "enr:invalid" addBootstrapNode(enr, enrs) # Then the enr is not added to the list check: enrs.len == 0