2022-07-01 20:19:57 +02:00
|
|
|
# Nim-LibP2P
|
2023-01-20 15:47:40 +01:00
|
|
|
# Copyright (c) 2023 Status Research & Development GmbH
|
2022-07-01 20:19:57 +02: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.
|
2019-09-09 20:14:24 -06:00
|
|
|
|
2023-06-07 13:12:49 +02:00
|
|
|
{.push raises: [].}
|
2021-05-21 10:27:01 -06:00
|
|
|
|
2023-03-06 16:36:10 +01:00
|
|
|
import std/[sets, hashes, tables]
|
2022-06-16 10:08:52 +02:00
|
|
|
import chronos, chronicles, metrics
|
2020-09-04 08:10:32 +02:00
|
|
|
import
|
|
|
|
./pubsub,
|
|
|
|
./pubsubpeer,
|
|
|
|
./timedcache,
|
|
|
|
./peertable,
|
2023-09-22 16:45:08 +02:00
|
|
|
./rpc/[message, messages, protobuf],
|
2024-05-01 18:38:24 +02:00
|
|
|
nimcrypto/[hash, sha2],
|
2021-10-25 10:26:32 +02:00
|
|
|
../../crypto/crypto,
|
2020-06-19 11:29:43 -06:00
|
|
|
../../stream/connection,
|
2020-07-01 15:25:09 +09:00
|
|
|
../../peerid,
|
2020-09-04 08:10:32 +02:00
|
|
|
../../peerinfo,
|
|
|
|
../../utility
|
2019-09-09 20:14:24 -06:00
|
|
|
|
2022-07-01 20:19:57 +02:00
|
|
|
## Simple flood-based publishing.
|
|
|
|
|
2019-09-09 20:14:24 -06:00
|
|
|
logScope:
|
2020-12-01 11:34:27 -06:00
|
|
|
topics = "libp2p floodsub"
|
2019-09-09 20:14:24 -06:00
|
|
|
|
|
|
|
const FloodSubCodec* = "/floodsub/1.0.0"
|
|
|
|
|
2022-07-01 20:19:57 +02:00
|
|
|
type FloodSub* {.public.} = ref object of PubSub
|
2020-07-15 13:18:55 -06:00
|
|
|
floodsub*: PeerTable # topic to remote peer map
|
2024-05-01 18:38:24 +02:00
|
|
|
seen*: TimedCache[SaltedId]
|
|
|
|
# Early filter for messages recently observed on the network
|
|
|
|
# We use a salted id because the messages in this cache have not yet
|
|
|
|
# been validated meaning that an attacker has greater control over the
|
|
|
|
# hash key and therefore could poison the table
|
|
|
|
seenSalt*: sha256
|
|
|
|
# The salt in this case is a partially updated SHA256 context pre-seeded
|
|
|
|
# with some random data
|
|
|
|
|
|
|
|
proc salt*(f: FloodSub, msgId: MessageId): SaltedId =
|
|
|
|
var tmp = f.seenSalt
|
|
|
|
tmp.update(msgId)
|
|
|
|
SaltedId(data: tmp.finish())
|
|
|
|
|
|
|
|
proc hasSeen*(f: FloodSub, saltedId: SaltedId): bool =
|
|
|
|
saltedId in f.seen
|
|
|
|
|
|
|
|
proc addSeen*(f: FloodSub, saltedId: SaltedId): bool =
|
2021-04-18 10:08:33 +02:00
|
|
|
# Return true if the message has already been seen
|
2024-05-01 18:38:24 +02:00
|
|
|
f.seen.put(saltedId)
|
2019-12-05 20:16:18 -06:00
|
|
|
|
2024-05-01 18:38:24 +02:00
|
|
|
proc firstSeen*(f: FloodSub, saltedId: SaltedId): Moment =
|
|
|
|
f.seen.addedAt(saltedId)
|
2022-05-11 10:38:43 +02:00
|
|
|
|
2024-05-15 18:57:15 +02:00
|
|
|
proc handleSubscribe(f: FloodSub, peer: PubSubPeer, topic: string, subscribe: bool) =
|
2020-12-16 03:32:11 +09:00
|
|
|
logScope:
|
|
|
|
peer
|
|
|
|
topic
|
|
|
|
|
2020-12-19 23:43:32 +09:00
|
|
|
# this is a workaround for a race condition
|
2020-12-16 03:32:11 +09:00
|
|
|
# that can happen if we disconnect the peer very early
|
2020-12-19 23:43:32 +09:00
|
|
|
# in the future we might use this as a test case
|
2020-12-16 03:32:11 +09:00
|
|
|
# and eventually remove this workaround
|
|
|
|
if subscribe and peer.peerId notin f.peers:
|
|
|
|
trace "ignoring unknown peer"
|
|
|
|
return
|
|
|
|
|
2021-01-13 23:49:44 +09:00
|
|
|
if subscribe and not (isNil(f.subscriptionValidator)) and
|
|
|
|
not (f.subscriptionValidator(topic)):
|
|
|
|
# this is a violation, so warn should be in order
|
|
|
|
warn "ignoring invalid topic subscription", topic, peer
|
|
|
|
return
|
2019-12-10 14:50:35 -06:00
|
|
|
|
|
|
|
if subscribe:
|
2020-09-06 10:31:47 +02:00
|
|
|
trace "adding subscription for topic", peer, topic
|
2021-01-13 23:49:44 +09:00
|
|
|
|
2019-12-10 14:50:35 -06:00
|
|
|
# subscribe the peer to the topic
|
2021-05-07 00:43:45 +02:00
|
|
|
f.floodsub.mgetOrPut(topic, HashSet[PubSubPeer]()).incl(peer)
|
2019-12-10 14:50:35 -06:00
|
|
|
else:
|
2021-05-07 00:43:45 +02:00
|
|
|
f.floodsub.withValue(topic, peers):
|
|
|
|
trace "removing subscription for topic", peer, topic
|
2021-01-13 23:49:44 +09:00
|
|
|
|
2021-05-07 00:43:45 +02:00
|
|
|
# unsubscribe the peer from the topic
|
|
|
|
peers[].excl(peer)
|
2019-12-10 14:50:35 -06:00
|
|
|
|
2021-12-16 11:05:20 +01:00
|
|
|
method unsubscribePeer*(f: FloodSub, peer: PeerId) =
|
2019-12-05 20:16:18 -06:00
|
|
|
## handle peer disconnects
|
2020-08-06 11:12:52 +09:00
|
|
|
##
|
2020-09-06 10:31:47 +02:00
|
|
|
trace "unsubscribing floodsub peer", peer
|
2020-08-11 18:05:49 -06:00
|
|
|
let pubSubPeer = f.peers.getOrDefault(peer)
|
|
|
|
if pubSubPeer.isNil:
|
|
|
|
return
|
|
|
|
|
2020-09-04 08:10:32 +02:00
|
|
|
for _, v in f.floodsub.mpairs():
|
|
|
|
v.excl(pubSubPeer)
|
2020-08-11 18:05:49 -06:00
|
|
|
|
|
|
|
procCall PubSub(f).unsubscribePeer(peer)
|
2020-08-06 11:12:52 +09:00
|
|
|
|
2019-12-05 20:16:18 -06:00
|
|
|
method rpcHandler*(f: FloodSub, peer: PubSubPeer, data: seq[byte]) {.async.} =
|
2023-09-22 16:45:08 +02:00
|
|
|
var rpcMsg = decodeRpcMsg(data).valueOr:
|
|
|
|
debug "failed to decode msg from peer", peer, err = error
|
2024-05-15 18:57:15 +02:00
|
|
|
raise newException(CatchableError, "Peer msg couldn't be decoded")
|
2023-09-22 16:45:08 +02:00
|
|
|
|
2024-08-14 17:19:54 +02:00
|
|
|
trace "decoded msg from peer", peer, payload = rpcMsg.shortLog
|
2023-09-22 16:45:08 +02:00
|
|
|
# trigger hooks
|
|
|
|
peer.recvObservers(rpcMsg)
|
|
|
|
|
2021-05-07 00:43:45 +02:00
|
|
|
for i in 0 ..< min(f.topicsHigh, rpcMsg.subscriptions.len):
|
|
|
|
template sub(): untyped =
|
|
|
|
rpcMsg.subscriptions[i]
|
2024-06-11 17:18:06 +02:00
|
|
|
|
2021-05-07 00:43:45 +02:00
|
|
|
f.handleSubscribe(peer, sub.topic, sub.subscribe)
|
2020-09-01 09:33:03 +02:00
|
|
|
|
2020-09-04 08:10:32 +02:00
|
|
|
for msg in rpcMsg.messages: # for every message
|
2022-02-22 02:04:17 +11:00
|
|
|
let msgIdResult = f.msgIdProvider(msg)
|
|
|
|
if msgIdResult.isErr:
|
|
|
|
debug "Dropping message due to failed message id generation",
|
|
|
|
error = msgIdResult.error
|
|
|
|
# TODO: descore peers due to error during message validation (malicious?)
|
|
|
|
continue
|
|
|
|
|
2024-05-01 18:38:24 +02:00
|
|
|
let
|
|
|
|
msgId = msgIdResult.get
|
|
|
|
saltedId = f.salt(msgId)
|
2020-09-01 09:33:03 +02:00
|
|
|
|
2024-05-01 18:38:24 +02:00
|
|
|
if f.addSeen(saltedId):
|
2020-09-06 10:31:47 +02:00
|
|
|
trace "Dropping already-seen message", msgId, peer
|
2020-09-04 08:10:32 +02:00
|
|
|
continue
|
2020-09-01 09:33:03 +02:00
|
|
|
|
2020-09-24 00:56:33 +09:00
|
|
|
if (msg.signature.len > 0 or f.verifySignature) and not msg.verify():
|
|
|
|
# always validate if signature is present or required
|
2020-09-06 10:31:47 +02:00
|
|
|
debug "Dropping message due to failed signature verification", msgId, peer
|
2020-09-04 08:10:32 +02:00
|
|
|
continue
|
2020-09-01 09:33:03 +02:00
|
|
|
|
2020-09-24 00:56:33 +09:00
|
|
|
if msg.seqno.len > 0 and msg.seqno.len != 8:
|
|
|
|
# if we have seqno should be 8 bytes long
|
|
|
|
debug "Dropping message due to invalid seqno length", msgId, peer
|
|
|
|
continue
|
|
|
|
|
|
|
|
# g.anonymize needs no evaluation when receiving messages
|
|
|
|
# as we have a "lax" policy and allow signed messages
|
|
|
|
|
2020-10-21 12:25:42 +09:00
|
|
|
let validation = await f.validate(msg)
|
|
|
|
case validation
|
2020-11-12 01:42:12 +09:00
|
|
|
of ValidationResult.Reject:
|
|
|
|
debug "Dropping message after validation, reason: reject", msgId, peer
|
|
|
|
continue
|
|
|
|
of ValidationResult.Ignore:
|
|
|
|
debug "Dropping message after validation, reason: ignore", msgId, peer
|
2020-09-04 08:10:32 +02:00
|
|
|
continue
|
2020-10-21 12:25:42 +09:00
|
|
|
of ValidationResult.Accept:
|
|
|
|
discard
|
2020-09-01 09:33:03 +02:00
|
|
|
|
2020-09-04 08:10:32 +02:00
|
|
|
var toSendPeers = initHashSet[PubSubPeer]()
|
2024-03-25 12:06:34 +01:00
|
|
|
let topic = msg.topic
|
|
|
|
if topic notin f.topics:
|
|
|
|
debug "Dropping message due to topic not in floodsub topics", topic, msgId, peer
|
|
|
|
continue
|
|
|
|
|
|
|
|
f.floodsub.withValue(topic, peers):
|
|
|
|
toSendPeers.incl(peers[])
|
2020-09-01 09:33:03 +02:00
|
|
|
|
2024-03-25 12:06:34 +01:00
|
|
|
await handleData(f, topic, msg.data)
|
2020-09-01 09:33:03 +02:00
|
|
|
|
2020-09-04 08:10:32 +02:00
|
|
|
# In theory, if topics are the same in all messages, we could batch - we'd
|
|
|
|
# also have to be careful to only include validated messages
|
2024-03-05 16:05:21 +01:00
|
|
|
f.broadcast(toSendPeers, RPCMsg(messages: @[msg]), isHighPriority = false)
|
2020-09-04 08:10:32 +02:00
|
|
|
trace "Forwared message to peers", peers = toSendPeers.len
|
2019-09-09 20:15:52 -06:00
|
|
|
|
2021-05-07 00:43:45 +02:00
|
|
|
f.updateMetrics(rpcMsg)
|
|
|
|
|
2020-06-13 07:54:12 +08:00
|
|
|
method init*(f: FloodSub) =
|
2019-12-16 23:24:03 -06:00
|
|
|
proc handler(conn: Connection, proto: string) {.async.} =
|
2019-09-12 04:08:11 -06:00
|
|
|
## main protocol handler that gets triggered on every
|
2019-09-09 20:15:52 -06:00
|
|
|
## connection for a protocol string
|
|
|
|
## e.g. ``/floodsub/1.0.0``, etc...
|
|
|
|
##
|
2020-09-04 19:30:45 +03:00
|
|
|
try:
|
|
|
|
await f.handleConn(conn, proto)
|
|
|
|
except CancelledError:
|
|
|
|
# This is top-level procedure which will work as separate task, so it
|
2020-11-23 15:02:23 -06:00
|
|
|
# do not need to propagate CancelledError.
|
2020-09-06 10:31:47 +02:00
|
|
|
trace "Unexpected cancellation in floodsub handler", conn
|
2020-09-04 19:30:45 +03:00
|
|
|
except CatchableError as exc:
|
2024-08-14 17:19:54 +02:00
|
|
|
trace "FloodSub handler leaks an error", description = exc.msg, conn
|
2019-09-09 20:14:24 -06:00
|
|
|
|
|
|
|
f.handler = handler
|
2019-09-09 20:15:52 -06:00
|
|
|
f.codec = FloodSubCodec
|
|
|
|
|
2020-09-01 09:33:03 +02:00
|
|
|
method publish*(f: FloodSub, topic: string, data: seq[byte]): Future[int] {.async.} =
|
2020-07-07 18:33:05 -06:00
|
|
|
# base returns always 0
|
2020-09-01 09:33:03 +02:00
|
|
|
discard await procCall PubSub(f).publish(topic, data)
|
2019-10-03 16:22:49 -06:00
|
|
|
|
2020-09-06 10:31:47 +02:00
|
|
|
trace "Publishing message on topic", data = data.shortLog, topic
|
2020-09-04 08:10:32 +02:00
|
|
|
|
|
|
|
if topic.len <= 0: # data could be 0/empty
|
2020-09-06 10:31:47 +02:00
|
|
|
debug "Empty topic, skipping publish", topic
|
2020-09-01 09:33:03 +02:00
|
|
|
return 0
|
2019-12-05 20:16:18 -06:00
|
|
|
|
2021-04-18 10:08:33 +02:00
|
|
|
let peers = f.floodsub.getOrDefault(topic)
|
2020-09-04 08:10:32 +02:00
|
|
|
|
|
|
|
if peers.len == 0:
|
2020-09-06 10:31:47 +02:00
|
|
|
debug "No peers for topic, skipping publish", topic
|
2020-09-04 08:10:32 +02:00
|
|
|
return 0
|
2019-12-05 20:16:18 -06:00
|
|
|
|
2020-09-01 09:33:03 +02:00
|
|
|
let
|
2020-09-24 00:56:33 +09:00
|
|
|
msg =
|
|
|
|
if f.anonymize:
|
|
|
|
Message.init(none(PeerInfo), data, topic, none(uint64), false)
|
|
|
|
else:
|
2022-06-30 09:56:49 +02:00
|
|
|
inc f.msgSeqno
|
2020-09-24 00:56:33 +09:00
|
|
|
Message.init(some(f.peerInfo), data, topic, some(f.msgSeqno), f.sign)
|
2022-06-30 09:56:49 +02:00
|
|
|
msgId = f.msgIdProvider(msg).valueOr:
|
|
|
|
trace "Error generating message id, skipping publish", error = error
|
|
|
|
return 0
|
2020-07-17 13:46:24 -06:00
|
|
|
|
2024-08-14 17:19:54 +02:00
|
|
|
trace "Created new message", payload = shortLog(msg), peers = peers.len, topic, msgId
|
2020-09-04 08:10:32 +02:00
|
|
|
|
2024-05-01 18:38:24 +02:00
|
|
|
if f.addSeen(f.salt(msgId)):
|
2020-09-04 08:10:32 +02:00
|
|
|
# custom msgid providers might cause this
|
2020-09-06 10:31:47 +02:00
|
|
|
trace "Dropping already-seen message", msgId, topic
|
2020-09-04 08:10:32 +02:00
|
|
|
return 0
|
|
|
|
|
|
|
|
# Try to send to all peers that are known to be interested
|
2024-03-05 16:05:21 +01:00
|
|
|
f.broadcast(peers, RPCMsg(messages: @[msg]), isHighPriority = true)
|
2019-09-09 20:14:24 -06:00
|
|
|
|
2020-08-05 01:27:59 +02:00
|
|
|
when defined(libp2p_expensive_metrics):
|
|
|
|
libp2p_pubsub_messages_published.inc(labelValues = [topic])
|
2020-06-16 22:14:02 -06:00
|
|
|
|
2020-09-06 10:31:47 +02:00
|
|
|
trace "Published message to peers", msgId, topic
|
2020-09-04 08:10:32 +02:00
|
|
|
|
2020-09-01 09:33:03 +02:00
|
|
|
return peers.len
|
2020-07-07 18:33:05 -06:00
|
|
|
|
2023-06-07 13:12:49 +02:00
|
|
|
method initPubSub*(f: FloodSub) {.raises: [InitializationError].} =
|
2020-04-30 22:22:31 +09:00
|
|
|
procCall PubSub(f).initPubSub()
|
2024-05-01 18:38:24 +02:00
|
|
|
f.seen = TimedCache[SaltedId].init(2.minutes)
|
|
|
|
f.seenSalt.init()
|
|
|
|
|
|
|
|
var tmp: array[32, byte]
|
|
|
|
hmacDrbgGenerate(f.rng[], tmp)
|
|
|
|
f.seenSalt.update(tmp)
|
2021-04-18 10:08:33 +02:00
|
|
|
|
2019-10-03 16:22:49 -06:00
|
|
|
f.init()
|