2026-02-20 09:34:40 -03:00
|
|
|
{.used.}
|
|
|
|
|
|
2026-02-20 19:49:02 -03:00
|
|
|
import std/[strutils, net, options]
|
2026-02-20 09:34:40 -03:00
|
|
|
import chronos, testutils/unittests, stew/byteutils
|
2026-02-20 19:49:02 -03:00
|
|
|
import libp2p/[peerid, peerinfo, multiaddress, crypto/crypto]
|
|
|
|
|
import ../testlib/[common, wakucore, wakunode, testasync]
|
|
|
|
|
|
2026-02-20 09:34:40 -03:00
|
|
|
import
|
2026-02-24 19:48:34 -03:00
|
|
|
waku,
|
|
|
|
|
waku/[
|
|
|
|
|
waku_node,
|
|
|
|
|
waku_core,
|
|
|
|
|
common/broker/broker_context,
|
|
|
|
|
events/message_events,
|
|
|
|
|
waku_relay/protocol,
|
|
|
|
|
]
|
2026-02-20 09:34:40 -03:00
|
|
|
import waku/api/api_conf, waku/factory/waku_conf
|
|
|
|
|
|
2026-02-24 19:48:34 -03:00
|
|
|
# TODO: Edge testing (after EdgeDriver is completed)
|
|
|
|
|
|
2026-02-20 19:49:02 -03:00
|
|
|
const TestTimeout = chronos.seconds(10)
|
2026-02-24 19:48:34 -03:00
|
|
|
const NegativeTestTimeout = chronos.seconds(2)
|
2026-02-20 19:49:02 -03:00
|
|
|
const DefaultShard = PubsubTopic("/waku/2/rs/1/0")
|
|
|
|
|
|
2026-02-20 09:34:40 -03:00
|
|
|
type ReceiveEventListenerManager = ref object
|
|
|
|
|
brokerCtx: BrokerContext
|
|
|
|
|
receivedListener: MessageReceivedEventListener
|
2026-02-20 19:49:02 -03:00
|
|
|
receivedEvent: AsyncEvent
|
2026-02-20 09:34:40 -03:00
|
|
|
receivedMessages: seq[WakuMessage]
|
2026-02-20 19:49:02 -03:00
|
|
|
targetCount: int
|
2026-02-20 09:34:40 -03:00
|
|
|
|
|
|
|
|
proc newReceiveEventListenerManager(
|
2026-02-20 19:49:02 -03:00
|
|
|
brokerCtx: BrokerContext, expectedCount: int = 1
|
2026-02-20 09:34:40 -03:00
|
|
|
): ReceiveEventListenerManager =
|
2026-02-20 19:49:02 -03:00
|
|
|
let manager = ReceiveEventListenerManager(
|
|
|
|
|
brokerCtx: brokerCtx, receivedMessages: @[], targetCount: expectedCount
|
|
|
|
|
)
|
|
|
|
|
manager.receivedEvent = newAsyncEvent()
|
2026-02-20 09:34:40 -03:00
|
|
|
|
2026-02-20 19:49:02 -03:00
|
|
|
manager.receivedListener = MessageReceivedEvent
|
|
|
|
|
.listen(
|
|
|
|
|
brokerCtx,
|
|
|
|
|
proc(event: MessageReceivedEvent) {.async: (raises: []).} =
|
|
|
|
|
manager.receivedMessages.add(event.message)
|
2026-02-20 09:34:40 -03:00
|
|
|
|
2026-02-20 19:49:02 -03:00
|
|
|
if manager.receivedMessages.len >= manager.targetCount:
|
|
|
|
|
manager.receivedEvent.fire()
|
|
|
|
|
,
|
|
|
|
|
)
|
|
|
|
|
.expect("Failed to listen to MessageReceivedEvent")
|
2026-02-20 09:34:40 -03:00
|
|
|
|
|
|
|
|
return manager
|
|
|
|
|
|
|
|
|
|
proc teardown(manager: ReceiveEventListenerManager) =
|
|
|
|
|
MessageReceivedEvent.dropListener(manager.brokerCtx, manager.receivedListener)
|
|
|
|
|
|
2026-02-20 19:49:02 -03:00
|
|
|
proc waitForEvents(
|
2026-02-20 09:34:40 -03:00
|
|
|
manager: ReceiveEventListenerManager, timeout: Duration
|
|
|
|
|
): Future[bool] {.async.} =
|
2026-02-20 19:49:02 -03:00
|
|
|
return await manager.receivedEvent.wait().withTimeout(timeout)
|
2026-02-20 09:34:40 -03:00
|
|
|
|
|
|
|
|
proc createApiNodeConf(mode: WakuMode = WakuMode.Core): NodeConfig =
|
|
|
|
|
let netConf = NetworkingConfig(listenIpv4: "0.0.0.0", p2pTcpPort: 0, discv5UdpPort: 0)
|
|
|
|
|
result = NodeConfig.init(
|
|
|
|
|
mode = mode,
|
|
|
|
|
protocolsConfig = ProtocolsConfig.init(
|
|
|
|
|
entryNodes = @[],
|
|
|
|
|
clusterId = 1,
|
|
|
|
|
autoShardingConfig = AutoShardingConfig(numShardsInCluster: 1),
|
|
|
|
|
),
|
|
|
|
|
networkingConfig = netConf,
|
|
|
|
|
p2pReliability = true,
|
|
|
|
|
)
|
|
|
|
|
|
2026-02-20 19:49:02 -03:00
|
|
|
proc setupSubscriberNode(conf: NodeConfig): Future[Waku] {.async.} =
|
|
|
|
|
var node: Waku
|
|
|
|
|
lockNewGlobalBrokerContext:
|
|
|
|
|
node = (await createNode(conf)).expect("Failed to create subscriber node")
|
|
|
|
|
(await startWaku(addr node)).expect("Failed to start subscriber node")
|
|
|
|
|
return node
|
|
|
|
|
|
2026-02-24 19:48:34 -03:00
|
|
|
proc waitForMesh*(node: WakuNode, shard: PubsubTopic) {.async.} =
|
|
|
|
|
for _ in 0 ..< 50:
|
|
|
|
|
if node.wakuRelay.getNumPeersInMesh(shard).valueOr(0) > 0:
|
|
|
|
|
return
|
|
|
|
|
await sleepAsync(100.milliseconds)
|
|
|
|
|
raise newException(ValueError, "GossipSub Mesh failed to stabilize")
|
|
|
|
|
|
2026-02-20 19:49:02 -03:00
|
|
|
proc publishWhenMeshReady(
|
|
|
|
|
publisher: WakuNode,
|
|
|
|
|
pubsubTopic: PubsubTopic,
|
|
|
|
|
contentTopic: ContentTopic,
|
|
|
|
|
payload: seq[byte],
|
|
|
|
|
): Future[Result[int, string]] {.async.} =
|
2026-02-24 19:48:34 -03:00
|
|
|
await waitForMesh(publisher, pubsubTopic)
|
2026-02-20 19:49:02 -03:00
|
|
|
|
2026-02-24 19:48:34 -03:00
|
|
|
let msg = WakuMessage(
|
|
|
|
|
payload: payload, contentTopic: contentTopic, version: 0, timestamp: now()
|
|
|
|
|
)
|
|
|
|
|
return await publisher.publish(some(pubsubTopic), msg)
|
2026-02-20 19:49:02 -03:00
|
|
|
|
|
|
|
|
suite "Messaging API, SubscriptionService":
|
|
|
|
|
var
|
|
|
|
|
publisherNode {.threadvar.}: WakuNode
|
|
|
|
|
publisherPeerInfo {.threadvar.}: RemotePeerInfo
|
|
|
|
|
publisherPeerId {.threadvar.}: PeerId
|
|
|
|
|
|
|
|
|
|
subscriberNode {.threadvar.}: Waku
|
|
|
|
|
|
|
|
|
|
asyncSetup:
|
2026-02-20 09:34:40 -03:00
|
|
|
lockNewGlobalBrokerContext:
|
2026-02-20 19:49:02 -03:00
|
|
|
publisherNode =
|
|
|
|
|
newTestWakuNode(generateSecp256k1Key(), parseIpAddress("0.0.0.0"), Port(0))
|
|
|
|
|
|
|
|
|
|
publisherNode.mountMetadata(1, @[0'u16]).expect("Failed to mount metadata")
|
|
|
|
|
(await publisherNode.mountRelay()).expect("Failed to mount relay")
|
|
|
|
|
await publisherNode.mountLibp2pPing()
|
|
|
|
|
await publisherNode.start()
|
|
|
|
|
|
|
|
|
|
publisherPeerInfo = publisherNode.peerInfo.toRemotePeerInfo()
|
|
|
|
|
publisherPeerId = publisherNode.peerInfo.peerId
|
|
|
|
|
|
2026-02-24 19:48:34 -03:00
|
|
|
proc dummyHandler(topic: PubsubTopic, msg: WakuMessage) {.async, gcsafe.} =
|
2026-02-20 19:49:02 -03:00
|
|
|
discard
|
|
|
|
|
|
|
|
|
|
publisherNode.subscribe((kind: PubsubSub, topic: DefaultShard), dummyHandler).expect(
|
|
|
|
|
"Failed to subscribe publisherNode"
|
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
asyncTeardown:
|
|
|
|
|
if not subscriberNode.isNil():
|
|
|
|
|
(await subscriberNode.stop()).expect("Failed to stop subscriber node")
|
|
|
|
|
subscriberNode = nil
|
|
|
|
|
|
|
|
|
|
if not publisherNode.isNil():
|
|
|
|
|
await publisherNode.stop()
|
|
|
|
|
publisherNode = nil
|
|
|
|
|
|
|
|
|
|
asyncTest "Subscription API, relay node auto subscribe and receive message":
|
|
|
|
|
subscriberNode = await setupSubscriberNode(createApiNodeConf(WakuMode.Core))
|
|
|
|
|
await subscriberNode.node.connectToNodes(@[publisherPeerInfo])
|
2026-02-20 09:34:40 -03:00
|
|
|
let testTopic = ContentTopic("/waku/2/test-content/proto")
|
|
|
|
|
|
2026-02-20 19:49:02 -03:00
|
|
|
(await subscriberNode.subscribe(testTopic)).expect(
|
|
|
|
|
"subscriberNode failed to subscribe"
|
|
|
|
|
)
|
2026-02-20 09:34:40 -03:00
|
|
|
|
2026-02-20 19:49:02 -03:00
|
|
|
let eventManager = newReceiveEventListenerManager(subscriberNode.brokerCtx, 1)
|
2026-02-20 09:34:40 -03:00
|
|
|
defer:
|
|
|
|
|
eventManager.teardown()
|
|
|
|
|
|
2026-02-24 19:48:34 -03:00
|
|
|
discard (
|
|
|
|
|
await publishWhenMeshReady(
|
|
|
|
|
publisherNode, DefaultShard, testTopic, "Hello, world!".toBytes()
|
|
|
|
|
)
|
|
|
|
|
).expect("Publish failed")
|
|
|
|
|
|
|
|
|
|
require await eventManager.waitForEvents(TestTimeout)
|
|
|
|
|
require eventManager.receivedMessages.len == 1
|
|
|
|
|
check eventManager.receivedMessages[0].contentTopic == testTopic
|
|
|
|
|
|
|
|
|
|
asyncTest "Subscription API, relay node ignores unsubscribed content topics on same shard":
|
|
|
|
|
subscriberNode = await setupSubscriberNode(createApiNodeConf(WakuMode.Core))
|
|
|
|
|
await subscriberNode.node.connectToNodes(@[publisherPeerInfo])
|
|
|
|
|
|
|
|
|
|
let subbedTopic = ContentTopic("/waku/2/subbed-topic/proto")
|
|
|
|
|
let ignoredTopic = ContentTopic("/waku/2/ignored-topic/proto")
|
|
|
|
|
(await subscriberNode.subscribe(subbedTopic)).expect("failed to subscribe")
|
|
|
|
|
|
|
|
|
|
let eventManager = newReceiveEventListenerManager(subscriberNode.brokerCtx, 1)
|
|
|
|
|
defer:
|
|
|
|
|
eventManager.teardown()
|
|
|
|
|
|
|
|
|
|
discard (
|
|
|
|
|
await publishWhenMeshReady(
|
|
|
|
|
publisherNode, DefaultShard, ignoredTopic, "Ghost Msg".toBytes()
|
|
|
|
|
)
|
|
|
|
|
).expect("Publish failed")
|
|
|
|
|
|
|
|
|
|
check not await eventManager.waitForEvents(NegativeTestTimeout)
|
|
|
|
|
check eventManager.receivedMessages.len == 0
|
|
|
|
|
|
|
|
|
|
asyncTest "Subscription API, relay node unsubscribe stops message receipt":
|
|
|
|
|
subscriberNode = await setupSubscriberNode(createApiNodeConf(WakuMode.Core))
|
|
|
|
|
await subscriberNode.node.connectToNodes(@[publisherPeerInfo])
|
|
|
|
|
let testTopic = ContentTopic("/waku/2/unsub-test/proto")
|
|
|
|
|
|
|
|
|
|
(await subscriberNode.subscribe(testTopic)).expect("failed to subscribe")
|
|
|
|
|
subscriberNode.unsubscribe(testTopic).expect("failed to unsubscribe")
|
|
|
|
|
|
|
|
|
|
let eventManager = newReceiveEventListenerManager(subscriberNode.brokerCtx, 1)
|
|
|
|
|
defer:
|
|
|
|
|
eventManager.teardown()
|
2026-02-20 19:49:02 -03:00
|
|
|
|
|
|
|
|
discard (
|
2026-02-24 19:48:34 -03:00
|
|
|
await publishWhenMeshReady(
|
|
|
|
|
publisherNode, DefaultShard, testTopic, "Should be dropped".toBytes()
|
|
|
|
|
)
|
|
|
|
|
).expect("Publish failed")
|
|
|
|
|
|
|
|
|
|
check not await eventManager.waitForEvents(NegativeTestTimeout)
|
|
|
|
|
check eventManager.receivedMessages.len == 0
|
|
|
|
|
|
|
|
|
|
asyncTest "Subscription API, overlapping topics on same shard maintain correct isolation":
|
|
|
|
|
subscriberNode = await setupSubscriberNode(createApiNodeConf(WakuMode.Core))
|
|
|
|
|
await subscriberNode.node.connectToNodes(@[publisherPeerInfo])
|
2026-02-20 09:34:40 -03:00
|
|
|
|
2026-02-24 19:48:34 -03:00
|
|
|
let topicA = ContentTopic("/waku/2/topic-a/proto")
|
|
|
|
|
let topicB = ContentTopic("/waku/2/topic-b/proto")
|
|
|
|
|
(await subscriberNode.subscribe(topicA)).expect("failed to sub A")
|
|
|
|
|
(await subscriberNode.subscribe(topicB)).expect("failed to sub B")
|
2026-02-20 09:34:40 -03:00
|
|
|
|
2026-02-24 19:48:34 -03:00
|
|
|
let eventManager = newReceiveEventListenerManager(subscriberNode.brokerCtx, 1)
|
|
|
|
|
defer:
|
|
|
|
|
eventManager.teardown()
|
|
|
|
|
|
|
|
|
|
await waitForMesh(publisherNode, DefaultShard)
|
|
|
|
|
|
|
|
|
|
subscriberNode.unsubscribe(topicA).expect("failed to unsub A")
|
|
|
|
|
|
|
|
|
|
discard (
|
|
|
|
|
await publisherNode.publish(
|
|
|
|
|
some(DefaultShard),
|
|
|
|
|
WakuMessage(
|
|
|
|
|
payload: "Dropped Message".toBytes(),
|
|
|
|
|
contentTopic: topicA,
|
|
|
|
|
version: 0,
|
|
|
|
|
timestamp: now(),
|
|
|
|
|
),
|
|
|
|
|
)
|
|
|
|
|
).expect("Publish A failed")
|
|
|
|
|
|
|
|
|
|
discard (
|
|
|
|
|
await publisherNode.publish(
|
|
|
|
|
some(DefaultShard),
|
|
|
|
|
WakuMessage(
|
|
|
|
|
payload: "Kept Msg".toBytes(),
|
|
|
|
|
contentTopic: topicB,
|
|
|
|
|
version: 0,
|
|
|
|
|
timestamp: now(),
|
|
|
|
|
),
|
|
|
|
|
)
|
|
|
|
|
).expect("Publish B failed")
|
|
|
|
|
|
|
|
|
|
require await eventManager.waitForEvents(TestTimeout)
|
2026-02-20 19:49:02 -03:00
|
|
|
require eventManager.receivedMessages.len == 1
|
2026-02-24 19:48:34 -03:00
|
|
|
check eventManager.receivedMessages[0].contentTopic == topicB
|
|
|
|
|
|
|
|
|
|
asyncTest "Subscription API, redundant subs tolerated and subs are removed":
|
|
|
|
|
subscriberNode = await setupSubscriberNode(createApiNodeConf(WakuMode.Core))
|
|
|
|
|
await subscriberNode.node.connectToNodes(@[publisherPeerInfo])
|
|
|
|
|
let glitchTopic = ContentTopic("/waku/2/glitch/proto")
|
|
|
|
|
|
|
|
|
|
(await subscriberNode.subscribe(glitchTopic)).expect("failed to sub")
|
|
|
|
|
(await subscriberNode.subscribe(glitchTopic)).expect("failed to double sub")
|
|
|
|
|
subscriberNode.unsubscribe(glitchTopic).expect("failed to unsub")
|
|
|
|
|
|
|
|
|
|
let eventManager = newReceiveEventListenerManager(subscriberNode.brokerCtx, 1)
|
|
|
|
|
defer:
|
|
|
|
|
eventManager.teardown()
|
|
|
|
|
|
|
|
|
|
discard (
|
|
|
|
|
await publishWhenMeshReady(
|
|
|
|
|
publisherNode, DefaultShard, glitchTopic, "Ghost Msg".toBytes()
|
|
|
|
|
)
|
|
|
|
|
).expect("Publish failed")
|
|
|
|
|
|
|
|
|
|
check not await eventManager.waitForEvents(NegativeTestTimeout)
|
|
|
|
|
check eventManager.receivedMessages.len == 0
|
|
|
|
|
|
|
|
|
|
asyncTest "Subscription API, resubscribe to an unsubscribed topic":
|
|
|
|
|
subscriberNode = await setupSubscriberNode(createApiNodeConf(WakuMode.Core))
|
|
|
|
|
await subscriberNode.node.connectToNodes(@[publisherPeerInfo])
|
|
|
|
|
let testTopic = ContentTopic("/waku/2/resub-test/proto")
|
|
|
|
|
|
|
|
|
|
# Subscribe
|
|
|
|
|
(await subscriberNode.subscribe(testTopic)).expect("Initial sub failed")
|
|
|
|
|
|
|
|
|
|
var eventManager = newReceiveEventListenerManager(subscriberNode.brokerCtx, 1)
|
|
|
|
|
discard (
|
|
|
|
|
await publishWhenMeshReady(
|
|
|
|
|
publisherNode, DefaultShard, testTopic, "Msg 1".toBytes()
|
|
|
|
|
)
|
|
|
|
|
).expect("Pub 1 failed")
|
|
|
|
|
|
|
|
|
|
require await eventManager.waitForEvents(TestTimeout)
|
|
|
|
|
eventManager.teardown()
|
|
|
|
|
|
|
|
|
|
# Unsubscribe and verify teardown
|
|
|
|
|
subscriberNode.unsubscribe(testTopic).expect("Unsub failed")
|
|
|
|
|
eventManager = newReceiveEventListenerManager(subscriberNode.brokerCtx, 1)
|
|
|
|
|
|
|
|
|
|
discard (
|
|
|
|
|
await publisherNode.publish(
|
|
|
|
|
some(DefaultShard),
|
|
|
|
|
WakuMessage(
|
|
|
|
|
payload: "Ghost".toBytes(),
|
|
|
|
|
contentTopic: testTopic,
|
|
|
|
|
version: 0,
|
|
|
|
|
timestamp: now(),
|
|
|
|
|
),
|
|
|
|
|
)
|
|
|
|
|
).expect("Ghost pub failed")
|
|
|
|
|
|
|
|
|
|
check not await eventManager.waitForEvents(NegativeTestTimeout)
|
|
|
|
|
eventManager.teardown()
|
|
|
|
|
|
|
|
|
|
# Resubscribe
|
|
|
|
|
(await subscriberNode.subscribe(testTopic)).expect("Resub failed")
|
|
|
|
|
eventManager = newReceiveEventListenerManager(subscriberNode.brokerCtx, 1)
|
|
|
|
|
|
|
|
|
|
discard (
|
|
|
|
|
await publishWhenMeshReady(
|
|
|
|
|
publisherNode, DefaultShard, testTopic, "Msg 2".toBytes()
|
|
|
|
|
)
|
|
|
|
|
).expect("Pub 2 failed")
|
2026-02-20 09:34:40 -03:00
|
|
|
|
2026-02-24 19:48:34 -03:00
|
|
|
require await eventManager.waitForEvents(TestTimeout)
|
|
|
|
|
check eventManager.receivedMessages[0].payload == "Msg 2".toBytes()
|