{.used.} import std/[sequtils, tempfiles], stew/byteutils, stew/shims/net as stewNet, testutils/unittests, chronos, libp2p/switch, libp2p/protocols/pubsub/pubsub from std/times import epochTime import waku/[node/waku_node, node/peer_manager, waku_core, waku_node, waku_rln_relay], ../waku_store/store_utils, ../waku_archive/archive_utils, ../waku_relay/utils, ../waku_rln_relay/test_rln_group_manager_onchain, ../testlib/[wakucore, wakunode, testasync, futures], ../resources/payloads suite "Waku RlnRelay - End to End": var pubsubTopic {.threadvar.}: PubsubTopic contentTopic {.threadvar.}: ContentTopic var server {.threadvar.}: WakuNode client {.threadvar.}: WakuNode var serverRemotePeerInfo {.threadvar.}: RemotePeerInfo clientPeerId {.threadvar.}: PeerId asyncSetup: pubsubTopic = DefaultPubsubTopic contentTopic = DefaultContentTopic let serverKey = generateSecp256k1Key() clientKey = generateSecp256k1Key() server = newTestWakuNode(serverKey, ValidIpAddress.init("0.0.0.0"), Port(0)) client = newTestWakuNode(clientKey, ValidIpAddress.init("0.0.0.0"), Port(0)) await allFutures(server.start(), client.start()) serverRemotePeerInfo = server.switch.peerInfo.toRemotePeerInfo() clientPeerId = client.switch.peerInfo.toRemotePeerInfo().peerId asyncTeardown: await allFutures(client.stop(), server.stop()) suite "Mount": asyncTest "Can't mount if relay is not mounted": # Given Relay and RLN are not mounted check: server.wakuRelay == nil server.wakuRlnRelay == nil # When RlnRelay is mounted let catchRes = catch: await server.setupRln(1) # Then Relay and RLN are not mounted,and the process fails check: server.wakuRelay == nil server.wakuRlnRelay == nil catchRes.error()[].msg == "WakuRelay protocol is not mounted, cannot mount WakuRlnRelay" asyncTest "Pubsub topics subscribed before mounting RlnRelay are added to it": # Given the node enables Relay and Rln while subscribing to a pubsub topic await server.setupRelayWithRln(1.uint, @[pubsubTopic]) await client.setupRelayWithRln(2.uint, @[pubsubTopic]) check: server.wakuRelay != nil server.wakuRlnRelay != nil client.wakuRelay != nil client.wakuRlnRelay != nil # And the nodes are connected await client.connectToNodes(@[serverRemotePeerInfo]) # And the node registers the completion handler var completionFuture = subscribeCompletionHandler(server, pubsubTopic) # When the client sends a valid RLN message let isCompleted1 = await sendRlnMessage(client, pubsubTopic, contentTopic, completionFuture) # Then the valid RLN message is relayed check: isCompleted1 completionFuture.read() # When the client sends an invalid RLN message completionFuture = newBoolFuture() let isCompleted2 = await sendRlnMessageWithInvalidProof( client, pubsubTopic, contentTopic, completionFuture ) # Then the invalid RLN message is not relayed check: not isCompleted2 asyncTest "Pubsub topics subscribed after mounting RlnRelay are added to it": # Given the node enables Relay and Rln without subscribing to a pubsub topic await server.setupRelayWithRln(1.uint, @[]) await client.setupRelayWithRln(2.uint, @[]) # And the nodes are connected await client.connectToNodes(@[serverRemotePeerInfo]) # await sleepAsync(FUTURE_TIMEOUT) # And the node registers the completion handler var completionFuture = subscribeCompletionHandler(server, pubsubTopic) await sleepAsync(FUTURE_TIMEOUT) # When the client sends a valid RLN message let isCompleted1 = await sendRlnMessage(client, pubsubTopic, contentTopic, completionFuture) # Then the valid RLN message is relayed check: isCompleted1 completionFuture.read() # When the client sends an invalid RLN message completionFuture = newBoolFuture() let isCompleted2 = await sendRlnMessageWithInvalidProof( client, pubsubTopic, contentTopic, completionFuture ) # Then the invalid RLN message is not relayed check: not isCompleted2 asyncTest "rln-relay-max-message-limit testing": let nodekey = generateSecp256k1Key() node = newTestWakuNode(nodekey, parseIpAddress("0.0.0.0"), Port(0)) await node.mountRelay(@[DefaultPubsubTopic]) let contractAddress = await uploadRLNContract(EthClient) let wakuRlnConfig = WakuRlnConfig( rlnRelayDynamic: true, rlnRelayCredIndex: some(0.uint), rlnRelayUserMessageLimit: 111, rlnRelayTreepath: genTempPath("rln_tree", "wakunode_0"), rlnRelayEthClientAddress: EthClient, rlnRelayEthContractAddress: $contractAddress, rlnRelayChainId: 1337, onFatalErrorAction: proc(errStr: string) = raiseAssert errStr , ) try: await node.mountRlnRelay(wakuRlnConfig) except CatchableError as e: check e.msg == "failed to mount WakuRlnRelay: rln-relay-user-message-limit can't exceed the MAX_MESSAGE_LIMIT in the rln contract" suite "Analysis of Bandwith Limitations": asyncTest "Valid Payload Sizes": # Given the node enables Relay and Rln while subscribing to a pubsub topic await server.setupRelayWithRln(1.uint, @[pubsubTopic]) await client.setupRelayWithRln(2.uint, @[pubsubTopic]) # And the nodes are connected await client.connectToNodes(@[serverRemotePeerInfo]) # Register Relay Handler var completionFut = newPushHandlerFuture() proc relayHandler( topic: PubsubTopic, msg: WakuMessage ): Future[void] {.async, gcsafe.} = if topic == pubsubTopic: completionFut.complete((topic, msg)) let subscriptionEvent = (kind: PubsubSub, topic: pubsubTopic) server.subscribe(subscriptionEvent, some(relayHandler)) await sleepAsync(FUTURE_TIMEOUT) # Generate Messages let epoch = epochTime() payload1b = getByteSequence(1) payload1kib = getByteSequence(1024) overhead: uint64 = 419 payload150kib = getByteSequence((150 * 1024) - overhead) payload150kibPlus = getByteSequence((150 * 1024) - overhead + 1) var message1b = WakuMessage(payload: @payload1b, contentTopic: contentTopic) message1kib = WakuMessage(payload: @payload1kib, contentTopic: contentTopic) message150kib = WakuMessage(payload: @payload150kib, contentTopic: contentTopic) message151kibPlus = WakuMessage(payload: @payload150kibPlus, contentTopic: contentTopic) doAssert( client.wakuRlnRelay .appendRLNProof( message1b, epoch + float64(client.wakuRlnRelay.rlnEpochSizeSec * 0) ) .isOk() ) doAssert( client.wakuRlnRelay .appendRLNProof( message1kib, epoch + float64(client.wakuRlnRelay.rlnEpochSizeSec * 1) ) .isOk() ) doAssert( client.wakuRlnRelay .appendRLNProof( message150kib, epoch + float64(client.wakuRlnRelay.rlnEpochSizeSec * 2) ) .isOk() ) doAssert( client.wakuRlnRelay .appendRLNProof( message151kibPlus, epoch + float64(client.wakuRlnRelay.rlnEpochSizeSec * 3) ) .isOk() ) # When sending the 1B message discard await client.publish(some(pubsubTopic), message1b) discard await completionFut.withTimeout(FUTURE_TIMEOUT_LONG) # Then the message is relayed check completionFut.read() == (pubsubTopic, message1b) # When sending the 1KiB message completionFut = newPushHandlerFuture() # Reset Future discard await client.publish(some(pubsubTopic), message1kib) discard await completionFut.withTimeout(FUTURE_TIMEOUT_LONG) # Then the message is relayed check completionFut.read() == (pubsubTopic, message1kib) # When sending the 150KiB message completionFut = newPushHandlerFuture() # Reset Future discard await client.publish(some(pubsubTopic), message150kib) discard await completionFut.withTimeout(FUTURE_TIMEOUT_LONG) # Then the message is relayed check completionFut.read() == (pubsubTopic, message150kib) # When sending the 150KiB plus message completionFut = newPushHandlerFuture() # Reset Future discard await client.publish(some(pubsubTopic), message151kibPlus) # Then the message is not relayed check not await completionFut.withTimeout(FUTURE_TIMEOUT_LONG) asyncTest "Invalid Payload Sizes": # Given the node enables Relay and Rln while subscribing to a pubsub topic await server.setupRelayWithRln(1.uint, @[pubsubTopic]) await client.setupRelayWithRln(2.uint, @[pubsubTopic]) # And the nodes are connected await client.connectToNodes(@[serverRemotePeerInfo]) # Register Relay Handler var completionFut = newPushHandlerFuture() proc relayHandler( topic: PubsubTopic, msg: WakuMessage ): Future[void] {.async, gcsafe.} = if topic == pubsubTopic: completionFut.complete((topic, msg)) let subscriptionEvent = (kind: PubsubSub, topic: pubsubTopic) server.subscribe(subscriptionEvent, some(relayHandler)) await sleepAsync(FUTURE_TIMEOUT) # Generate Messages let epoch = epochTime() overhead: uint64 = 419 payload150kibPlus = getByteSequence((150 * 1024) - overhead + 1) var message151kibPlus = WakuMessage(payload: @payload150kibPlus, contentTopic: contentTopic) doAssert( client.wakuRlnRelay .appendRLNProof( message151kibPlus, epoch + float64(client.wakuRlnRelay.rlnEpochSizeSec * 3) ) .isOk() ) # When sending the 150KiB plus message completionFut = newPushHandlerFuture() # Reset Future discard await client.publish(some(pubsubTopic), message151kibPlus) # Then the message is not relayed check not await completionFut.withTimeout(FUTURE_TIMEOUT_LONG)