2022-07-01 20:19:57 +02:00
|
|
|
# Nim-Libp2p
|
|
|
|
# Copyright (c) 2022 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.
|
2019-12-05 20:16:18 -06:00
|
|
|
|
2020-05-08 22:58:23 +02:00
|
|
|
{.used.}
|
|
|
|
|
2021-05-21 10:27:01 -06:00
|
|
|
import sequtils, options, tables, sets
|
2020-06-02 20:21:11 -06:00
|
|
|
import chronos, stew/byteutils
|
2020-04-21 10:24:42 +09:00
|
|
|
import chronicles
|
|
|
|
import utils, ../../libp2p/[errors,
|
2020-07-01 15:25:09 +09:00
|
|
|
peerid,
|
2019-12-05 20:16:18 -06:00
|
|
|
peerinfo,
|
2020-06-19 11:29:43 -06:00
|
|
|
stream/connection,
|
2020-08-02 23:20:11 -06:00
|
|
|
stream/bufferstream,
|
2019-12-05 20:16:18 -06:00
|
|
|
crypto/crypto,
|
|
|
|
protocols/pubsub/pubsub,
|
2020-12-19 23:43:32 +09:00
|
|
|
protocols/pubsub/gossipsub,
|
2020-08-02 23:20:11 -06:00
|
|
|
protocols/pubsub/pubsubpeer,
|
2020-07-15 13:18:55 -06:00
|
|
|
protocols/pubsub/peertable,
|
2021-11-14 09:08:05 +01:00
|
|
|
protocols/pubsub/timedcache,
|
2019-12-16 23:24:03 -06:00
|
|
|
protocols/pubsub/rpc/messages]
|
2022-02-22 02:04:17 +11:00
|
|
|
import ../../libp2p/protocols/pubsub/errors as pubsub_errors
|
2020-05-08 22:10:06 +02:00
|
|
|
import ../helpers
|
2020-04-21 10:24:42 +09:00
|
|
|
|
2021-04-18 10:08:33 +02:00
|
|
|
proc `$`(peer: PubSubPeer): string = shortLog(peer)
|
|
|
|
|
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
|
2020-08-11 18:05:49 -06:00
|
|
|
let fsub = GossipSub(sender)
|
2020-09-21 18:16:29 +09:00
|
|
|
let ev = newAsyncEvent()
|
|
|
|
fsub.heartbeatEvents.add(ev)
|
|
|
|
|
|
|
|
# await first heartbeat
|
|
|
|
await ev.wait()
|
|
|
|
ev.clear()
|
|
|
|
|
2020-04-21 10:24:42 +09:00
|
|
|
while (not fsub.gossipsub.hasKey(key) or
|
2021-12-16 11:05:20 +01:00
|
|
|
not fsub.gossipsub.hasPeerId(key, receiver.peerInfo.peerId)) and
|
2020-04-21 10:24:42 +09:00
|
|
|
(not fsub.mesh.hasKey(key) or
|
2021-12-16 11:05:20 +01:00
|
|
|
not fsub.mesh.hasPeerId(key, receiver.peerInfo.peerId)) and
|
2020-04-21 10:24:42 +09:00
|
|
|
(not fsub.fanout.hasKey(key) or
|
2021-12-16 11:05:20 +01:00
|
|
|
not fsub.fanout.hasPeerId(key , receiver.peerInfo.peerId)):
|
2020-05-27 12:33:49 -06:00
|
|
|
trace "waitSub sleeping..."
|
2020-11-12 21:44:02 -06:00
|
|
|
|
2020-09-21 18:16:29 +09:00
|
|
|
# await more heartbeats
|
|
|
|
await ev.wait()
|
|
|
|
ev.clear()
|
|
|
|
|
2020-04-21 10:24:42 +09:00
|
|
|
dec ceil
|
|
|
|
doAssert(ceil > 0, "waitSub timeout!")
|
|
|
|
|
2020-07-07 18:33:05 -06:00
|
|
|
template tryPublish(call: untyped, require: int, wait: Duration = 1.seconds, times: int = 10): untyped =
|
|
|
|
var
|
|
|
|
limit = times
|
|
|
|
pubs = 0
|
|
|
|
while pubs < require and limit > 0:
|
|
|
|
pubs = pubs + call
|
|
|
|
await sleepAsync(wait)
|
|
|
|
limit.dec()
|
|
|
|
if limit == 0:
|
|
|
|
doAssert(false, "Failed to publish!")
|
|
|
|
|
2019-12-05 20:16:18 -06:00
|
|
|
suite "GossipSub":
|
2020-04-21 10:24:42 +09:00
|
|
|
teardown:
|
2020-09-21 19:48:19 +02:00
|
|
|
checkTrackers()
|
2020-04-21 10:24:42 +09:00
|
|
|
|
2020-11-12 21:44:02 -06:00
|
|
|
asyncTest "GossipSub validation should succeed":
|
|
|
|
var handlerFut = newFuture[bool]()
|
|
|
|
proc handler(topic: string, data: seq[byte]) {.async, gcsafe.} =
|
|
|
|
check topic == "foobar"
|
|
|
|
handlerFut.complete(true)
|
2019-12-16 23:24:03 -06:00
|
|
|
|
2020-11-12 21:44:02 -06:00
|
|
|
let
|
|
|
|
nodes = generateNodes(2, gossip = true)
|
2020-07-27 13:33:51 -06:00
|
|
|
|
2020-11-12 21:44:02 -06:00
|
|
|
# start switches
|
|
|
|
nodesFut = await allFinished(
|
|
|
|
nodes[0].switch.start(),
|
|
|
|
nodes[1].switch.start(),
|
2020-08-11 18:05:49 -06:00
|
|
|
)
|
2020-07-27 13:33:51 -06:00
|
|
|
|
2020-11-12 21:44:02 -06:00
|
|
|
await subscribeNodes(nodes)
|
2019-12-16 23:24:03 -06:00
|
|
|
|
2020-12-19 23:43:32 +09:00
|
|
|
nodes[0].subscribe("foobar", handler)
|
|
|
|
nodes[1].subscribe("foobar", handler)
|
2019-12-16 23:24:03 -06:00
|
|
|
|
2020-11-12 21:44:02 -06:00
|
|
|
var subs: seq[Future[void]]
|
|
|
|
subs &= waitSub(nodes[1], nodes[0], "foobar")
|
|
|
|
subs &= waitSub(nodes[0], nodes[1], "foobar")
|
2020-08-11 18:05:49 -06:00
|
|
|
|
2020-11-12 21:44:02 -06:00
|
|
|
await allFuturesThrowing(subs)
|
2019-12-16 23:24:03 -06:00
|
|
|
|
2020-11-12 21:44:02 -06:00
|
|
|
var validatorFut = newFuture[bool]()
|
|
|
|
proc validator(topic: string,
|
|
|
|
message: Message):
|
|
|
|
Future[ValidationResult] {.async.} =
|
|
|
|
check topic == "foobar"
|
|
|
|
validatorFut.complete(true)
|
|
|
|
result = ValidationResult.Accept
|
2020-08-11 18:05:49 -06:00
|
|
|
|
2020-11-12 21:44:02 -06:00
|
|
|
nodes[1].addValidator("foobar", validator)
|
|
|
|
tryPublish await nodes[0].publish("foobar", "Hello!".toBytes()), 1
|
2020-04-21 10:24:42 +09:00
|
|
|
|
2020-11-12 21:44:02 -06:00
|
|
|
check (await validatorFut) and (await handlerFut)
|
2019-12-16 23:24:03 -06:00
|
|
|
|
2020-11-12 21:44:02 -06:00
|
|
|
await allFuturesThrowing(
|
|
|
|
nodes[0].switch.stop(),
|
|
|
|
nodes[1].switch.stop()
|
|
|
|
)
|
2020-09-21 18:16:29 +09:00
|
|
|
|
2020-11-12 21:44:02 -06:00
|
|
|
await allFuturesThrowing(nodesFut.concat())
|
2020-09-21 18:16:29 +09:00
|
|
|
|
2020-12-15 10:25:22 +09:00
|
|
|
asyncTest "GossipSub validation should fail (reject)":
|
2020-11-12 21:44:02 -06:00
|
|
|
proc handler(topic: string, data: seq[byte]) {.async, gcsafe.} =
|
|
|
|
check false # if we get here, it should fail
|
2019-12-16 23:24:03 -06:00
|
|
|
|
2020-11-12 21:44:02 -06:00
|
|
|
let
|
|
|
|
nodes = generateNodes(2, gossip = true)
|
2019-12-16 23:24:03 -06:00
|
|
|
|
2020-11-12 21:44:02 -06:00
|
|
|
# start switches
|
|
|
|
nodesFut = await allFinished(
|
|
|
|
nodes[0].switch.start(),
|
|
|
|
nodes[1].switch.start(),
|
2020-08-11 18:05:49 -06:00
|
|
|
)
|
2020-07-27 13:33:51 -06:00
|
|
|
|
2020-11-12 21:44:02 -06:00
|
|
|
await subscribeNodes(nodes)
|
2019-12-16 23:24:03 -06:00
|
|
|
|
2020-12-19 23:43:32 +09:00
|
|
|
nodes[0].subscribe("foobar", handler)
|
|
|
|
nodes[1].subscribe("foobar", handler)
|
2020-08-11 18:05:49 -06:00
|
|
|
|
2020-11-12 21:44:02 -06:00
|
|
|
var subs: seq[Future[void]]
|
|
|
|
subs &= waitSub(nodes[1], nodes[0], "foobar")
|
|
|
|
subs &= waitSub(nodes[0], nodes[1], "foobar")
|
2020-08-11 18:05:49 -06:00
|
|
|
|
2020-11-12 21:44:02 -06:00
|
|
|
await allFuturesThrowing(subs)
|
2020-08-11 18:05:49 -06:00
|
|
|
|
2020-11-12 21:44:02 -06:00
|
|
|
let gossip1 = GossipSub(nodes[0])
|
|
|
|
let gossip2 = GossipSub(nodes[1])
|
2019-12-16 23:24:03 -06:00
|
|
|
|
2020-11-12 21:44:02 -06:00
|
|
|
check:
|
|
|
|
gossip1.mesh["foobar"].len == 1 and "foobar" notin gossip1.fanout
|
|
|
|
gossip2.mesh["foobar"].len == 1 and "foobar" notin gossip2.fanout
|
|
|
|
|
|
|
|
var validatorFut = newFuture[bool]()
|
|
|
|
proc validator(topic: string,
|
|
|
|
message: Message):
|
|
|
|
Future[ValidationResult] {.async.} =
|
|
|
|
result = ValidationResult.Reject
|
|
|
|
validatorFut.complete(true)
|
|
|
|
|
|
|
|
nodes[1].addValidator("foobar", validator)
|
|
|
|
tryPublish await nodes[0].publish("foobar", "Hello!".toBytes()), 1
|
|
|
|
|
|
|
|
check (await validatorFut) == true
|
2020-12-15 10:25:22 +09:00
|
|
|
|
|
|
|
await allFuturesThrowing(
|
|
|
|
nodes[0].switch.stop(),
|
|
|
|
nodes[1].switch.stop()
|
|
|
|
)
|
|
|
|
|
|
|
|
await allFuturesThrowing(nodesFut.concat())
|
|
|
|
|
|
|
|
asyncTest "GossipSub validation should fail (ignore)":
|
|
|
|
proc handler(topic: string, data: seq[byte]) {.async, gcsafe.} =
|
|
|
|
check false # if we get here, it should fail
|
|
|
|
|
|
|
|
let
|
|
|
|
nodes = generateNodes(2, gossip = true)
|
|
|
|
|
|
|
|
# start switches
|
|
|
|
nodesFut = await allFinished(
|
|
|
|
nodes[0].switch.start(),
|
|
|
|
nodes[1].switch.start(),
|
|
|
|
)
|
|
|
|
|
|
|
|
await subscribeNodes(nodes)
|
|
|
|
|
2020-12-19 23:43:32 +09:00
|
|
|
nodes[0].subscribe("foobar", handler)
|
|
|
|
nodes[1].subscribe("foobar", handler)
|
2020-12-15 10:25:22 +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])
|
|
|
|
let gossip2 = GossipSub(nodes[1])
|
|
|
|
|
|
|
|
check:
|
|
|
|
gossip1.mesh["foobar"].len == 1 and "foobar" notin gossip1.fanout
|
|
|
|
gossip2.mesh["foobar"].len == 1 and "foobar" notin gossip2.fanout
|
|
|
|
|
|
|
|
var validatorFut = newFuture[bool]()
|
|
|
|
proc validator(topic: string,
|
|
|
|
message: Message):
|
|
|
|
Future[ValidationResult] {.async.} =
|
|
|
|
result = ValidationResult.Ignore
|
|
|
|
validatorFut.complete(true)
|
|
|
|
|
|
|
|
nodes[1].addValidator("foobar", validator)
|
|
|
|
tryPublish await nodes[0].publish("foobar", "Hello!".toBytes()), 1
|
|
|
|
|
|
|
|
check (await validatorFut) == true
|
2020-11-12 21:44:02 -06:00
|
|
|
|
|
|
|
await allFuturesThrowing(
|
|
|
|
nodes[0].switch.stop(),
|
|
|
|
nodes[1].switch.stop()
|
|
|
|
)
|
|
|
|
|
|
|
|
await allFuturesThrowing(nodesFut.concat())
|
|
|
|
|
|
|
|
asyncTest "GossipSub validation one fails and one succeeds":
|
|
|
|
var handlerFut = newFuture[bool]()
|
|
|
|
proc handler(topic: string, data: seq[byte]) {.async, gcsafe.} =
|
|
|
|
check topic == "foo"
|
|
|
|
handlerFut.complete(true)
|
|
|
|
|
|
|
|
let
|
|
|
|
nodes = generateNodes(2, gossip = true)
|
|
|
|
|
|
|
|
# start switches
|
|
|
|
nodesFut = await allFinished(
|
|
|
|
nodes[0].switch.start(),
|
|
|
|
nodes[1].switch.start(),
|
2020-08-11 18:05:49 -06:00
|
|
|
)
|
2020-08-02 23:20:11 -06:00
|
|
|
|
2020-11-12 21:44:02 -06:00
|
|
|
await subscribeNodes(nodes)
|
2020-08-02 23:20:11 -06:00
|
|
|
|
2020-12-19 23:43:32 +09:00
|
|
|
nodes[1].subscribe("foo", handler)
|
|
|
|
nodes[1].subscribe("bar", handler)
|
2019-12-05 20:16:18 -06:00
|
|
|
|
2020-11-12 21:44:02 -06:00
|
|
|
var passed, failed: Future[bool] = newFuture[bool]()
|
|
|
|
proc validator(topic: string,
|
|
|
|
message: Message):
|
|
|
|
Future[ValidationResult] {.async.} =
|
|
|
|
result = if topic == "foo":
|
|
|
|
passed.complete(true)
|
|
|
|
ValidationResult.Accept
|
|
|
|
else:
|
|
|
|
failed.complete(true)
|
|
|
|
ValidationResult.Reject
|
2019-12-05 20:16:18 -06:00
|
|
|
|
2020-11-12 21:44:02 -06:00
|
|
|
nodes[1].addValidator("foo", "bar", validator)
|
|
|
|
tryPublish await nodes[0].publish("foo", "Hello!".toBytes()), 1
|
|
|
|
tryPublish await nodes[0].publish("bar", "Hello!".toBytes()), 1
|
2019-12-05 20:16:18 -06:00
|
|
|
|
2020-11-12 21:44:02 -06:00
|
|
|
check ((await passed) and (await failed) and (await handlerFut))
|
2019-12-05 20:16:18 -06:00
|
|
|
|
2020-11-12 21:44:02 -06:00
|
|
|
let gossip1 = GossipSub(nodes[0])
|
|
|
|
let gossip2 = GossipSub(nodes[1])
|
2020-07-27 13:33:51 -06:00
|
|
|
|
2020-11-12 21:44:02 -06:00
|
|
|
check:
|
|
|
|
"foo" notin gossip1.mesh and gossip1.fanout["foo"].len == 1
|
|
|
|
"foo" notin gossip2.mesh and "foo" notin gossip2.fanout
|
|
|
|
"bar" notin gossip1.mesh and gossip1.fanout["bar"].len == 1
|
|
|
|
"bar" notin gossip2.mesh and "bar" notin gossip2.fanout
|
|
|
|
|
|
|
|
await allFuturesThrowing(
|
|
|
|
nodes[0].switch.stop(),
|
|
|
|
nodes[1].switch.stop()
|
|
|
|
)
|
|
|
|
|
|
|
|
await allFuturesThrowing(nodesFut.concat())
|
|
|
|
|
2021-12-02 15:47:40 +01:00
|
|
|
asyncTest "GossipSub unsub - resub faster than backoff":
|
|
|
|
var handlerFut = newFuture[bool]()
|
|
|
|
proc handler(topic: string, data: seq[byte]) {.async, gcsafe.} =
|
|
|
|
check topic == "foobar"
|
|
|
|
handlerFut.complete(true)
|
|
|
|
|
|
|
|
let
|
|
|
|
nodes = generateNodes(2, gossip = true)
|
|
|
|
|
|
|
|
# start switches
|
|
|
|
nodesFut = await allFinished(
|
|
|
|
nodes[0].switch.start(),
|
|
|
|
nodes[1].switch.start(),
|
|
|
|
)
|
|
|
|
|
|
|
|
await subscribeNodes(nodes)
|
|
|
|
|
|
|
|
nodes[0].subscribe("foobar", handler)
|
|
|
|
nodes[1].subscribe("foobar", handler)
|
|
|
|
|
|
|
|
var subs: seq[Future[void]]
|
|
|
|
subs &= waitSub(nodes[1], nodes[0], "foobar")
|
|
|
|
subs &= waitSub(nodes[0], nodes[1], "foobar")
|
|
|
|
|
|
|
|
await allFuturesThrowing(subs)
|
|
|
|
|
|
|
|
nodes[0].unsubscribe("foobar", handler)
|
|
|
|
nodes[0].subscribe("foobar", handler)
|
|
|
|
|
|
|
|
# regular backoff is 60 seconds, so we must not wait that long
|
|
|
|
await (waitSub(nodes[0], nodes[1], "foobar") and waitSub(nodes[1], nodes[0], "foobar")).wait(30.seconds)
|
|
|
|
|
|
|
|
var validatorFut = newFuture[bool]()
|
|
|
|
proc validator(topic: string,
|
|
|
|
message: Message):
|
|
|
|
Future[ValidationResult] {.async.} =
|
|
|
|
check topic == "foobar"
|
|
|
|
validatorFut.complete(true)
|
|
|
|
result = ValidationResult.Accept
|
|
|
|
|
|
|
|
nodes[1].addValidator("foobar", validator)
|
|
|
|
tryPublish await nodes[0].publish("foobar", "Hello!".toBytes()), 1
|
|
|
|
|
|
|
|
check (await validatorFut) and (await handlerFut)
|
|
|
|
|
|
|
|
await allFuturesThrowing(
|
|
|
|
nodes[0].switch.stop(),
|
|
|
|
nodes[1].switch.stop()
|
|
|
|
)
|
|
|
|
|
|
|
|
await allFuturesThrowing(nodesFut.concat())
|
|
|
|
|
2020-11-12 21:44:02 -06:00
|
|
|
asyncTest "e2e - GossipSub should add remote peer topic subscriptions":
|
|
|
|
proc handler(topic: string, data: seq[byte]) {.async, gcsafe.} =
|
|
|
|
discard
|
|
|
|
|
|
|
|
let
|
|
|
|
nodes = generateNodes(
|
|
|
|
2,
|
2021-02-08 14:33:34 -06:00
|
|
|
gossip = true)
|
2020-11-12 21:44:02 -06:00
|
|
|
|
|
|
|
# start switches
|
|
|
|
nodesFut = await allFinished(
|
|
|
|
nodes[0].switch.start(),
|
|
|
|
nodes[1].switch.start(),
|
2020-08-11 18:05:49 -06:00
|
|
|
)
|
2019-12-05 20:16:18 -06:00
|
|
|
|
2020-11-12 21:44:02 -06:00
|
|
|
await subscribeNodes(nodes)
|
2019-12-05 20:16:18 -06:00
|
|
|
|
2020-12-19 23:43:32 +09:00
|
|
|
nodes[1].subscribe("foobar", handler)
|
2020-11-12 21:44:02 -06:00
|
|
|
await sleepAsync(10.seconds)
|
2019-12-05 20:16:18 -06:00
|
|
|
|
2020-11-12 21:44:02 -06:00
|
|
|
let gossip1 = GossipSub(nodes[0])
|
|
|
|
let gossip2 = GossipSub(nodes[1])
|
2020-08-11 18:05:49 -06:00
|
|
|
|
2020-11-12 21:44:02 -06:00
|
|
|
check:
|
|
|
|
"foobar" in gossip2.topics
|
|
|
|
"foobar" in gossip1.gossipsub
|
2021-12-16 11:05:20 +01:00
|
|
|
gossip1.gossipsub.hasPeerId("foobar", gossip2.peerInfo.peerId)
|
2020-11-12 21:44:02 -06:00
|
|
|
|
|
|
|
await allFuturesThrowing(
|
|
|
|
nodes[0].switch.stop(),
|
|
|
|
nodes[1].switch.stop()
|
|
|
|
)
|
|
|
|
|
|
|
|
await allFuturesThrowing(nodesFut.concat())
|
|
|
|
|
|
|
|
asyncTest "e2e - GossipSub should add remote peer topic subscriptions if both peers are subscribed":
|
|
|
|
proc handler(topic: string, data: seq[byte]) {.async, gcsafe.} =
|
|
|
|
discard
|
|
|
|
|
|
|
|
let
|
|
|
|
nodes = generateNodes(
|
|
|
|
2,
|
2021-02-08 14:33:34 -06:00
|
|
|
gossip = true)
|
2020-11-12 21:44:02 -06:00
|
|
|
|
|
|
|
# start switches
|
|
|
|
nodesFut = await allFinished(
|
|
|
|
nodes[0].switch.start(),
|
|
|
|
nodes[1].switch.start(),
|
2020-08-11 18:05:49 -06:00
|
|
|
)
|
2019-12-05 20:16:18 -06:00
|
|
|
|
2020-11-12 21:44:02 -06:00
|
|
|
await subscribeNodes(nodes)
|
2020-08-11 18:05:49 -06:00
|
|
|
|
2020-12-19 23:43:32 +09:00
|
|
|
nodes[0].subscribe("foobar", handler)
|
|
|
|
nodes[1].subscribe("foobar", handler)
|
2020-08-11 18:05:49 -06:00
|
|
|
|
2020-11-12 21:44:02 -06:00
|
|
|
var subs: seq[Future[void]]
|
|
|
|
subs &= waitSub(nodes[1], nodes[0], "foobar")
|
|
|
|
subs &= waitSub(nodes[0], nodes[1], "foobar")
|
2019-12-05 20:16:18 -06:00
|
|
|
|
2020-11-12 21:44:02 -06:00
|
|
|
await allFuturesThrowing(subs)
|
2019-12-05 20:16:18 -06:00
|
|
|
|
2020-11-12 21:44:02 -06:00
|
|
|
let
|
|
|
|
gossip1 = GossipSub(nodes[0])
|
|
|
|
gossip2 = GossipSub(nodes[1])
|
2020-04-30 22:22:31 +09:00
|
|
|
|
2020-11-12 21:44:02 -06:00
|
|
|
check:
|
|
|
|
"foobar" in gossip1.topics
|
|
|
|
"foobar" in gossip2.topics
|
|
|
|
|
|
|
|
"foobar" in gossip1.gossipsub
|
|
|
|
"foobar" in gossip2.gossipsub
|
|
|
|
|
2021-12-16 11:05:20 +01:00
|
|
|
gossip1.gossipsub.hasPeerId("foobar", gossip2.peerInfo.peerId) or
|
|
|
|
gossip1.mesh.hasPeerId("foobar", gossip2.peerInfo.peerId)
|
2020-11-12 21:44:02 -06:00
|
|
|
|
2021-12-16 11:05:20 +01:00
|
|
|
gossip2.gossipsub.hasPeerId("foobar", gossip1.peerInfo.peerId) or
|
|
|
|
gossip2.mesh.hasPeerId("foobar", gossip1.peerInfo.peerId)
|
2020-11-12 21:44:02 -06:00
|
|
|
|
|
|
|
await allFuturesThrowing(
|
|
|
|
nodes[0].switch.stop(),
|
|
|
|
nodes[1].switch.stop()
|
|
|
|
)
|
|
|
|
|
|
|
|
await allFuturesThrowing(nodesFut.concat())
|
|
|
|
|
|
|
|
asyncTest "e2e - GossipSub send over fanout A -> B":
|
|
|
|
var passed = newFuture[void]()
|
|
|
|
proc handler(topic: string, data: seq[byte]) {.async, gcsafe.} =
|
|
|
|
check topic == "foobar"
|
|
|
|
passed.complete()
|
|
|
|
|
|
|
|
let
|
|
|
|
nodes = generateNodes(
|
|
|
|
2,
|
2021-02-08 14:33:34 -06:00
|
|
|
gossip = true)
|
2020-11-12 21:44:02 -06:00
|
|
|
|
|
|
|
# start switches
|
|
|
|
nodesFut = await allFinished(
|
|
|
|
nodes[0].switch.start(),
|
|
|
|
nodes[1].switch.start(),
|
|
|
|
)
|
2019-12-05 20:16:18 -06:00
|
|
|
|
2020-11-12 21:44:02 -06:00
|
|
|
await subscribeNodes(nodes)
|
2020-04-21 10:24:42 +09:00
|
|
|
|
2020-12-19 23:43:32 +09:00
|
|
|
nodes[1].subscribe("foobar", handler)
|
2020-11-12 21:44:02 -06:00
|
|
|
await waitSub(nodes[0], nodes[1], "foobar")
|
2020-07-27 13:33:51 -06:00
|
|
|
|
2020-11-12 21:44:02 -06:00
|
|
|
var observed = 0
|
|
|
|
let
|
|
|
|
obs1 = PubSubObserver(onRecv: proc(peer: PubSubPeer; msgs: var RPCMsg) =
|
|
|
|
inc observed
|
2020-08-11 18:05:49 -06:00
|
|
|
)
|
2020-11-12 21:44:02 -06:00
|
|
|
obs2 = PubSubObserver(onSend: proc(peer: PubSubPeer; msgs: var RPCMsg) =
|
|
|
|
inc observed
|
2020-08-11 18:05:49 -06:00
|
|
|
)
|
2019-12-05 20:16:18 -06:00
|
|
|
|
2020-11-12 21:44:02 -06:00
|
|
|
nodes[1].addObserver(obs1)
|
|
|
|
nodes[0].addObserver(obs2)
|
2020-08-11 18:05:49 -06:00
|
|
|
|
2020-11-12 21:44:02 -06:00
|
|
|
tryPublish await nodes[0].publish("foobar", "Hello!".toBytes()), 1
|
2019-12-05 20:16:18 -06:00
|
|
|
|
2020-11-12 21:44:02 -06:00
|
|
|
var gossip1: GossipSub = GossipSub(nodes[0])
|
|
|
|
var gossip2: GossipSub = GossipSub(nodes[1])
|
2019-12-05 20:16:18 -06:00
|
|
|
|
2020-11-12 21:44:02 -06:00
|
|
|
check:
|
|
|
|
"foobar" in gossip1.gossipsub
|
2021-12-16 11:05:20 +01:00
|
|
|
gossip1.fanout.hasPeerId("foobar", gossip2.peerInfo.peerId)
|
|
|
|
not gossip1.mesh.hasPeerId("foobar", gossip2.peerInfo.peerId)
|
2020-11-12 21:44:02 -06:00
|
|
|
|
|
|
|
await passed.wait(2.seconds)
|
|
|
|
|
|
|
|
trace "test done, stopping..."
|
|
|
|
|
|
|
|
await allFuturesThrowing(
|
|
|
|
nodes[0].switch.stop(),
|
|
|
|
nodes[1].switch.stop()
|
|
|
|
)
|
|
|
|
|
|
|
|
await allFuturesThrowing(nodesFut.concat())
|
|
|
|
check observed == 2
|
|
|
|
|
2022-02-21 16:22:08 +01:00
|
|
|
asyncTest "e2e - GossipSub send over fanout A -> B for subscribed topic":
|
|
|
|
var passed = newFuture[void]()
|
|
|
|
proc handler(topic: string, data: seq[byte]) {.async, gcsafe.} =
|
|
|
|
check topic == "foobar"
|
|
|
|
passed.complete()
|
|
|
|
|
|
|
|
let
|
|
|
|
nodes = generateNodes(
|
|
|
|
2,
|
2022-02-21 18:14:43 +01:00
|
|
|
gossip = true,
|
|
|
|
unsubscribeBackoff = 10.minutes)
|
2022-02-21 16:22:08 +01:00
|
|
|
|
|
|
|
# start switches
|
|
|
|
nodesFut = await allFinished(
|
|
|
|
nodes[0].switch.start(),
|
|
|
|
nodes[1].switch.start(),
|
|
|
|
)
|
|
|
|
|
|
|
|
await subscribeNodes(nodes)
|
|
|
|
|
|
|
|
nodes[1].subscribe("foobar", handler)
|
|
|
|
nodes[0].subscribe("foobar", handler)
|
|
|
|
await waitSub(nodes[0], nodes[1], "foobar")
|
|
|
|
await waitSub(nodes[1], nodes[0], "foobar")
|
|
|
|
|
|
|
|
nodes[0].unsubscribe("foobar", handler)
|
|
|
|
|
|
|
|
let gsNode = GossipSub(nodes[1])
|
|
|
|
check await checkExpiring(gsNode.mesh.getOrDefault("foobar").len == 0)
|
|
|
|
|
|
|
|
nodes[0].subscribe("foobar", handler)
|
|
|
|
|
|
|
|
check GossipSub(nodes[0]).mesh.getOrDefault("foobar").len == 0
|
|
|
|
|
|
|
|
tryPublish await nodes[0].publish("foobar", "Hello!".toBytes()), 1
|
|
|
|
|
|
|
|
check:
|
|
|
|
GossipSub(nodes[0]).fanout.getOrDefault("foobar").len > 0
|
|
|
|
GossipSub(nodes[0]).mesh.getOrDefault("foobar").len == 0
|
|
|
|
|
|
|
|
await passed.wait(2.seconds)
|
|
|
|
|
|
|
|
trace "test done, stopping..."
|
|
|
|
|
|
|
|
await allFuturesThrowing(
|
|
|
|
nodes[0].switch.stop(),
|
|
|
|
nodes[1].switch.stop()
|
|
|
|
)
|
|
|
|
|
|
|
|
await allFuturesThrowing(nodesFut.concat())
|
|
|
|
|
2020-11-12 21:44:02 -06:00
|
|
|
asyncTest "e2e - GossipSub send over mesh A -> B":
|
|
|
|
var passed: Future[bool] = newFuture[bool]()
|
|
|
|
proc handler(topic: string, data: seq[byte]) {.async, gcsafe.} =
|
|
|
|
check topic == "foobar"
|
|
|
|
passed.complete(true)
|
|
|
|
|
|
|
|
let
|
|
|
|
nodes = generateNodes(
|
|
|
|
2,
|
2021-02-08 14:33:34 -06:00
|
|
|
gossip = true)
|
2020-11-12 21:44:02 -06:00
|
|
|
|
|
|
|
# start switches
|
|
|
|
nodesFut = await allFinished(
|
|
|
|
nodes[0].switch.start(),
|
|
|
|
nodes[1].switch.start(),
|
|
|
|
)
|
2020-08-11 18:05:49 -06:00
|
|
|
|
2020-11-12 21:44:02 -06:00
|
|
|
await subscribeNodes(nodes)
|
2019-12-05 20:16:18 -06:00
|
|
|
|
2020-12-19 23:43:32 +09:00
|
|
|
nodes[0].subscribe("foobar", handler)
|
|
|
|
nodes[1].subscribe("foobar", handler)
|
2020-11-12 21:44:02 -06:00
|
|
|
await waitSub(nodes[0], nodes[1], "foobar")
|
2020-01-07 02:06:27 -06:00
|
|
|
|
2020-11-12 21:44:02 -06:00
|
|
|
tryPublish await nodes[0].publish("foobar", "Hello!".toBytes()), 1
|
2019-12-05 20:16:18 -06:00
|
|
|
|
2020-11-12 21:44:02 -06:00
|
|
|
check await passed
|
2020-04-21 10:24:42 +09:00
|
|
|
|
2020-11-12 21:44:02 -06:00
|
|
|
var gossip1: GossipSub = GossipSub(nodes[0])
|
|
|
|
var gossip2: GossipSub = GossipSub(nodes[1])
|
2019-12-05 20:16:18 -06:00
|
|
|
|
2020-11-12 21:44:02 -06:00
|
|
|
check:
|
|
|
|
"foobar" in gossip1.gossipsub
|
|
|
|
"foobar" in gossip2.gossipsub
|
2021-12-16 11:05:20 +01:00
|
|
|
gossip1.mesh.hasPeerId("foobar", gossip2.peerInfo.peerId)
|
|
|
|
not gossip1.fanout.hasPeerId("foobar", gossip2.peerInfo.peerId)
|
|
|
|
gossip2.mesh.hasPeerId("foobar", gossip1.peerInfo.peerId)
|
|
|
|
not gossip2.fanout.hasPeerId("foobar", gossip1.peerInfo.peerId)
|
2020-11-12 21:44:02 -06:00
|
|
|
|
|
|
|
await allFuturesThrowing(
|
|
|
|
nodes[0].switch.stop(),
|
|
|
|
nodes[1].switch.stop()
|
|
|
|
)
|
|
|
|
|
|
|
|
await allFuturesThrowing(nodesFut.concat())
|
|
|
|
|
2021-11-14 09:08:05 +01:00
|
|
|
asyncTest "e2e - GossipSub should not send to source & peers who already seen":
|
|
|
|
# 3 nodes: A, B, C
|
|
|
|
# A publishes, B relays, C is having a long validation
|
|
|
|
# so C should not send to anyone
|
|
|
|
|
|
|
|
let
|
|
|
|
nodes = generateNodes(
|
|
|
|
3,
|
|
|
|
gossip = true)
|
|
|
|
|
|
|
|
# start switches
|
|
|
|
nodesFut = await allFinished(
|
|
|
|
nodes[0].switch.start(),
|
|
|
|
nodes[1].switch.start(),
|
|
|
|
nodes[2].switch.start(),
|
|
|
|
)
|
|
|
|
|
|
|
|
await subscribeNodes(nodes)
|
|
|
|
|
|
|
|
var cRelayed: Future[void] = newFuture[void]()
|
|
|
|
var bFinished: Future[void] = newFuture[void]()
|
|
|
|
var
|
|
|
|
aReceived = 0
|
|
|
|
cReceived = 0
|
|
|
|
proc handlerA(topic: string, data: seq[byte]) {.async, gcsafe.} =
|
|
|
|
inc aReceived
|
|
|
|
check aReceived < 2
|
|
|
|
proc handlerB(topic: string, data: seq[byte]) {.async, gcsafe.} = discard
|
|
|
|
proc handlerC(topic: string, data: seq[byte]) {.async, gcsafe.} =
|
|
|
|
inc cReceived
|
|
|
|
check cReceived < 2
|
|
|
|
cRelayed.complete()
|
|
|
|
|
|
|
|
nodes[0].subscribe("foobar", handlerA)
|
|
|
|
nodes[1].subscribe("foobar", handlerB)
|
|
|
|
nodes[2].subscribe("foobar", handlerC)
|
|
|
|
await waitSub(nodes[0], nodes[1], "foobar")
|
|
|
|
await waitSub(nodes[0], nodes[2], "foobar")
|
|
|
|
await waitSub(nodes[2], nodes[1], "foobar")
|
|
|
|
await waitSub(nodes[1], nodes[2], "foobar")
|
|
|
|
|
|
|
|
var gossip1: GossipSub = GossipSub(nodes[0])
|
|
|
|
var gossip2: GossipSub = GossipSub(nodes[1])
|
|
|
|
var gossip3: GossipSub = GossipSub(nodes[2])
|
|
|
|
|
|
|
|
proc slowValidator(topic: string, message: Message): Future[ValidationResult] {.async.} =
|
|
|
|
await cRelayed
|
|
|
|
# Empty A & C caches to detect duplicates
|
|
|
|
gossip1.seen = TimedCache[MessageId].init()
|
|
|
|
gossip3.seen = TimedCache[MessageId].init()
|
|
|
|
let msgId = toSeq(gossip2.validationSeen.keys)[0]
|
|
|
|
check await checkExpiring(try: gossip2.validationSeen[msgId].len > 0 except: false)
|
|
|
|
result = ValidationResult.Accept
|
|
|
|
bFinished.complete()
|
|
|
|
|
|
|
|
nodes[1].addValidator("foobar", slowValidator)
|
|
|
|
|
|
|
|
tryPublish await nodes[0].publish("foobar", "Hello!".toBytes()), 1
|
|
|
|
|
|
|
|
await bFinished
|
|
|
|
|
|
|
|
await allFuturesThrowing(
|
|
|
|
nodes[0].switch.stop(),
|
|
|
|
nodes[1].switch.stop(),
|
|
|
|
nodes[2].switch.stop()
|
|
|
|
)
|
|
|
|
|
|
|
|
await allFuturesThrowing(nodesFut.concat())
|
|
|
|
|
2021-04-22 18:51:22 +09:00
|
|
|
asyncTest "e2e - GossipSub send over floodPublish A -> B":
|
|
|
|
var passed: Future[bool] = newFuture[bool]()
|
|
|
|
proc handler(topic: string, data: seq[byte]) {.async, gcsafe.} =
|
|
|
|
check topic == "foobar"
|
|
|
|
passed.complete(true)
|
2020-11-12 21:44:02 -06:00
|
|
|
|
|
|
|
let
|
2021-04-22 18:51:22 +09:00
|
|
|
nodes = generateNodes(
|
|
|
|
2,
|
|
|
|
gossip = true)
|
2020-11-12 21:44:02 -06:00
|
|
|
|
2021-04-22 18:51:22 +09:00
|
|
|
# start switches
|
|
|
|
nodesFut = await allFinished(
|
|
|
|
nodes[0].switch.start(),
|
|
|
|
nodes[1].switch.start(),
|
|
|
|
)
|
2020-11-12 21:44:02 -06:00
|
|
|
|
2021-04-22 18:51:22 +09:00
|
|
|
var gossip1: GossipSub = GossipSub(nodes[0])
|
|
|
|
gossip1.parameters.floodPublish = true
|
|
|
|
var gossip2: GossipSub = GossipSub(nodes[1])
|
|
|
|
gossip2.parameters.floodPublish = true
|
2020-11-12 21:44:02 -06:00
|
|
|
|
2021-04-22 18:51:22 +09:00
|
|
|
await subscribeNodes(nodes)
|
2020-11-12 21:44:02 -06:00
|
|
|
|
2021-04-22 18:51:22 +09:00
|
|
|
# nodes[0].subscribe("foobar", handler)
|
|
|
|
nodes[1].subscribe("foobar", handler)
|
|
|
|
await waitSub(nodes[0], nodes[1], "foobar")
|
2020-11-12 21:44:02 -06:00
|
|
|
|
2021-04-22 18:51:22 +09:00
|
|
|
tryPublish await nodes[0].publish("foobar", "Hello!".toBytes()), 1
|
2020-07-20 15:55:00 +09:00
|
|
|
|
2021-04-22 18:51:22 +09:00
|
|
|
check await passed
|
|
|
|
|
|
|
|
check:
|
|
|
|
"foobar" in gossip1.gossipsub
|
|
|
|
"foobar" notin gossip2.gossipsub
|
2021-12-16 11:05:20 +01:00
|
|
|
not gossip1.mesh.hasPeerId("foobar", gossip2.peerInfo.peerId)
|
|
|
|
not gossip1.fanout.hasPeerId("foobar", gossip2.peerInfo.peerId)
|
2020-11-12 21:44:02 -06:00
|
|
|
|
|
|
|
await allFuturesThrowing(
|
2021-04-22 18:51:22 +09:00
|
|
|
nodes[0].switch.stop(),
|
|
|
|
nodes[1].switch.stop()
|
|
|
|
)
|
2020-11-12 21:44:02 -06:00
|
|
|
|
2021-04-22 18:51:22 +09:00
|
|
|
await allFuturesThrowing(nodesFut.concat())
|
|
|
|
|
|
|
|
asyncTest "e2e - GossipSub with multiple peers":
|
2020-11-12 21:44:02 -06:00
|
|
|
var runs = 10
|
|
|
|
|
|
|
|
let
|
|
|
|
nodes = generateNodes(runs, gossip = true, triggerSelf = true)
|
|
|
|
nodesFut = nodes.mapIt(it.switch.start())
|
|
|
|
|
2021-04-22 18:51:22 +09:00
|
|
|
await subscribeNodes(nodes)
|
2020-11-12 21:44:02 -06:00
|
|
|
|
|
|
|
var seen: Table[string, int]
|
|
|
|
var seenFut = newFuture[void]()
|
2020-12-14 15:22:53 -06:00
|
|
|
for i in 0..<nodes.len:
|
|
|
|
let dialer = nodes[i]
|
2020-11-12 21:44:02 -06:00
|
|
|
var handler: TopicHandler
|
|
|
|
closureScope:
|
|
|
|
var peerName = $dialer.peerInfo.peerId
|
|
|
|
handler = proc(topic: string, data: seq[byte]) {.async, gcsafe, closure.} =
|
|
|
|
if peerName notin seen:
|
|
|
|
seen[peerName] = 0
|
|
|
|
seen[peerName].inc
|
|
|
|
check topic == "foobar"
|
|
|
|
if not seenFut.finished() and seen.len >= runs:
|
|
|
|
seenFut.complete()
|
|
|
|
|
2020-12-19 23:43:32 +09:00
|
|
|
dialer.subscribe("foobar", handler)
|
2020-11-12 21:44:02 -06:00
|
|
|
await waitSub(nodes[0], dialer, "foobar")
|
|
|
|
|
|
|
|
tryPublish await wait(nodes[0].publish("foobar",
|
|
|
|
toBytes("from node " &
|
|
|
|
$nodes[0].peerInfo.peerId)),
|
|
|
|
1.minutes), 1, 5.seconds
|
|
|
|
|
2021-04-22 18:51:22 +09:00
|
|
|
await wait(seenFut, 2.minutes)
|
2020-11-12 21:44:02 -06:00
|
|
|
check: seen.len >= runs
|
|
|
|
for k, v in seen.pairs:
|
|
|
|
check: v >= 1
|
|
|
|
|
|
|
|
for node in nodes:
|
|
|
|
var gossip = GossipSub(node)
|
2021-04-22 18:51:22 +09:00
|
|
|
|
2020-11-12 21:44:02 -06:00
|
|
|
check:
|
|
|
|
"foobar" in gossip.gossipsub
|
2020-07-27 13:33:51 -06:00
|
|
|
|
2020-11-12 21:44:02 -06:00
|
|
|
await allFuturesThrowing(
|
|
|
|
nodes.mapIt(
|
|
|
|
allFutures(
|
|
|
|
it.switch.stop())))
|
2020-01-09 21:59:27 -06:00
|
|
|
|
2020-11-12 21:44:02 -06:00
|
|
|
await allFuturesThrowing(nodesFut)
|
2021-01-13 23:49:44 +09:00
|
|
|
|
2021-04-22 18:51:22 +09:00
|
|
|
asyncTest "e2e - GossipSub with multiple peers (sparse)":
|
2021-02-26 14:15:58 +09:00
|
|
|
var runs = 10
|
|
|
|
|
|
|
|
let
|
|
|
|
nodes = generateNodes(runs, gossip = true, triggerSelf = true)
|
|
|
|
nodesFut = nodes.mapIt(it.switch.start())
|
|
|
|
|
2021-04-22 18:51:22 +09:00
|
|
|
await subscribeSparseNodes(nodes)
|
2021-02-26 14:15:58 +09:00
|
|
|
|
|
|
|
var seen: Table[string, int]
|
|
|
|
var seenFut = newFuture[void]()
|
|
|
|
for i in 0..<nodes.len:
|
|
|
|
let dialer = nodes[i]
|
|
|
|
var handler: TopicHandler
|
|
|
|
closureScope:
|
|
|
|
var peerName = $dialer.peerInfo.peerId
|
|
|
|
handler = proc(topic: string, data: seq[byte]) {.async, gcsafe, closure.} =
|
|
|
|
if peerName notin seen:
|
|
|
|
seen[peerName] = 0
|
|
|
|
seen[peerName].inc
|
|
|
|
check topic == "foobar"
|
|
|
|
if not seenFut.finished() and seen.len >= runs:
|
|
|
|
seenFut.complete()
|
|
|
|
|
|
|
|
dialer.subscribe("foobar", handler)
|
|
|
|
await waitSub(nodes[0], dialer, "foobar")
|
|
|
|
|
|
|
|
tryPublish await wait(nodes[0].publish("foobar",
|
|
|
|
toBytes("from node " &
|
|
|
|
$nodes[0].peerInfo.peerId)),
|
|
|
|
1.minutes), 1, 5.seconds
|
|
|
|
|
|
|
|
await wait(seenFut, 5.minutes)
|
|
|
|
check: seen.len >= runs
|
|
|
|
for k, v in seen.pairs:
|
|
|
|
check: v >= 1
|
|
|
|
|
|
|
|
for node in nodes:
|
|
|
|
var gossip = GossipSub(node)
|
|
|
|
check:
|
|
|
|
"foobar" in gossip.gossipsub
|
|
|
|
gossip.fanout.len == 0
|
|
|
|
gossip.mesh["foobar"].len > 0
|
|
|
|
|
|
|
|
await allFuturesThrowing(
|
|
|
|
nodes.mapIt(
|
|
|
|
allFutures(
|
|
|
|
it.switch.stop())))
|
|
|
|
|
|
|
|
await allFuturesThrowing(nodesFut)
|
2022-03-14 09:39:30 +01:00
|
|
|
|
|
|
|
asyncTest "e2e - GossipSub peer exchange":
|
|
|
|
# A, B & C are subscribed to something
|
|
|
|
# B unsubcribe from it, it should send
|
|
|
|
# PX to A & C
|
|
|
|
#
|
|
|
|
# C sent his SPR, not A
|
|
|
|
proc handler(topic: string, data: seq[byte]) {.async, gcsafe.} =
|
|
|
|
discard # not used in this test
|
|
|
|
|
|
|
|
let
|
|
|
|
nodes = generateNodes(
|
|
|
|
2,
|
2022-05-10 10:39:43 +02:00
|
|
|
gossip = true,
|
|
|
|
enablePX = true) &
|
2022-03-14 09:39:30 +01:00
|
|
|
generateNodes(1, gossip = true, sendSignedPeerRecord = true)
|
|
|
|
|
|
|
|
# start switches
|
|
|
|
nodesFut = await allFinished(
|
|
|
|
nodes[0].switch.start(),
|
|
|
|
nodes[1].switch.start(),
|
|
|
|
nodes[2].switch.start(),
|
|
|
|
)
|
|
|
|
|
|
|
|
var
|
|
|
|
gossip0 = GossipSub(nodes[0])
|
|
|
|
gossip1 = GossipSub(nodes[1])
|
|
|
|
gossip2 = GossipSub(nodes[1])
|
|
|
|
|
|
|
|
await subscribeNodes(nodes)
|
|
|
|
|
|
|
|
nodes[0].subscribe("foobar", handler)
|
|
|
|
nodes[1].subscribe("foobar", handler)
|
|
|
|
nodes[2].subscribe("foobar", handler)
|
|
|
|
for x in 0..<3:
|
|
|
|
for y in 0..<3:
|
|
|
|
if x != y:
|
|
|
|
await waitSub(nodes[x], nodes[y], "foobar")
|
|
|
|
|
|
|
|
var passed: Future[void] = newFuture[void]()
|
|
|
|
gossip0.routingRecordsHandler.add(proc(peer: PeerId, tag: string, peers: seq[RoutingRecordsPair]) =
|
|
|
|
check:
|
|
|
|
tag == "foobar"
|
|
|
|
peers.len == 2
|
|
|
|
peers[0].record.isSome() xor peers[1].record.isSome()
|
|
|
|
passed.complete()
|
|
|
|
)
|
|
|
|
nodes[1].unsubscribe("foobar", handler)
|
|
|
|
|
2022-05-10 10:39:43 +02:00
|
|
|
await passed.wait(5.seconds)
|
2022-03-14 09:39:30 +01:00
|
|
|
|
|
|
|
await allFuturesThrowing(
|
|
|
|
nodes[0].switch.stop(),
|
|
|
|
nodes[1].switch.stop(),
|
|
|
|
nodes[2].switch.stop()
|
|
|
|
)
|
|
|
|
|
|
|
|
await allFuturesThrowing(nodesFut.concat())
|