2019-12-05 20:16:18 -06:00
|
|
|
## Nim-Libp2p
|
|
|
|
## Copyright (c) 2018 Status Research & Development GmbH
|
|
|
|
## Licensed under either of
|
|
|
|
## * Apache License, version 2.0, ([LICENSE-APACHE](LICENSE-APACHE))
|
|
|
|
## * MIT license ([LICENSE-MIT](LICENSE-MIT))
|
|
|
|
## at your option.
|
|
|
|
## This file may not be copied, modified, or distributed except according to
|
|
|
|
## those terms.
|
|
|
|
|
2020-01-09 21:59:27 -06:00
|
|
|
import unittest, sequtils, options, tables, sets
|
2019-12-05 20:16:18 -06:00
|
|
|
import chronos
|
2020-04-21 10:24:42 +09:00
|
|
|
import chronicles
|
|
|
|
import utils, ../../libp2p/[errors,
|
|
|
|
peer,
|
2019-12-05 20:16:18 -06:00
|
|
|
peerinfo,
|
|
|
|
connection,
|
|
|
|
crypto/crypto,
|
|
|
|
stream/bufferstream,
|
|
|
|
protocols/pubsub/pubsub,
|
2019-12-16 23:24:03 -06:00
|
|
|
protocols/pubsub/gossipsub,
|
|
|
|
protocols/pubsub/rpc/messages]
|
|
|
|
|
2020-04-21 10:24:42 +09:00
|
|
|
const
|
|
|
|
StreamTransportTrackerName = "stream.transport"
|
|
|
|
StreamServerTrackerName = "stream.server"
|
|
|
|
|
2019-12-05 20:16:18 -06:00
|
|
|
proc createGossipSub(): GossipSub =
|
2019-12-07 10:36:39 -06:00
|
|
|
var peerInfo = PeerInfo.init(PrivateKey.random(RSA))
|
2019-12-05 20:16:18 -06:00
|
|
|
result = newPubSub(GossipSub, peerInfo)
|
|
|
|
|
2020-04-21 10:24:42 +09:00
|
|
|
proc waitSub(sender, receiver: auto; key: string) {.async, gcsafe.} =
|
|
|
|
if sender == receiver:
|
|
|
|
return
|
|
|
|
# turn things deterministic
|
|
|
|
# this is for testing purposes only
|
|
|
|
# peers can be inside `mesh` and `fanout`, not just `gossipsub`
|
|
|
|
var ceil = 15
|
|
|
|
let fsub = cast[GossipSub](sender.pubSub.get())
|
|
|
|
while (not fsub.gossipsub.hasKey(key) or
|
|
|
|
not fsub.gossipsub[key].contains(receiver.peerInfo.id)) and
|
|
|
|
(not fsub.mesh.hasKey(key) or
|
|
|
|
not fsub.mesh[key].contains(receiver.peerInfo.id)) and
|
|
|
|
(not fsub.fanout.hasKey(key) or
|
|
|
|
not fsub.fanout[key].contains(receiver.peerInfo.id)):
|
|
|
|
trace "waitSub sleeping...", peers=fsub.gossipsub[key]
|
|
|
|
await sleepAsync(100.millis)
|
|
|
|
dec ceil
|
|
|
|
doAssert(ceil > 0, "waitSub timeout!")
|
|
|
|
|
2019-12-05 20:16:18 -06:00
|
|
|
suite "GossipSub":
|
2020-04-21 10:24:42 +09:00
|
|
|
teardown:
|
|
|
|
let
|
|
|
|
trackers = [
|
|
|
|
getTracker(BufferStreamTrackerName),
|
|
|
|
getTracker(AsyncStreamWriterTrackerName),
|
|
|
|
getTracker(AsyncStreamReaderTrackerName),
|
|
|
|
getTracker(StreamTransportTrackerName),
|
|
|
|
getTracker(StreamServerTrackerName)
|
|
|
|
]
|
|
|
|
for tracker in trackers:
|
|
|
|
if not isNil(tracker):
|
|
|
|
# echo tracker.dump()
|
|
|
|
check tracker.isLeaked() == false
|
|
|
|
|
2019-12-16 23:24:03 -06:00
|
|
|
test "GossipSub validation should succeed":
|
|
|
|
proc runTests(): Future[bool] {.async.} =
|
|
|
|
var handlerFut = newFuture[bool]()
|
|
|
|
proc handler(topic: string, data: seq[byte]) {.async, gcsafe.} =
|
|
|
|
check topic == "foobar"
|
|
|
|
handlerFut.complete(true)
|
|
|
|
|
|
|
|
var nodes = generateNodes(2, true)
|
|
|
|
var awaiters: seq[Future[void]]
|
|
|
|
awaiters.add((await nodes[0].start()))
|
|
|
|
awaiters.add((await nodes[1].start()))
|
|
|
|
|
2020-04-21 10:24:42 +09:00
|
|
|
await subscribeNodes(nodes)
|
|
|
|
|
2020-01-07 02:06:27 -06:00
|
|
|
await nodes[0].subscribe("foobar", handler)
|
2020-04-21 10:24:42 +09:00
|
|
|
await waitSub(nodes[1], nodes[0], "foobar")
|
2019-12-16 23:24:03 -06:00
|
|
|
await nodes[1].subscribe("foobar", handler)
|
2020-04-21 10:24:42 +09:00
|
|
|
await waitSub(nodes[0], nodes[1], "foobar")
|
2019-12-16 23:24:03 -06:00
|
|
|
|
|
|
|
var validatorFut = newFuture[bool]()
|
|
|
|
proc validator(topic: string,
|
|
|
|
message: Message):
|
|
|
|
Future[bool] {.async.} =
|
|
|
|
check topic == "foobar"
|
|
|
|
validatorFut.complete(true)
|
|
|
|
result = true
|
|
|
|
|
|
|
|
nodes[1].addValidator("foobar", validator)
|
|
|
|
await nodes[0].publish("foobar", cast[seq[byte]]("Hello!"))
|
|
|
|
|
2020-01-07 02:06:27 -06:00
|
|
|
result = (await validatorFut) and (await handlerFut)
|
2020-04-21 10:24:42 +09:00
|
|
|
await allFuturesThrowing(nodes[0].stop(), nodes[1].stop())
|
|
|
|
await allFuturesThrowing(awaiters)
|
2020-01-07 02:06:27 -06:00
|
|
|
|
2019-12-16 23:24:03 -06:00
|
|
|
check:
|
|
|
|
waitFor(runTests()) == true
|
|
|
|
|
|
|
|
test "GossipSub validation should fail":
|
|
|
|
proc runTests(): Future[bool] {.async.} =
|
|
|
|
proc handler(topic: string, data: seq[byte]) {.async, gcsafe.} =
|
|
|
|
check false # if we get here, it should fail
|
|
|
|
|
|
|
|
var nodes = generateNodes(2, true)
|
|
|
|
var awaiters: seq[Future[void]]
|
|
|
|
awaiters.add((await nodes[0].start()))
|
|
|
|
awaiters.add((await nodes[1].start()))
|
|
|
|
|
|
|
|
await subscribeNodes(nodes)
|
2020-04-21 10:24:42 +09:00
|
|
|
|
2019-12-16 23:24:03 -06:00
|
|
|
await nodes[1].subscribe("foobar", handler)
|
2020-04-21 10:24:42 +09:00
|
|
|
await waitSub(nodes[0], nodes[1], "foobar")
|
2019-12-16 23:24:03 -06:00
|
|
|
|
|
|
|
var validatorFut = newFuture[bool]()
|
|
|
|
proc validator(topic: string,
|
|
|
|
message: Message):
|
|
|
|
Future[bool] {.async.} =
|
|
|
|
validatorFut.complete(true)
|
|
|
|
result = false
|
|
|
|
|
|
|
|
nodes[1].addValidator("foobar", validator)
|
|
|
|
await nodes[0].publish("foobar", cast[seq[byte]]("Hello!"))
|
|
|
|
|
2020-01-07 02:06:27 -06:00
|
|
|
result = await validatorFut
|
2020-04-21 10:24:42 +09:00
|
|
|
await allFuturesThrowing(nodes[0].stop(), nodes[1].stop())
|
|
|
|
await allFuturesThrowing(awaiters)
|
2019-12-16 23:24:03 -06:00
|
|
|
|
|
|
|
check:
|
|
|
|
waitFor(runTests()) == true
|
|
|
|
|
|
|
|
test "GossipSub validation one fails and one succeeds":
|
|
|
|
proc runTests(): Future[bool] {.async.} =
|
|
|
|
var handlerFut = newFuture[bool]()
|
|
|
|
proc handler(topic: string, data: seq[byte]) {.async, gcsafe.} =
|
|
|
|
check topic == "foo"
|
|
|
|
handlerFut.complete(true)
|
|
|
|
|
|
|
|
var nodes = generateNodes(2, true)
|
|
|
|
var awaiters: seq[Future[void]]
|
|
|
|
awaiters.add((await nodes[0].start()))
|
|
|
|
awaiters.add((await nodes[1].start()))
|
|
|
|
|
2020-04-21 10:24:42 +09:00
|
|
|
await subscribeNodes(nodes)
|
2019-12-16 23:24:03 -06:00
|
|
|
await nodes[1].subscribe("foo", handler)
|
2020-04-21 10:24:42 +09:00
|
|
|
await waitSub(nodes[0], nodes[1], "foo")
|
2019-12-16 23:24:03 -06:00
|
|
|
await nodes[1].subscribe("bar", handler)
|
2020-04-21 10:24:42 +09:00
|
|
|
await waitSub(nodes[0], nodes[1], "bar")
|
2019-12-16 23:24:03 -06:00
|
|
|
|
2020-01-07 02:06:27 -06:00
|
|
|
var passed, failed: Future[bool] = newFuture[bool]()
|
2019-12-16 23:24:03 -06:00
|
|
|
proc validator(topic: string,
|
|
|
|
message: Message):
|
|
|
|
Future[bool] {.async.} =
|
2020-01-07 02:06:27 -06:00
|
|
|
result = if topic == "foo":
|
|
|
|
passed.complete(true)
|
|
|
|
true
|
2019-12-16 23:24:03 -06:00
|
|
|
else:
|
2020-01-07 02:06:27 -06:00
|
|
|
failed.complete(true)
|
|
|
|
false
|
2019-12-16 23:24:03 -06:00
|
|
|
|
|
|
|
nodes[1].addValidator("foo", "bar", validator)
|
|
|
|
await nodes[0].publish("foo", cast[seq[byte]]("Hello!"))
|
|
|
|
await nodes[0].publish("bar", cast[seq[byte]]("Hello!"))
|
|
|
|
|
2020-01-07 02:06:27 -06:00
|
|
|
result = ((await passed) and (await failed) and (await handlerFut))
|
2020-04-21 10:24:42 +09:00
|
|
|
await allFuturesThrowing(nodes[0].stop(), nodes[1].stop())
|
|
|
|
await allFuturesThrowing(awaiters)
|
2019-12-16 23:24:03 -06:00
|
|
|
result = true
|
|
|
|
check:
|
|
|
|
waitFor(runTests()) == true
|
|
|
|
|
2020-01-07 02:06:27 -06:00
|
|
|
test "GossipSub should add remote peer topic subscriptions":
|
2019-12-16 23:24:03 -06:00
|
|
|
proc runTests(): Future[bool] {.async.} =
|
2019-12-10 14:50:35 -06:00
|
|
|
proc handler(topic: string, data: seq[byte]) {.async, gcsafe.} =
|
2019-12-05 20:16:18 -06:00
|
|
|
discard
|
|
|
|
|
|
|
|
let gossip1 = createGossipSub()
|
|
|
|
let gossip2 = createGossipSub()
|
|
|
|
|
|
|
|
var buf1 = newBufferStream()
|
|
|
|
var conn1 = newConnection(buf1)
|
2019-12-10 14:50:35 -06:00
|
|
|
conn1.peerInfo = gossip1.peerInfo
|
2019-12-05 20:16:18 -06:00
|
|
|
|
|
|
|
var buf2 = newBufferStream()
|
|
|
|
var conn2 = newConnection(buf2)
|
2019-12-10 14:50:35 -06:00
|
|
|
conn2.peerInfo = gossip2.peerInfo
|
2019-12-05 20:16:18 -06:00
|
|
|
|
|
|
|
buf1 = buf1 | buf2 | buf1
|
|
|
|
|
|
|
|
await gossip1.subscribeToPeer(conn2)
|
|
|
|
asyncCheck gossip2.handleConn(conn1, GossipSubCodec)
|
|
|
|
|
|
|
|
await gossip1.subscribe("foobar", handler)
|
2020-04-21 10:24:42 +09:00
|
|
|
await sleepAsync(1.seconds)
|
2019-12-05 20:16:18 -06:00
|
|
|
|
|
|
|
check:
|
2020-01-07 02:06:27 -06:00
|
|
|
"foobar" in gossip2.gossipsub
|
|
|
|
gossip1.peerInfo.id in gossip2.gossipsub["foobar"]
|
2019-12-10 14:50:35 -06:00
|
|
|
|
2020-04-21 10:24:42 +09:00
|
|
|
await allFuturesThrowing(
|
|
|
|
buf1.close(),
|
|
|
|
buf2.close()
|
|
|
|
)
|
|
|
|
|
2019-12-05 20:16:18 -06:00
|
|
|
result = true
|
|
|
|
|
|
|
|
check:
|
2019-12-16 23:24:03 -06:00
|
|
|
waitFor(runTests()) == true
|
2019-12-05 20:16:18 -06:00
|
|
|
|
2020-01-07 02:06:27 -06:00
|
|
|
test "e2e - GossipSub should add remote peer topic subscriptions":
|
2019-12-05 20:16:18 -06:00
|
|
|
proc testBasicGossipSub(): Future[bool] {.async.} =
|
|
|
|
proc handler(topic: string, data: seq[byte]) {.async, gcsafe.} =
|
|
|
|
discard
|
|
|
|
|
|
|
|
var nodes: seq[Switch] = newSeq[Switch]()
|
|
|
|
for i in 0..<2:
|
2019-12-08 23:06:58 +02:00
|
|
|
nodes.add newStandardSwitch(gossip = true)
|
2019-12-05 20:16:18 -06:00
|
|
|
|
|
|
|
var awaitters: seq[Future[void]]
|
|
|
|
for node in nodes:
|
|
|
|
awaitters.add(await node.start())
|
|
|
|
|
|
|
|
await subscribeNodes(nodes)
|
2020-04-21 10:24:42 +09:00
|
|
|
await nodes[1].subscribe("foobar", handler)
|
|
|
|
await sleepAsync(1.seconds)
|
2019-12-05 20:16:18 -06:00
|
|
|
|
|
|
|
let gossip1 = GossipSub(nodes[0].pubSub.get())
|
|
|
|
let gossip2 = GossipSub(nodes[1].pubSub.get())
|
|
|
|
|
|
|
|
check:
|
|
|
|
"foobar" in gossip2.topics
|
|
|
|
"foobar" in gossip1.gossipsub
|
2019-12-07 10:36:39 -06:00
|
|
|
gossip2.peerInfo.id in gossip1.gossipsub["foobar"]
|
2019-12-05 20:16:18 -06:00
|
|
|
|
2020-04-21 10:24:42 +09:00
|
|
|
await allFuturesThrowing(nodes.mapIt(it.stop()))
|
|
|
|
await allFuturesThrowing(awaitters)
|
2019-12-05 20:16:18 -06:00
|
|
|
|
|
|
|
result = true
|
|
|
|
|
|
|
|
check:
|
|
|
|
waitFor(testBasicGossipSub()) == true
|
|
|
|
|
2020-01-07 02:06:27 -06:00
|
|
|
test "GossipSub should add remote peer topic subscriptions if both peers are subscribed":
|
2019-12-16 23:24:03 -06:00
|
|
|
proc runTests(): Future[bool] {.async.} =
|
2019-12-10 14:50:35 -06:00
|
|
|
proc handler(topic: string, data: seq[byte]) {.async, gcsafe.} =
|
2019-12-05 20:16:18 -06:00
|
|
|
discard
|
|
|
|
|
|
|
|
let gossip1 = createGossipSub()
|
|
|
|
let gossip2 = createGossipSub()
|
|
|
|
|
|
|
|
var buf1 = newBufferStream()
|
|
|
|
var conn1 = newConnection(buf1)
|
2019-12-10 14:50:35 -06:00
|
|
|
conn1.peerInfo = gossip1.peerInfo
|
2019-12-05 20:16:18 -06:00
|
|
|
|
|
|
|
var buf2 = newBufferStream()
|
|
|
|
var conn2 = newConnection(buf2)
|
2019-12-10 14:50:35 -06:00
|
|
|
conn2.peerInfo = gossip2.peerInfo
|
|
|
|
|
2019-12-05 20:16:18 -06:00
|
|
|
buf1 = buf1 | buf2 | buf1
|
|
|
|
|
|
|
|
await gossip1.subscribeToPeer(conn2)
|
|
|
|
asyncCheck gossip1.handleConn(conn1, GossipSubCodec)
|
|
|
|
|
|
|
|
await gossip2.subscribeToPeer(conn1)
|
|
|
|
asyncCheck gossip2.handleConn(conn2, GossipSubCodec)
|
|
|
|
|
|
|
|
await gossip1.subscribe("foobar", handler)
|
|
|
|
await gossip2.subscribe("foobar", handler)
|
2020-04-21 10:24:42 +09:00
|
|
|
await sleepAsync(1.seconds)
|
2019-12-05 20:16:18 -06:00
|
|
|
|
|
|
|
check:
|
2020-01-07 02:06:27 -06:00
|
|
|
"foobar" in gossip1.topics
|
|
|
|
"foobar" in gossip2.topics
|
2019-12-05 20:16:18 -06:00
|
|
|
|
2020-01-07 02:06:27 -06:00
|
|
|
"foobar" in gossip1.gossipsub
|
|
|
|
"foobar" in gossip2.gossipsub
|
2019-12-05 20:16:18 -06:00
|
|
|
|
2020-01-07 02:06:27 -06:00
|
|
|
# TODO: in a real setting, we would be checking for the peerId from
|
|
|
|
# gossip1 in gossip2 and vice versa, but since we're doing some mockery
|
|
|
|
# with connection piping and such, this is fine - do not change!
|
|
|
|
gossip1.peerInfo.id in gossip1.gossipsub["foobar"]
|
|
|
|
gossip2.peerInfo.id in gossip2.gossipsub["foobar"]
|
2019-12-05 20:16:18 -06:00
|
|
|
|
2020-04-21 10:24:42 +09:00
|
|
|
await allFuturesThrowing(
|
|
|
|
buf1.close(),
|
|
|
|
buf2.close()
|
|
|
|
)
|
|
|
|
|
2019-12-05 20:16:18 -06:00
|
|
|
result = true
|
|
|
|
|
|
|
|
check:
|
2019-12-16 23:24:03 -06:00
|
|
|
waitFor(runTests()) == true
|
2019-12-05 20:16:18 -06:00
|
|
|
|
2020-01-07 02:06:27 -06:00
|
|
|
test "e2e - GossipSub should add remote peer topic subscriptions if both peers are subscribed":
|
2019-12-05 20:16:18 -06:00
|
|
|
proc testBasicGossipSub(): Future[bool] {.async.} =
|
|
|
|
proc handler(topic: string, data: seq[byte]) {.async, gcsafe.} =
|
|
|
|
discard
|
|
|
|
|
|
|
|
var nodes: seq[Switch] = newSeq[Switch]()
|
|
|
|
for i in 0..<2:
|
2019-12-08 23:06:58 +02:00
|
|
|
nodes.add newStandardSwitch(gossip = true)
|
2019-12-05 20:16:18 -06:00
|
|
|
|
|
|
|
var awaitters: seq[Future[void]]
|
|
|
|
for node in nodes:
|
|
|
|
awaitters.add(await node.start())
|
|
|
|
|
2020-04-21 10:24:42 +09:00
|
|
|
await subscribeNodes(nodes)
|
|
|
|
|
2019-12-05 20:16:18 -06:00
|
|
|
await nodes[0].subscribe("foobar", handler)
|
|
|
|
await nodes[1].subscribe("foobar", handler)
|
|
|
|
|
2020-04-21 10:24:42 +09:00
|
|
|
var subs: seq[Future[void]]
|
|
|
|
subs &= waitSub(nodes[1], nodes[0], "foobar")
|
|
|
|
subs &= waitSub(nodes[0], nodes[1], "foobar")
|
|
|
|
await allFuturesThrowing(subs)
|
|
|
|
|
|
|
|
let
|
|
|
|
gossip1 = GossipSub(nodes[0].pubSub.get())
|
|
|
|
gossip2 = GossipSub(nodes[1].pubSub.get())
|
2019-12-05 20:16:18 -06:00
|
|
|
|
|
|
|
check:
|
|
|
|
"foobar" in gossip1.topics
|
|
|
|
"foobar" in gossip2.topics
|
|
|
|
|
|
|
|
"foobar" in gossip1.gossipsub
|
|
|
|
"foobar" in gossip2.gossipsub
|
|
|
|
|
2020-04-21 10:24:42 +09:00
|
|
|
gossip2.peerInfo.id in gossip1.gossipsub["foobar"] or
|
|
|
|
gossip2.peerInfo.id in gossip1.mesh["foobar"]
|
2019-12-05 20:16:18 -06:00
|
|
|
|
2020-04-21 10:24:42 +09:00
|
|
|
gossip1.peerInfo.id in gossip2.gossipsub["foobar"] or
|
|
|
|
gossip1.peerInfo.id in gossip2.mesh["foobar"]
|
|
|
|
|
|
|
|
await allFuturesThrowing(nodes.mapIt(it.stop()))
|
|
|
|
await allFuturesThrowing(awaitters)
|
2019-12-05 20:16:18 -06:00
|
|
|
|
|
|
|
result = true
|
|
|
|
|
|
|
|
check:
|
|
|
|
waitFor(testBasicGossipSub()) == true
|
|
|
|
|
|
|
|
# test "send over fanout A -> B":
|
2019-12-16 23:24:03 -06:00
|
|
|
# proc runTests(): Future[bool] {.async.} =
|
2019-12-05 20:16:18 -06:00
|
|
|
# var handlerFut = newFuture[bool]()
|
2019-12-10 14:50:35 -06:00
|
|
|
# proc handler(topic: string, data: seq[byte]) {.async, gcsafe.} =
|
2019-12-05 20:16:18 -06:00
|
|
|
# check:
|
|
|
|
# topic == "foobar"
|
|
|
|
# cast[string](data) == "Hello!"
|
|
|
|
|
|
|
|
# handlerFut.complete(true)
|
|
|
|
|
|
|
|
# let gossip1 = createGossipSub()
|
|
|
|
# let gossip2 = createGossipSub()
|
|
|
|
|
|
|
|
# var buf1 = newBufferStream()
|
|
|
|
# var conn1 = newConnection(buf1)
|
|
|
|
|
|
|
|
# var buf2 = newBufferStream()
|
|
|
|
# var conn2 = newConnection(buf2)
|
|
|
|
|
|
|
|
# conn1.peerInfo = gossip2.peerInfo
|
|
|
|
# conn2.peerInfo = gossip1.peerInfo
|
|
|
|
|
|
|
|
# buf1 = buf1 | buf2 | buf1
|
|
|
|
|
|
|
|
# await gossip1.subscribeToPeer(conn2)
|
|
|
|
# asyncCheck gossip1.handleConn(conn1, GossipSubCodec)
|
|
|
|
|
|
|
|
# await gossip2.subscribeToPeer(conn1)
|
|
|
|
# asyncCheck gossip2.handleConn(conn2, GossipSubCodec)
|
|
|
|
|
|
|
|
# await gossip1.subscribe("foobar", handler)
|
|
|
|
# await sleepAsync(1.seconds)
|
|
|
|
# await gossip2.publish("foobar", cast[seq[byte]]("Hello!"))
|
|
|
|
# await sleepAsync(1.seconds)
|
|
|
|
|
|
|
|
# result = await handlerFut
|
|
|
|
|
|
|
|
# check:
|
2019-12-16 23:24:03 -06:00
|
|
|
# waitFor(runTests()) == true
|
2019-12-05 20:16:18 -06:00
|
|
|
|
2020-01-07 02:06:27 -06:00
|
|
|
test "e2e - GossipSub send over fanout A -> B":
|
2019-12-16 23:24:03 -06:00
|
|
|
proc runTests(): Future[bool] {.async.} =
|
2020-04-21 10:24:42 +09:00
|
|
|
var passed = newFuture[void]()
|
2019-12-10 14:50:35 -06:00
|
|
|
proc handler(topic: string, data: seq[byte]) {.async, gcsafe.} =
|
2019-12-05 20:16:18 -06:00
|
|
|
check topic == "foobar"
|
2020-04-21 10:24:42 +09:00
|
|
|
passed.complete()
|
2019-12-05 20:16:18 -06:00
|
|
|
|
|
|
|
var nodes = generateNodes(2, true)
|
|
|
|
var wait = newSeq[Future[void]]()
|
|
|
|
wait.add(await nodes[0].start())
|
|
|
|
wait.add(await nodes[1].start())
|
|
|
|
|
|
|
|
await subscribeNodes(nodes)
|
|
|
|
|
|
|
|
await nodes[1].subscribe("foobar", handler)
|
2020-04-21 10:24:42 +09:00
|
|
|
await waitSub(nodes[0], nodes[1], "foobar")
|
2019-12-05 20:16:18 -06:00
|
|
|
await nodes[0].publish("foobar", cast[seq[byte]]("Hello!"))
|
|
|
|
|
|
|
|
var gossipSub1: GossipSub = GossipSub(nodes[0].pubSub.get())
|
|
|
|
|
|
|
|
check:
|
|
|
|
"foobar" in gossipSub1.gossipsub
|
|
|
|
|
2020-04-21 10:24:42 +09:00
|
|
|
await passed.wait(5.seconds)
|
|
|
|
|
|
|
|
trace "test done, stopping..."
|
|
|
|
|
2019-12-05 20:16:18 -06:00
|
|
|
await nodes[0].stop()
|
2020-04-21 10:24:42 +09:00
|
|
|
await nodes[1].stop()
|
|
|
|
await allFuturesThrowing(wait)
|
2019-12-05 20:16:18 -06:00
|
|
|
|
2020-04-21 10:24:42 +09:00
|
|
|
result = true
|
2019-12-05 20:16:18 -06:00
|
|
|
|
|
|
|
check:
|
2019-12-16 23:24:03 -06:00
|
|
|
waitFor(runTests()) == true
|
2019-12-05 20:16:18 -06:00
|
|
|
|
|
|
|
# test "send over mesh A -> B":
|
2019-12-16 23:24:03 -06:00
|
|
|
# proc runTests(): Future[bool] {.async.} =
|
2020-04-21 10:24:42 +09:00
|
|
|
# var passed = newFuture[void]()
|
2019-12-10 14:50:35 -06:00
|
|
|
# proc handler(topic: string, data: seq[byte]) {.async, gcsafe.} =
|
2019-12-05 20:16:18 -06:00
|
|
|
# check:
|
|
|
|
# topic == "foobar"
|
|
|
|
# cast[string](data) == "Hello!"
|
|
|
|
|
2020-04-21 10:24:42 +09:00
|
|
|
# passed.complete()
|
2019-12-05 20:16:18 -06:00
|
|
|
|
|
|
|
# let gossip1 = createGossipSub()
|
|
|
|
# let gossip2 = createGossipSub()
|
|
|
|
|
|
|
|
# var buf1 = newBufferStream()
|
|
|
|
# var conn1 = newConnection(buf1)
|
|
|
|
# conn1.peerInfo = gossip1.peerInfo
|
|
|
|
|
|
|
|
# var buf2 = newBufferStream()
|
|
|
|
# var conn2 = newConnection(buf2)
|
|
|
|
# conn2.peerInfo = gossip2.peerInfo
|
|
|
|
|
|
|
|
# buf1 = buf1 | buf2 | buf1
|
|
|
|
|
|
|
|
# await gossip1.subscribeToPeer(conn2)
|
|
|
|
# await gossip2.subscribeToPeer(conn1)
|
|
|
|
|
|
|
|
# await gossip1.subscribe("foobar", handler)
|
|
|
|
# await sleepAsync(1.seconds)
|
|
|
|
|
|
|
|
# await gossip2.subscribe("foobar", handler)
|
|
|
|
# await sleepAsync(1.seconds)
|
|
|
|
|
|
|
|
# await gossip2.publish("foobar", cast[seq[byte]]("Hello!"))
|
|
|
|
# await sleepAsync(1.seconds)
|
2020-04-21 10:24:42 +09:00
|
|
|
|
|
|
|
# await passed.wait(5.seconds)
|
|
|
|
# result = true
|
|
|
|
|
|
|
|
# await allFuturesThrowing(buf1.close(), buf2.close())
|
2019-12-05 20:16:18 -06:00
|
|
|
|
|
|
|
# check:
|
2019-12-16 23:24:03 -06:00
|
|
|
# waitFor(runTests()) == true
|
2019-12-05 20:16:18 -06:00
|
|
|
|
2020-01-09 21:59:27 -06:00
|
|
|
test "e2e - GossipSub send over mesh A -> B":
|
|
|
|
proc runTests(): Future[bool] {.async.} =
|
|
|
|
var passed: Future[bool] = newFuture[bool]()
|
|
|
|
proc handler(topic: string, data: seq[byte]) {.async, gcsafe.} =
|
|
|
|
check topic == "foobar"
|
|
|
|
passed.complete(true)
|
2019-12-05 20:16:18 -06:00
|
|
|
|
2020-01-09 21:59:27 -06:00
|
|
|
var nodes = generateNodes(2, true)
|
|
|
|
var wait: seq[Future[void]]
|
|
|
|
wait.add(await nodes[0].start())
|
|
|
|
wait.add(await nodes[1].start())
|
2019-12-05 20:16:18 -06:00
|
|
|
|
2020-01-09 21:59:27 -06:00
|
|
|
await subscribeNodes(nodes)
|
2020-01-07 02:06:27 -06:00
|
|
|
|
2020-01-09 21:59:27 -06:00
|
|
|
await nodes[1].subscribe("foobar", handler)
|
2020-04-21 10:24:42 +09:00
|
|
|
await waitSub(nodes[0], nodes[1], "foobar")
|
2019-12-05 20:16:18 -06:00
|
|
|
|
2020-01-09 21:59:27 -06:00
|
|
|
await nodes[0].publish("foobar", cast[seq[byte]]("Hello!"))
|
2020-04-21 10:24:42 +09:00
|
|
|
|
2020-01-09 21:59:27 -06:00
|
|
|
result = await passed
|
2019-12-05 20:16:18 -06:00
|
|
|
|
2020-01-09 21:59:27 -06:00
|
|
|
await nodes[0].stop()
|
|
|
|
await nodes[1].stop()
|
2020-04-21 10:24:42 +09:00
|
|
|
await allFuturesThrowing(wait)
|
2020-01-09 21:59:27 -06:00
|
|
|
|
|
|
|
check:
|
|
|
|
waitFor(runTests()) == true
|
2019-12-05 20:16:18 -06:00
|
|
|
|
2019-12-10 14:50:35 -06:00
|
|
|
# test "with multiple peers":
|
2019-12-16 23:24:03 -06:00
|
|
|
# proc runTests(): Future[bool] {.async.} =
|
2019-12-05 20:16:18 -06:00
|
|
|
# var nodes: seq[GossipSub]
|
|
|
|
# for i in 0..<10:
|
|
|
|
# nodes.add(createGossipSub())
|
|
|
|
|
|
|
|
# var pending: seq[Future[void]]
|
|
|
|
# var awaitters: seq[Future[void]]
|
|
|
|
# var seen: Table[string, int]
|
|
|
|
# for dialer in nodes:
|
|
|
|
# var handler: TopicHandler
|
|
|
|
# closureScope:
|
|
|
|
# var dialerNode = dialer
|
|
|
|
# handler = proc(topic: string, data: seq[byte]) {.async, gcsafe, closure.} =
|
|
|
|
# if dialerNode.peerInfo.peerId.get().pretty notin seen:
|
|
|
|
# seen[dialerNode.peerInfo.peerId.get().pretty] = 0
|
|
|
|
# seen[dialerNode.peerInfo.peerId.get().pretty].inc
|
|
|
|
# check topic == "foobar"
|
|
|
|
|
|
|
|
# await dialer.subscribe("foobar", handler)
|
|
|
|
# await sleepAsync(20.millis)
|
|
|
|
|
|
|
|
# for i, node in nodes:
|
|
|
|
# if dialer.peerInfo.peerId != node.peerInfo.peerId:
|
|
|
|
# var buf1 = newBufferStream()
|
|
|
|
# var conn1 = newConnection(buf1)
|
|
|
|
# conn1.peerInfo = dialer.peerInfo
|
|
|
|
|
|
|
|
# var buf2 = newBufferStream()
|
|
|
|
# var conn2 = newConnection(buf2)
|
|
|
|
# conn2.peerInfo = node.peerInfo
|
|
|
|
|
|
|
|
# buf1 = buf2 | buf1
|
|
|
|
# buf2 = buf1 | buf2
|
|
|
|
|
|
|
|
# pending.add(dialer.subscribeToPeer(conn2))
|
|
|
|
# pending.add(node.subscribeToPeer(conn1))
|
|
|
|
# await sleepAsync(10.millis)
|
|
|
|
|
|
|
|
# awaitters.add(dialer.start())
|
|
|
|
|
2019-12-10 14:50:35 -06:00
|
|
|
# await nodes[0].publish("foobar",
|
|
|
|
# cast[seq[byte]]("from node " &
|
2019-12-05 20:16:18 -06:00
|
|
|
# nodes[1].peerInfo.peerId.get().pretty))
|
|
|
|
|
|
|
|
# await sleepAsync(1000.millis)
|
2020-04-21 10:24:42 +09:00
|
|
|
# await allFuturesThrowing(nodes.mapIt(it.stop()))
|
|
|
|
# await allFuturesThrowing(awaitters)
|
2019-12-05 20:16:18 -06:00
|
|
|
|
|
|
|
# check: seen.len == 9
|
|
|
|
# for k, v in seen.pairs:
|
|
|
|
# check: v == 1
|
|
|
|
|
|
|
|
# result = true
|
|
|
|
|
|
|
|
# check:
|
2019-12-16 23:24:03 -06:00
|
|
|
# waitFor(runTests()) == true
|
2019-12-05 20:16:18 -06:00
|
|
|
|
2020-01-07 02:06:27 -06:00
|
|
|
test "e2e - GossipSub with multiple peers":
|
2019-12-16 23:24:03 -06:00
|
|
|
proc runTests(): Future[bool] {.async.} =
|
2019-12-05 20:16:18 -06:00
|
|
|
var nodes: seq[Switch] = newSeq[Switch]()
|
|
|
|
var awaitters: seq[Future[void]]
|
|
|
|
|
2020-01-07 02:06:27 -06:00
|
|
|
for i in 0..<11:
|
2019-12-08 23:06:58 +02:00
|
|
|
nodes.add newStandardSwitch(triggerSelf = true, gossip = true)
|
2019-12-05 20:16:18 -06:00
|
|
|
awaitters.add((await nodes[i].start()))
|
|
|
|
|
2020-04-21 10:24:42 +09:00
|
|
|
await subscribeNodes(nodes)
|
|
|
|
|
2019-12-05 20:16:18 -06:00
|
|
|
var seen: Table[string, int]
|
2020-01-07 02:06:27 -06:00
|
|
|
var subs: seq[Future[void]]
|
|
|
|
var seenFut = newFuture[void]()
|
2019-12-05 20:16:18 -06:00
|
|
|
for dialer in nodes:
|
|
|
|
var handler: TopicHandler
|
|
|
|
closureScope:
|
|
|
|
var dialerNode = dialer
|
|
|
|
handler = proc(topic: string, data: seq[byte]) {.async, gcsafe, closure.} =
|
2019-12-07 10:36:39 -06:00
|
|
|
if dialerNode.peerInfo.id notin seen:
|
|
|
|
seen[dialerNode.peerInfo.id] = 0
|
|
|
|
seen[dialerNode.peerInfo.id].inc
|
2019-12-05 20:16:18 -06:00
|
|
|
check topic == "foobar"
|
2020-01-07 02:06:27 -06:00
|
|
|
if not seenFut.finished() and seen.len == 10:
|
|
|
|
seenFut.complete()
|
2019-12-05 20:16:18 -06:00
|
|
|
|
2020-04-21 10:24:42 +09:00
|
|
|
subs.add(allFutures(dialer.subscribe("foobar", handler), waitSub(nodes[0], dialer, "foobar")))
|
|
|
|
|
|
|
|
await allFuturesThrowing(subs)
|
2019-12-05 20:16:18 -06:00
|
|
|
|
2020-01-07 02:06:27 -06:00
|
|
|
await wait(nodes[0].publish("foobar",
|
|
|
|
cast[seq[byte]]("from node " &
|
|
|
|
nodes[1].peerInfo.id)),
|
|
|
|
1.minutes)
|
2019-12-05 20:16:18 -06:00
|
|
|
|
2020-02-25 17:05:12 -06:00
|
|
|
await wait(seenFut, 2.minutes)
|
2020-01-07 02:06:27 -06:00
|
|
|
check: seen.len >= 10
|
2019-12-05 20:16:18 -06:00
|
|
|
for k, v in seen.pairs:
|
|
|
|
check: v == 1
|
|
|
|
|
2020-04-21 10:24:42 +09:00
|
|
|
await allFuturesThrowing(nodes.mapIt(it.stop()))
|
|
|
|
await allFuturesThrowing(awaitters)
|
2019-12-05 20:16:18 -06:00
|
|
|
result = true
|
|
|
|
|
|
|
|
check:
|
2019-12-16 23:24:03 -06:00
|
|
|
waitFor(runTests()) == true
|