2019-09-09 20:15:52 -06:00
|
|
|
## Nim-LibP2P
|
2019-09-24 11:48:23 -06:00
|
|
|
## Copyright (c) 2019 Status Research & Development GmbH
|
2019-09-09 20:15:52 -06:00
|
|
|
## 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-07-16 12:06:57 +02:00
|
|
|
import std/[tables, sequtils, sets]
|
2020-07-17 09:36:48 -06:00
|
|
|
import chronos, chronicles, metrics
|
2019-09-09 20:15:52 -06:00
|
|
|
import pubsubpeer,
|
2020-06-28 17:56:38 +02:00
|
|
|
rpc/[message, messages],
|
2020-08-11 18:05:49 -06:00
|
|
|
../../switch,
|
2019-09-09 20:15:52 -06:00
|
|
|
../protocol,
|
2020-06-19 11:29:43 -06:00
|
|
|
../../stream/connection,
|
2020-07-01 15:25:09 +09:00
|
|
|
../../peerid,
|
2020-07-17 09:36:48 -06:00
|
|
|
../../peerinfo,
|
|
|
|
../../errors
|
2019-09-09 20:15:52 -06:00
|
|
|
|
|
|
|
export PubSubPeer
|
2020-04-30 22:22:31 +09:00
|
|
|
export PubSubObserver
|
2019-09-09 20:15:52 -06:00
|
|
|
|
2019-09-11 20:10:38 -06:00
|
|
|
logScope:
|
2020-06-10 11:48:01 +03:00
|
|
|
topics = "pubsub"
|
2019-09-11 20:10:38 -06:00
|
|
|
|
2020-06-07 16:15:21 +09:00
|
|
|
declareGauge(libp2p_pubsub_peers, "pubsub peer instances")
|
|
|
|
declareGauge(libp2p_pubsub_topics, "pubsub subscribed topics")
|
2020-06-07 16:41:23 +09:00
|
|
|
declareCounter(libp2p_pubsub_validation_success, "pubsub successfully validated messages")
|
|
|
|
declareCounter(libp2p_pubsub_validation_failure, "pubsub failed validated messages")
|
2020-08-05 01:27:59 +02:00
|
|
|
when defined(libp2p_expensive_metrics):
|
|
|
|
declarePublicCounter(libp2p_pubsub_messages_published, "published messages", labels = ["topic"])
|
2020-06-07 16:15:21 +09:00
|
|
|
|
2019-09-09 20:15:52 -06:00
|
|
|
type
|
2019-12-16 23:24:03 -06:00
|
|
|
TopicHandler* = proc(topic: string,
|
|
|
|
data: seq[byte]): Future[void] {.gcsafe.}
|
|
|
|
|
|
|
|
ValidatorHandler* = proc(topic: string,
|
2020-03-24 09:48:05 +02:00
|
|
|
message: Message): Future[bool] {.gcsafe, closure.}
|
2019-09-24 10:16:39 -06:00
|
|
|
|
|
|
|
TopicPair* = tuple[topic: string, handler: TopicHandler]
|
2019-09-11 20:10:38 -06:00
|
|
|
|
2020-06-28 17:56:38 +02:00
|
|
|
MsgIdProvider* =
|
|
|
|
proc(m: Message): string {.noSideEffect, raises: [Defect], nimcall, gcsafe.}
|
|
|
|
|
2019-09-09 20:15:52 -06:00
|
|
|
Topic* = object
|
|
|
|
name*: string
|
2019-09-24 10:16:39 -06:00
|
|
|
handler*: seq[TopicHandler]
|
2019-09-09 20:15:52 -06:00
|
|
|
|
|
|
|
PubSub* = ref object of LPProtocol
|
2020-08-11 18:05:49 -06:00
|
|
|
switch*: Switch # the switch used to dial/connect to peers
|
2020-07-17 13:46:24 -06:00
|
|
|
peerInfo*: PeerInfo # this peer's info
|
|
|
|
topics*: Table[string, Topic] # local topics
|
2020-08-11 18:05:49 -06:00
|
|
|
peers*: Table[PeerID, PubSubPeer] # peerid to peer map
|
2020-07-17 13:46:24 -06:00
|
|
|
triggerSelf*: bool # trigger own local handler on publish
|
|
|
|
verifySignature*: bool # enable signature verification
|
|
|
|
sign*: bool # enable message signing
|
2019-12-05 20:16:18 -06:00
|
|
|
cleanupLock: AsyncLock
|
2019-12-16 23:24:03 -06:00
|
|
|
validators*: Table[string, HashSet[ValidatorHandler]]
|
2020-08-11 18:05:49 -06:00
|
|
|
observers: ref seq[PubSubObserver] # ref as in smart_ptr
|
|
|
|
msgIdProvider*: MsgIdProvider # Turn message into message id (not nil)
|
2020-07-15 12:51:33 +09:00
|
|
|
msgSeqno*: uint64
|
2020-08-11 18:05:49 -06:00
|
|
|
lifetimeFut*: Future[void] # pubsub liftime future
|
2019-09-09 20:15:52 -06:00
|
|
|
|
2020-08-11 18:05:49 -06:00
|
|
|
method unsubscribePeer*(p: PubSub, peerId: PeerID) {.base.} =
|
2020-07-07 18:33:05 -06:00
|
|
|
## handle peer disconnects
|
|
|
|
##
|
|
|
|
|
2020-08-11 18:05:49 -06:00
|
|
|
trace "unsubscribing pubsub peer", peer = $peerId
|
|
|
|
if peerId in p.peers:
|
|
|
|
p.peers.del(peerId)
|
2020-07-07 18:33:05 -06:00
|
|
|
|
2020-08-11 18:05:49 -06:00
|
|
|
libp2p_pubsub_peers.set(p.peers.len.int64)
|
2020-07-17 13:46:24 -06:00
|
|
|
|
2020-08-11 18:05:49 -06:00
|
|
|
proc send*(
|
|
|
|
p: PubSub,
|
|
|
|
peer: PubSubPeer,
|
|
|
|
msg: RPCMsg,
|
|
|
|
timeout: Duration) {.async.} =
|
|
|
|
## send to remote peer
|
|
|
|
##
|
2020-07-17 13:46:24 -06:00
|
|
|
|
2020-08-15 21:50:31 +02:00
|
|
|
trace "sending pubsub message to peer", peer = $peer, msg = shortLog(msg)
|
2020-08-11 18:05:49 -06:00
|
|
|
try:
|
|
|
|
await peer.send(msg, timeout)
|
2020-07-17 13:46:24 -06:00
|
|
|
except CancelledError as exc:
|
|
|
|
raise exc
|
|
|
|
except CatchableError as exc:
|
2020-08-15 21:50:31 +02:00
|
|
|
trace "exception sending pubsub message to peer",
|
|
|
|
peer = $peer, msg = shortLog(msg)
|
2020-08-11 18:05:49 -06:00
|
|
|
p.unsubscribePeer(peer.peerId)
|
|
|
|
raise exc
|
|
|
|
|
|
|
|
proc broadcast*(
|
|
|
|
p: PubSub,
|
|
|
|
sendPeers: seq[PubSubPeer],
|
|
|
|
msg: RPCMsg,
|
|
|
|
timeout: Duration): Future[int] {.async.} =
|
|
|
|
## send messages and cleanup failed peers
|
|
|
|
##
|
|
|
|
|
2020-08-15 21:50:31 +02:00
|
|
|
trace "broadcasting messages to peers",
|
|
|
|
peers = sendPeers.len, message = shortLog(msg)
|
2020-08-11 18:05:49 -06:00
|
|
|
let sent = await allFinished(
|
|
|
|
sendPeers.mapIt( p.send(it, msg, timeout) ))
|
|
|
|
return sent.filterIt( it.finished and it.error.isNil ).len
|
2020-07-17 13:46:24 -06:00
|
|
|
|
2019-12-05 20:16:18 -06:00
|
|
|
proc sendSubs*(p: PubSub,
|
|
|
|
peer: PubSubPeer,
|
|
|
|
topics: seq[string],
|
2019-12-16 23:24:03 -06:00
|
|
|
subscribe: bool) {.async.} =
|
2019-12-05 20:16:18 -06:00
|
|
|
## send subscriptions to remote peer
|
2020-08-11 18:05:49 -06:00
|
|
|
await p.send(
|
|
|
|
peer,
|
|
|
|
RPCMsg(
|
|
|
|
subscriptions: topics.mapIt(SubOpts(subscribe: subscribe, topic: it))),
|
|
|
|
DefaultSendTimeout)
|
2019-12-05 20:16:18 -06:00
|
|
|
|
2019-12-16 23:24:03 -06:00
|
|
|
method subscribeTopic*(p: PubSub,
|
|
|
|
topic: string,
|
|
|
|
subscribe: bool,
|
2020-08-17 12:10:22 +02:00
|
|
|
peer: PubSubPeer) {.base.} =
|
2020-07-16 12:06:57 +02:00
|
|
|
# called when remote peer subscribes to a topic
|
2020-07-07 18:33:05 -06:00
|
|
|
discard
|
2019-12-16 23:24:03 -06:00
|
|
|
|
2019-12-05 20:16:18 -06:00
|
|
|
method rpcHandler*(p: PubSub,
|
|
|
|
peer: PubSubPeer,
|
2019-12-16 23:24:03 -06:00
|
|
|
rpcMsgs: seq[RPCMsg]) {.async, base.} =
|
2019-12-05 20:16:18 -06:00
|
|
|
## handle rpc messages
|
2020-03-23 15:03:36 +09:00
|
|
|
trace "processing RPC message", peer = peer.id, msgs = rpcMsgs.len
|
2020-04-30 22:22:31 +09:00
|
|
|
|
2019-12-16 23:24:03 -06:00
|
|
|
for m in rpcMsgs: # for all RPC messages
|
2020-03-23 15:03:36 +09:00
|
|
|
trace "processing messages", msg = m.shortLog
|
2019-12-16 23:24:03 -06:00
|
|
|
if m.subscriptions.len > 0: # if there are any subscriptions
|
|
|
|
for s in m.subscriptions: # subscribe/unsubscribe the peer for each topic
|
2020-05-27 12:33:49 -06:00
|
|
|
trace "about to subscribe to topic", topicId = s.topic
|
2020-08-17 12:10:22 +02:00
|
|
|
p.subscribeTopic(s.topic, s.subscribe, peer)
|
2019-12-16 23:24:03 -06:00
|
|
|
|
2020-08-11 18:05:49 -06:00
|
|
|
proc getOrCreatePeer*(
|
|
|
|
p: PubSub,
|
|
|
|
peer: PeerID,
|
|
|
|
proto: string): PubSubPeer =
|
|
|
|
if peer in p.peers:
|
|
|
|
return p.peers[peer]
|
2019-12-05 20:16:18 -06:00
|
|
|
|
|
|
|
# create new pubsub peer
|
2020-08-11 18:05:49 -06:00
|
|
|
let pubSubPeer = newPubSubPeer(peer, p.switch, proto)
|
|
|
|
trace "created new pubsub peer", peerId = $peer
|
2019-12-05 20:16:18 -06:00
|
|
|
|
2020-08-11 18:05:49 -06:00
|
|
|
p.peers[peer] = pubSubPeer
|
|
|
|
pubSubPeer.observers = p.observers
|
2020-07-13 16:15:27 +02:00
|
|
|
|
|
|
|
libp2p_pubsub_peers.set(p.peers.len.int64)
|
2020-08-11 18:05:49 -06:00
|
|
|
return pubSubPeer
|
2019-12-05 20:16:18 -06:00
|
|
|
|
|
|
|
method handleConn*(p: PubSub,
|
|
|
|
conn: Connection,
|
2019-12-16 23:24:03 -06:00
|
|
|
proto: string) {.base, async.} =
|
2019-12-05 20:16:18 -06:00
|
|
|
## handle incoming connections
|
|
|
|
##
|
|
|
|
## this proc will:
|
|
|
|
## 1) register a new PubSubPeer for the connection
|
|
|
|
## 2) register a handler with the peer;
|
|
|
|
## this handler gets called on every rpc message
|
|
|
|
## that the peer receives
|
|
|
|
## 3) ask the peer to subscribe us to every topic
|
|
|
|
## that we're interested in
|
|
|
|
##
|
|
|
|
|
2020-07-07 18:33:05 -06:00
|
|
|
if isNil(conn.peerInfo):
|
|
|
|
trace "no valid PeerId for peer"
|
|
|
|
await conn.close()
|
|
|
|
return
|
2020-05-21 09:01:36 -06:00
|
|
|
|
2020-07-07 18:33:05 -06:00
|
|
|
proc handler(peer: PubSubPeer, msgs: seq[RPCMsg]) {.async.} =
|
|
|
|
# call pubsub rpc handler
|
|
|
|
await p.rpcHandler(peer, msgs)
|
2020-05-21 09:01:36 -06:00
|
|
|
|
2020-08-11 18:05:49 -06:00
|
|
|
let peer = p.getOrCreatePeer(conn.peerInfo.peerId, proto)
|
2020-08-02 12:22:49 +02:00
|
|
|
if p.topics.len > 0:
|
|
|
|
await p.sendSubs(peer, toSeq(p.topics.keys), true)
|
2020-05-21 09:01:36 -06:00
|
|
|
|
2020-07-07 18:33:05 -06:00
|
|
|
try:
|
2020-05-21 09:01:36 -06:00
|
|
|
peer.handler = handler
|
|
|
|
await peer.handle(conn) # spawn peer read loop
|
2020-07-07 18:33:05 -06:00
|
|
|
trace "pubsub peer handler ended", peer = peer.id
|
2020-06-29 09:15:31 -06:00
|
|
|
except CancelledError as exc:
|
|
|
|
raise exc
|
2020-05-21 09:01:36 -06:00
|
|
|
except CatchableError as exc:
|
|
|
|
trace "exception ocurred in pubsub handle", exc = exc.msg
|
2020-07-07 18:33:05 -06:00
|
|
|
finally:
|
2020-06-29 09:15:31 -06:00
|
|
|
await conn.close()
|
2020-04-07 09:49:43 -06:00
|
|
|
|
2020-08-11 18:05:49 -06:00
|
|
|
method subscribePeer*(p: PubSub, peer: PeerID) {.base.} =
|
|
|
|
## subscribe to remote peer to receive/send pubsub
|
|
|
|
## messages
|
|
|
|
##
|
2020-07-07 18:33:05 -06:00
|
|
|
|
2020-08-11 18:05:49 -06:00
|
|
|
let pubsubPeer = p.getOrCreatePeer(peer, p.codec)
|
|
|
|
if p.topics.len > 0:
|
|
|
|
asyncCheck p.sendSubs(pubsubPeer, toSeq(p.topics.keys), true)
|
2019-12-05 20:16:18 -06:00
|
|
|
|
2020-08-11 18:05:49 -06:00
|
|
|
pubsubPeer.subscribed = true
|
2019-12-05 20:16:18 -06:00
|
|
|
|
2019-09-24 10:16:39 -06:00
|
|
|
method unsubscribe*(p: PubSub,
|
2019-12-16 23:24:03 -06:00
|
|
|
topics: seq[TopicPair]) {.base, async.} =
|
2019-09-09 20:15:52 -06:00
|
|
|
## unsubscribe from a list of ``topic`` strings
|
2019-09-24 10:16:39 -06:00
|
|
|
for t in topics:
|
|
|
|
for i, h in p.topics[t.topic].handler:
|
|
|
|
if h == t.handler:
|
|
|
|
p.topics[t.topic].handler.del(i)
|
|
|
|
|
2020-07-21 01:16:13 +09:00
|
|
|
# make sure we delete the topic if
|
|
|
|
# no more handlers are left
|
|
|
|
if p.topics[t.topic].handler.len <= 0:
|
|
|
|
p.topics.del(t.topic)
|
|
|
|
# metrics
|
2020-07-27 13:33:51 -06:00
|
|
|
libp2p_pubsub_topics.set(p.topics.len.int64)
|
2020-07-09 14:21:47 -06:00
|
|
|
|
2020-07-16 21:26:57 +02:00
|
|
|
proc unsubscribe*(p: PubSub,
|
2020-07-27 13:33:51 -06:00
|
|
|
topic: string,
|
|
|
|
handler: TopicHandler): Future[void] =
|
2019-09-24 10:16:39 -06:00
|
|
|
## unsubscribe from a ``topic`` string
|
2020-07-27 13:33:51 -06:00
|
|
|
##
|
|
|
|
|
2020-06-07 16:15:21 +09:00
|
|
|
p.unsubscribe(@[(topic, handler)])
|
2019-09-09 20:15:52 -06:00
|
|
|
|
2020-07-21 01:16:13 +09:00
|
|
|
method unsubscribeAll*(p: PubSub, topic: string) {.base, async.} =
|
|
|
|
p.topics.del(topic)
|
2020-07-27 13:33:51 -06:00
|
|
|
libp2p_pubsub_topics.set(p.topics.len.int64)
|
2020-07-21 01:16:13 +09:00
|
|
|
|
2019-09-24 10:16:39 -06:00
|
|
|
method subscribe*(p: PubSub,
|
|
|
|
topic: string,
|
2019-12-16 23:24:03 -06:00
|
|
|
handler: TopicHandler) {.base, async.} =
|
2019-09-09 20:15:52 -06:00
|
|
|
## subscribe to a topic
|
|
|
|
##
|
|
|
|
## ``topic`` - a string topic to subscribe to
|
|
|
|
##
|
2019-12-10 14:50:35 -06:00
|
|
|
## ``handler`` - is a user provided proc
|
|
|
|
## that will be triggered
|
2019-09-09 20:15:52 -06:00
|
|
|
## on every received message
|
|
|
|
##
|
2019-12-05 20:16:18 -06:00
|
|
|
if topic notin p.topics:
|
2019-09-28 13:55:35 -06:00
|
|
|
trace "subscribing to topic", name = topic
|
2019-09-24 10:16:39 -06:00
|
|
|
p.topics[topic] = Topic(name: topic)
|
2019-09-28 13:55:35 -06:00
|
|
|
|
2019-09-24 10:16:39 -06:00
|
|
|
p.topics[topic].handler.add(handler)
|
2019-09-09 20:15:52 -06:00
|
|
|
|
2020-07-17 09:36:48 -06:00
|
|
|
var sent: seq[Future[void]]
|
2020-07-07 18:33:05 -06:00
|
|
|
for peer in toSeq(p.peers.values):
|
2020-07-17 09:36:48 -06:00
|
|
|
sent.add(p.sendSubs(peer, @[topic], true))
|
|
|
|
|
|
|
|
checkFutures(await allFinished(sent))
|
2019-12-05 20:16:18 -06:00
|
|
|
|
2020-06-07 16:15:21 +09:00
|
|
|
# metrics
|
2020-07-27 13:33:51 -06:00
|
|
|
libp2p_pubsub_topics.set(p.topics.len.int64)
|
2020-06-07 16:15:21 +09:00
|
|
|
|
2019-12-05 20:16:18 -06:00
|
|
|
method publish*(p: PubSub,
|
|
|
|
topic: string,
|
2020-08-02 23:20:11 -06:00
|
|
|
data: seq[byte],
|
|
|
|
timeout: Duration = InfiniteDuration): Future[int] {.base, async.} =
|
2019-09-09 20:15:52 -06:00
|
|
|
## publish to a ``topic``
|
2019-10-03 16:22:49 -06:00
|
|
|
if p.triggerSelf and topic in p.topics:
|
|
|
|
for h in p.topics[topic].handler:
|
2019-12-23 12:45:12 -06:00
|
|
|
trace "triggering handler", topicID = topic
|
2020-05-15 05:56:56 +02:00
|
|
|
try:
|
|
|
|
await h(topic, data)
|
|
|
|
except CancelledError as exc:
|
|
|
|
raise exc
|
|
|
|
except CatchableError as exc:
|
|
|
|
# TODO these exceptions are ignored since it's likely that if writes are
|
|
|
|
# are failing, the underlying connection is already closed - this needs
|
|
|
|
# more cleanup though
|
|
|
|
debug "Could not write to pubsub connection", msg = exc.msg
|
2019-10-03 16:22:49 -06:00
|
|
|
|
2020-07-07 18:33:05 -06:00
|
|
|
return 0
|
|
|
|
|
2019-12-05 20:16:18 -06:00
|
|
|
method initPubSub*(p: PubSub) {.base.} =
|
2020-05-27 12:33:49 -06:00
|
|
|
## perform pubsub initialization
|
2020-04-30 22:22:31 +09:00
|
|
|
p.observers = new(seq[PubSubObserver])
|
2020-06-28 17:56:38 +02:00
|
|
|
if p.msgIdProvider == nil:
|
|
|
|
p.msgIdProvider = defaultMsgIdProvider
|
2019-12-05 20:16:18 -06:00
|
|
|
|
|
|
|
method start*(p: PubSub) {.async, base.} =
|
|
|
|
## start pubsub
|
|
|
|
discard
|
|
|
|
|
|
|
|
method stop*(p: PubSub) {.async, base.} =
|
|
|
|
## stopt pubsub
|
2019-09-09 20:15:52 -06:00
|
|
|
discard
|
2019-10-03 16:22:49 -06:00
|
|
|
|
2019-12-16 23:24:03 -06:00
|
|
|
method addValidator*(p: PubSub,
|
|
|
|
topic: varargs[string],
|
|
|
|
hook: ValidatorHandler) {.base.} =
|
|
|
|
for t in topic:
|
|
|
|
if t notin p.validators:
|
|
|
|
p.validators[t] = initHashSet[ValidatorHandler]()
|
|
|
|
|
|
|
|
trace "adding validator for topic", topicId = t
|
|
|
|
p.validators[t].incl(hook)
|
|
|
|
|
|
|
|
method removeValidator*(p: PubSub,
|
|
|
|
topic: varargs[string],
|
|
|
|
hook: ValidatorHandler) {.base.} =
|
|
|
|
for t in topic:
|
|
|
|
if t in p.validators:
|
|
|
|
p.validators[t].excl(hook)
|
|
|
|
|
|
|
|
method validate*(p: PubSub, message: Message): Future[bool] {.async, base.} =
|
|
|
|
var pending: seq[Future[bool]]
|
|
|
|
trace "about to validate message"
|
|
|
|
for topic in message.topicIDs:
|
|
|
|
trace "looking for validators on topic", topicID = topic,
|
|
|
|
registered = toSeq(p.validators.keys)
|
|
|
|
if topic in p.validators:
|
|
|
|
trace "running validators for topic", topicID = topic
|
|
|
|
# TODO: add timeout to validator
|
|
|
|
pending.add(p.validators[topic].mapIt(it(topic, message)))
|
|
|
|
|
2020-04-11 13:08:25 +09:00
|
|
|
let futs = await allFinished(pending)
|
|
|
|
result = futs.allIt(not it.failed and it.read())
|
2020-06-07 16:15:21 +09:00
|
|
|
if result:
|
|
|
|
libp2p_pubsub_validation_success.inc()
|
|
|
|
else:
|
|
|
|
libp2p_pubsub_validation_failure.inc()
|
2019-12-16 23:24:03 -06:00
|
|
|
|
2020-08-11 18:05:49 -06:00
|
|
|
proc init*(
|
|
|
|
P: typedesc[PubSub],
|
|
|
|
switch: Switch,
|
|
|
|
triggerSelf: bool = false,
|
|
|
|
verifySignature: bool = true,
|
|
|
|
sign: bool = true,
|
|
|
|
msgIdProvider: MsgIdProvider = defaultMsgIdProvider): P =
|
|
|
|
result = P(switch: switch,
|
|
|
|
peerInfo: switch.peerInfo,
|
2020-05-06 03:26:08 -06:00
|
|
|
triggerSelf: triggerSelf,
|
|
|
|
verifySignature: verifySignature,
|
|
|
|
sign: sign,
|
2020-08-11 18:05:49 -06:00
|
|
|
peers: initTable[PeerID, PubSubPeer](),
|
|
|
|
topics: initTable[string, Topic](),
|
2020-06-28 17:56:38 +02:00
|
|
|
cleanupLock: newAsyncLock(),
|
|
|
|
msgIdProvider: msgIdProvider)
|
2019-10-03 16:22:49 -06:00
|
|
|
result.initPubSub()
|
2020-04-30 22:22:31 +09:00
|
|
|
|
2020-06-29 09:15:31 -06:00
|
|
|
proc addObserver*(p: PubSub; observer: PubSubObserver) =
|
|
|
|
p.observers[] &= observer
|
2020-04-30 22:22:31 +09:00
|
|
|
|
|
|
|
proc removeObserver*(p: PubSub; observer: PubSubObserver) =
|
|
|
|
let idx = p.observers[].find(observer)
|
|
|
|
if idx != -1:
|
|
|
|
p.observers[].del(idx)
|