mirror of
https://github.com/waku-org/nwaku.git
synced 2025-01-27 07:06:42 +00:00
396 lines
12 KiB
Nim
396 lines
12 KiB
Nim
{.used.}
|
|
|
|
import
|
|
std/[options, sequtils],
|
|
stew/shims/net as stewNet,
|
|
testutils/unittests,
|
|
chronicles,
|
|
libp2p/crypto/crypto,
|
|
json_rpc/[rpcserver, rpcclient]
|
|
import
|
|
../../../waku/common/base64,
|
|
../../../waku/v2/node/peer_manager,
|
|
../../../waku/v2/node/message_cache,
|
|
../../../waku/v2/node/waku_node,
|
|
../../../waku/v2/node/jsonrpc/relay/handlers as relay_api,
|
|
../../../waku/v2/node/jsonrpc/relay/client as relay_api_client,
|
|
../../../waku/v2/protocol/waku_message,
|
|
../../../waku/v2/protocol/waku_relay,
|
|
../../../waku/v2/utils/compat,
|
|
../../../waku/v2/utils/peers,
|
|
../testlib/common,
|
|
../testlib/waku2
|
|
|
|
|
|
proc newTestMessageCache(): relay_api.MessageCache =
|
|
relay_api.MessageCache.init(capacity=30)
|
|
|
|
|
|
suite "Waku v2 JSON-RPC API - Relay":
|
|
|
|
asyncTest "subscribe and unsubscribe from topics":
|
|
## Setup
|
|
let node = WakuNode.new(generateSecp256k1Key(), ValidIpAddress.init("0.0.0.0"), Port(0))
|
|
|
|
await node.start()
|
|
await node.mountRelay(topics = @[DefaultPubsubTopic])
|
|
|
|
# JSON-RPC server
|
|
let
|
|
rpcPort = Port(8547)
|
|
ta = initTAddress(ValidIpAddress.init("0.0.0.0"), rpcPort)
|
|
server = newRpcHttpServer([ta])
|
|
|
|
installRelayApiHandlers(node, server, newTestMessageCache())
|
|
server.start()
|
|
|
|
# JSON-RPC client
|
|
let client = newRpcHttpClient()
|
|
await client.connect("127.0.0.1", rpcPort, false)
|
|
|
|
## Given
|
|
let newTopics = @["test-topic1","test-topic2","test-topic3"]
|
|
|
|
## When
|
|
# Subscribe to new topics
|
|
let subResp = await client.post_waku_v2_relay_v1_subscriptions(newTopics)
|
|
|
|
let subTopics = node.wakuRelay.subscribedTopics.toSeq()
|
|
|
|
# Unsubscribe from new topics
|
|
let unsubResp = await client.delete_waku_v2_relay_v1_subscriptions(newTopics)
|
|
|
|
let unsubTopics = node.wakuRelay.subscribedTopics.toSeq()
|
|
|
|
## Then
|
|
check:
|
|
subResp == true
|
|
check:
|
|
# Node is now subscribed to default + new topics
|
|
subTopics.len == 1 + newTopics.len
|
|
DefaultPubsubTopic in subTopics
|
|
newTopics.allIt(it in subTopics)
|
|
|
|
check:
|
|
unsubResp == true
|
|
check:
|
|
# Node is now unsubscribed from new topics
|
|
unsubTopics.len == 1
|
|
DefaultPubsubTopic in unsubTopics
|
|
newTopics.allIt(it notin unsubTopics)
|
|
|
|
await server.stop()
|
|
await server.closeWait()
|
|
|
|
await node.stop()
|
|
|
|
asyncTest "publish message to topic":
|
|
## Setup
|
|
let
|
|
pubSubTopic = "test-jsonrpc-pubsub-topic"
|
|
contentTopic = "test-jsonrpc-content-topic"
|
|
|
|
# Relay nodes setup
|
|
let
|
|
srcNode = WakuNode.new(generateSecp256k1Key(), ValidIpAddress.init("0.0.0.0"), Port(0))
|
|
dstNode = WakuNode.new(generateSecp256k1Key(), ValidIpAddress.init("0.0.0.0"), Port(0))
|
|
|
|
await allFutures(srcNode.start(), dstNode.start())
|
|
|
|
await srcNode.mountRelay(@[pubSubTopic])
|
|
await dstNode.mountRelay(@[pubSubTopic])
|
|
|
|
await srcNode.connectToNodes(@[dstNode.peerInfo.toRemotePeerInfo()])
|
|
|
|
|
|
# RPC server (source node)
|
|
let
|
|
rpcPort = Port(8548)
|
|
ta = initTAddress(ValidIpAddress.init("0.0.0.0"), rpcPort)
|
|
server = newRpcHttpServer([ta])
|
|
|
|
installRelayApiHandlers(srcNode, server, newTestMessageCache())
|
|
server.start()
|
|
|
|
# JSON-RPC client
|
|
let client = newRpcHttpClient()
|
|
await client.connect("127.0.0.1", rpcPort, false)
|
|
|
|
## Given
|
|
let message = fakeWakuMessage( payload= @[byte 72], contentTopic=contentTopic)
|
|
|
|
let dstHandlerFut = newFuture[(PubsubTopic, WakuMessage)]()
|
|
proc dstHandler(topic: PubsubTopic, msg: WakuMessage) {.async, gcsafe.} =
|
|
dstHandlerFut.complete((topic, msg))
|
|
|
|
dstNode.subscribe(pubSubTopic, dstHandler)
|
|
|
|
## When
|
|
let rpcMessage = WakuMessageRPC(
|
|
payload: Base64String.encode(message.payload),
|
|
contentTopic: some(message.contentTopic),
|
|
timestamp: some(message.timestamp),
|
|
version: some(message.version)
|
|
)
|
|
let response = await client.post_waku_v2_relay_v1_message(pubSubTopic, rpcMessage)
|
|
|
|
## Then
|
|
require:
|
|
response == true
|
|
await dstHandlerFut.withTimeout(chronos.seconds(5))
|
|
|
|
let (topic, msg) = dstHandlerFut.read()
|
|
check:
|
|
topic == pubSubTopic
|
|
msg == message
|
|
|
|
|
|
## Cleanup
|
|
await server.stop()
|
|
await server.closeWait()
|
|
await allFutures(srcNode.stop(), dstNode.stop())
|
|
|
|
asyncTest "get latest messages received from topics cache":
|
|
## Setup
|
|
let
|
|
pubSubTopic = "test-jsonrpc-pubsub-topic"
|
|
contentTopic = "test-jsonrpc-content-topic"
|
|
|
|
# Relay nodes setup
|
|
let
|
|
srcNode = WakuNode.new(generateSecp256k1Key(), ValidIpAddress.init("0.0.0.0"), Port(0))
|
|
dstNode = WakuNode.new(generateSecp256k1Key(), ValidIpAddress.init("0.0.0.0"), Port(0))
|
|
|
|
await allFutures(srcNode.start(), dstNode.start())
|
|
|
|
await srcNode.mountRelay(@[pubSubTopic])
|
|
await dstNode.mountRelay(@[pubSubTopic])
|
|
|
|
await srcNode.connectToNodes(@[dstNode.peerInfo.toRemotePeerInfo()])
|
|
|
|
|
|
# RPC server (destination node)
|
|
let
|
|
rpcPort = Port(8548)
|
|
ta = initTAddress(ValidIpAddress.init("0.0.0.0"), rpcPort)
|
|
server = newRpcHttpServer([ta])
|
|
|
|
installRelayApiHandlers(dstNode, server, newTestMessageCache())
|
|
server.start()
|
|
|
|
# JSON-RPC client
|
|
let client = newRpcHttpClient()
|
|
await client.connect("127.0.0.1", rpcPort, false)
|
|
|
|
## Given
|
|
let messages = @[
|
|
fakeWakuMessage(payload= @[byte 70], contentTopic=contentTopic),
|
|
fakeWakuMessage(payload= @[byte 71], contentTopic=contentTopic),
|
|
fakeWakuMessage(payload= @[byte 72], contentTopic=contentTopic),
|
|
fakeWakuMessage(payload= @[byte 73], contentTopic=contentTopic)
|
|
]
|
|
|
|
## When
|
|
for msg in messages:
|
|
await srcNode.publish(pubSubTopic, msg)
|
|
|
|
await sleepAsync(200.millis)
|
|
|
|
let dstMessages = await client.get_waku_v2_relay_v1_messages(pubSubTopic)
|
|
|
|
## Then
|
|
check:
|
|
dstMessages.len == 4
|
|
dstMessages[2].payload == Base64String.encode(messages[2].payload)
|
|
dstMessages[2].contentTopic.get() == messages[2].contentTopic
|
|
dstMessages[2].timestamp.get() == messages[2].timestamp
|
|
dstMessages[2].version.get() == messages[2].version
|
|
|
|
## Cleanup
|
|
await server.stop()
|
|
await server.closeWait()
|
|
await allFutures(srcNode.stop(), dstNode.stop())
|
|
|
|
|
|
suite "Waku v2 JSON-RPC API - Relay (Private)":
|
|
|
|
asyncTest "generate symmetric keys and encrypt/decrypt communication":
|
|
let
|
|
pubSubTopic = "test-relay-pubsub-topic"
|
|
contentTopic = "test-relay-content-topic"
|
|
|
|
let
|
|
srcNode = WakuNode.new(generateSecp256k1Key(), ValidIpAddress.init("127.0.0.1"), Port(0))
|
|
relNode = WakuNode.new(generateSecp256k1Key(), ValidIpAddress.init("127.0.0.1"), Port(0))
|
|
dstNode = WakuNode.new(generateSecp256k1Key(), ValidIpAddress.init("127.0.0.1"), Port(0))
|
|
|
|
await allFutures(srcNode.start(), relNode.start(), dstNode.start())
|
|
|
|
await srcNode.mountRelay(@[pubSubTopic])
|
|
await relNode.mountRelay(@[pubSubTopic])
|
|
await dstNode.mountRelay(@[pubSubTopic])
|
|
|
|
await srcNode.connectToNodes(@[relNode.peerInfo.toRemotePeerInfo()])
|
|
await relNode.connectToNodes(@[dstNode.peerInfo.toRemotePeerInfo()])
|
|
|
|
# Setup two servers so we can see both sides of encrypted communication
|
|
let
|
|
srcRpcPort = Port(8554)
|
|
srcTa = initTAddress(ValidIpAddress.init("127.0.0.1"), srcRpcPort)
|
|
srcServer = newRpcHttpServer([srcTa])
|
|
|
|
let srcMessageCache = newTestMessageCache()
|
|
installRelayApiHandlers(srcNode, srcServer, srcMessageCache)
|
|
installRelayPrivateApiHandlers(srcNode, srcServer, srcMessageCache)
|
|
srcServer.start()
|
|
|
|
let
|
|
dstRpcPort = Port(8555)
|
|
dstTa = initTAddress(ValidIpAddress.init("127.0.0.1"), dstRpcPort)
|
|
dstServer = newRpcHttpServer([dstTa])
|
|
|
|
let dstMessageCache = newTestMessageCache()
|
|
installRelayApiHandlers(dstNode, dstServer, dstMessageCache)
|
|
installRelayPrivateApiHandlers(dstNode, dstServer, dstMessageCache)
|
|
dstServer.start()
|
|
|
|
|
|
let srcClient = newRpcHttpClient()
|
|
await srcClient.connect("127.0.0.1", srcRpcPort, false)
|
|
|
|
let dstClient = newRpcHttpClient()
|
|
await dstClient.connect("127.0.0.1", dstRpcPort, false)
|
|
|
|
## Given
|
|
let
|
|
payload = @[byte 38]
|
|
payloadBase64 = Base64String.encode(payload)
|
|
|
|
let message = WakuMessageRPC(
|
|
payload: payloadBase64,
|
|
contentTopic: some(contentTopic),
|
|
timestamp: some(now()),
|
|
)
|
|
|
|
## When
|
|
let symkey = await dstClient.get_waku_v2_private_v1_symmetric_key()
|
|
|
|
let posted = await srcCLient.post_waku_v2_private_v1_symmetric_message(pubSubTopic, message, symkey = (%symkey).getStr())
|
|
require:
|
|
posted
|
|
|
|
await sleepAsync(100.millis)
|
|
|
|
# Let's see if we can receive, and decrypt, this message on dstNode
|
|
var messages = await dstClient.get_waku_v2_private_v1_symmetric_messages(pubSubTopic, symkey = (%symkey).getStr())
|
|
check:
|
|
messages.len == 1
|
|
messages[0].payload == payloadBase64
|
|
messages[0].contentTopic == message.contentTopic
|
|
messages[0].timestamp == message.timestamp
|
|
messages[0].version.get() == 1'u32
|
|
|
|
# Ensure that read messages are cleared from cache
|
|
messages = await dstClient.get_waku_v2_private_v1_symmetric_messages(pubSubTopic, symkey = (%symkey).getStr())
|
|
check:
|
|
messages.len == 0
|
|
|
|
## Cleanup
|
|
await srcServer.stop()
|
|
await srcServer.closeWait()
|
|
await dstServer.stop()
|
|
await dstServer.closeWait()
|
|
|
|
await allFutures(srcNode.stop(), relNode.stop(), dstNode.stop())
|
|
|
|
asyncTest "generate asymmetric keys and encrypt/decrypt communication":
|
|
let
|
|
pubSubTopic = "test-relay-pubsub-topic"
|
|
contentTopic = "test-relay-content-topic"
|
|
|
|
let
|
|
srcNode = WakuNode.new(generateSecp256k1Key(), ValidIpAddress.init("127.0.0.1"), Port(0))
|
|
relNode = WakuNode.new(generateSecp256k1Key(), ValidIpAddress.init("127.0.0.1"), Port(0))
|
|
dstNode = WakuNode.new(generateSecp256k1Key(), ValidIpAddress.init("127.0.0.1"), Port(0))
|
|
|
|
await allFutures(srcNode.start(), relNode.start(), dstNode.start())
|
|
|
|
await srcNode.mountRelay(@[pubSubTopic])
|
|
await relNode.mountRelay(@[pubSubTopic])
|
|
await dstNode.mountRelay(@[pubSubTopic])
|
|
|
|
await srcNode.connectToNodes(@[relNode.peerInfo.toRemotePeerInfo()])
|
|
await relNode.connectToNodes(@[dstNode.peerInfo.toRemotePeerInfo()])
|
|
|
|
# Setup two servers so we can see both sides of encrypted communication
|
|
let
|
|
srcRpcPort = Port(8554)
|
|
srcTa = initTAddress(ValidIpAddress.init("127.0.0.1"), srcRpcPort)
|
|
srcServer = newRpcHttpServer([srcTa])
|
|
|
|
let srcMessageCache = newTestMessageCache()
|
|
installRelayApiHandlers(srcNode, srcServer, srcMessageCache)
|
|
installRelayPrivateApiHandlers(srcNode, srcServer, srcMessageCache)
|
|
srcServer.start()
|
|
|
|
let
|
|
dstRpcPort = Port(8555)
|
|
dstTa = initTAddress(ValidIpAddress.init("127.0.0.1"), dstRpcPort)
|
|
dstServer = newRpcHttpServer([dstTa])
|
|
|
|
let dstMessageCache = newTestMessageCache()
|
|
installRelayApiHandlers(dstNode, dstServer, dstMessageCache)
|
|
installRelayPrivateApiHandlers(dstNode, dstServer, dstMessageCache)
|
|
dstServer.start()
|
|
|
|
|
|
let srcClient = newRpcHttpClient()
|
|
await srcClient.connect("127.0.0.1", srcRpcPort, false)
|
|
|
|
let dstClient = newRpcHttpClient()
|
|
await dstClient.connect("127.0.0.1", dstRpcPort, false)
|
|
|
|
## Given
|
|
let
|
|
payload = @[byte 38]
|
|
payloadBase64 = Base64String.encode(payload)
|
|
|
|
let message = WakuMessageRPC(
|
|
payload: payloadBase64,
|
|
contentTopic: some(contentTopic),
|
|
timestamp: some(now()),
|
|
)
|
|
|
|
## When
|
|
let keypair = await dstClient.get_waku_v2_private_v1_asymmetric_keypair()
|
|
|
|
# Now publish and encrypt a message on srcNode using dstNode's public key
|
|
let posted = await srcClient.post_waku_v2_private_v1_asymmetric_message(pubSubTopic, message, publicKey = (%keypair.pubkey).getStr())
|
|
require:
|
|
posted
|
|
|
|
await sleepAsync(100.millis)
|
|
|
|
# Let's see if we can receive, and decrypt, this message on dstNode
|
|
var messages = await dstClient.get_waku_v2_private_v1_asymmetric_messages(pubSubTopic, privateKey = (%keypair.seckey).getStr())
|
|
check:
|
|
messages.len == 1
|
|
messages[0].payload == payloadBase64
|
|
messages[0].contentTopic == message.contentTopic
|
|
messages[0].timestamp == message.timestamp
|
|
messages[0].version.get() == 1'u32
|
|
|
|
# Ensure that read messages are cleared from cache
|
|
messages = await dstClient.get_waku_v2_private_v1_asymmetric_messages(pubSubTopic, privateKey = (%keypair.seckey).getStr())
|
|
check:
|
|
messages.len == 0
|
|
|
|
## Cleanup
|
|
await srcServer.stop()
|
|
await srcServer.closeWait()
|
|
await dstServer.stop()
|
|
await dstServer.closeWait()
|
|
|
|
await allFutures(srcNode.stop(), relNode.stop(), dstNode.stop())
|
|
|