2020-08-31 03:32:41 +00:00
|
|
|
{.used.}
|
|
|
|
|
|
|
|
import
|
2020-09-01 02:09:54 +00:00
|
|
|
std/unittest,
|
|
|
|
chronicles, chronos, stew/shims/net as stewNet, stew/byteutils,
|
2020-08-31 03:32:41 +00:00
|
|
|
libp2p/crypto/crypto,
|
|
|
|
libp2p/crypto/secp,
|
2020-09-02 03:15:25 +00:00
|
|
|
libp2p/switch,
|
2020-08-31 03:32:41 +00:00
|
|
|
eth/keys,
|
2020-10-02 12:48:56 +00:00
|
|
|
../../waku/protocol/v2/[waku_relay, waku_store, waku_filter, message_notifier],
|
2020-09-01 02:09:54 +00:00
|
|
|
../../waku/node/v2/[wakunode2, waku_types],
|
2020-08-31 03:32:41 +00:00
|
|
|
../test_helpers
|
2020-07-29 13:24:01 +00:00
|
|
|
|
|
|
|
procSuite "WakuNode":
|
2020-09-02 03:15:25 +00:00
|
|
|
let rng = keys.newRng()
|
2020-07-29 13:24:01 +00:00
|
|
|
asyncTest "Message published with content filter is retrievable":
|
2020-09-01 02:09:54 +00:00
|
|
|
let
|
|
|
|
nodeKey = crypto.PrivateKey.random(Secp256k1, rng[])[]
|
|
|
|
node = WakuNode.init(nodeKey, ValidIpAddress.init("0.0.0.0"),
|
|
|
|
Port(60000))
|
2020-09-02 03:15:25 +00:00
|
|
|
pubSubTopic = "chat"
|
|
|
|
contentTopic = "foobar"
|
2020-10-02 12:48:56 +00:00
|
|
|
filterRequest = FilterRequest(topic: pubSubTopic, contentFilters: @[ContentFilter(topics: @[contentTopic])])
|
2020-09-02 03:15:25 +00:00
|
|
|
message = WakuMessage(payload: "hello world".toBytes(),
|
|
|
|
contentTopic: contentTopic)
|
|
|
|
|
|
|
|
# This could/should become a more fixed handler (at least default) that
|
|
|
|
# would be enforced on WakuNode level.
|
|
|
|
proc relayHandler(topic: string, data: seq[byte]) {.async, gcsafe.} =
|
|
|
|
let msg = WakuMessage.init(data)
|
|
|
|
if msg.isOk():
|
|
|
|
check:
|
|
|
|
topic == "chat"
|
2020-10-02 12:48:56 +00:00
|
|
|
node.filters.notify(msg.value(), topic)
|
2020-09-02 03:15:25 +00:00
|
|
|
|
|
|
|
var completionFut = newFuture[bool]()
|
|
|
|
|
|
|
|
# This would be the actual application handler
|
2020-10-02 12:48:56 +00:00
|
|
|
proc contentHandler(msg: WakuMessage) {.gcsafe, closure.} =
|
|
|
|
let message = string.fromBytes(msg.payload)
|
2020-09-02 03:15:25 +00:00
|
|
|
check:
|
2020-10-02 12:48:56 +00:00
|
|
|
message == "hello world"
|
2020-09-02 03:15:25 +00:00
|
|
|
completionFut.complete(true)
|
2020-07-29 13:24:01 +00:00
|
|
|
|
2020-09-01 02:09:54 +00:00
|
|
|
await node.start()
|
2020-07-29 13:24:01 +00:00
|
|
|
|
2020-09-02 03:15:25 +00:00
|
|
|
# Subscribe our node to the pubSubTopic where all chat data go onto.
|
2020-09-16 04:23:10 +00:00
|
|
|
await node.subscribe(pubSubTopic, relayHandler)
|
2020-10-02 12:48:56 +00:00
|
|
|
|
2020-09-02 03:15:25 +00:00
|
|
|
# Subscribe a contentFilter to trigger a specific application handler when
|
|
|
|
# WakuMessages with that content are received
|
2020-10-02 12:48:56 +00:00
|
|
|
# node2.wakuFilter.setPeer(node1.peerInfo)
|
|
|
|
await node.subscribe(filterRequest, contentHandler)
|
|
|
|
|
|
|
|
await sleepAsync(2000.millis)
|
2020-09-02 03:15:25 +00:00
|
|
|
|
|
|
|
node.publish(pubSubTopic, message)
|
|
|
|
|
|
|
|
check:
|
|
|
|
(await completionFut.withTimeout(5.seconds)) == true
|
|
|
|
|
|
|
|
await node.stop()
|
|
|
|
|
|
|
|
asyncTest "Content filtered publishing over network":
|
2020-09-01 02:09:54 +00:00
|
|
|
let
|
2020-09-02 03:15:25 +00:00
|
|
|
nodeKey1 = crypto.PrivateKey.random(Secp256k1, rng[])[]
|
|
|
|
node1 = WakuNode.init(nodeKey1, ValidIpAddress.init("0.0.0.0"),
|
|
|
|
Port(60000))
|
|
|
|
nodeKey2 = crypto.PrivateKey.random(Secp256k1, rng[])[]
|
|
|
|
node2 = WakuNode.init(nodeKey2, ValidIpAddress.init("0.0.0.0"),
|
|
|
|
Port(60002))
|
|
|
|
pubSubTopic = "chat"
|
2020-09-01 15:20:38 +00:00
|
|
|
contentTopic = "foobar"
|
2020-10-02 12:48:56 +00:00
|
|
|
filterRequest = FilterRequest(topic: pubSubTopic, contentFilters: @[ContentFilter(topics: @[contentTopic])])
|
2020-09-02 03:15:25 +00:00
|
|
|
message = WakuMessage(payload: "hello world".toBytes(),
|
2020-09-01 15:20:38 +00:00
|
|
|
contentTopic: contentTopic)
|
2020-07-29 13:24:01 +00:00
|
|
|
|
2020-09-02 03:15:25 +00:00
|
|
|
var completionFut = newFuture[bool]()
|
|
|
|
|
|
|
|
# This could/should become a more fixed handler (at least default) that
|
|
|
|
# would be enforced on WakuNode level.
|
|
|
|
proc relayHandler(topic: string, data: seq[byte]) {.async, gcsafe.} =
|
|
|
|
let msg = WakuMessage.init(data)
|
|
|
|
if msg.isOk():
|
|
|
|
check:
|
|
|
|
topic == "chat"
|
2020-10-02 12:48:56 +00:00
|
|
|
node1.filters.notify(msg.value(), topic)
|
2020-09-02 03:15:25 +00:00
|
|
|
|
|
|
|
# This would be the actual application handler
|
2020-10-02 12:48:56 +00:00
|
|
|
proc contentHandler(msg: WakuMessage) {.gcsafe, closure.} =
|
|
|
|
let message = string.fromBytes(msg.payload)
|
2020-09-02 03:15:25 +00:00
|
|
|
check:
|
2020-10-02 12:48:56 +00:00
|
|
|
message == "hello world"
|
2020-09-02 03:15:25 +00:00
|
|
|
completionFut.complete(true)
|
|
|
|
|
|
|
|
await allFutures([node1.start(), node2.start()])
|
|
|
|
|
|
|
|
# Subscribe our node to the pubSubTopic where all chat data go onto.
|
2020-09-16 04:23:10 +00:00
|
|
|
await node1.subscribe(pubSubTopic, relayHandler)
|
2020-09-02 03:15:25 +00:00
|
|
|
# Subscribe a contentFilter to trigger a specific application handler when
|
|
|
|
# WakuMessages with that content are received
|
2020-10-02 12:48:56 +00:00
|
|
|
node1.wakuFilter.setPeer(node2.peerInfo)
|
|
|
|
await node1.subscribe(filterRequest, contentHandler)
|
|
|
|
await sleepAsync(2000.millis)
|
|
|
|
|
2020-09-02 03:15:25 +00:00
|
|
|
# Connect peers by dialing from node2 to node1
|
|
|
|
let conn = await node2.switch.dial(node1.peerInfo, WakuRelayCodec)
|
2020-10-02 12:48:56 +00:00
|
|
|
|
2020-09-16 04:23:10 +00:00
|
|
|
# We need to sleep to allow the subscription to go through
|
|
|
|
info "Going to sleep to allow subscribe to go through"
|
|
|
|
await sleepAsync(2000.millis)
|
|
|
|
|
|
|
|
info "Waking up and publishing"
|
2020-09-02 03:15:25 +00:00
|
|
|
node2.publish(pubSubTopic, message)
|
|
|
|
|
|
|
|
check:
|
|
|
|
(await completionFut.withTimeout(5.seconds)) == true
|
2020-09-24 02:16:25 +00:00
|
|
|
await node1.stop()
|
|
|
|
await node2.stop()
|
|
|
|
|
|
|
|
asyncTest "Store protocol returns expected message":
|
|
|
|
let
|
|
|
|
nodeKey1 = crypto.PrivateKey.random(Secp256k1, rng[])[]
|
|
|
|
node1 = WakuNode.init(nodeKey1, ValidIpAddress.init("0.0.0.0"),
|
|
|
|
Port(60000))
|
|
|
|
nodeKey2 = crypto.PrivateKey.random(Secp256k1, rng[])[]
|
|
|
|
node2 = WakuNode.init(nodeKey2, ValidIpAddress.init("0.0.0.0"),
|
|
|
|
Port(60002))
|
|
|
|
contentTopic = "foobar"
|
|
|
|
message = WakuMessage(payload: "hello world".toBytes(), contentTopic: contentTopic)
|
|
|
|
|
|
|
|
var completionFut = newFuture[bool]()
|
|
|
|
|
|
|
|
await node1.start()
|
|
|
|
await node2.start()
|
|
|
|
|
|
|
|
await node2.subscriptions.notify("waku", message)
|
|
|
|
|
|
|
|
await sleepAsync(2000.millis)
|
|
|
|
|
|
|
|
node1.wakuStore.setPeer(node2.peerInfo)
|
|
|
|
|
|
|
|
proc storeHandler(response: HistoryResponse) {.gcsafe, closure.} =
|
|
|
|
check:
|
|
|
|
response.messages[0] == message
|
|
|
|
completionFut.complete(true)
|
|
|
|
|
2020-09-25 14:02:13 +00:00
|
|
|
await node1.query(HistoryQuery(topics: @[contentTopic]), storeHandler)
|
2020-09-24 02:16:25 +00:00
|
|
|
|
|
|
|
check:
|
|
|
|
(await completionFut.withTimeout(5.seconds)) == true
|
|
|
|
await node1.stop()
|
|
|
|
await node2.stop()
|
2020-10-02 12:48:56 +00:00
|
|
|
|
|
|
|
asyncTest "Filter protocol returns expected message":
|
|
|
|
let
|
|
|
|
nodeKey1 = crypto.PrivateKey.random(Secp256k1, rng[])[]
|
|
|
|
node1 = WakuNode.init(nodeKey1, ValidIpAddress.init("0.0.0.0"),
|
|
|
|
Port(60000))
|
|
|
|
nodeKey2 = crypto.PrivateKey.random(Secp256k1, rng[])[]
|
|
|
|
node2 = WakuNode.init(nodeKey2, ValidIpAddress.init("0.0.0.0"),
|
|
|
|
Port(60002))
|
|
|
|
contentTopic = "foobar"
|
|
|
|
message = WakuMessage(payload: "hello world".toBytes(), contentTopic: contentTopic)
|
|
|
|
|
|
|
|
var completionFut = newFuture[bool]()
|
|
|
|
|
|
|
|
await node1.start()
|
|
|
|
await node2.start()
|
|
|
|
|
|
|
|
node1.wakuFilter.setPeer(node2.peerInfo)
|
|
|
|
|
|
|
|
proc handler(msg: WakuMessage) {.gcsafe, closure.} =
|
|
|
|
check:
|
|
|
|
msg == message
|
|
|
|
completionFut.complete(true)
|
|
|
|
|
|
|
|
await node1.subscribe(FilterRequest(topic: "waku", contentFilters: @[ContentFilter(topics: @[contentTopic])]), handler)
|
|
|
|
|
|
|
|
await sleepAsync(2000.millis)
|
|
|
|
|
|
|
|
await node2.subscriptions.notify("waku", message)
|
|
|
|
|
|
|
|
await sleepAsync(2000.millis)
|
|
|
|
|
|
|
|
check:
|
|
|
|
(await completionFut.withTimeout(5.seconds)) == true
|
|
|
|
await node1.stop()
|
|
|
|
await node2.stop()
|