2024-06-28 16:04:57 +05:30
|
|
|
|
{.push raises: [].}
|
2022-11-04 09:40:13 +01:00
|
|
|
|
|
2021-02-04 12:32:58 +02:00
|
|
|
|
import
|
2024-01-30 07:28:21 -05:00
|
|
|
|
std/[options, sets, sequtils, times, strutils, math, random],
|
2022-11-24 14:11:23 +01:00
|
|
|
|
chronos,
|
|
|
|
|
chronicles,
|
2022-11-04 09:40:13 +01:00
|
|
|
|
metrics,
|
2023-04-12 13:05:34 +02:00
|
|
|
|
libp2p/multistream,
|
2023-05-31 09:47:56 +02:00
|
|
|
|
libp2p/muxers/muxer,
|
2024-03-19 16:18:52 +01:00
|
|
|
|
libp2p/nameresolving/nameresolver,
|
|
|
|
|
libp2p/peerstore
|
|
|
|
|
|
2022-11-04 09:40:13 +01:00
|
|
|
|
import
|
2023-08-09 18:11:50 +01:00
|
|
|
|
../../common/nimchronos,
|
2023-12-07 07:21:18 -05:00
|
|
|
|
../../common/enr,
|
2023-04-24 16:37:54 +02:00
|
|
|
|
../../waku_core,
|
2023-04-18 15:22:10 +02:00
|
|
|
|
../../waku_relay,
|
2023-09-22 15:13:50 -04:00
|
|
|
|
../../waku_enr/sharding,
|
2024-01-30 07:28:21 -05:00
|
|
|
|
../../waku_enr/capabilities,
|
2023-10-11 08:58:45 +02:00
|
|
|
|
../../waku_metadata,
|
2022-11-04 09:40:13 +01:00
|
|
|
|
./peer_store/peer_storage,
|
|
|
|
|
./waku_peer_store
|
2021-02-04 12:32:58 +02:00
|
|
|
|
|
2021-10-06 14:29:08 +02:00
|
|
|
|
export waku_peer_store, peer_storage, peers
|
2021-02-05 12:49:11 +02:00
|
|
|
|
|
|
|
|
|
declareCounter waku_peers_dials, "Number of peer dials", ["outcome"]
|
2022-12-14 16:04:11 +01:00
|
|
|
|
# TODO: Populate from PeerStore.Source when ready
|
2024-03-16 00:08:47 +01:00
|
|
|
|
declarePublicCounter waku_node_conns_initiated,
|
|
|
|
|
"Number of connections initiated", ["source"]
|
2021-03-26 10:49:51 +02:00
|
|
|
|
declarePublicGauge waku_peers_errors, "Number of peer manager errors", ["type"]
|
2024-03-16 00:08:47 +01:00
|
|
|
|
declarePublicGauge waku_connected_peers,
|
|
|
|
|
"Number of physical connections per direction and protocol",
|
|
|
|
|
labels = ["direction", "protocol"]
|
|
|
|
|
declarePublicGauge waku_streams_peers,
|
|
|
|
|
"Number of streams per direction and protocol", labels = ["direction", "protocol"]
|
2023-01-31 13:24:49 +01:00
|
|
|
|
declarePublicGauge waku_peer_store_size, "Number of peers managed by the peer store"
|
2024-03-16 00:08:47 +01:00
|
|
|
|
declarePublicGauge waku_service_peers,
|
|
|
|
|
"Service peer protocol and multiaddress ", labels = ["protocol", "peerId"]
|
2024-07-26 16:18:14 -04:00
|
|
|
|
declarePublicGauge waku_total_unique_peers, "total number of unique peers"
|
2022-11-04 09:40:13 +01:00
|
|
|
|
|
2021-02-05 12:49:11 +02:00
|
|
|
|
logScope:
|
2022-11-03 16:36:24 +01:00
|
|
|
|
topics = "waku node peer_manager"
|
2021-02-05 12:49:11 +02:00
|
|
|
|
|
2024-01-30 07:28:21 -05:00
|
|
|
|
randomize()
|
|
|
|
|
|
2022-12-14 16:04:11 +01:00
|
|
|
|
const
|
|
|
|
|
# TODO: Make configurable
|
2024-05-13 17:25:44 +02:00
|
|
|
|
DefaultDialTimeout* = chronos.seconds(10)
|
2021-02-05 12:49:11 +02:00
|
|
|
|
|
2023-01-23 21:24:46 +01:00
|
|
|
|
# Max attempts before removing the peer
|
|
|
|
|
MaxFailedAttempts = 5
|
|
|
|
|
|
|
|
|
|
# Time to wait before attempting to dial again is calculated as:
|
|
|
|
|
# initialBackoffInSec*(backoffFactor^(failedAttempts-1))
|
|
|
|
|
# 120s, 480s, 1920, 7680s
|
|
|
|
|
InitialBackoffInSec = 120
|
|
|
|
|
BackoffFactor = 4
|
|
|
|
|
|
2023-01-26 10:20:20 +01:00
|
|
|
|
# Limit the amount of paralel dials
|
2024-01-30 07:28:21 -05:00
|
|
|
|
MaxParallelDials = 10
|
2023-01-18 15:17:56 +01:00
|
|
|
|
|
2023-01-26 10:20:20 +01:00
|
|
|
|
# Delay between consecutive relayConnectivityLoop runs
|
2024-07-09 17:33:18 +03:00
|
|
|
|
ConnectivityLoopInterval = chronos.seconds(30)
|
2023-01-18 15:17:56 +01:00
|
|
|
|
|
2023-01-31 13:24:49 +01:00
|
|
|
|
# How often the peer store is pruned
|
2023-09-08 13:36:26 +02:00
|
|
|
|
PrunePeerStoreInterval = chronos.minutes(10)
|
2023-01-31 13:24:49 +01:00
|
|
|
|
|
2023-05-31 09:47:56 +02:00
|
|
|
|
# How often metrics and logs are shown/updated
|
2023-07-04 13:31:18 +02:00
|
|
|
|
LogAndMetricsInterval = chronos.minutes(3)
|
2023-04-12 13:05:34 +02:00
|
|
|
|
|
2023-05-31 09:47:56 +02:00
|
|
|
|
# Max peers that we allow from the same IP
|
2024-01-02 14:01:18 +01:00
|
|
|
|
DefaultColocationLimit* = 5
|
2023-05-18 09:40:14 +02:00
|
|
|
|
|
2024-03-16 00:08:47 +01:00
|
|
|
|
type PeerManager* = ref object of RootObj
|
|
|
|
|
switch*: Switch
|
2024-09-27 18:16:46 +05:30
|
|
|
|
wakuPeerStore*: WakuPeerStore
|
2024-03-16 00:08:47 +01:00
|
|
|
|
wakuMetadata*: WakuMetadata
|
|
|
|
|
initialBackoffInSec*: int
|
|
|
|
|
backoffFactor*: int
|
|
|
|
|
maxFailedAttempts*: int
|
|
|
|
|
storage*: PeerStorage
|
|
|
|
|
serviceSlots*: Table[string, RemotePeerInfo]
|
|
|
|
|
maxRelayPeers*: int
|
|
|
|
|
outRelayPeersTarget: int
|
|
|
|
|
inRelayPeersTarget: int
|
|
|
|
|
ipTable*: Table[string, seq[PeerId]]
|
|
|
|
|
colocationLimit*: int
|
|
|
|
|
started: bool
|
|
|
|
|
shardedPeerManagement: bool # temp feature flag
|
2023-01-23 21:24:46 +01:00
|
|
|
|
|
2024-10-04 15:23:20 +05:30
|
|
|
|
#~~~~~~~~~~~~~~~~~~~#
|
|
|
|
|
# Helper Functions #
|
|
|
|
|
#~~~~~~~~~~~~~~~~~~~#
|
|
|
|
|
|
|
|
|
|
proc calculateBackoff(
|
|
|
|
|
initialBackoffInSec: int, backoffFactor: int, failedAttempts: int
|
|
|
|
|
): timer.Duration =
|
|
|
|
|
if failedAttempts == 0:
|
|
|
|
|
return chronos.seconds(0)
|
|
|
|
|
return chronos.seconds(initialBackoffInSec * (backoffFactor ^ (failedAttempts - 1)))
|
|
|
|
|
|
2023-02-27 18:24:31 +01:00
|
|
|
|
proc protocolMatcher*(codec: string): Matcher =
|
|
|
|
|
## Returns a protocol matcher function for the provided codec
|
|
|
|
|
proc match(proto: string): bool {.gcsafe.} =
|
|
|
|
|
## Matches a proto with any postfix to the provided codec.
|
|
|
|
|
## E.g. if the codec is `/vac/waku/filter/2.0.0` it matches the protos:
|
|
|
|
|
## `/vac/waku/filter/2.0.0`, `/vac/waku/filter/2.0.0-beta3`, `/vac/waku/filter/2.0.0-actualnonsense`
|
|
|
|
|
return proto.startsWith(codec)
|
|
|
|
|
|
|
|
|
|
return match
|
|
|
|
|
|
2024-10-04 15:23:20 +05:30
|
|
|
|
#~~~~~~~~~~~~~~~~~~~~~~~~~~#
|
|
|
|
|
# Peer Storage Management #
|
|
|
|
|
#~~~~~~~~~~~~~~~~~~~~~~~~~~#
|
2021-03-26 10:49:51 +02:00
|
|
|
|
|
2024-07-09 13:14:28 +02:00
|
|
|
|
proc insertOrReplace(ps: PeerStorage, remotePeerInfo: RemotePeerInfo) {.gcsafe.} =
|
2023-11-27 08:08:58 -05:00
|
|
|
|
## Insert peer entry into persistent storage, or replace existing entry with updated info
|
|
|
|
|
ps.put(remotePeerInfo).isOkOr:
|
|
|
|
|
warn "failed to store peers", err = error
|
2021-03-26 10:49:51 +02:00
|
|
|
|
waku_peers_errors.inc(labelValues = ["storage_failure"])
|
2023-11-27 08:08:58 -05:00
|
|
|
|
return
|
2021-03-26 10:49:51 +02:00
|
|
|
|
|
2024-07-09 13:14:28 +02:00
|
|
|
|
proc addPeer*(
|
|
|
|
|
pm: PeerManager, remotePeerInfo: RemotePeerInfo, origin = UnknownOrigin
|
|
|
|
|
) {.gcsafe.} =
|
2023-11-27 08:08:58 -05:00
|
|
|
|
## Adds peer to manager for the specified protocol
|
2023-03-28 13:29:48 +02:00
|
|
|
|
|
|
|
|
|
if remotePeerInfo.peerId == pm.switch.peerInfo.peerId:
|
2024-05-16 22:30:51 +02:00
|
|
|
|
trace "skipping to manage our unmanageable self"
|
2023-03-28 13:29:48 +02:00
|
|
|
|
return
|
|
|
|
|
|
2024-09-27 18:16:46 +05:30
|
|
|
|
if pm.wakuPeerStore[AddressBook][remotePeerInfo.peerId] == remotePeerInfo.addrs and
|
|
|
|
|
pm.wakuPeerStore[KeyBook][remotePeerInfo.peerId] == remotePeerInfo.publicKey and
|
|
|
|
|
pm.wakuPeerStore[ENRBook][remotePeerInfo.peerId].raw.len > 0:
|
2024-06-19 17:29:55 +02:00
|
|
|
|
let incomingEnr = remotePeerInfo.enr.valueOr:
|
|
|
|
|
trace "peer already managed and incoming ENR is empty",
|
|
|
|
|
remote_peer_id = $remotePeerInfo.peerId
|
|
|
|
|
return
|
|
|
|
|
|
2024-09-27 18:16:46 +05:30
|
|
|
|
if pm.wakuPeerStore[ENRBook][remotePeerInfo.peerId].raw == incomingEnr.raw or
|
|
|
|
|
pm.wakuPeerStore[ENRBook][remotePeerInfo.peerId].seqNum > incomingEnr.seqNum:
|
2024-06-19 17:29:55 +02:00
|
|
|
|
trace "peer already managed and ENR info is already saved",
|
|
|
|
|
remote_peer_id = $remotePeerInfo.peerId
|
|
|
|
|
return
|
2023-03-28 13:29:48 +02:00
|
|
|
|
|
2024-03-16 00:08:47 +01:00
|
|
|
|
trace "Adding peer to manager",
|
|
|
|
|
peerId = remotePeerInfo.peerId, addresses = remotePeerInfo.addrs
|
|
|
|
|
|
2024-07-26 16:18:14 -04:00
|
|
|
|
waku_total_unique_peers.inc()
|
|
|
|
|
|
2024-09-27 18:16:46 +05:30
|
|
|
|
pm.wakuPeerStore[AddressBook][remotePeerInfo.peerId] = remotePeerInfo.addrs
|
|
|
|
|
pm.wakuPeerStore[KeyBook][remotePeerInfo.peerId] = remotePeerInfo.publicKey
|
|
|
|
|
pm.wakuPeerStore[SourceBook][remotePeerInfo.peerId] = origin
|
|
|
|
|
pm.wakuPeerStore[ProtoVersionBook][remotePeerInfo.peerId] =
|
|
|
|
|
remotePeerInfo.protoVersion
|
|
|
|
|
pm.wakuPeerStore[AgentBook][remotePeerInfo.peerId] = remotePeerInfo.agent
|
2024-03-16 00:08:47 +01:00
|
|
|
|
|
2024-01-30 07:28:21 -05:00
|
|
|
|
if remotePeerInfo.protocols.len > 0:
|
2024-09-27 18:16:46 +05:30
|
|
|
|
pm.wakuPeerStore[ProtoBook][remotePeerInfo.peerId] = remotePeerInfo.protocols
|
2024-03-16 00:08:47 +01:00
|
|
|
|
|
2023-04-19 16:12:00 +02:00
|
|
|
|
if remotePeerInfo.enr.isSome():
|
2024-09-27 18:16:46 +05:30
|
|
|
|
pm.wakuPeerStore[ENRBook][remotePeerInfo.peerId] = remotePeerInfo.enr.get()
|
2023-03-28 13:29:48 +02:00
|
|
|
|
|
|
|
|
|
# Add peer to storage. Entry will subsequently be updated with connectedness information
|
|
|
|
|
if not pm.storage.isNil:
|
2024-09-27 18:16:46 +05:30
|
|
|
|
# Reading from the db (pm.storage) is only done on startup, hence you need to connect to all saved peers.
|
2024-05-13 17:25:44 +02:00
|
|
|
|
# `remotePeerInfo.connectedness` should already be `NotConnected`, but both we reset it to `NotConnected` just in case.
|
|
|
|
|
# This reset is also done when reading from storage, I believe, to ensure the `connectedness` state is the correct one.
|
|
|
|
|
# So many resets are likely redudant, but I haven't verified whether this is the case or not.
|
2023-11-27 08:08:58 -05:00
|
|
|
|
remotePeerInfo.connectedness = NotConnected
|
|
|
|
|
|
|
|
|
|
pm.storage.insertOrReplace(remotePeerInfo)
|
2023-03-28 13:29:48 +02:00
|
|
|
|
|
2024-10-04 15:23:20 +05:30
|
|
|
|
proc loadFromStorage(pm: PeerManager) {.gcsafe.} =
|
|
|
|
|
## Load peers from storage, if available
|
|
|
|
|
|
|
|
|
|
trace "loading peers from storage"
|
|
|
|
|
|
|
|
|
|
var amount = 0
|
|
|
|
|
|
|
|
|
|
proc onData(remotePeerInfo: RemotePeerInfo) =
|
|
|
|
|
let peerId = remotePeerInfo.peerId
|
|
|
|
|
|
|
|
|
|
if pm.switch.peerInfo.peerId == peerId:
|
|
|
|
|
# Do not manage self
|
|
|
|
|
return
|
|
|
|
|
|
|
|
|
|
trace "loading peer",
|
|
|
|
|
peerId = peerId,
|
|
|
|
|
address = remotePeerInfo.addrs,
|
|
|
|
|
protocols = remotePeerInfo.protocols,
|
|
|
|
|
agent = remotePeerInfo.agent,
|
|
|
|
|
version = remotePeerInfo.protoVersion
|
|
|
|
|
|
|
|
|
|
# nim-libp2p books
|
|
|
|
|
pm.wakuPeerStore[AddressBook][peerId] = remotePeerInfo.addrs
|
|
|
|
|
pm.wakuPeerStore[ProtoBook][peerId] = remotePeerInfo.protocols
|
|
|
|
|
pm.wakuPeerStore[KeyBook][peerId] = remotePeerInfo.publicKey
|
|
|
|
|
pm.wakuPeerStore[AgentBook][peerId] = remotePeerInfo.agent
|
|
|
|
|
pm.wakuPeerStore[ProtoVersionBook][peerId] = remotePeerInfo.protoVersion
|
|
|
|
|
|
|
|
|
|
# custom books
|
|
|
|
|
pm.wakuPeerStore[ConnectionBook][peerId] = NotConnected # Reset connectedness state
|
|
|
|
|
pm.wakuPeerStore[DisconnectBook][peerId] = remotePeerInfo.disconnectTime
|
|
|
|
|
pm.wakuPeerStore[SourceBook][peerId] = remotePeerInfo.origin
|
|
|
|
|
|
|
|
|
|
if remotePeerInfo.enr.isSome():
|
|
|
|
|
pm.wakuPeerStore[ENRBook][peerId] = remotePeerInfo.enr.get()
|
|
|
|
|
|
|
|
|
|
amount.inc()
|
|
|
|
|
|
|
|
|
|
pm.storage.getAll(onData).isOkOr:
|
|
|
|
|
warn "loading peers from storage failed", err = error
|
|
|
|
|
waku_peers_errors.inc(labelValues = ["storage_load_failure"])
|
|
|
|
|
return
|
|
|
|
|
|
|
|
|
|
trace "recovered peers from storage", amount = amount
|
|
|
|
|
|
|
|
|
|
proc selectPeer*(
|
|
|
|
|
pm: PeerManager, proto: string, shard: Option[PubsubTopic] = none(PubsubTopic)
|
|
|
|
|
): Option[RemotePeerInfo] =
|
|
|
|
|
trace "Selecting peer from peerstore", protocol = proto
|
|
|
|
|
|
|
|
|
|
# Selects the best peer for a given protocol
|
|
|
|
|
var peers = pm.wakuPeerStore.getPeersByProtocol(proto)
|
|
|
|
|
|
|
|
|
|
if shard.isSome():
|
|
|
|
|
peers.keepItIf((it.enr.isSome() and it.enr.get().containsShard(shard.get())))
|
|
|
|
|
|
2024-10-16 15:18:47 -04:00
|
|
|
|
shuffle(peers)
|
|
|
|
|
|
2024-10-04 15:23:20 +05:30
|
|
|
|
# No criteria for selecting a peer for WakuRelay, random one
|
|
|
|
|
if proto == WakuRelayCodec:
|
|
|
|
|
# TODO: proper heuristic here that compares peer scores and selects "best" one. For now the first peer for the given protocol is returned
|
|
|
|
|
if peers.len > 0:
|
|
|
|
|
trace "Got peer from peerstore",
|
|
|
|
|
peerId = peers[0].peerId, multi = peers[0].addrs[0], protocol = proto
|
|
|
|
|
return some(peers[0])
|
|
|
|
|
trace "No peer found for protocol", protocol = proto
|
|
|
|
|
return none(RemotePeerInfo)
|
|
|
|
|
|
|
|
|
|
# For other protocols, we select the peer that is slotted for the given protocol
|
|
|
|
|
pm.serviceSlots.withValue(proto, serviceSlot):
|
|
|
|
|
trace "Got peer from service slots",
|
|
|
|
|
peerId = serviceSlot[].peerId, multi = serviceSlot[].addrs[0], protocol = proto
|
|
|
|
|
return some(serviceSlot[])
|
|
|
|
|
|
|
|
|
|
# If not slotted, we select a random peer for the given protocol
|
|
|
|
|
if peers.len > 0:
|
|
|
|
|
trace "Got peer from peerstore",
|
|
|
|
|
peerId = peers[0].peerId, multi = peers[0].addrs[0], protocol = proto
|
|
|
|
|
return some(peers[0])
|
|
|
|
|
trace "No peer found for protocol", protocol = proto
|
|
|
|
|
return none(RemotePeerInfo)
|
|
|
|
|
|
|
|
|
|
# Adds a peer to the service slots, which is a list of peers that are slotted for a given protocol
|
|
|
|
|
proc addServicePeer*(pm: PeerManager, remotePeerInfo: RemotePeerInfo, proto: string) =
|
|
|
|
|
# Do not add relay peers
|
|
|
|
|
if proto == WakuRelayCodec:
|
|
|
|
|
warn "Can't add relay peer to service peers slots"
|
|
|
|
|
return
|
|
|
|
|
|
|
|
|
|
info "Adding peer to service slots",
|
|
|
|
|
peerId = remotePeerInfo.peerId, addr = remotePeerInfo.addrs[0], service = proto
|
|
|
|
|
waku_service_peers.set(1, labelValues = [$proto, $remotePeerInfo.addrs[0]])
|
|
|
|
|
|
|
|
|
|
# Set peer for service slot
|
|
|
|
|
pm.serviceSlots[proto] = remotePeerInfo
|
|
|
|
|
|
|
|
|
|
pm.addPeer(remotePeerInfo)
|
|
|
|
|
|
|
|
|
|
#~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~#
|
|
|
|
|
# Connection Lifecycle Management #
|
|
|
|
|
#~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~#
|
|
|
|
|
|
|
|
|
|
# require pre-connection
|
|
|
|
|
proc pruneInRelayConns(pm: PeerManager, amount: int) {.async.}
|
|
|
|
|
|
2023-03-28 13:29:48 +02:00
|
|
|
|
# Connects to a given node. Note that this function uses `connect` and
|
|
|
|
|
# does not provide a protocol. Streams for relay (gossipsub) are created
|
|
|
|
|
# automatically without the needing to dial.
|
2024-10-29 18:37:07 +02:00
|
|
|
|
proc connectPeer*(
|
2024-03-16 00:08:47 +01:00
|
|
|
|
pm: PeerManager,
|
|
|
|
|
peer: RemotePeerInfo,
|
|
|
|
|
dialTimeout = DefaultDialTimeout,
|
|
|
|
|
source = "api",
|
|
|
|
|
): Future[bool] {.async.} =
|
2023-03-28 13:29:48 +02:00
|
|
|
|
let peerId = peer.peerId
|
2022-12-14 16:04:11 +01:00
|
|
|
|
|
|
|
|
|
# Do not attempt to dial self
|
|
|
|
|
if peerId == pm.switch.peerInfo.peerId:
|
2023-03-28 13:29:48 +02:00
|
|
|
|
return false
|
2022-12-14 16:04:11 +01:00
|
|
|
|
|
2024-10-29 18:37:07 +02:00
|
|
|
|
if not pm.wakuPeerStore.peerExists(peerId):
|
2023-03-28 13:29:48 +02:00
|
|
|
|
pm.addPeer(peer)
|
2021-03-26 10:49:51 +02:00
|
|
|
|
|
2024-09-27 18:16:46 +05:30
|
|
|
|
let failedAttempts = pm.wakuPeerStore[NumberFailedConnBook][peerId]
|
2024-10-29 18:37:07 +02:00
|
|
|
|
trace "Connecting to peer",
|
2024-03-16 00:08:47 +01:00
|
|
|
|
wireAddr = peer.addrs, peerId = peerId, failedAttempts = failedAttempts
|
2021-03-26 10:49:51 +02:00
|
|
|
|
|
2023-03-28 13:29:48 +02:00
|
|
|
|
var deadline = sleepAsync(dialTimeout)
|
2024-01-30 07:28:21 -05:00
|
|
|
|
let workfut = pm.switch.connect(peerId, peer.addrs)
|
2024-03-16 00:08:47 +01:00
|
|
|
|
|
2024-01-30 07:28:21 -05:00
|
|
|
|
# Can't use catch: with .withTimeout() in this case
|
2024-03-16 00:08:47 +01:00
|
|
|
|
let res = catch:
|
|
|
|
|
await workfut or deadline
|
2024-01-30 07:28:21 -05:00
|
|
|
|
|
2024-03-16 00:08:47 +01:00
|
|
|
|
let reasonFailed =
|
2024-01-30 07:28:21 -05:00
|
|
|
|
if not workfut.finished():
|
|
|
|
|
await workfut.cancelAndWait()
|
|
|
|
|
"timed out"
|
2024-03-16 00:08:47 +01:00
|
|
|
|
elif res.isErr():
|
|
|
|
|
res.error.msg
|
|
|
|
|
else:
|
2023-03-28 13:29:48 +02:00
|
|
|
|
if not deadline.finished():
|
2024-01-30 07:28:21 -05:00
|
|
|
|
await deadline.cancelAndWait()
|
2024-03-16 00:08:47 +01:00
|
|
|
|
|
2021-03-26 10:49:51 +02:00
|
|
|
|
waku_peers_dials.inc(labelValues = ["successful"])
|
2022-12-14 16:04:11 +01:00
|
|
|
|
waku_node_conns_initiated.inc(labelValues = [source])
|
2024-01-30 07:28:21 -05:00
|
|
|
|
|
2024-09-27 18:16:46 +05:30
|
|
|
|
pm.wakuPeerStore[NumberFailedConnBook][peerId] = 0
|
2023-12-20 15:23:41 +01:00
|
|
|
|
|
2024-01-30 07:28:21 -05:00
|
|
|
|
return true
|
2024-03-16 00:08:47 +01:00
|
|
|
|
|
2023-01-23 21:24:46 +01:00
|
|
|
|
# Dial failed
|
2024-09-27 18:16:46 +05:30
|
|
|
|
pm.wakuPeerStore[NumberFailedConnBook][peerId] =
|
|
|
|
|
pm.wakuPeerStore[NumberFailedConnBook][peerId] + 1
|
|
|
|
|
pm.wakuPeerStore[LastFailedConnBook][peerId] = Moment.init(getTime().toUnix, Second)
|
|
|
|
|
pm.wakuPeerStore[ConnectionBook][peerId] = CannotConnect
|
2022-11-24 14:11:23 +01:00
|
|
|
|
|
2024-10-29 18:37:07 +02:00
|
|
|
|
trace "Connecting peer failed",
|
2024-03-16 00:08:47 +01:00
|
|
|
|
peerId = peerId,
|
|
|
|
|
reason = reasonFailed,
|
2024-09-27 18:16:46 +05:30
|
|
|
|
failedAttempts = pm.wakuPeerStore[NumberFailedConnBook][peerId]
|
2023-01-23 21:24:46 +01:00
|
|
|
|
waku_peers_dials.inc(labelValues = [reasonFailed])
|
2022-11-24 14:11:23 +01:00
|
|
|
|
|
2023-03-28 13:29:48 +02:00
|
|
|
|
return false
|
|
|
|
|
|
2024-10-04 15:23:20 +05:30
|
|
|
|
proc connectToNodes*(
|
|
|
|
|
pm: PeerManager,
|
|
|
|
|
nodes: seq[string] | seq[RemotePeerInfo],
|
|
|
|
|
dialTimeout = DefaultDialTimeout,
|
|
|
|
|
source = "api",
|
|
|
|
|
) {.async.} =
|
|
|
|
|
if nodes.len == 0:
|
|
|
|
|
return
|
|
|
|
|
|
|
|
|
|
info "Dialing multiple peers", numOfPeers = nodes.len, nodes = $nodes
|
|
|
|
|
|
|
|
|
|
var futConns: seq[Future[bool]]
|
|
|
|
|
var connectedPeers: seq[RemotePeerInfo]
|
|
|
|
|
for node in nodes:
|
|
|
|
|
let node = parsePeerInfo(node)
|
|
|
|
|
if node.isOk():
|
2024-10-29 18:37:07 +02:00
|
|
|
|
futConns.add(pm.connectPeer(node.value))
|
2024-10-04 15:23:20 +05:30
|
|
|
|
connectedPeers.add(node.value)
|
|
|
|
|
else:
|
|
|
|
|
error "Couldn't parse node info", error = node.error
|
|
|
|
|
|
|
|
|
|
await allFutures(futConns)
|
|
|
|
|
|
|
|
|
|
# Filtering successful connectedPeers based on futConns
|
|
|
|
|
let combined = zip(connectedPeers, futConns)
|
|
|
|
|
connectedPeers = combined.filterIt(it[1].read() == true).mapIt(it[0])
|
|
|
|
|
|
|
|
|
|
when defined(debugDiscv5):
|
|
|
|
|
let peerIds = connectedPeers.mapIt(it.peerId)
|
|
|
|
|
let origin = connectedPeers.mapIt(it.origin)
|
|
|
|
|
if peerIds.len > 0:
|
|
|
|
|
notice "established connections with found peers",
|
|
|
|
|
peerIds = peerIds.mapIt(shortLog(it)), origin = origin
|
|
|
|
|
else:
|
|
|
|
|
notice "could not connect to new peers", attempted = nodes.len
|
|
|
|
|
|
|
|
|
|
info "Finished dialing multiple peers",
|
|
|
|
|
successfulConns = connectedPeers.len, attempted = nodes.len
|
|
|
|
|
|
|
|
|
|
# The issue seems to be around peers not being fully connected when
|
|
|
|
|
# trying to subscribe. So what we do is sleep to guarantee nodes are
|
|
|
|
|
# fully connected.
|
|
|
|
|
#
|
|
|
|
|
# This issue was known to Dmitiry on nim-libp2p and may be resolvable
|
|
|
|
|
# later.
|
|
|
|
|
await sleepAsync(chronos.seconds(5))
|
|
|
|
|
|
2024-10-15 15:32:02 +03:00
|
|
|
|
proc disconnectNode*(pm: PeerManager, peerId: PeerId) {.async.} =
|
|
|
|
|
await pm.switch.disconnect(peerId)
|
|
|
|
|
|
2024-09-24 18:20:29 +02:00
|
|
|
|
proc disconnectNode*(pm: PeerManager, peer: RemotePeerInfo) {.async.} =
|
|
|
|
|
let peerId = peer.peerId
|
2024-10-15 15:32:02 +03:00
|
|
|
|
await pm.disconnectNode(peerId)
|
2024-09-24 18:20:29 +02:00
|
|
|
|
|
2023-03-28 13:29:48 +02:00
|
|
|
|
# Dialing should be used for just protocols that require a stream to write and read
|
|
|
|
|
# This shall not be used to dial Relay protocols, since that would create
|
|
|
|
|
# unneccesary unused streams.
|
2024-03-16 00:08:47 +01:00
|
|
|
|
proc dialPeer(
|
|
|
|
|
pm: PeerManager,
|
|
|
|
|
peerId: PeerID,
|
|
|
|
|
addrs: seq[MultiAddress],
|
|
|
|
|
proto: string,
|
|
|
|
|
dialTimeout = DefaultDialTimeout,
|
|
|
|
|
source = "api",
|
|
|
|
|
): Future[Option[Connection]] {.async.} =
|
2023-03-28 13:29:48 +02:00
|
|
|
|
if peerId == pm.switch.peerInfo.peerId:
|
|
|
|
|
error "could not dial self"
|
|
|
|
|
return none(Connection)
|
|
|
|
|
|
|
|
|
|
if proto == WakuRelayCodec:
|
|
|
|
|
error "dial shall not be used to connect to relays"
|
|
|
|
|
return none(Connection)
|
|
|
|
|
|
2024-03-16 00:08:47 +01:00
|
|
|
|
trace "Dialing peer", wireAddr = addrs, peerId = peerId, proto = proto
|
2023-03-28 13:29:48 +02:00
|
|
|
|
|
|
|
|
|
# Dial Peer
|
|
|
|
|
let dialFut = pm.switch.dial(peerId, addrs, proto)
|
2024-01-30 07:28:21 -05:00
|
|
|
|
|
|
|
|
|
let res = catch:
|
|
|
|
|
if await dialFut.withTimeout(dialTimeout):
|
2023-03-28 13:29:48 +02:00
|
|
|
|
return some(dialFut.read())
|
2024-03-16 00:08:47 +01:00
|
|
|
|
else:
|
|
|
|
|
await cancelAndWait(dialFut)
|
2024-01-30 07:28:21 -05:00
|
|
|
|
|
2024-03-16 00:08:47 +01:00
|
|
|
|
let reasonFailed = if res.isOk: "timed out" else: res.error.msg
|
2023-03-28 13:29:48 +02:00
|
|
|
|
|
2024-03-16 00:08:47 +01:00
|
|
|
|
trace "Dialing peer failed", peerId = peerId, reason = reasonFailed, proto = proto
|
2022-11-24 14:11:23 +01:00
|
|
|
|
|
2023-01-23 21:24:46 +01:00
|
|
|
|
return none(Connection)
|
|
|
|
|
|
2024-10-04 15:23:20 +05:30
|
|
|
|
proc dialPeer*(
|
|
|
|
|
pm: PeerManager,
|
|
|
|
|
remotePeerInfo: RemotePeerInfo,
|
|
|
|
|
proto: string,
|
|
|
|
|
dialTimeout = DefaultDialTimeout,
|
|
|
|
|
source = "api",
|
|
|
|
|
): Future[Option[Connection]] {.async.} =
|
|
|
|
|
# Dial a given peer and add it to the list of known peers
|
|
|
|
|
# TODO: check peer validity and score before continuing. Limit number of peers to be managed.
|
2024-03-16 00:08:47 +01:00
|
|
|
|
|
2024-10-04 15:23:20 +05:30
|
|
|
|
# First add dialed peer info to peer store, if it does not exist yet..
|
|
|
|
|
# TODO: nim libp2p peerstore already adds them
|
|
|
|
|
if not pm.wakuPeerStore.hasPeer(remotePeerInfo.peerId, proto):
|
|
|
|
|
trace "Adding newly dialed peer to manager",
|
|
|
|
|
peerId = $remotePeerInfo.peerId, address = $remotePeerInfo.addrs[0], proto = proto
|
|
|
|
|
pm.addPeer(remotePeerInfo)
|
2022-11-24 14:11:23 +01:00
|
|
|
|
|
2024-10-04 15:23:20 +05:30
|
|
|
|
return await pm.dialPeer(
|
|
|
|
|
remotePeerInfo.peerId, remotePeerInfo.addrs, proto, dialTimeout, source
|
|
|
|
|
)
|
2023-10-11 08:58:45 +02:00
|
|
|
|
|
2024-10-04 15:23:20 +05:30
|
|
|
|
proc dialPeer*(
|
|
|
|
|
pm: PeerManager,
|
|
|
|
|
peerId: PeerID,
|
|
|
|
|
proto: string,
|
|
|
|
|
dialTimeout = DefaultDialTimeout,
|
|
|
|
|
source = "api",
|
|
|
|
|
): Future[Option[Connection]] {.async.} =
|
|
|
|
|
# Dial an existing peer by looking up it's existing addrs in the switch's peerStore
|
|
|
|
|
# TODO: check peer validity and score before continuing. Limit number of peers to be managed.
|
2023-10-11 08:58:45 +02:00
|
|
|
|
|
2024-10-04 15:23:20 +05:30
|
|
|
|
let addrs = pm.switch.peerStore[AddressBook][peerId]
|
|
|
|
|
return await pm.dialPeer(peerId, addrs, proto, dialTimeout, source)
|
2022-11-24 14:11:23 +01:00
|
|
|
|
|
2024-03-16 00:08:47 +01:00
|
|
|
|
proc canBeConnected*(pm: PeerManager, peerId: PeerId): bool =
|
2023-04-14 15:12:22 +02:00
|
|
|
|
# Returns if we can try to connect to this peer, based on past failed attempts
|
|
|
|
|
# It uses an exponential backoff. Each connection attempt makes us
|
|
|
|
|
# wait more before trying again.
|
2024-09-27 18:16:46 +05:30
|
|
|
|
let failedAttempts = pm.wakuPeerStore[NumberFailedConnBook][peerId]
|
2023-04-14 15:12:22 +02:00
|
|
|
|
|
|
|
|
|
# if it never errored, we can try to connect
|
|
|
|
|
if failedAttempts == 0:
|
|
|
|
|
return true
|
|
|
|
|
|
|
|
|
|
# if there are too many failed attempts, do not reconnect
|
|
|
|
|
if failedAttempts >= pm.maxFailedAttempts:
|
|
|
|
|
return false
|
|
|
|
|
|
|
|
|
|
# If it errored we wait an exponential backoff from last connection
|
|
|
|
|
# the more failed attempts, the greater the backoff since last attempt
|
|
|
|
|
let now = Moment.init(getTime().toUnix, Second)
|
2024-09-27 18:16:46 +05:30
|
|
|
|
let lastFailed = pm.wakuPeerStore[LastFailedConnBook][peerId]
|
2024-03-16 00:08:47 +01:00
|
|
|
|
let backoff =
|
|
|
|
|
calculateBackoff(pm.initialBackoffInSec, pm.backoffFactor, failedAttempts)
|
|
|
|
|
|
2024-01-30 07:28:21 -05:00
|
|
|
|
return now >= (lastFailed + backoff)
|
2023-04-14 15:12:22 +02:00
|
|
|
|
|
2024-10-04 15:23:20 +05:30
|
|
|
|
proc connectedPeers*(
|
|
|
|
|
pm: PeerManager, protocol: string = ""
|
|
|
|
|
): (seq[PeerId], seq[PeerId]) =
|
|
|
|
|
## Returns the peerIds of physical connections (in and out)
|
|
|
|
|
## If a protocol is specified, only returns peers with at least one stream of that protocol
|
2021-03-26 10:49:51 +02:00
|
|
|
|
|
2024-10-04 15:23:20 +05:30
|
|
|
|
var inPeers: seq[PeerId]
|
|
|
|
|
var outPeers: seq[PeerId]
|
2024-01-30 07:28:21 -05:00
|
|
|
|
|
2024-10-04 15:23:20 +05:30
|
|
|
|
for peerId, muxers in pm.switch.connManager.getConnections():
|
|
|
|
|
for peerConn in muxers:
|
|
|
|
|
let streams = peerConn.getStreams()
|
|
|
|
|
if protocol.len == 0 or streams.anyIt(it.protocol == protocol):
|
|
|
|
|
if peerConn.connection.transportDir == Direction.In:
|
|
|
|
|
inPeers.add(peerId)
|
|
|
|
|
elif peerConn.connection.transportDir == Direction.Out:
|
|
|
|
|
outPeers.add(peerId)
|
2024-01-30 07:28:21 -05:00
|
|
|
|
|
2024-10-04 15:23:20 +05:30
|
|
|
|
return (inPeers, outPeers)
|
2024-01-30 07:28:21 -05:00
|
|
|
|
|
2024-10-04 15:23:20 +05:30
|
|
|
|
proc connectToRelayPeers*(pm: PeerManager) {.async.} =
|
|
|
|
|
var (inRelayPeers, outRelayPeers) = pm.connectedPeers(WakuRelayCodec)
|
|
|
|
|
let totalRelayPeers = inRelayPeers.len + outRelayPeers.len
|
|
|
|
|
|
|
|
|
|
if inRelayPeers.len > pm.inRelayPeersTarget:
|
|
|
|
|
await pm.pruneInRelayConns(inRelayPeers.len - pm.inRelayPeersTarget)
|
|
|
|
|
|
|
|
|
|
if outRelayPeers.len >= pm.outRelayPeersTarget:
|
|
|
|
|
return
|
|
|
|
|
|
|
|
|
|
let notConnectedPeers = pm.wakuPeerStore.getDisconnectedPeers()
|
|
|
|
|
|
|
|
|
|
var outsideBackoffPeers = notConnectedPeers.filterIt(pm.canBeConnected(it.peerId))
|
|
|
|
|
|
|
|
|
|
shuffle(outsideBackoffPeers)
|
|
|
|
|
|
|
|
|
|
var index = 0
|
|
|
|
|
var numPendingConnReqs =
|
|
|
|
|
min(outsideBackoffPeers.len, pm.outRelayPeersTarget - outRelayPeers.len)
|
|
|
|
|
## number of outstanding connection requests
|
|
|
|
|
|
|
|
|
|
while numPendingConnReqs > 0 and outRelayPeers.len < pm.outRelayPeersTarget:
|
|
|
|
|
let numPeersToConnect = min(numPendingConnReqs, MaxParallelDials)
|
|
|
|
|
await pm.connectToNodes(outsideBackoffPeers[index ..< (index + numPeersToConnect)])
|
|
|
|
|
|
|
|
|
|
(inRelayPeers, outRelayPeers) = pm.connectedPeers(WakuRelayCodec)
|
|
|
|
|
|
|
|
|
|
index += numPeersToConnect
|
|
|
|
|
numPendingConnReqs -= numPeersToConnect
|
|
|
|
|
|
|
|
|
|
proc reconnectPeers*(
|
|
|
|
|
pm: PeerManager, proto: string, backoffTime: chronos.Duration = chronos.seconds(0)
|
|
|
|
|
) {.async.} =
|
|
|
|
|
## Reconnect to peers registered for this protocol. This will update connectedness.
|
|
|
|
|
## Especially useful to resume connections from persistent storage after a restart.
|
|
|
|
|
|
|
|
|
|
debug "Reconnecting peers", proto = proto
|
|
|
|
|
|
|
|
|
|
# Proto is not persisted, we need to iterate over all peers.
|
|
|
|
|
for peerInfo in pm.wakuPeerStore.peers(protocolMatcher(proto)):
|
|
|
|
|
# Check that the peer can be connected
|
|
|
|
|
if peerInfo.connectedness == CannotConnect:
|
|
|
|
|
error "Not reconnecting to unreachable or non-existing peer",
|
|
|
|
|
peerId = peerInfo.peerId
|
|
|
|
|
continue
|
|
|
|
|
|
|
|
|
|
if backoffTime > ZeroDuration:
|
|
|
|
|
debug "Backing off before reconnect",
|
|
|
|
|
peerId = peerInfo.peerId, backoffTime = backoffTime
|
|
|
|
|
# We disconnected recently and still need to wait for a backoff period before connecting
|
|
|
|
|
await sleepAsync(backoffTime)
|
|
|
|
|
|
|
|
|
|
await pm.connectToNodes(@[peerInfo])
|
|
|
|
|
|
|
|
|
|
proc getNumStreams*(pm: PeerManager, protocol: string): (int, int) =
|
|
|
|
|
var
|
|
|
|
|
numStreamsIn = 0
|
|
|
|
|
numStreamsOut = 0
|
|
|
|
|
for peerId, muxers in pm.switch.connManager.getConnections():
|
|
|
|
|
for peerConn in muxers:
|
|
|
|
|
for stream in peerConn.getStreams():
|
|
|
|
|
if stream.protocol == protocol:
|
|
|
|
|
if stream.dir == Direction.In:
|
|
|
|
|
numStreamsIn += 1
|
|
|
|
|
elif stream.dir == Direction.Out:
|
|
|
|
|
numStreamsOut += 1
|
|
|
|
|
return (numStreamsIn, numStreamsOut)
|
|
|
|
|
|
|
|
|
|
proc getPeerIp(pm: PeerManager, peerId: PeerId): Option[string] =
|
|
|
|
|
if not pm.switch.connManager.getConnections().hasKey(peerId):
|
|
|
|
|
return none(string)
|
|
|
|
|
|
|
|
|
|
let conns = pm.switch.connManager.getConnections().getOrDefault(peerId)
|
|
|
|
|
if conns.len == 0:
|
|
|
|
|
return none(string)
|
|
|
|
|
|
|
|
|
|
let obAddr = conns[0].connection.observedAddr.valueOr:
|
|
|
|
|
return none(string)
|
|
|
|
|
|
|
|
|
|
# TODO: think if circuit relay ips should be handled differently
|
2024-03-16 00:08:47 +01:00
|
|
|
|
|
2024-01-30 07:28:21 -05:00
|
|
|
|
return some(obAddr.getHostname())
|
2023-06-28 09:14:11 +02:00
|
|
|
|
|
2024-10-04 15:23:20 +05:30
|
|
|
|
#~~~~~~~~~~~~~~~~~#
|
|
|
|
|
# Event Handling #
|
|
|
|
|
#~~~~~~~~~~~~~~~~~#
|
2024-01-30 07:28:21 -05:00
|
|
|
|
|
|
|
|
|
proc onPeerMetadata(pm: PeerManager, peerId: PeerId) {.async.} =
|
2024-03-16 00:08:47 +01:00
|
|
|
|
let res = catch:
|
|
|
|
|
await pm.switch.dial(peerId, WakuMetadataCodec)
|
2024-01-30 07:28:21 -05:00
|
|
|
|
|
|
|
|
|
var reason: string
|
|
|
|
|
block guardClauses:
|
|
|
|
|
let conn = res.valueOr:
|
|
|
|
|
reason = "dial failed: " & error.msg
|
|
|
|
|
break guardClauses
|
2024-03-16 00:08:47 +01:00
|
|
|
|
|
2024-01-30 07:28:21 -05:00
|
|
|
|
let metadata = (await pm.wakuMetadata.request(conn)).valueOr:
|
|
|
|
|
reason = "waku metatdata request failed: " & error
|
|
|
|
|
break guardClauses
|
|
|
|
|
|
|
|
|
|
let clusterId = metadata.clusterId.valueOr:
|
|
|
|
|
reason = "empty cluster-id reported"
|
|
|
|
|
break guardClauses
|
|
|
|
|
|
|
|
|
|
if pm.wakuMetadata.clusterId != clusterId:
|
2024-03-16 00:08:47 +01:00
|
|
|
|
reason =
|
|
|
|
|
"different clusterId reported: " & $pm.wakuMetadata.clusterId & " vs " &
|
|
|
|
|
$clusterId
|
2024-01-30 07:28:21 -05:00
|
|
|
|
break guardClauses
|
|
|
|
|
|
2024-03-19 16:18:52 +01:00
|
|
|
|
if (
|
2024-09-27 18:16:46 +05:30
|
|
|
|
pm.wakuPeerStore.hasPeer(peerId, WakuRelayCodec) and
|
2024-03-19 16:18:52 +01:00
|
|
|
|
not metadata.shards.anyIt(pm.wakuMetadata.shards.contains(it))
|
|
|
|
|
):
|
2024-06-14 18:29:42 +05:30
|
|
|
|
let myShardsString = "[ " & toSeq(pm.wakuMetadata.shards).join(", ") & " ]"
|
|
|
|
|
let otherShardsString = "[ " & metadata.shards.join(", ") & " ]"
|
2024-06-10 13:40:18 +05:30
|
|
|
|
reason =
|
|
|
|
|
"no shards in common: my_shards = " & myShardsString & " others_shards = " &
|
|
|
|
|
otherShardsString
|
2024-01-30 07:28:21 -05:00
|
|
|
|
break guardClauses
|
|
|
|
|
|
|
|
|
|
return
|
2024-03-16 00:08:47 +01:00
|
|
|
|
|
|
|
|
|
info "disconnecting from peer", peerId = peerId, reason = reason
|
2024-01-30 07:28:21 -05:00
|
|
|
|
asyncSpawn(pm.switch.disconnect(peerId))
|
2024-09-27 18:16:46 +05:30
|
|
|
|
pm.wakuPeerStore.delete(peerId)
|
2023-02-14 15:38:32 +01:00
|
|
|
|
|
2024-10-04 15:23:20 +05:30
|
|
|
|
# called when a connection i) is created or ii) is closed
|
|
|
|
|
proc onConnEvent(pm: PeerManager, peerId: PeerID, event: ConnEvent) {.async.} =
|
|
|
|
|
case event.kind
|
|
|
|
|
of ConnEventKind.Connected:
|
|
|
|
|
#let direction = if event.incoming: Inbound else: Outbound
|
|
|
|
|
discard
|
|
|
|
|
of ConnEventKind.Disconnected:
|
|
|
|
|
discard
|
2024-09-27 19:35:18 +03:00
|
|
|
|
|
2023-05-31 09:47:56 +02:00
|
|
|
|
# called when a peer i) first connects to us ii) disconnects all connections from us
|
2023-02-14 15:38:32 +01:00
|
|
|
|
proc onPeerEvent(pm: PeerManager, peerId: PeerId, event: PeerEvent) {.async.} =
|
2024-01-30 07:28:21 -05:00
|
|
|
|
if not pm.wakuMetadata.isNil() and event.kind == PeerEventKind.Joined:
|
|
|
|
|
await pm.onPeerMetadata(peerId)
|
|
|
|
|
|
2023-05-31 09:47:56 +02:00
|
|
|
|
var direction: PeerDirection
|
|
|
|
|
var connectedness: Connectedness
|
|
|
|
|
|
2024-03-16 00:08:47 +01:00
|
|
|
|
case event.kind
|
|
|
|
|
of Joined:
|
|
|
|
|
direction = if event.initiator: Outbound else: Inbound
|
|
|
|
|
connectedness = Connected
|
|
|
|
|
|
2024-09-27 19:35:18 +03:00
|
|
|
|
## Check max allowed in-relay peers
|
|
|
|
|
let inRelayPeers = pm.connectedPeers(WakuRelayCodec)[0]
|
|
|
|
|
if inRelayPeers.len > pm.inRelayPeersTarget and
|
|
|
|
|
pm.wakuPeerStore.hasPeer(peerId, WakuRelayCodec):
|
|
|
|
|
debug "disconnecting relay peer because reached max num in-relay peers",
|
|
|
|
|
peerId = peerId,
|
|
|
|
|
inRelayPeers = inRelayPeers.len,
|
|
|
|
|
inRelayPeersTarget = pm.inRelayPeersTarget
|
|
|
|
|
await pm.switch.disconnect(peerId)
|
|
|
|
|
|
2024-10-04 15:23:20 +05:30
|
|
|
|
## Apply max ip colocation limit
|
|
|
|
|
if (let ip = pm.getPeerIp(peerId); ip.isSome()):
|
|
|
|
|
pm.ipTable.mgetOrPut(ip.get, newSeq[PeerId]()).add(peerId)
|
2023-05-18 09:40:14 +02:00
|
|
|
|
|
2024-10-04 15:23:20 +05:30
|
|
|
|
# in theory this should always be one, but just in case
|
|
|
|
|
let peersBehindIp = pm.ipTable[ip.get]
|
2023-05-31 09:47:56 +02:00
|
|
|
|
|
2024-10-04 15:23:20 +05:30
|
|
|
|
# pm.colocationLimit == 0 disables the ip colocation limit
|
|
|
|
|
if pm.colocationLimit != 0 and peersBehindIp.len > pm.colocationLimit:
|
|
|
|
|
for peerId in peersBehindIp[0 ..< (peersBehindIp.len - pm.colocationLimit)]:
|
|
|
|
|
debug "Pruning connection due to ip colocation", peerId = peerId, ip = ip
|
|
|
|
|
asyncSpawn(pm.switch.disconnect(peerId))
|
|
|
|
|
pm.wakuPeerStore.delete(peerId)
|
|
|
|
|
of Left:
|
|
|
|
|
direction = UnknownDirection
|
|
|
|
|
connectedness = CanConnect
|
2023-05-18 09:40:14 +02:00
|
|
|
|
|
2024-10-04 15:23:20 +05:30
|
|
|
|
# note we cant access the peerId ip here as the connection was already closed
|
|
|
|
|
for ip, peerIds in pm.ipTable.pairs:
|
|
|
|
|
if peerIds.contains(peerId):
|
|
|
|
|
pm.ipTable[ip] = pm.ipTable[ip].filterIt(it != peerId)
|
|
|
|
|
if pm.ipTable[ip].len == 0:
|
|
|
|
|
pm.ipTable.del(ip)
|
|
|
|
|
break
|
|
|
|
|
of Identified:
|
|
|
|
|
debug "event identified", peerId = peerId
|
2023-01-18 15:17:56 +01:00
|
|
|
|
|
2024-10-04 15:23:20 +05:30
|
|
|
|
pm.wakuPeerStore[ConnectionBook][peerId] = connectedness
|
|
|
|
|
pm.wakuPeerStore[DirectionBook][peerId] = direction
|
2023-01-18 15:17:56 +01:00
|
|
|
|
|
2024-10-04 15:23:20 +05:30
|
|
|
|
if not pm.storage.isNil:
|
|
|
|
|
var remotePeerInfo = pm.wakuPeerStore.getPeer(peerId)
|
2023-01-18 15:17:56 +01:00
|
|
|
|
|
2024-10-04 15:23:20 +05:30
|
|
|
|
if event.kind == PeerEventKind.Left:
|
|
|
|
|
remotePeerInfo.disconnectTime = getTime().toUnix
|
2024-03-19 19:07:03 +01:00
|
|
|
|
|
2024-10-04 15:23:20 +05:30
|
|
|
|
pm.storage.insertOrReplace(remotePeerInfo)
|
2024-03-19 19:07:03 +01:00
|
|
|
|
|
2024-10-04 15:23:20 +05:30
|
|
|
|
#~~~~~~~~~~~~~~~~~#
|
|
|
|
|
# Metrics Logging #
|
|
|
|
|
#~~~~~~~~~~~~~~~~~#
|
2024-03-19 19:07:03 +01:00
|
|
|
|
|
2024-10-04 15:23:20 +05:30
|
|
|
|
proc logAndMetrics(pm: PeerManager) {.async.} =
|
|
|
|
|
heartbeat "Scheduling log and metrics run", LogAndMetricsInterval:
|
|
|
|
|
# log metrics
|
|
|
|
|
let (inRelayPeers, outRelayPeers) = pm.connectedPeers(WakuRelayCodec)
|
|
|
|
|
let maxConnections = pm.switch.connManager.inSema.size
|
|
|
|
|
let notConnectedPeers = pm.wakuPeerStore.getDisconnectedPeers().mapIt(
|
|
|
|
|
RemotePeerInfo.init(it.peerId, it.addrs)
|
|
|
|
|
)
|
|
|
|
|
let outsideBackoffPeers = notConnectedPeers.filterIt(pm.canBeConnected(it.peerId))
|
|
|
|
|
let totalConnections = pm.switch.connManager.getConnections().len
|
2024-03-19 19:07:03 +01:00
|
|
|
|
|
2024-10-04 15:23:20 +05:30
|
|
|
|
info "Relay peer connections",
|
|
|
|
|
inRelayConns = $inRelayPeers.len & "/" & $pm.inRelayPeersTarget,
|
|
|
|
|
outRelayConns = $outRelayPeers.len & "/" & $pm.outRelayPeersTarget,
|
|
|
|
|
totalConnections = $totalConnections & "/" & $maxConnections,
|
|
|
|
|
notConnectedPeers = notConnectedPeers.len,
|
|
|
|
|
outsideBackoffPeers = outsideBackoffPeers.len
|
2024-03-19 19:07:03 +01:00
|
|
|
|
|
2024-10-04 15:23:20 +05:30
|
|
|
|
# update prometheus metrics
|
|
|
|
|
for proto in pm.wakuPeerStore.getWakuProtos():
|
|
|
|
|
let (protoConnsIn, protoConnsOut) = pm.connectedPeers(proto)
|
|
|
|
|
let (protoStreamsIn, protoStreamsOut) = pm.getNumStreams(proto)
|
|
|
|
|
waku_connected_peers.set(
|
|
|
|
|
protoConnsIn.len.float64, labelValues = [$Direction.In, proto]
|
|
|
|
|
)
|
|
|
|
|
waku_connected_peers.set(
|
|
|
|
|
protoConnsOut.len.float64, labelValues = [$Direction.Out, proto]
|
|
|
|
|
)
|
|
|
|
|
waku_streams_peers.set(
|
|
|
|
|
protoStreamsIn.float64, labelValues = [$Direction.In, proto]
|
|
|
|
|
)
|
|
|
|
|
waku_streams_peers.set(
|
|
|
|
|
protoStreamsOut.float64, labelValues = [$Direction.Out, proto]
|
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
#~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~#
|
|
|
|
|
# Pruning and Maintenance (Stale Peers Management) #
|
|
|
|
|
#~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~#
|
2023-01-26 10:20:20 +01:00
|
|
|
|
|
2024-01-30 07:28:21 -05:00
|
|
|
|
proc manageRelayPeers*(pm: PeerManager) {.async.} =
|
|
|
|
|
if pm.wakuMetadata.shards.len == 0:
|
|
|
|
|
return
|
2024-03-16 00:08:47 +01:00
|
|
|
|
|
2024-01-30 07:28:21 -05:00
|
|
|
|
var peersToConnect: HashSet[PeerId] # Can't use RemotePeerInfo as they are ref objects
|
|
|
|
|
var peersToDisconnect: int
|
|
|
|
|
|
|
|
|
|
# Get all connected peers for Waku Relay
|
|
|
|
|
var (inPeers, outPeers) = pm.connectedPeers(WakuRelayCodec)
|
|
|
|
|
|
|
|
|
|
# Calculate in/out target number of peers for each shards
|
|
|
|
|
let inTarget = pm.inRelayPeersTarget div pm.wakuMetadata.shards.len
|
|
|
|
|
let outTarget = pm.outRelayPeersTarget div pm.wakuMetadata.shards.len
|
|
|
|
|
|
|
|
|
|
for shard in pm.wakuMetadata.shards.items:
|
|
|
|
|
# Filter out peer not on this shard
|
|
|
|
|
let connectedInPeers = inPeers.filterIt(
|
2024-09-27 18:16:46 +05:30
|
|
|
|
pm.wakuPeerStore.hasShard(it, uint16(pm.wakuMetadata.clusterId), uint16(shard))
|
2024-03-16 00:08:47 +01:00
|
|
|
|
)
|
2024-01-30 07:28:21 -05:00
|
|
|
|
|
|
|
|
|
let connectedOutPeers = outPeers.filterIt(
|
2024-09-27 18:16:46 +05:30
|
|
|
|
pm.wakuPeerStore.hasShard(it, uint16(pm.wakuMetadata.clusterId), uint16(shard))
|
2024-03-16 00:08:47 +01:00
|
|
|
|
)
|
2024-01-30 07:28:21 -05:00
|
|
|
|
|
|
|
|
|
# Calculate the difference between current values and targets
|
|
|
|
|
let inPeerDiff = connectedInPeers.len - inTarget
|
|
|
|
|
let outPeerDiff = outTarget - connectedOutPeers.len
|
|
|
|
|
|
|
|
|
|
if inPeerDiff > 0:
|
|
|
|
|
peersToDisconnect += inPeerDiff
|
|
|
|
|
|
|
|
|
|
if outPeerDiff <= 0:
|
|
|
|
|
continue
|
|
|
|
|
|
|
|
|
|
# Get all peers for this shard
|
2024-03-16 00:08:47 +01:00
|
|
|
|
var connectablePeers =
|
2024-09-27 18:16:46 +05:30
|
|
|
|
pm.wakuPeerStore.getPeersByShard(uint16(pm.wakuMetadata.clusterId), uint16(shard))
|
2024-03-16 00:08:47 +01:00
|
|
|
|
|
2024-01-30 07:28:21 -05:00
|
|
|
|
let shardCount = connectablePeers.len
|
|
|
|
|
|
|
|
|
|
connectablePeers.keepItIf(
|
2024-09-27 18:16:46 +05:30
|
|
|
|
not pm.wakuPeerStore.isConnected(it.peerId) and pm.canBeConnected(it.peerId)
|
2024-03-16 00:08:47 +01:00
|
|
|
|
)
|
2024-01-30 07:28:21 -05:00
|
|
|
|
|
|
|
|
|
let connectableCount = connectablePeers.len
|
|
|
|
|
|
2024-09-27 18:16:46 +05:30
|
|
|
|
connectablePeers.keepItIf(pm.wakuPeerStore.hasCapability(it.peerId, Relay))
|
2024-01-30 07:28:21 -05:00
|
|
|
|
|
|
|
|
|
let relayCount = connectablePeers.len
|
|
|
|
|
|
|
|
|
|
debug "Sharded Peer Management",
|
|
|
|
|
shard = shard,
|
|
|
|
|
connectable = $connectableCount & "/" & $shardCount,
|
|
|
|
|
relayConnectable = $relayCount & "/" & $shardCount,
|
|
|
|
|
relayInboundTarget = $connectedInPeers.len & "/" & $inTarget,
|
|
|
|
|
relayOutboundTarget = $connectedOutPeers.len & "/" & $outTarget
|
2024-03-16 00:08:47 +01:00
|
|
|
|
|
2024-01-30 07:28:21 -05:00
|
|
|
|
# Always pick random connectable relay peers
|
|
|
|
|
shuffle(connectablePeers)
|
|
|
|
|
|
|
|
|
|
let length = min(outPeerDiff, connectablePeers.len)
|
2024-03-16 00:08:47 +01:00
|
|
|
|
for peer in connectablePeers[0 ..< length]:
|
2024-01-30 07:28:21 -05:00
|
|
|
|
trace "Peer To Connect To", peerId = $peer.peerId
|
|
|
|
|
peersToConnect.incl(peer.peerId)
|
|
|
|
|
|
|
|
|
|
await pm.pruneInRelayConns(peersToDisconnect)
|
2024-03-16 00:08:47 +01:00
|
|
|
|
|
2024-01-30 07:28:21 -05:00
|
|
|
|
if peersToConnect.len == 0:
|
|
|
|
|
return
|
|
|
|
|
|
2024-09-27 18:16:46 +05:30
|
|
|
|
let uniquePeers = toSeq(peersToConnect).mapIt(pm.wakuPeerStore.getPeer(it))
|
2024-01-30 07:28:21 -05:00
|
|
|
|
|
|
|
|
|
# Connect to all nodes
|
|
|
|
|
for i in countup(0, uniquePeers.len, MaxParallelDials):
|
|
|
|
|
let stop = min(i + MaxParallelDials, uniquePeers.len)
|
2024-03-16 00:08:47 +01:00
|
|
|
|
trace "Connecting to Peers", peerIds = $uniquePeers[i ..< stop]
|
|
|
|
|
await pm.connectToNodes(uniquePeers[i ..< stop])
|
2024-01-30 07:28:21 -05:00
|
|
|
|
|
2023-01-31 13:24:49 +01:00
|
|
|
|
proc prunePeerStore*(pm: PeerManager) =
|
2024-09-27 18:16:46 +05:30
|
|
|
|
let numPeers = pm.wakuPeerStore[AddressBook].book.len
|
|
|
|
|
let capacity = pm.wakuPeerStore.getCapacity()
|
2023-12-07 07:21:18 -05:00
|
|
|
|
if numPeers <= capacity:
|
2023-01-31 13:24:49 +01:00
|
|
|
|
return
|
|
|
|
|
|
2023-12-12 16:00:18 +01:00
|
|
|
|
trace "Peer store capacity exceeded", numPeers = numPeers, capacity = capacity
|
2023-12-07 07:21:18 -05:00
|
|
|
|
let pruningCount = numPeers - capacity
|
|
|
|
|
var peersToPrune: HashSet[PeerId]
|
|
|
|
|
|
|
|
|
|
# prune failed connections
|
2024-09-27 18:16:46 +05:30
|
|
|
|
for peerId, count in pm.wakuPeerStore[NumberFailedConnBook].book.pairs:
|
2023-12-07 07:21:18 -05:00
|
|
|
|
if count < pm.maxFailedAttempts:
|
|
|
|
|
continue
|
|
|
|
|
|
|
|
|
|
if peersToPrune.len >= pruningCount:
|
2023-01-31 13:24:49 +01:00
|
|
|
|
break
|
|
|
|
|
|
2023-12-07 07:21:18 -05:00
|
|
|
|
peersToPrune.incl(peerId)
|
2024-03-16 00:08:47 +01:00
|
|
|
|
|
2024-09-27 18:16:46 +05:30
|
|
|
|
var notConnected = pm.wakuPeerStore.getDisconnectedPeers().mapIt(it.peerId)
|
2024-01-30 07:28:21 -05:00
|
|
|
|
|
|
|
|
|
# Always pick random non-connected peers
|
|
|
|
|
shuffle(notConnected)
|
2023-12-07 07:21:18 -05:00
|
|
|
|
|
|
|
|
|
var shardlessPeers: seq[PeerId]
|
|
|
|
|
var peersByShard = initTable[uint16, seq[PeerId]]()
|
|
|
|
|
|
|
|
|
|
for peer in notConnected:
|
2024-09-27 18:16:46 +05:30
|
|
|
|
if not pm.wakuPeerStore[ENRBook].contains(peer):
|
2023-12-07 07:21:18 -05:00
|
|
|
|
shardlessPeers.add(peer)
|
|
|
|
|
continue
|
|
|
|
|
|
2024-09-27 18:16:46 +05:30
|
|
|
|
let record = pm.wakuPeerStore[ENRBook][peer]
|
2023-12-07 07:21:18 -05:00
|
|
|
|
|
|
|
|
|
let rec = record.toTyped().valueOr:
|
|
|
|
|
shardlessPeers.add(peer)
|
|
|
|
|
continue
|
|
|
|
|
|
|
|
|
|
let rs = rec.relaySharding().valueOr:
|
|
|
|
|
shardlessPeers.add(peer)
|
|
|
|
|
continue
|
|
|
|
|
|
|
|
|
|
for shard in rs.shardIds:
|
2024-05-13 17:25:44 +02:00
|
|
|
|
peersByShard.mgetOrPut(shard, @[]).add(peer)
|
2023-12-07 07:21:18 -05:00
|
|
|
|
|
|
|
|
|
# prune not connected peers without shard
|
|
|
|
|
for peer in shardlessPeers:
|
|
|
|
|
if peersToPrune.len >= pruningCount:
|
2023-01-31 13:24:49 +01:00
|
|
|
|
break
|
|
|
|
|
|
2023-12-07 07:21:18 -05:00
|
|
|
|
peersToPrune.incl(peer)
|
|
|
|
|
|
|
|
|
|
# calculate the avg peers per shard
|
|
|
|
|
let total = sum(toSeq(peersByShard.values).mapIt(it.len))
|
|
|
|
|
let avg = min(1, total div max(1, peersByShard.len))
|
|
|
|
|
|
|
|
|
|
# prune peers from shard with higher than avg count
|
|
|
|
|
for shard, peers in peersByShard.pairs:
|
|
|
|
|
let count = max(peers.len - avg, 0)
|
2024-03-16 00:08:47 +01:00
|
|
|
|
for peer in peers[0 .. count]:
|
2023-12-07 07:21:18 -05:00
|
|
|
|
if peersToPrune.len >= pruningCount:
|
|
|
|
|
break
|
|
|
|
|
|
|
|
|
|
peersToPrune.incl(peer)
|
|
|
|
|
|
|
|
|
|
for peer in peersToPrune:
|
2024-09-27 18:16:46 +05:30
|
|
|
|
pm.wakuPeerStore.delete(peer)
|
2023-12-07 07:21:18 -05:00
|
|
|
|
|
2024-09-27 18:16:46 +05:30
|
|
|
|
let afterNumPeers = pm.wakuPeerStore[AddressBook].book.len
|
2023-12-07 07:21:18 -05:00
|
|
|
|
|
2024-03-16 00:08:47 +01:00
|
|
|
|
trace "Finished pruning peer store",
|
|
|
|
|
beforeNumPeers = numPeers,
|
|
|
|
|
afterNumPeers = afterNumPeers,
|
|
|
|
|
capacity = capacity,
|
|
|
|
|
pruned = peersToPrune.len
|
2023-01-31 13:24:49 +01:00
|
|
|
|
|
2023-02-27 18:24:31 +01:00
|
|
|
|
# Prunes peers from peerstore to remove old/stale ones
|
2024-03-16 00:08:47 +01:00
|
|
|
|
proc prunePeerStoreLoop(pm: PeerManager) {.async.} =
|
2023-12-12 16:00:18 +01:00
|
|
|
|
trace "Starting prune peerstore loop"
|
2023-02-27 18:24:31 +01:00
|
|
|
|
while pm.started:
|
|
|
|
|
pm.prunePeerStore()
|
|
|
|
|
await sleepAsync(PrunePeerStoreInterval)
|
|
|
|
|
|
|
|
|
|
# Ensures a healthy amount of connected relay peers
|
|
|
|
|
proc relayConnectivityLoop*(pm: PeerManager) {.async.} =
|
2023-12-12 16:00:18 +01:00
|
|
|
|
trace "Starting relay connectivity loop"
|
2023-02-27 18:24:31 +01:00
|
|
|
|
while pm.started:
|
2024-01-30 07:28:21 -05:00
|
|
|
|
if pm.shardedPeerManagement:
|
|
|
|
|
await pm.manageRelayPeers()
|
2024-03-16 00:08:47 +01:00
|
|
|
|
else:
|
|
|
|
|
await pm.connectToRelayPeers()
|
2024-09-12 22:49:47 +02:00
|
|
|
|
let
|
|
|
|
|
(inRelayPeers, outRelayPeers) = pm.connectedPeers(WakuRelayCodec)
|
|
|
|
|
excessInConns = max(inRelayPeers.len - pm.inRelayPeersTarget, 0)
|
|
|
|
|
|
2024-09-27 18:16:46 +05:30
|
|
|
|
# One minus the percentage of excess connections relative to the target, limited to 100%
|
2024-09-12 22:49:47 +02:00
|
|
|
|
# We calculate one minus this percentage because we want the factor to be inversely proportional to the number of excess peers
|
|
|
|
|
inFactor = 1 - min(excessInConns / pm.inRelayPeersTarget, 1)
|
|
|
|
|
# Percentage of out relay peers relative to the target
|
|
|
|
|
outFactor = min(outRelayPeers.len / pm.outRelayPeersTarget, 1)
|
|
|
|
|
factor = min(outFactor, inFactor)
|
|
|
|
|
dynamicSleepInterval =
|
|
|
|
|
chronos.seconds(int(float(ConnectivityLoopInterval.seconds()) * factor))
|
|
|
|
|
|
|
|
|
|
# Shorten the connectivity loop interval dynamically based on percentage of peers to fill or connections to prune
|
2024-10-22 20:09:25 +02:00
|
|
|
|
await sleepAsync(max(dynamicSleepInterval, chronos.seconds(1)))
|
2023-02-27 18:24:31 +01:00
|
|
|
|
|
2024-10-04 15:23:20 +05:30
|
|
|
|
proc pruneInRelayConns(pm: PeerManager, amount: int) {.async.} =
|
|
|
|
|
if amount <= 0:
|
|
|
|
|
return
|
2023-05-18 09:40:14 +02:00
|
|
|
|
|
2024-10-04 15:23:20 +05:30
|
|
|
|
let (inRelayPeers, _) = pm.connectedPeers(WakuRelayCodec)
|
|
|
|
|
let connsToPrune = min(amount, inRelayPeers.len)
|
2023-05-18 09:40:14 +02:00
|
|
|
|
|
2024-10-04 15:23:20 +05:30
|
|
|
|
for p in inRelayPeers[0 ..< connsToPrune]:
|
|
|
|
|
trace "Pruning Peer", Peer = $p
|
|
|
|
|
asyncSpawn(pm.switch.disconnect(p))
|
|
|
|
|
|
|
|
|
|
#~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~#
|
|
|
|
|
# Initialization and Constructor #
|
|
|
|
|
#~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~#
|
2023-04-12 13:05:34 +02:00
|
|
|
|
|
2023-01-26 10:20:20 +01:00
|
|
|
|
proc start*(pm: PeerManager) =
|
|
|
|
|
pm.started = true
|
|
|
|
|
asyncSpawn pm.relayConnectivityLoop()
|
2023-01-31 13:24:49 +01:00
|
|
|
|
asyncSpawn pm.prunePeerStoreLoop()
|
2023-05-31 09:47:56 +02:00
|
|
|
|
asyncSpawn pm.logAndMetrics()
|
2023-01-26 10:20:20 +01:00
|
|
|
|
|
|
|
|
|
proc stop*(pm: PeerManager) =
|
2024-03-16 00:08:47 +01:00
|
|
|
|
pm.started = false
|
2024-10-04 15:23:20 +05:30
|
|
|
|
|
|
|
|
|
proc new*(
|
|
|
|
|
T: type PeerManager,
|
|
|
|
|
switch: Switch,
|
|
|
|
|
wakuMetadata: WakuMetadata = nil,
|
|
|
|
|
maxRelayPeers: Option[int] = none(int),
|
|
|
|
|
storage: PeerStorage = nil,
|
|
|
|
|
initialBackoffInSec = InitialBackoffInSec,
|
|
|
|
|
backoffFactor = BackoffFactor,
|
|
|
|
|
maxFailedAttempts = MaxFailedAttempts,
|
|
|
|
|
colocationLimit = DefaultColocationLimit,
|
|
|
|
|
shardedPeerManagement = false,
|
|
|
|
|
): PeerManager {.gcsafe.} =
|
|
|
|
|
let capacity = switch.peerStore.capacity
|
|
|
|
|
let maxConnections = switch.connManager.inSema.size
|
|
|
|
|
if maxConnections > capacity:
|
|
|
|
|
error "Max number of connections can't be greater than PeerManager capacity",
|
|
|
|
|
capacity = capacity, maxConnections = maxConnections
|
|
|
|
|
raise newException(
|
|
|
|
|
Defect, "Max number of connections can't be greater than PeerManager capacity"
|
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
var maxRelayPeersValue = 0
|
|
|
|
|
if maxRelayPeers.isSome():
|
|
|
|
|
if maxRelayPeers.get() > maxConnections:
|
|
|
|
|
error "Max number of relay peers can't be greater than the max amount of connections",
|
|
|
|
|
maxConnections = maxConnections, maxRelayPeers = maxRelayPeers.get()
|
|
|
|
|
raise newException(
|
|
|
|
|
Defect,
|
|
|
|
|
"Max number of relay peers can't be greater than the max amount of connections",
|
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
if maxRelayPeers.get() == maxConnections:
|
|
|
|
|
warn "Max number of relay peers is equal to max amount of connections, peer won't be contributing to service peers",
|
|
|
|
|
maxConnections = maxConnections, maxRelayPeers = maxRelayPeers.get()
|
|
|
|
|
maxRelayPeersValue = maxRelayPeers.get()
|
|
|
|
|
else:
|
|
|
|
|
# Leave by default 20% of connections for service peers
|
|
|
|
|
maxRelayPeersValue = maxConnections - (maxConnections div 5)
|
|
|
|
|
|
|
|
|
|
# attempt to calculate max backoff to prevent potential overflows or unreasonably high values
|
|
|
|
|
let backoff = calculateBackoff(initialBackoffInSec, backoffFactor, maxFailedAttempts)
|
|
|
|
|
if backoff.weeks() > 1:
|
|
|
|
|
error "Max backoff time can't be over 1 week", maxBackoff = backoff
|
|
|
|
|
raise newException(Defect, "Max backoff time can't be over 1 week")
|
|
|
|
|
|
|
|
|
|
let outRelayPeersTarget = maxRelayPeersValue div 3
|
|
|
|
|
|
|
|
|
|
let pm = PeerManager(
|
|
|
|
|
switch: switch,
|
|
|
|
|
wakuMetadata: wakuMetadata,
|
|
|
|
|
wakuPeerStore: createWakuPeerStore(switch.peerStore),
|
|
|
|
|
storage: storage,
|
|
|
|
|
initialBackoffInSec: initialBackoffInSec,
|
|
|
|
|
backoffFactor: backoffFactor,
|
|
|
|
|
outRelayPeersTarget: outRelayPeersTarget,
|
|
|
|
|
inRelayPeersTarget: maxRelayPeersValue - outRelayPeersTarget,
|
|
|
|
|
maxRelayPeers: maxRelayPeersValue,
|
|
|
|
|
maxFailedAttempts: maxFailedAttempts,
|
|
|
|
|
colocationLimit: colocationLimit,
|
|
|
|
|
shardedPeerManagement: shardedPeerManagement,
|
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
proc connHook(peerId: PeerID, event: ConnEvent): Future[void] {.gcsafe.} =
|
|
|
|
|
onConnEvent(pm, peerId, event)
|
|
|
|
|
|
|
|
|
|
proc peerHook(peerId: PeerId, event: PeerEvent): Future[void] {.gcsafe.} =
|
|
|
|
|
onPeerEvent(pm, peerId, event)
|
|
|
|
|
|
|
|
|
|
proc peerStoreChanged(peerId: PeerId) {.gcsafe.} =
|
|
|
|
|
waku_peer_store_size.set(toSeq(pm.wakuPeerStore[AddressBook].book.keys).len.int64)
|
|
|
|
|
|
|
|
|
|
# currently disabled
|
|
|
|
|
#pm.switch.addConnEventHandler(connHook, ConnEventKind.Connected)
|
|
|
|
|
#pm.switch.addConnEventHandler(connHook, ConnEventKind.Disconnected)
|
|
|
|
|
|
|
|
|
|
pm.switch.addPeerEventHandler(peerHook, PeerEventKind.Joined)
|
|
|
|
|
pm.switch.addPeerEventHandler(peerHook, PeerEventKind.Left)
|
|
|
|
|
|
|
|
|
|
# called every time the peerstore is updated
|
|
|
|
|
pm.wakuPeerStore[AddressBook].addHandler(peerStoreChanged)
|
|
|
|
|
|
|
|
|
|
pm.serviceSlots = initTable[string, RemotePeerInfo]()
|
|
|
|
|
pm.ipTable = initTable[string, seq[PeerId]]()
|
|
|
|
|
|
|
|
|
|
if not storage.isNil():
|
|
|
|
|
trace "found persistent peer storage"
|
|
|
|
|
pm.loadFromStorage() # Load previously managed peers.
|
|
|
|
|
else:
|
|
|
|
|
trace "no peer storage found"
|
|
|
|
|
|
|
|
|
|
return pm
|