2021-03-22 09:44:45 +00:00
|
|
|
{.used.}
|
|
|
|
|
|
|
|
import
|
2022-06-27 12:16:15 +00:00
|
|
|
std/[sequtils, strutils, tables],
|
2021-03-26 09:52:04 +00:00
|
|
|
testutils/unittests,
|
2021-04-08 09:55:19 +00:00
|
|
|
chronicles, chronos, stew/shims/net as stewNet, stew/[byteutils, objects],
|
2021-03-22 09:44:45 +00:00
|
|
|
libp2p/crypto/crypto,
|
|
|
|
libp2p/crypto/secp,
|
|
|
|
libp2p/peerid,
|
|
|
|
libp2p/multiaddress,
|
|
|
|
libp2p/switch,
|
|
|
|
libp2p/protocols/pubsub/rpc/messages,
|
|
|
|
libp2p/protocols/pubsub/pubsub,
|
|
|
|
eth/p2p,
|
|
|
|
eth/keys,
|
|
|
|
../../waku/common/wakubridge,
|
|
|
|
../../waku/v1/protocol/waku_protocol,
|
2021-03-23 08:04:51 +00:00
|
|
|
../../waku/v2/protocol/waku_message,
|
2021-03-22 09:44:45 +00:00
|
|
|
../../waku/v2/protocol/waku_store/waku_store,
|
|
|
|
../../waku/v2/protocol/waku_filter/waku_filter,
|
2021-03-23 08:04:51 +00:00
|
|
|
../../waku/v2/node/[wakunode2, waku_payload],
|
2021-10-06 12:29:08 +00:00
|
|
|
../../waku/v2/utils/peers,
|
2021-03-22 09:44:45 +00:00
|
|
|
../test_helpers
|
|
|
|
|
|
|
|
procSuite "WakuBridge":
|
2021-05-20 16:03:56 +00:00
|
|
|
###############
|
|
|
|
# Suite setup #
|
|
|
|
###############
|
|
|
|
|
2021-06-17 10:59:28 +00:00
|
|
|
const DefaultBridgeTopic = "/waku/2/default-bridge/proto"
|
|
|
|
|
2021-05-20 16:03:56 +00:00
|
|
|
let
|
|
|
|
rng = keys.newRng()
|
|
|
|
|
|
|
|
# Bridge
|
|
|
|
nodev1Key = keys.KeyPair.random(rng[])
|
|
|
|
nodev2Key = crypto.PrivateKey.random(Secp256k1, rng[])[]
|
|
|
|
bridge = WakuBridge.new(
|
|
|
|
nodev1Key= nodev1Key,
|
2022-06-27 12:16:15 +00:00
|
|
|
nodev1Address = localAddress(30302),
|
2021-05-20 16:03:56 +00:00
|
|
|
powRequirement = 0.002,
|
|
|
|
rng = rng,
|
|
|
|
nodev2Key = nodev2Key,
|
2021-06-17 10:59:28 +00:00
|
|
|
nodev2BindIp = ValidIpAddress.init("0.0.0.0"), nodev2BindPort= Port(60000),
|
|
|
|
nodev2PubsubTopic = DefaultBridgeTopic)
|
2021-05-20 16:03:56 +00:00
|
|
|
|
|
|
|
# Waku v1 node
|
|
|
|
v1Node = setupTestNode(rng, Waku)
|
|
|
|
|
|
|
|
# Waku v2 node
|
|
|
|
v2NodeKey = crypto.PrivateKey.random(Secp256k1, rng[])[]
|
2021-07-14 17:58:46 +00:00
|
|
|
v2Node = WakuNode.new(v2NodeKey, ValidIpAddress.init("0.0.0.0"), Port(60002))
|
2021-05-20 16:03:56 +00:00
|
|
|
|
2021-08-26 12:24:00 +00:00
|
|
|
contentTopic = ContentTopic("/waku/1/0x1a2b3c4d/rfc26")
|
2021-06-18 10:30:24 +00:00
|
|
|
topic = [byte 0x1a, byte 0x2b, byte 0x3c, byte 0x4d]
|
2021-05-20 16:03:56 +00:00
|
|
|
payloadV1 = "hello from V1".toBytes()
|
|
|
|
payloadV2 = "hello from V2".toBytes()
|
2021-08-26 12:24:00 +00:00
|
|
|
encodedPayloadV2 = Payload(payload: payloadV2, dst: some(nodev1Key.pubKey))
|
|
|
|
message = WakuMessage(payload: encodedPayloadV2.encode(1, rng[]).get(), contentTopic: contentTopic, version: 1)
|
2021-05-20 16:03:56 +00:00
|
|
|
|
|
|
|
########################
|
|
|
|
# Tests setup/teardown #
|
|
|
|
########################
|
|
|
|
|
2021-06-18 10:30:24 +00:00
|
|
|
# setup:
|
|
|
|
# # Runs before each test
|
|
|
|
|
|
|
|
# teardown:
|
|
|
|
# # Runs after each test
|
|
|
|
|
|
|
|
###############
|
|
|
|
# Suite tests #
|
|
|
|
###############
|
|
|
|
|
|
|
|
asyncTest "Topics are correctly converted between Waku v1 and Waku v2":
|
|
|
|
# Expected cases
|
|
|
|
|
|
|
|
check:
|
2021-08-26 12:24:00 +00:00
|
|
|
toV1Topic(ContentTopic("/waku/1/0x00000000/rfc26")) == [byte 0x00, byte 0x00, byte 0x00, byte 0x00]
|
|
|
|
toV2ContentTopic([byte 0x00, byte 0x00, byte 0x00, byte 0x00]) == ContentTopic("/waku/1/0x00000000/rfc26")
|
|
|
|
toV1Topic(ContentTopic("/waku/1/0xffffffff/rfc26")) == [byte 0xff, byte 0xff, byte 0xff, byte 0xff]
|
|
|
|
toV2ContentTopic([byte 0xff, byte 0xff, byte 0xff, byte 0xff]) == ContentTopic("/waku/1/0xffffffff/rfc26")
|
|
|
|
toV1Topic(ContentTopic("/waku/1/0x1a2b3c4d/rfc26")) == [byte 0x1a, byte 0x2b, byte 0x3c, byte 0x4d]
|
|
|
|
toV2ContentTopic([byte 0x1a, byte 0x2b, byte 0x3c, byte 0x4d]) == ContentTopic("/waku/1/0x1a2b3c4d/rfc26")
|
2021-07-21 12:22:40 +00:00
|
|
|
# Topic conversion should still work where '0x' prefix is omitted from <v1 topic byte array>
|
2021-08-26 12:24:00 +00:00
|
|
|
toV1Topic(ContentTopic("/waku/1/1a2b3c4d/rfc26")) == [byte 0x1a, byte 0x2b, byte 0x3c, byte 0x4d]
|
2021-07-21 12:22:40 +00:00
|
|
|
|
2021-06-18 10:30:24 +00:00
|
|
|
# Invalid cases
|
2021-07-16 15:13:36 +00:00
|
|
|
|
|
|
|
expect LPError:
|
2021-06-18 10:30:24 +00:00
|
|
|
# Content topic not namespaced
|
|
|
|
discard toV1Topic(ContentTopic("this-is-my-content"))
|
|
|
|
|
|
|
|
expect ValueError:
|
|
|
|
# Content topic name too short
|
2021-08-26 12:24:00 +00:00
|
|
|
discard toV1Topic(ContentTopic("/waku/1/0x112233/rfc26"))
|
2021-06-18 10:30:24 +00:00
|
|
|
|
|
|
|
expect ValueError:
|
|
|
|
# Content topic name not hex
|
2021-08-26 12:24:00 +00:00
|
|
|
discard toV1Topic(ContentTopic("/waku/1/my-content/rfc26"))
|
2022-03-08 10:48:17 +00:00
|
|
|
|
|
|
|
asyncTest "Verify that WakuMessages are on bridgeable content topics":
|
|
|
|
let
|
|
|
|
validCT = ContentTopic("/waku/1/my-content/rfc26")
|
|
|
|
unnamespacedCT = ContentTopic("just_a_bunch_of_words")
|
|
|
|
invalidAppCT = ContentTopic("/facebook/1/my-content/rfc26")
|
|
|
|
invalidVersionCT = ContentTopic("/waku/2/my-content/rfc26")
|
|
|
|
|
|
|
|
check:
|
|
|
|
WakuMessage(contentTopic: validCT).isBridgeable() == true
|
|
|
|
WakuMessage(contentTopic: unnamespacedCT).isBridgeable() == false
|
|
|
|
WakuMessage(contentTopic: invalidAppCT).isBridgeable() == false
|
|
|
|
WakuMessage(contentTopic: invalidVersionCT).isBridgeable() == false
|
2021-06-18 10:30:24 +00:00
|
|
|
|
|
|
|
asyncTest "Messages are bridged between Waku v1 and Waku v2":
|
|
|
|
# Setup test
|
|
|
|
|
2021-05-20 16:03:56 +00:00
|
|
|
waitFor bridge.start()
|
|
|
|
|
|
|
|
waitFor v2Node.start()
|
2021-08-26 12:24:00 +00:00
|
|
|
v2Node.mountRelay(@[DefaultBridgeTopic], triggerSelf = false)
|
2021-05-20 16:03:56 +00:00
|
|
|
|
|
|
|
discard waitFor v1Node.rlpxConnect(newNode(bridge.nodev1.toENode()))
|
2022-01-10 15:07:35 +00:00
|
|
|
waitFor v2Node.connectToNodes(@[bridge.nodev2.switch.peerInfo.toRemotePeerInfo()])
|
2021-03-22 09:44:45 +00:00
|
|
|
|
|
|
|
var completionFut = newFuture[bool]()
|
2021-05-20 16:03:56 +00:00
|
|
|
|
|
|
|
proc relayHandler(topic: string, data: seq[byte]) {.async, gcsafe.} =
|
2021-03-22 09:44:45 +00:00
|
|
|
let msg = WakuMessage.init(data)
|
2021-05-20 16:03:56 +00:00
|
|
|
|
2021-03-22 09:44:45 +00:00
|
|
|
if msg.isOk() and msg.value().version == 1:
|
2021-03-23 08:04:51 +00:00
|
|
|
check:
|
|
|
|
# Message fields are as expected
|
|
|
|
msg.value().contentTopic == contentTopic # Topic translation worked
|
|
|
|
string.fromBytes(msg.value().payload).contains("from V1")
|
2021-05-20 16:03:56 +00:00
|
|
|
|
2021-03-22 09:44:45 +00:00
|
|
|
completionFut.complete(true)
|
|
|
|
|
2021-05-20 16:03:56 +00:00
|
|
|
v2Node.subscribe(DefaultBridgeTopic, relayHandler)
|
2021-03-22 09:44:45 +00:00
|
|
|
|
|
|
|
await sleepAsync(2000.millis)
|
|
|
|
|
|
|
|
# Test bridging from V2 to V1
|
2021-05-20 16:03:56 +00:00
|
|
|
await v2Node.publish(DefaultBridgeTopic, message)
|
2021-03-22 09:44:45 +00:00
|
|
|
|
|
|
|
await sleepAsync(2000.millis)
|
|
|
|
|
|
|
|
check:
|
|
|
|
# v1Node received message published by v2Node
|
|
|
|
v1Node.protocolState(Waku).queue.items.len == 1
|
2021-03-23 08:04:51 +00:00
|
|
|
|
2021-08-26 12:24:00 +00:00
|
|
|
let
|
|
|
|
msg = v1Node.protocolState(Waku).queue.items[0]
|
|
|
|
decodedPayload = msg.env.data.decode(some(nodev1Key.seckey), none[SymKey]()).get()
|
2021-03-23 08:04:51 +00:00
|
|
|
|
|
|
|
check:
|
|
|
|
# Message fields are as expected
|
|
|
|
msg.env.topic == topic # Topic translation worked
|
2021-08-26 12:24:00 +00:00
|
|
|
string.fromBytes(decodedPayload.payload).contains("from V2")
|
2021-03-22 09:44:45 +00:00
|
|
|
|
|
|
|
# Test bridging from V1 to V2
|
|
|
|
check:
|
|
|
|
v1Node.postMessage(ttl = 5,
|
|
|
|
topic = topic,
|
|
|
|
payload = payloadV1) == true
|
|
|
|
|
|
|
|
# v2Node received payload published by v1Node
|
|
|
|
await completionFut.withTimeout(5.seconds)
|
|
|
|
|
2021-05-20 16:03:56 +00:00
|
|
|
# Test filtering of WakuMessage duplicates
|
|
|
|
v1Node.resetMessageQueue()
|
|
|
|
|
|
|
|
await v2Node.publish(DefaultBridgeTopic, message)
|
|
|
|
|
|
|
|
await sleepAsync(2000.millis)
|
|
|
|
|
|
|
|
check:
|
|
|
|
# v1Node did not receive duplicate of previous message
|
|
|
|
v1Node.protocolState(Waku).queue.items.len == 0
|
2021-06-18 10:30:24 +00:00
|
|
|
|
|
|
|
# Teardown test
|
|
|
|
|
|
|
|
bridge.nodeV1.resetMessageQueue()
|
|
|
|
v1Node.resetMessageQueue()
|
|
|
|
waitFor allFutures([bridge.stop(), v2Node.stop()])
|
2022-06-27 12:16:15 +00:00
|
|
|
|
|
|
|
asyncTest "Bridge manages its v1 connections":
|
|
|
|
# Given
|
|
|
|
let
|
|
|
|
# Waku v1 node
|
|
|
|
v1NodePool = @[setupTestNode(rng, Waku),
|
|
|
|
setupTestNode(rng, Waku),
|
|
|
|
setupTestNode(rng, Waku)]
|
|
|
|
targetV1Peers = v1NodePool.len() - 1
|
|
|
|
|
|
|
|
# Bridge
|
|
|
|
v1Bridge = WakuBridge.new(
|
|
|
|
nodev1Key= nodev1Key,
|
|
|
|
nodev1Address = localAddress(30303),
|
|
|
|
powRequirement = 0.002,
|
|
|
|
rng = rng,
|
|
|
|
nodev2Key = nodev2Key,
|
|
|
|
nodev2BindIp = ValidIpAddress.init("0.0.0.0"), nodev2BindPort= Port(60000),
|
|
|
|
nodev2PubsubTopic = DefaultBridgeTopic,
|
|
|
|
v1Pool = v1NodePool.mapIt(newNode(it.toEnode())),
|
|
|
|
targetV1Peers = targetV1Peers)
|
|
|
|
|
|
|
|
for node in v1NodePool:
|
|
|
|
node.startListening()
|
|
|
|
|
|
|
|
# When
|
|
|
|
waitFor v1Bridge.start()
|
|
|
|
await sleepAsync(2000.millis) # Give peers some time to connect
|
|
|
|
|
|
|
|
# Then
|
|
|
|
check:
|
|
|
|
v1Bridge.nodev1.peerPool.connectedNodes.len() == targetV1Peers
|
|
|
|
|
|
|
|
# When
|
|
|
|
let connected = v1Bridge.nodev1.peerPool.connectedNodes
|
|
|
|
for peer in connected.values():
|
|
|
|
waitFor peer.disconnect(SubprotocolReason)
|
|
|
|
|
|
|
|
# Then
|
|
|
|
check:
|
|
|
|
v1Bridge.nodev1.peerPool.connectedNodes.len() == 0
|
|
|
|
|
|
|
|
# When
|
|
|
|
discard v1Bridge.maintenanceLoop() # Forces one more run of the maintenance loop
|
|
|
|
await sleepAsync(2000.millis) # Give peers some time to connect
|
|
|
|
|
|
|
|
# Then
|
|
|
|
check:
|
|
|
|
v1Bridge.nodev1.peerPool.connectedNodes.len() == targetV1Peers
|
|
|
|
|
|
|
|
# Cleanup
|
|
|
|
v1Bridge.nodev1.resetMessageQueue()
|
|
|
|
|
|
|
|
for node in v1NodePool:
|
|
|
|
node.resetMessageQueue()
|
|
|
|
|
|
|
|
waitFor v1Bridge.stop()
|