2021-07-14 17:58:46 +00:00
{. push raises : [ Defect ] . }
2020-04-29 04:49:27 +00:00
import
2022-01-10 14:07:01 +00:00
std / [ hashes , options , tables , strutils , sequtils , os ] ,
2021-06-09 14:37:08 +00:00
chronos , chronicles , metrics ,
stew / shims / net as stewNet ,
2021-09-17 17:31:25 +00:00
stew / byteutils ,
2020-09-01 02:09:54 +00:00
eth / keys ,
2022-06-17 22:00:19 +00:00
nimcrypto ,
2021-08-12 08:51:38 +00:00
eth / p2p / discoveryv5 / enr ,
2020-05-15 04:11:14 +00:00
libp2p / crypto / crypto ,
2021-06-15 08:55:47 +00:00
libp2p / protocols / ping ,
2022-01-10 14:07:01 +00:00
libp2p / protocols / pubsub / [ gossipsub , rpc / messages ] ,
2022-02-16 16:12:09 +00:00
libp2p / nameresolving / nameresolver ,
2022-01-10 14:07:01 +00:00
libp2p / [ builders , multihash ] ,
2022-07-25 11:01:37 +00:00
libp2p / transports / [ transport , tcptransport , wstransport ]
import
2021-07-13 07:18:51 +00:00
.. / protocol / [ waku_relay , waku_message ] ,
2022-07-25 11:01:37 +00:00
.. / protocol / waku_store ,
2020-11-23 02:27:45 +00:00
.. / protocol / waku_swap / waku_swap ,
2022-08-12 10:15:51 +00:00
.. / protocol / waku_filter ,
2022-08-02 23:47:42 +00:00
.. / protocol / waku_lightpush ,
2022-09-07 01:51:10 +00:00
.. / protocol / waku_rln_relay / waku_rln_relay_types ,
2022-09-08 09:02:50 +00:00
.. / utils / [ peers , requests , wakuenr ] ,
2021-08-12 08:51:38 +00:00
. / peer_manager / peer_manager ,
2022-09-08 09:02:50 +00:00
. / storage / message / waku_store_queue ,
2022-07-25 11:01:37 +00:00
. / storage / message / message_store ,
2022-09-16 10:55:22 +00:00
. / storage / message / message_retention_policy ,
. / storage / message / message_retention_policy_capacity ,
. / storage / message / message_retention_policy_time ,
2021-11-01 18:02:39 +00:00
. / dnsdisc / waku_dnsdisc ,
2022-06-27 19:35:26 +00:00
. / discv5 / waku_discv5 ,
2022-09-08 09:02:50 +00:00
. / wakuswitch ,
2022-07-25 11:01:37 +00:00
. / wakunode2_types
2020-10-21 09:54:29 +00:00
2021-07-16 15:13:36 +00:00
export
2022-06-27 19:35:26 +00:00
wakunode2_types
2021-07-16 15:13:36 +00:00
2021-06-08 18:56:32 +00:00
when defined ( rln ) :
2022-06-27 19:35:26 +00:00
import .. / protocol / waku_rln_relay / waku_rln_relay_utils
2021-06-08 18:56:32 +00:00
2022-03-15 12:39:32 +00:00
declarePublicCounter waku_node_messages , " number of messages received " , [ " type " ]
2021-01-29 08:42:41 +00:00
declarePublicGauge waku_node_filters , " number of content filter subscriptions "
declarePublicGauge waku_node_errors , " number of wakunode errors " , [ " type " ]
2022-01-26 11:02:57 +00:00
declarePublicCounter waku_node_conns_initiated , " number of connections initiated by this node " , [ " source " ]
2021-01-29 08:42:41 +00:00
2020-09-16 04:23:10 +00:00
logScope :
topics = " wakunode "
2020-09-11 04:16:45 +00:00
# Default clientId
const clientId * = " Nimbus Waku v2 node "
2020-04-29 04:49:27 +00:00
2021-03-11 07:48:59 +00:00
# Default topic
2022-03-29 08:09:48 +00:00
const defaultTopic * = " /waku/2/default-waku/proto "
2021-03-11 07:48:59 +00:00
2021-12-08 18:35:47 +00:00
# Default Waku Filter Timeout
const WakuFilterTimeout : Duration = 1 . days
2021-07-27 06:48:56 +00:00
proc protocolMatcher ( codec : string ) : Matcher =
2021-07-02 08:49:41 +00:00
## 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 )
2021-07-27 06:48:56 +00:00
return match
2021-11-30 09:12:09 +00:00
proc updateSwitchPeerInfo ( node : WakuNode ) =
## TODO: remove this when supported upstream
##
## nim-libp2p does not yet support announcing addrs
## different from bound addrs.
##
## This is a temporary workaround to replace
## peer info addrs in switch to announced
## addresses.
##
## WARNING: this should only be called once the switch
## has already been started.
if node . announcedAddresses . len > 0 :
node . switch . peerInfo . addrs = node . announcedAddresses
2022-02-18 11:10:38 +00:00
template ip4TcpEndPoint ( address , port ) : MultiAddress =
2020-07-24 01:39:58 +00:00
MultiAddress . init ( address , tcpProtocol , port )
2022-02-18 11:10:38 +00:00
template dns4Ma ( dns4DomainName : string ) : MultiAddress =
MultiAddress . init ( " /dns4/ " & dns4DomainName ) . tryGet ( )
template tcpPortMa ( port : Port ) : MultiAddress =
MultiAddress . init ( " /tcp/ " & $ port ) . tryGet ( )
template dns4TcpEndPoint ( dns4DomainName : string , port : Port ) : MultiAddress =
dns4Ma ( dns4DomainName ) & tcpPortMa ( port )
template wsFlag ( wssEnabled : bool ) : MultiAddress =
2021-12-06 19:51:37 +00:00
if wssEnabled : MultiAddress . init ( " /wss " ) . tryGet ( )
else : MultiAddress . init ( " /ws " ) . tryGet ( )
2020-09-01 02:09:54 +00:00
2021-07-14 17:58:46 +00:00
proc new * ( T : type WakuNode , nodeKey : crypto . PrivateKey ,
2020-09-01 02:09:54 +00:00
bindIp : ValidIpAddress , bindPort : Port ,
2022-06-09 15:46:21 +00:00
extIp = none ( ValidIpAddress ) , extPort = none ( Port ) ,
2021-10-12 12:48:48 +00:00
peerStorage : PeerStorage = nil ,
2021-11-02 10:29:11 +00:00
maxConnections = builders . MaxConnections ,
wsBindPort : Port = ( Port ) 8000 ,
2021-11-10 12:05:36 +00:00
wsEnabled : bool = false ,
wssEnabled : bool = false ,
secureKey : string = " " ,
2021-12-06 19:51:37 +00:00
secureCert : string = " " ,
2022-02-16 16:12:09 +00:00
wakuFlags = none ( WakuEnrBitfield ) ,
nameResolver : NameResolver = nil ,
2022-03-29 08:09:48 +00:00
sendSignedPeerRecord = false ,
2022-06-09 15:46:21 +00:00
dns4DomainName = none ( string ) ,
discv5UdpPort = none ( Port )
2021-12-06 19:51:37 +00:00
) : T
{. raises : [ Defect , LPError , IOError , TLSStreamProtocolError ] . } =
2020-10-06 03:33:28 +00:00
## Creates a Waku Node.
##
## Status: Implemented.
##
2021-12-06 19:51:37 +00:00
## Initialize addresses
2020-04-29 04:49:27 +00:00
let
2021-12-06 19:51:37 +00:00
# Bind addresses
2022-02-18 11:10:38 +00:00
hostAddress = ip4TcpEndPoint ( bindIp , bindPort )
wsHostAddress = if wsEnabled or wssEnabled : some ( ip4TcpEndPoint ( bindIp , wsbindPort ) & wsFlag ( wssEnabled ) )
2021-12-06 19:51:37 +00:00
else : none ( MultiAddress )
2022-02-18 11:10:38 +00:00
# Setup external addresses, if available
var
hostExtAddress , wsExtAddress = none ( MultiAddress )
if ( dns4DomainName . isSome ( ) ) :
# Use dns4 for externally announced addresses
hostExtAddress = some ( dns4TcpEndPoint ( dns4DomainName . get ( ) , extPort . get ( ) ) )
if ( wsHostAddress . isSome ( ) ) :
wsExtAddress = some ( dns4TcpEndPoint ( dns4DomainName . get ( ) , wsBindPort ) & wsFlag ( wssEnabled ) )
else :
# No public domain name, use ext IP if available
if extIp . isSome ( ) and extPort . isSome ( ) :
hostExtAddress = some ( ip4TcpEndPoint ( extIp . get ( ) , extPort . get ( ) ) )
if ( wsHostAddress . isSome ( ) ) :
wsExtAddress = some ( ip4TcpEndPoint ( extIp . get ( ) , wsBindPort ) & wsFlag ( wssEnabled ) )
2021-12-06 19:51:37 +00:00
var announcedAddresses : seq [ MultiAddress ]
if hostExtAddress . isSome :
announcedAddresses . add ( hostExtAddress . get ( ) )
else :
announcedAddresses . add ( hostAddress ) # We always have at least a bind address for the host
if wsExtAddress . isSome :
announcedAddresses . add ( wsExtAddress . get ( ) )
elif wsHostAddress . isSome :
announcedAddresses . add ( wsHostAddress . get ( ) )
## Initialize peer
let
rng = crypto . newRng ( )
2021-08-12 08:51:38 +00:00
enrIp = if extIp . isSome ( ) : extIp
else : some ( bindIp )
enrTcpPort = if extPort . isSome ( ) : extPort
else : some ( bindPort )
2021-12-06 19:51:37 +00:00
enrMultiaddrs = if wsExtAddress . isSome : @ [ wsExtAddress . get ( ) ] # Only add ws/wss to `multiaddrs` field
elif wsHostAddress . isSome : @ [ wsHostAddress . get ( ) ]
else : @ [ ]
enr = initEnr ( nodeKey ,
enrIp ,
2022-06-09 15:46:21 +00:00
enrTcpPort ,
discv5UdpPort ,
2021-12-06 19:51:37 +00:00
wakuFlags ,
enrMultiaddrs )
2021-08-12 08:51:38 +00:00
2022-01-10 15:07:35 +00:00
info " Initializing networking " , addrs = announcedAddresses
2021-11-10 12:05:36 +00:00
2021-11-02 10:29:11 +00:00
var switch = newWakuSwitch ( some ( nodekey ) ,
2021-12-06 19:51:37 +00:00
hostAddress ,
wsHostAddress ,
transportFlags = { ServerFlags . ReuseAddr } ,
rng = rng ,
maxConnections = maxConnections ,
wssEnabled = wssEnabled ,
secureKeyPath = secureKey ,
2022-02-16 16:12:09 +00:00
secureCertPath = secureCert ,
2022-03-29 08:09:48 +00:00
nameResolver = nameResolver ,
sendSignedPeerRecord = sendSignedPeerRecord )
2021-11-02 10:29:11 +00:00
2021-07-14 17:58:46 +00:00
let wakuNode = WakuNode (
2021-03-26 08:49:51 +00:00
peerManager : PeerManager . new ( switch , peerStorage ) ,
2020-09-28 21:44:14 +00:00
switch : switch ,
2022-01-10 15:07:35 +00:00
rng : rng ,
2021-08-12 08:51:38 +00:00
enr : enr ,
2022-08-12 10:15:51 +00:00
filters : Filters . init ( ) ,
2021-11-15 13:29:18 +00:00
announcedAddresses : announcedAddresses
2020-09-17 20:10:41 +00:00
)
2020-09-11 11:28:27 +00:00
2021-07-14 17:58:46 +00:00
return wakuNode
2021-04-13 05:56:49 +00:00
proc subscribe ( node : WakuNode , topic : Topic , handler : Option [ TopicHandler ] ) =
2021-04-29 04:54:31 +00:00
if node . wakuRelay . isNil :
error " Invalid API call to `subscribe`. WakuRelay not mounted. "
# @TODO improved error handling
return
2021-04-13 05:56:49 +00:00
info " subscribe " , topic = topic
proc defaultHandler ( topic : string , data : seq [ byte ] ) {. async , gcsafe . } =
# A default handler should be registered for all topics
trace " Hit default handler " , topic = topic , data = data
let msg = WakuMessage . init ( data )
if msg . isOk ( ) :
2021-07-13 07:18:51 +00:00
# Notify mounted protocols of new message
if ( not node . wakuFilter . isNil ) :
await node . wakuFilter . handleMessage ( topic , msg . value ( ) )
if ( not node . wakuStore . isNil ) :
await node . wakuStore . handleMessage ( topic , msg . value ( ) )
2022-03-15 12:39:32 +00:00
waku_node_messages . inc ( labelValues = [ " relay " ] )
2021-04-13 05:56:49 +00:00
let wakuRelay = node . wakuRelay
if topic notin PubSub ( wakuRelay ) . topics :
# Add default handler only for new topics
debug " Registering default handler " , topic = topic
wakuRelay . subscribe ( topic , defaultHandler )
if handler . isSome :
debug " Registering handler " , topic = topic
wakuRelay . subscribe ( topic , handler . get ( ) )
2021-02-02 11:33:59 +00:00
proc subscribe * ( node : WakuNode , topic : Topic , handler : TopicHandler ) =
2020-07-27 09:01:06 +00:00
## Subscribes to a PubSub topic. Triggers handler when receiving messages on
2020-08-27 10:15:46 +00:00
## this topic. TopicHandler is a method that takes a topic and some data.
2020-07-27 09:01:06 +00:00
##
2020-08-27 10:15:46 +00:00
## NOTE The data field SHOULD be decoded as a WakuMessage.
## Status: Implemented.
2021-04-13 05:56:49 +00:00
node . subscribe ( topic , some ( handler ) )
2020-07-27 09:01:06 +00:00
2020-10-02 12:48:56 +00:00
proc subscribe * ( node : WakuNode , request : FilterRequest , handler : ContentFilterHandler ) {. async , gcsafe . } =
## Registers for messages that match a specific filter. Triggers the handler whenever a message is received.
## FilterHandler is a method that takes a MessagePush.
##
## Status: Implemented.
2020-11-10 07:13:16 +00:00
# Sanity check for well-formed subscribe FilterRequest
doAssert ( request . subscribe , " invalid subscribe request " )
2020-10-02 12:48:56 +00:00
info " subscribe content " , filter = request
2020-07-27 09:01:06 +00:00
2020-10-20 02:36:27 +00:00
var id = generateRequestId ( node . rng )
2021-02-08 09:17:20 +00:00
2020-10-20 02:36:27 +00:00
if node . wakuFilter . isNil = = false :
2022-08-12 10:15:51 +00:00
let
pubsubTopic = request . pubsubTopic
contentTopics = request . contentFilters . mapIt ( it . contentTopic )
let resSubscription = await node . wakuFilter . subscribe ( pubsubTopic , contentTopics )
2020-07-27 09:01:06 +00:00
2022-08-12 10:15:51 +00:00
if resSubscription . isOk ( ) :
id = resSubscription . get ( )
2021-02-08 09:17:20 +00:00
else :
# Failed to subscribe
error " remote subscription to filter failed " , filter = request
waku_node_errors . inc ( labelValues = [ " subscribe_filter_failure " ] )
# Register handler for filter, whether remote subscription succeeded or not
2022-08-12 10:15:51 +00:00
node . filters . addContentFilters ( id , request . pubSubTopic , request . contentFilters , handler )
2021-01-29 08:42:41 +00:00
waku_node_filters . set ( node . filters . len . int64 )
2021-02-02 11:33:59 +00:00
proc unsubscribe * ( node : WakuNode , topic : Topic , handler : TopicHandler ) =
2020-10-27 01:13:56 +00:00
## Unsubscribes a handler from a PubSub topic.
2020-07-27 09:01:06 +00:00
##
2020-10-27 01:13:56 +00:00
## Status: Implemented.
2021-04-29 04:54:31 +00:00
if node . wakuRelay . isNil :
error " Invalid API call to `unsubscribe`. WakuRelay not mounted. "
# @TODO improved error handling
return
2020-10-27 01:13:56 +00:00
info " unsubscribe " , topic = topic
let wakuRelay = node . wakuRelay
2021-02-02 11:33:59 +00:00
wakuRelay . unsubscribe ( @ [ ( topic , handler ) ] )
2020-10-27 01:13:56 +00:00
2021-02-02 11:33:59 +00:00
proc unsubscribeAll * ( node : WakuNode , topic : Topic ) =
2020-10-27 01:13:56 +00:00
## Unsubscribes all handlers registered on a specific PubSub topic.
##
## Status: Implemented.
2021-04-29 04:54:31 +00:00
if node . wakuRelay . isNil :
error " Invalid API call to `unsubscribeAll`. WakuRelay not mounted. "
# @TODO improved error handling
return
2020-10-27 01:13:56 +00:00
info " unsubscribeAll " , topic = topic
let wakuRelay = node . wakuRelay
2021-02-02 11:33:59 +00:00
wakuRelay . unsubscribeAll ( topic )
2020-10-27 01:13:56 +00:00
2020-07-27 09:01:06 +00:00
2020-11-10 07:13:16 +00:00
proc unsubscribe * ( node : WakuNode , request : FilterRequest ) {. async , gcsafe . } =
2020-07-27 09:01:06 +00:00
## Unsubscribe from a content filter.
##
2020-11-10 07:13:16 +00:00
## Status: Implemented.
# Sanity check for well-formed unsubscribe FilterRequest
doAssert ( request . subscribe = = false , " invalid unsubscribe request " )
info " unsubscribe content " , filter = request
2022-08-12 10:15:51 +00:00
let
pubsubTopic = request . pubsubTopic
contentTopics = request . contentFilters . mapIt ( it . contentTopic )
discard await node . wakuFilter . unsubscribe ( pubsubTopic , contentTopics )
2020-11-10 07:13:16 +00:00
node . filters . removeContentFilters ( request . contentFilters )
2020-09-01 15:20:38 +00:00
2021-01-29 08:42:41 +00:00
waku_node_filters . set ( node . filters . len . int64 )
2020-11-16 09:55:49 +00:00
2021-10-20 00:37:29 +00:00
proc publish * ( node : WakuNode , topic : Topic , message : WakuMessage ) {. async , gcsafe . } =
2020-09-01 15:20:38 +00:00
## Publish a `WakuMessage` to a PubSub topic. `WakuMessage` should contain a
## `contentTopic` field for light node functionality. This field may be also
## be omitted.
2020-07-27 09:01:06 +00:00
##
2020-09-01 15:20:38 +00:00
## Status: Implemented.
2021-10-20 00:37:29 +00:00
2021-04-29 04:54:31 +00:00
if node . wakuRelay . isNil :
error " Invalid API call to `publish`. WakuRelay not mounted. Try `lightpush` instead. "
# @TODO improved error handling
return
2020-07-27 09:01:06 +00:00
2020-09-16 04:23:10 +00:00
let wakuRelay = node . wakuRelay
2022-02-28 17:16:50 +00:00
trace " publish " , topic = topic , contentTopic = message . contentTopic
2021-03-16 18:18:40 +00:00
var publishingMessage = message
2020-09-02 03:15:25 +00:00
let data = message . encode ( ) . buffer
2020-12-02 08:40:53 +00:00
discard await wakuRelay . publish ( topic , data )
2020-07-27 09:01:06 +00:00
2022-09-02 08:17:14 +00:00
proc lightpush * ( node : WakuNode , topic : Topic , message : WakuMessage ) : Future [ WakuLightpushResult [ PushResponse ] ] {. async , gcsafe . } =
2021-04-24 04:56:37 +00:00
## Pushes a `WakuMessage` to a node which relays it further on PubSub topic.
2022-09-02 08:17:14 +00:00
## Returns whether relaying was successful or not.
2021-04-24 04:56:37 +00:00
## `WakuMessage` should contain a `contentTopic` field for light node
2022-09-02 08:17:14 +00:00
## functionality.
2021-04-24 04:56:37 +00:00
debug " Publishing with lightpush " , topic = topic , contentTopic = message . contentTopic
let rpc = PushRequest ( pubSubTopic : topic , message : message )
2022-09-02 08:17:14 +00:00
return await node . wakuLightPush . request ( rpc )
2021-04-24 04:56:37 +00:00
2022-09-02 08:17:14 +00:00
proc lightpush * ( node : WakuNode , topic : Topic , message : WakuMessage , handler : PushResponseHandler ) {. async , gcsafe ,
deprecated : " Use the no-callback version of this method " . } =
## Pushes a `WakuMessage` to a node which relays it further on PubSub topic.
## Returns whether relaying was successful or not in `handler`.
## `WakuMessage` should contain a `contentTopic` field for light node
## functionality.
let rpc = PushRequest ( pubSubTopic : topic , message : message )
let res = await node . wakuLightPush . request ( rpc )
if res . isOk ( ) :
handler ( res . value )
else :
error " Message lightpush failed " , error = res . error ( )
proc query * ( node : WakuNode , query : HistoryQuery ) : Future [ WakuStoreResult [ HistoryResponse ] ] {. async , gcsafe . } =
## Queries known nodes for historical messages
2020-10-06 03:33:28 +00:00
2022-09-02 08:17:14 +00:00
# TODO: Once waku swap is less experimental, this can simplified
2020-11-21 05:31:48 +00:00
if node . wakuSwap . isNil :
debug " Using default query "
2022-09-02 08:17:14 +00:00
return await node . wakuStore . query ( query )
else :
debug " Using SWAP accounting query "
# TODO: wakuSwap now part of wakuStore object
return await node . wakuStore . queryWithAccounting ( query )
proc query * ( node : WakuNode , query : HistoryQuery , handler : QueryHandlerFunc ) {. async , gcsafe ,
deprecated : " Use the no-callback version of this method " . } =
## Queries known nodes for historical messages. Triggers the handler whenever a response is received.
## QueryHandlerFunc is a method that takes a HistoryResponse.
let res = await node . query ( query )
if res . isOk ( ) :
handler ( res . value )
2020-11-21 05:31:48 +00:00
else :
2022-09-02 08:17:14 +00:00
error " History query failed " , error = res . error ( )
2020-11-16 09:55:49 +00:00
2021-10-06 12:29:08 +00:00
proc resume * ( node : WakuNode , peerList : Option [ seq [ RemotePeerInfo ] ] = none ( seq [ RemotePeerInfo ] ) ) {. async , gcsafe . } =
2021-05-26 19:33:22 +00:00
## resume proc retrieves the history of waku messages published on the default waku pubsub topic since the last time the waku node has been online
## for resume to work properly the waku node must have the store protocol mounted in the full mode (i.e., persisting messages)
## messages are stored in the the wakuStore's messages field and in the message db
## the offline time window is measured as the difference between the current time and the timestamp of the most recent persisted waku message
## an offset of 20 second is added to the time window to count for nodes asynchrony
## peerList indicates the list of peers to query from. The history is fetched from the first available peer in this list. Such candidates should be found through a discovery method (to be developed).
## if no peerList is passed, one of the peers in the underlying peer manager unit of the store protocol is picked randomly to fetch the history from.
## The history gets fetched successfully if the dialed peer has been online during the queried time window.
if not node . wakuStore . isNil :
let retrievedMessages = await node . wakuStore . resume ( peerList )
if retrievedMessages . isOk :
info " the number of retrieved messages since the last online time: " , number = retrievedMessages . value
2020-10-06 03:33:28 +00:00
# TODO Extend with more relevant info: topics, peers, memory usage, online time, etc
proc info * ( node : WakuNode ) : WakuInfo =
## Returns information about the Node, such as what multiaddress it can be reached at.
##
## Status: Implemented.
##
2022-01-10 15:07:35 +00:00
let peerInfo = node . switch . peerInfo
2021-12-06 19:51:37 +00:00
2021-11-24 15:10:29 +00:00
var listenStr : seq [ string ]
for address in node . announcedAddresses :
var fulladdr = $ address & " /p2p/ " & $ peerInfo . peerId
listenStr & = fulladdr
2022-09-08 13:25:02 +00:00
let enrUri = node . enr . toUri ( )
2022-03-30 13:10:15 +00:00
let wakuInfo = WakuInfo ( listenAddresses : listenStr , enrUri : enrUri )
2020-10-06 03:33:28 +00:00
return wakuInfo
2020-07-27 09:01:06 +00:00
2022-09-07 15:31:27 +00:00
proc mountFilter * ( node : WakuNode , filterTimeout : Duration = WakuFilterTimeout ) {. async , raises : [ Defect , LPError ] } =
2020-10-20 02:36:27 +00:00
info " mounting filter "
2022-08-12 10:15:51 +00:00
proc filterHandler ( requestId : string , msg : MessagePush ) {. async , gcsafe . } =
2021-07-14 17:58:46 +00:00
2020-10-20 02:36:27 +00:00
info " push received "
for message in msg . messages :
2021-04-22 13:45:13 +00:00
node . filters . notify ( message , requestId ) # Trigger filter handlers on a light node
2022-05-16 10:51:10 +00:00
if not node . wakuStore . isNil and ( requestId in node . filters ) :
let pubSubTopic = node . filters [ requestId ] . pubSubTopic
await node . wakuStore . handleMessage ( pubSubTopic , message )
2022-03-15 12:39:32 +00:00
waku_node_messages . inc ( labelValues = [ " filter " ] )
2020-10-20 02:36:27 +00:00
2021-12-08 18:35:47 +00:00
node . wakuFilter = WakuFilter . init ( node . peerManager , node . rng , filterHandler , filterTimeout )
2022-09-07 15:31:27 +00:00
if node . started :
# Node has started already. Let's start filter too.
await node . wakuFilter . start ( )
2021-07-02 08:49:41 +00:00
node . switch . mount ( node . wakuFilter , protocolMatcher ( WakuFilterCodec ) )
2020-10-20 02:36:27 +00:00
2022-09-07 15:31:27 +00:00
2020-11-24 04:53:42 +00:00
# NOTE: If using the swap protocol, it must be mounted before store. This is
# because store is using a reference to the swap protocol.
2022-09-07 15:31:27 +00:00
proc mountSwap * ( node : WakuNode , swapConfig : SwapConfig = SwapConfig . init ( ) ) {. async , raises : [ Defect , LPError ] . } =
2021-06-15 02:06:36 +00:00
info " mounting swap " , mode = $ swapConfig . mode
2022-09-07 15:31:27 +00:00
2021-06-15 02:06:36 +00:00
node . wakuSwap = WakuSwap . init ( node . peerManager , node . rng , swapConfig )
2022-09-07 15:31:27 +00:00
if node . started :
# Node has started already. Let's start swap too.
await node . wakuSwap . start ( )
2021-07-02 08:49:41 +00:00
node . switch . mount ( node . wakuSwap , protocolMatcher ( WakuSwapCodec ) )
2020-11-18 12:45:51 +00:00
2022-09-16 10:55:22 +00:00
proc mountStore * ( node : WakuNode , store : MessageStore = nil , persistMessages : bool = false , capacity = StoreDefaultCapacity , retentionTime = StoreDefaultRetentionTime , isSqliteOnly = false ) {. async , raises : [ Defect , LPError ] . } =
if node . wakuSwap . isNil ( ) :
info " mounting waku store protocol (no waku swap) "
2020-11-24 04:53:42 +00:00
else :
2022-09-16 10:55:22 +00:00
info " mounting waku store protocol with waku swap support "
let retentionPolicy = if isSqliteOnly : TimeRetentionPolicy . init ( retentionTime )
else : CapacityRetentionPolicy . init ( capacity )
node . wakuStore = WakuStore . init (
node . peerManager ,
node . rng ,
store ,
wakuSwap = node . wakuSwap ,
persistMessages = persistMessages ,
capacity = capacity ,
isSqliteOnly = isSqliteOnly ,
retentionPolicy = some ( retentionPolicy )
)
2022-09-07 15:31:27 +00:00
if node . started :
# Node has started already. Let's start store too.
await node . wakuStore . start ( )
2020-11-24 04:53:42 +00:00
2021-07-02 08:49:41 +00:00
node . switch . mount ( node . wakuStore , protocolMatcher ( WakuStoreCodec ) )
2021-06-08 18:56:32 +00:00
2021-07-21 09:37:10 +00:00
2021-07-22 09:46:54 +00:00
proc startRelay * ( node : WakuNode ) {. async . } =
if node . wakuRelay . isNil :
trace " Failed to start relay. Not mounted. "
return
## Setup and start relay protocol
info " starting relay "
# Topic subscriptions
for topic in node . wakuRelay . defaultTopics :
node . subscribe ( topic , none ( TopicHandler ) )
# Resume previous relay connections
2021-07-27 06:48:56 +00:00
if node . peerManager . hasPeers ( protocolMatcher ( WakuRelayCodec ) ) :
2021-07-22 09:46:54 +00:00
info " Found previous WakuRelay peers. Reconnecting. "
# Reconnect to previous relay peers. This will respect a backoff period, if necessary
let backoffPeriod = node . wakuRelay . parameters . pruneBackoff + chronos . seconds ( BackoffSlackTime )
await node . peerManager . reconnectPeers ( WakuRelayCodec ,
2021-07-27 06:48:56 +00:00
protocolMatcher ( WakuRelayCodec ) ,
2021-07-22 09:46:54 +00:00
backoffPeriod )
# Start the WakuRelay protocol
await node . wakuRelay . start ( )
info " relay started successfully "
2021-05-04 12:11:41 +00:00
proc mountRelay * ( node : WakuNode ,
topics : seq [ string ] = newSeq [ string ] ( ) ,
2022-03-29 08:09:48 +00:00
triggerSelf = true ,
peerExchangeHandler = none ( RoutingRecordsHandler ) )
2021-07-14 17:58:46 +00:00
# @TODO: Better error handling: CatchableError is raised by `waitFor`
2022-09-07 15:31:27 +00:00
{. async , gcsafe , raises : [ Defect , InitializationError , LPError , CatchableError ] . } =
2022-01-10 14:07:01 +00:00
2022-03-02 12:24:48 +00:00
proc msgIdProvider ( m : messages . Message ) : Result [ MessageID , ValidationResult ] =
2022-01-10 14:07:01 +00:00
let mh = MultiHash . digest ( " sha2-256 " , m . data )
if mh . isOk ( ) :
2022-03-02 12:24:48 +00:00
return ok ( mh [ ] . data . buffer )
2022-01-10 14:07:01 +00:00
else :
2022-03-02 12:24:48 +00:00
return ok ( ( $ m . data . hash ) . toBytes ( ) )
2021-07-14 17:58:46 +00:00
2020-10-20 02:36:27 +00:00
let wakuRelay = WakuRelay . init (
switch = node . switch ,
2022-01-10 14:07:01 +00:00
msgIdProvider = msgIdProvider ,
2021-05-24 11:19:33 +00:00
triggerSelf = triggerSelf ,
2020-10-20 02:36:27 +00:00
sign = false ,
2022-01-06 12:42:37 +00:00
verifySignature = false ,
maxMessageSize = MaxWakuMessageSize
2020-10-20 02:36:27 +00:00
)
2021-05-04 12:11:41 +00:00
2022-09-07 15:31:27 +00:00
info " mounting relay "
2020-10-20 02:36:27 +00:00
2021-07-22 09:46:54 +00:00
## The default relay topics is the union of
## all configured topics plus the hard-coded defaultTopic(s)
wakuRelay . defaultTopics = concat ( @ [ defaultTopic ] , topics )
2022-03-29 08:09:48 +00:00
## Add peer exchange handler
2022-05-20 10:57:19 +00:00
if peerExchangeHandler . isSome ( ) :
wakuRelay . parameters . enablePX = true # Feature flag for peer exchange in nim-libp2p
2022-03-29 08:09:48 +00:00
wakuRelay . routingRecordsHandler . add ( peerExchangeHandler . get ( ) )
2022-09-07 15:31:27 +00:00
node . wakuRelay = wakuRelay
if node . started :
# Node has started already. Let's start relay too.
await node . startRelay ( )
2020-10-20 02:36:27 +00:00
2022-09-07 15:31:27 +00:00
node . switch . mount ( wakuRelay , protocolMatcher ( WakuRelayCodec ) )
2021-09-17 17:31:25 +00:00
2021-07-22 09:46:54 +00:00
info " relay mounted successfully "
2021-04-01 09:37:14 +00:00
2022-09-07 15:31:27 +00:00
proc mountLightPush * ( node : WakuNode ) {. async , raises : [ Defect , LPError ] . } =
2021-04-24 04:56:37 +00:00
info " mounting light push "
if node . wakuRelay . isNil :
debug " mounting lightpush without relay "
node . wakuLightPush = WakuLightPush . init ( node . peerManager , node . rng , nil )
else :
debug " mounting lightpush with relay "
node . wakuLightPush = WakuLightPush . init ( node . peerManager , node . rng , nil , node . wakuRelay )
2021-07-02 08:49:41 +00:00
2022-09-07 15:31:27 +00:00
if node . started :
# Node has started already. Let's start lightpush too.
await node . wakuLightPush . start ( )
2021-07-02 08:49:41 +00:00
node . switch . mount ( node . wakuLightPush , protocolMatcher ( WakuLightPushCodec ) )
2021-04-24 04:56:37 +00:00
2022-09-07 15:31:27 +00:00
proc mountLibp2pPing * ( node : WakuNode ) {. async , raises : [ Defect , LPError ] . } =
2021-06-15 08:55:47 +00:00
info " mounting libp2p ping protocol "
2021-06-02 07:53:34 +00:00
2021-07-14 17:58:46 +00:00
try :
node . libp2pPing = Ping . new ( rng = node . rng )
except Exception as e :
# This is necessary as `Ping.new*` does not have explicit `raises` requirement
# @TODO: remove exception handling once explicit `raises` in ping module
2021-07-22 09:46:54 +00:00
raise newException ( LPError , " Failed to initialize ping protocol " )
2022-09-07 15:31:27 +00:00
if node . started :
# Node has started already. Let's start ping too.
await node . libp2pPing . start ( )
2021-06-15 08:55:47 +00:00
node . switch . mount ( node . libp2pPing )
2021-06-02 07:53:34 +00:00
proc keepaliveLoop ( node : WakuNode , keepalive : chronos . Duration ) {. async . } =
while node . started :
2021-06-15 08:55:47 +00:00
# Keep all connected peers alive while running
2021-06-02 07:53:34 +00:00
trace " Running keepalive "
2021-06-15 08:55:47 +00:00
# First get a list of connected peer infos
let peers = node . peerManager . peers ( )
. filterIt ( node . peerManager . connectedness ( it . peerId ) = = Connected )
2021-10-06 12:29:08 +00:00
. mapIt ( it . toRemotePeerInfo ( ) )
2021-06-15 08:55:47 +00:00
# Attempt to retrieve and ping the active outgoing connection for each peer
for peer in peers :
let connOpt = await node . peerManager . dialPeer ( peer , PingCodec )
if connOpt . isNone :
# @TODO more sophisticated error handling here
debug " failed to connect to remote peer " , peer = peer
waku_node_errors . inc ( labelValues = [ " keep_alive_failure " ] )
return
discard await node . libp2pPing . ping ( connOpt . get ( ) ) # Ping connection
2021-06-02 07:53:34 +00:00
await sleepAsync ( keepalive )
proc startKeepalive * ( node : WakuNode ) =
2022-06-07 20:44:58 +00:00
let defaultKeepalive = 2 . minutes # 20% of the default chronosstream timeout duration
2021-06-02 07:53:34 +00:00
info " starting keepalive " , keepalive = defaultKeepalive
asyncSpawn node . keepaliveLoop ( defaultKeepalive )
2020-10-21 09:54:29 +00:00
## Helpers
2022-01-26 11:02:57 +00:00
proc connectToNode ( n : WakuNode , remotePeer : RemotePeerInfo , source = " api " ) {. async . } =
## `source` indicates source of node addrs (static config, api call, discovery, etc)
info " Connecting to node " , remotePeer = remotePeer , source = source
2020-10-21 09:54:29 +00:00
# NOTE This is dialing on WakuRelay protocol specifically
2022-01-26 11:02:57 +00:00
info " Attempting dial " , wireAddr = remotePeer . addrs [ 0 ] , peerId = remotePeer . peerId
let connOpt = await n . peerManager . dialPeer ( remotePeer , WakuRelayCodec )
if connOpt . isSome ( ) :
info " Successfully connected to peer " , wireAddr = remotePeer . addrs [ 0 ] , peerId = remotePeer . peerId
waku_node_conns_initiated . inc ( labelValues = [ source ] )
else :
error " Failed to connect to peer " , wireAddr = remotePeer . addrs [ 0 ] , peerId = remotePeer . peerId
waku_node_errors . inc ( labelValues = [ " conn_init_failure " ] )
2020-09-11 04:16:45 +00:00
2022-09-01 15:02:58 +00:00
proc setStorePeer * ( n : WakuNode , peer : RemotePeerInfo ) =
n . wakuStore . setPeer ( peer )
2021-07-14 17:58:46 +00:00
proc setStorePeer * ( n : WakuNode , address : string ) {. raises : [ Defect , ValueError , LPError ] . } =
2021-06-04 10:02:47 +00:00
info " Set store peer " , address = address
2020-09-11 04:16:45 +00:00
2022-09-01 15:02:58 +00:00
let peer = parseRemotePeerInfo ( address )
n . setStorePeer ( peer )
2020-09-25 11:35:32 +00:00
2022-09-01 15:02:58 +00:00
proc setFilterPeer * ( n : WakuNode , peer : RemotePeerInfo ) =
n . wakuFilter . setPeer ( peer )
2020-09-25 11:35:32 +00:00
2021-07-14 17:58:46 +00:00
proc setFilterPeer * ( n : WakuNode , address : string ) {. raises : [ Defect , ValueError , LPError ] . } =
2021-06-04 10:02:47 +00:00
info " Set filter peer " , address = address
2020-09-25 11:35:32 +00:00
2022-09-01 15:02:58 +00:00
let peer = parseRemotePeerInfo ( address )
n . setFilterPeer ( peer )
2020-10-09 13:58:50 +00:00
2022-09-01 15:02:58 +00:00
proc setLightPushPeer * ( n : WakuNode , peer : RemotePeerInfo ) =
n . wakuLightPush . setPeer ( peer )
2020-10-09 13:58:50 +00:00
2021-07-14 17:58:46 +00:00
proc setLightPushPeer * ( n : WakuNode , address : string ) {. raises : [ Defect , ValueError , LPError ] . } =
2021-06-04 10:02:47 +00:00
info " Set lightpush peer " , address = address
2022-09-01 15:02:58 +00:00
let peer = parseRemotePeerInfo ( address )
n . wakuLightPush . setPeer ( peer )
2021-06-04 10:02:47 +00:00
2022-01-26 11:02:57 +00:00
proc connectToNodes * ( n : WakuNode , nodes : seq [ string ] , source = " api " ) {. async . } =
## `source` indicates source of node addrs (static config, api call, discovery, etc)
info " connectToNodes " , len = nodes . len
2020-10-21 09:54:29 +00:00
for nodeId in nodes :
2022-01-26 11:02:57 +00:00
await connectToNode ( n , parseRemotePeerInfo ( nodeId ) , source )
2020-10-22 11:12:00 +00:00
# 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 ( 5 . seconds )
2022-01-26 11:02:57 +00:00
proc connectToNodes * ( n : WakuNode , nodes : seq [ RemotePeerInfo ] , source = " api " ) {. async . } =
## `source` indicates source of node addrs (static config, api call, discovery, etc)
info " connectToNodes " , len = nodes . len
2021-10-06 12:29:08 +00:00
for remotePeerInfo in nodes :
2022-01-26 11:02:57 +00:00
await connectToNode ( n , remotePeerInfo , source )
2020-10-22 11:12:00 +00:00
# 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 ( 5 . seconds )
2020-10-09 13:58:50 +00:00
2021-11-01 18:02:39 +00:00
proc runDiscv5Loop ( node : WakuNode ) {. async . } =
## Continuously add newly discovered nodes
## using Node Discovery v5
if ( node . wakuDiscv5 . isNil ) :
warn " Trying to run discovery v5 while it ' s disabled "
return
info " Starting discovery loop "
while node . wakuDiscv5 . listening :
trace " Running discovery loop "
## Query for a random target and collect all discovered nodes
## @TODO: we could filter nodes here
let discoveredPeers = await node . wakuDiscv5 . findRandomPeers ( )
if discoveredPeers . isOk :
## Let's attempt to connect to peers we
## have not encountered before
trace " Discovered peers " , count = discoveredPeers . get ( ) . len ( )
let newPeers = discoveredPeers . get ( ) . filterIt (
2022-06-01 09:49:41 +00:00
not node . switch . peerStore [ AddressBook ] . contains ( it . peerId ) )
2021-11-01 18:02:39 +00:00
if newPeers . len > 0 :
debug " Connecting to newly discovered peers " , count = newPeers . len ( )
2022-01-26 11:02:57 +00:00
await connectToNodes ( node , newPeers , " discv5 " )
2021-11-01 18:02:39 +00:00
# Discovery `queryRandom` can have a synchronous fast path for example
# when no peers are in the routing table. Don't run it in continuous loop.
#
# Also, give some time to dial the discovered nodes and update stats etc
await sleepAsync ( 5 . seconds )
proc startDiscv5 * ( node : WakuNode ) : Future [ bool ] {. async . } =
## Start Discovery v5 service
info " Starting discovery v5 service "
if not node . wakuDiscv5 . isNil :
## First start listening on configured port
try :
trace " Start listening on discv5 port "
node . wakuDiscv5 . open ( )
except CatchableError :
error " Failed to start discovery service. UDP port may be already in use "
return false
## Start Discovery v5
trace " Start discv5 service "
node . wakuDiscv5 . start ( )
trace " Start discovering new peers using discv5 "
asyncSpawn node . runDiscv5Loop ( )
debug " Successfully started discovery v5 service "
2022-03-30 13:10:15 +00:00
info " Discv5: discoverable ENR " , enr = node . wakuDiscV5 . protocol . localNode . record . toUri ( )
2021-11-01 18:02:39 +00:00
return true
return false
proc stopDiscv5 * ( node : WakuNode ) : Future [ bool ] {. async . } =
## Stop Discovery v5 service
if not node . wakuDiscv5 . isNil :
info " Stopping discovery v5 service "
## Stop Discovery v5 process and close listening port
if node . wakuDiscv5 . listening :
trace " Stop listening on discv5 port "
await node . wakuDiscv5 . closeWait ( )
debug " Successfully stopped discovery v5 service "
2021-07-22 09:46:54 +00:00
proc start * ( node : WakuNode ) {. async . } =
## Starts a created Waku Node and
## all its mounted protocols.
##
## Status: Implemented.
# TODO Get this from WakuNode obj
2022-01-10 15:07:35 +00:00
let peerInfo = node . switch . peerInfo
2021-07-22 09:46:54 +00:00
info " PeerInfo " , peerId = peerInfo . peerId , addrs = peerInfo . addrs
2021-11-15 13:29:18 +00:00
var listenStr = " "
for address in node . announcedAddresses :
var fulladdr = " [ " & $ address & " /p2p/ " & $ peerInfo . peerId & " ] "
listenStr & = fulladdr
2021-07-22 09:46:54 +00:00
## XXX: this should be /ip4..., / stripped?
info " Listening on " , full = listenStr
2022-03-30 13:10:15 +00:00
info " DNS: discoverable ENR " , enr = node . enr . toUri ( )
2021-07-22 09:46:54 +00:00
2021-11-30 09:12:09 +00:00
## Update switch peer info with announced addrs
node . updateSwitchPeerInfo ( )
2022-09-07 15:31:27 +00:00
# Start mounted protocols. For now we start each one explicitly
2021-07-22 09:46:54 +00:00
if not node . wakuRelay . isNil :
await node . startRelay ( )
2022-09-07 15:31:27 +00:00
if not node . wakuStore . isNil :
await node . wakuStore . start ( )
if not node . wakuFilter . isNil :
await node . wakuFilter . start ( )
if not node . wakuLightPush . isNil :
await node . wakuLightPush . start ( )
if not node . wakuSwap . isNil :
await node . wakuSwap . start ( )
if not node . libp2pPing . isNil :
await node . libp2pPing . start ( )
2021-07-22 09:46:54 +00:00
2022-09-07 15:31:27 +00:00
await node . switch . start ( )
2021-07-22 09:46:54 +00:00
node . started = true
2022-09-07 15:31:27 +00:00
info " Node started successfully "
2021-07-22 09:46:54 +00:00
proc stop * ( node : WakuNode ) {. async . } =
if not node . wakuRelay . isNil :
await node . wakuRelay . stop ( )
2021-11-01 18:02:39 +00:00
if not node . wakuDiscv5 . isNil :
discard await node . stopDiscv5 ( )
2021-07-22 09:46:54 +00:00
await node . switch . stop ( )
node . started = false
2021-07-14 17:58:46 +00:00
{. pop . } # @TODO confutils.nim(775, 17) Error: can raise an unlisted exception: ref IOError
2020-10-21 09:54:29 +00:00
when isMainModule :
2021-07-22 09:46:54 +00:00
## Node setup happens in 6 phases:
## 1. Set up storage
## 2. Initialize node
## 3. Mount and initialize configured protocols
## 4. Start node and mounted protocols
## 5. Start monitoring tools and external interfaces
## 6. Setup graceful shutdown hooks
2020-10-21 09:54:29 +00:00
import
2022-05-17 15:48:08 +00:00
confutils , toml_serialization ,
2021-04-15 08:18:14 +00:00
system / ansi_c ,
2022-02-16 16:12:09 +00:00
libp2p / nameresolving / dnsresolver ,
2021-07-22 09:46:54 +00:00
.. / .. / common / utils / nat ,
. / config ,
2022-07-19 11:11:46 +00:00
. / wakunode2_setup ,
2022-06-28 10:22:59 +00:00
. / wakunode2_setup_rest ,
2022-07-17 15:16:57 +00:00
. / wakunode2_setup_metrics ,
2022-07-17 14:25:21 +00:00
. / wakunode2_setup_rpc ,
2022-07-19 09:19:18 +00:00
. / wakunode2_setup_sql_migrations ,
2022-09-14 16:09:08 +00:00
. / storage / sqlite ,
2022-09-13 11:36:04 +00:00
. / storage / message / sqlite_store ,
2021-07-22 09:46:54 +00:00
. / storage / peer / waku_peer_storage
logScope :
topics = " wakunode.setup "
###################
# Setup functions #
###################
2022-03-17 16:33:17 +00:00
# 1/7 Setup storage
2021-07-22 09:46:54 +00:00
proc setupStorage ( conf : WakuNodeConf ) :
2022-09-13 11:36:04 +00:00
SetupResult [ tuple [ pStorage : WakuPeerStorage , mStorage : SqliteStore ] ] =
2021-07-22 09:46:54 +00:00
## Setup a SQLite Database for a wakunode based on a supplied
## configuration file and perform all necessary migration.
##
## If config allows, return peer storage and message store
## for use elsewhere.
var
sqliteDatabase : SqliteDatabase
2022-09-13 11:36:04 +00:00
storeTuple : tuple [ pStorage : WakuPeerStorage , mStorage : SqliteStore ]
2021-07-22 09:46:54 +00:00
# Setup DB
if conf . dbPath ! = " " :
let dbRes = SqliteDatabase . init ( conf . dbPath )
if dbRes . isErr :
warn " failed to init database " , err = dbRes . error
waku_node_errors . inc ( labelValues = [ " init_db_failure " ] )
return err ( " failed to init database " )
else :
sqliteDatabase = dbRes . value
2022-05-18 19:44:53 +00:00
if not sqliteDatabase . isNil and ( conf . persistPeers or conf . persistMessages ) :
2022-09-14 16:09:08 +00:00
## Database vacuuming
# TODO: Wrap and move this logic to the appropriate module
let
pageSize = ? sqliteDatabase . getPageSize ( )
pageCount = ? sqliteDatabase . getPageCount ( )
freelistCount = ? sqliteDatabase . getFreelistCount ( )
debug " sqlite database page stats " , pageSize = pageSize , pages = pageCount , freePages = freelistCount
# TODO: Run vacuuming conditionally based on database page stats
if conf . dbVacuum and ( pageCount > 0 and freelistCount > 0 ) :
debug " starting sqlite database vacuuming "
let resVacuum = sqliteDatabase . vacuum ( )
if resVacuum . isErr ( ) :
return err ( " failed to execute vacuum: " & resVacuum . error ( ) )
debug " finished sqlite database vacuuming "
2021-07-22 09:46:54 +00:00
# Database initialized. Let's set it up
sqliteDatabase . runMigrations ( conf ) # First migrate what we have
if conf . persistPeers :
# Peer persistence enable. Set up Peer table in storage
let res = WakuPeerStorage . new ( sqliteDatabase )
if res . isErr :
warn " failed to init new WakuPeerStorage " , err = res . error
waku_node_errors . inc ( labelValues = [ " init_store_failure " ] )
else :
storeTuple . pStorage = res . value
if conf . persistMessages :
# Historical message persistence enable. Set up Message table in storage
2022-09-16 10:55:22 +00:00
let res = SqliteStore . init ( sqliteDatabase )
2022-09-13 11:36:04 +00:00
if res . isErr ( ) :
warn " failed to init SqliteStore " , err = res . error
2021-07-22 09:46:54 +00:00
waku_node_errors . inc ( labelValues = [ " init_store_failure " ] )
else :
storeTuple . mStorage = res . value
2022-09-14 16:09:08 +00:00
2021-07-22 09:46:54 +00:00
ok ( storeTuple )
2022-03-17 16:33:17 +00:00
# 2/7 Retrieve dynamic bootstrap nodes
proc retrieveDynamicBootstrapNodes ( conf : WakuNodeConf ) : SetupResult [ seq [ RemotePeerInfo ] ] =
2022-06-08 09:20:18 +00:00
2022-03-17 16:33:17 +00:00
if conf . dnsDiscovery and conf . dnsDiscoveryUrl ! = " " :
2022-06-08 09:20:18 +00:00
# DNS discovery
2022-03-17 16:33:17 +00:00
debug " Discovering nodes using Waku DNS discovery " , url = conf . dnsDiscoveryUrl
var nameServers : seq [ TransportAddress ]
for ip in conf . dnsDiscoveryNameServers :
nameServers . add ( initTAddress ( ip , Port ( 53 ) ) ) # Assume all servers use port 53
let dnsResolver = DnsResolver . new ( nameServers )
proc resolver ( domain : string ) : Future [ string ] {. async , gcsafe . } =
trace " resolving " , domain = domain
let resolved = await dnsResolver . resolveTxt ( domain )
return resolved [ 0 ] # Use only first answer
var wakuDnsDiscovery = WakuDnsDiscovery . init ( conf . dnsDiscoveryUrl ,
resolver )
if wakuDnsDiscovery . isOk :
return wakuDnsDiscovery . get ( ) . findPeers ( )
. mapErr ( proc ( e : cstring ) : string = $ e )
else :
warn " Failed to init Waku DNS discovery "
2022-06-08 20:23:13 +00:00
debug " No method for retrieving dynamic bootstrap nodes specified. "
2022-06-08 09:20:18 +00:00
ok ( newSeq [ RemotePeerInfo ] ( ) ) # Return an empty seq by default
2022-03-17 16:33:17 +00:00
# 3/7 Initialize node
2021-07-22 09:46:54 +00:00
proc initNode ( conf : WakuNodeConf ,
2022-03-17 16:33:17 +00:00
pStorage : WakuPeerStorage = nil ,
dynamicBootstrapNodes : openArray [ RemotePeerInfo ] = @ [ ] ) : SetupResult [ WakuNode ] =
2021-07-22 09:46:54 +00:00
## Setup a basic Waku v2 node based on a supplied configuration
## file. Optionally include persistent peer storage.
## No protocols are mounted yet.
2020-09-11 04:16:45 +00:00
2022-02-16 16:12:09 +00:00
var dnsResolver : DnsResolver
if conf . dnsAddrs :
# Support for DNS multiaddrs
var nameServers : seq [ TransportAddress ]
for ip in conf . dnsAddrsNameServers :
nameServers . add ( initTAddress ( ip , Port ( 53 ) ) ) # Assume all servers use port 53
dnsResolver = DnsResolver . new ( nameServers )
2021-11-01 18:02:39 +00:00
let
## `udpPort` is only supplied to satisfy underlying APIs but is not
## actually a supported transport for libp2p traffic.
udpPort = conf . tcpPort
2021-07-22 09:46:54 +00:00
( extIp , extTcpPort , extUdpPort ) = setupNat ( conf . nat ,
clientId ,
Port ( uint16 ( conf . tcpPort ) + conf . portsShift ) ,
2021-10-12 11:43:01 +00:00
Port ( uint16 ( udpPort ) + conf . portsShift ) )
2022-05-18 08:15:03 +00:00
dns4DomainName = if conf . dns4DomainName ! = " " : some ( conf . dns4DomainName )
else : none ( string )
2022-06-09 15:46:21 +00:00
discv5UdpPort = if conf . discv5Discovery : some ( Port ( uint16 ( conf . discv5UdpPort ) + conf . portsShift ) )
else : none ( Port )
2022-05-18 08:15:03 +00:00
2021-07-22 09:46:54 +00:00
## @TODO: the NAT setup assumes a manual port mapping configuration if extIp config is set. This probably
## implies adding manual config item for extPort as well. The following heuristic assumes that, in absence of manual
## config, the external port is the same as the bind port.
2022-05-18 08:15:03 +00:00
extPort = if ( extIp . isSome ( ) or dns4DomainName . isSome ( ) ) and extTcpPort . isNone ( ) :
2021-07-22 09:46:54 +00:00
some ( Port ( uint16 ( conf . tcpPort ) + conf . portsShift ) )
else :
extTcpPort
2021-12-06 19:51:37 +00:00
wakuFlags = initWakuFlags ( conf . lightpush ,
conf . filter ,
conf . store ,
conf . relay )
2021-11-01 18:02:39 +00:00
2021-11-10 12:05:36 +00:00
node = WakuNode . new ( conf . nodekey ,
2022-02-16 16:12:09 +00:00
conf . listenAddress , Port ( uint16 ( conf . tcpPort ) + conf . portsShift ) ,
extIp , extPort ,
pStorage ,
conf . maxConnections . int ,
Port ( uint16 ( conf . websocketPort ) + conf . portsShift ) ,
conf . websocketSupport ,
conf . websocketSecureSupport ,
conf . websocketSecureKeyPath ,
conf . websocketSecureCertPath ,
some ( wakuFlags ) ,
2022-02-18 11:10:38 +00:00
dnsResolver ,
2022-03-29 08:09:48 +00:00
conf . relayPeerExchange , # We send our own signed peer record when peer exchange enabled
2022-06-09 15:46:21 +00:00
dns4DomainName ,
discv5UdpPort
2022-02-16 16:12:09 +00:00
)
2021-11-01 18:02:39 +00:00
if conf . discv5Discovery :
2022-03-01 14:11:56 +00:00
let
discoveryConfig = DiscoveryConfig . init (
conf . discv5TableIpLimit , conf . discv5BucketIpLimit , conf . discv5BitsPerHop )
2021-11-01 18:02:39 +00:00
2022-03-17 16:33:17 +00:00
# select dynamic bootstrap nodes that have an ENR containing a udp port.
# Discv5 only supports UDP https://github.com/ethereum/devp2p/blob/master/discv5/discv5-theory.md)
var discv5BootstrapEnrs : seq [ enr . Record ]
for n in dynamicBootstrapNodes :
if n . enr . isSome ( ) :
let
enr = n . enr . get ( )
tenrRes = enr . toTypedRecord ( )
if tenrRes . isOk ( ) and ( tenrRes . get ( ) . udp . isSome ( ) or tenrRes . get ( ) . udp6 . isSome ( ) ) :
discv5BootstrapEnrs . add ( enr )
# parse enrURIs from the configuration and add the resulting ENRs to the discv5BootstrapEnrs seq
for enrUri in conf . discv5BootstrapNodes :
addBootstrapNode ( enrUri , discv5BootstrapEnrs )
2021-11-01 18:02:39 +00:00
node . wakuDiscv5 = WakuDiscoveryV5 . new (
2022-06-09 15:46:21 +00:00
extIP , extPort , discv5UdpPort ,
2021-11-01 18:02:39 +00:00
conf . listenAddress ,
2022-06-09 15:46:21 +00:00
discv5UdpPort . get ( ) ,
2022-03-17 16:33:17 +00:00
discv5BootstrapEnrs ,
2021-11-01 18:02:39 +00:00
conf . discv5EnrAutoUpdate ,
keys . PrivateKey ( conf . nodekey . skkey ) ,
2021-12-06 19:51:37 +00:00
wakuFlags ,
2021-11-01 18:02:39 +00:00
[ ] , # Empty enr fields, for now
2022-03-01 14:11:56 +00:00
node . rng ,
discoveryConfig
2021-11-01 18:02:39 +00:00
)
2021-07-22 09:46:54 +00:00
ok ( node )
2022-03-17 16:33:17 +00:00
# 4/7 Mount and initialize configured protocols
2022-03-29 08:09:48 +00:00
proc setupProtocols ( node : WakuNode ,
2021-11-01 18:02:39 +00:00
conf : WakuNodeConf ,
2022-09-13 11:36:04 +00:00
mStorage : SqliteStore = nil ) : SetupResult [ bool ] =
2020-12-24 08:02:30 +00:00
2021-07-22 09:46:54 +00:00
## Setup configured protocols on an existing Waku v2 node.
## Optionally include persistent message storage.
## No protocols are started yet.
2020-12-24 08:02:30 +00:00
2021-07-22 09:46:54 +00:00
# Mount relay on all nodes
2022-03-29 08:09:48 +00:00
var peerExchangeHandler = none ( RoutingRecordsHandler )
if conf . relayPeerExchange :
proc handlePeerExchange ( peer : PeerId , topic : string ,
peers : seq [ RoutingRecordsPair ] ) {. gcsafe , raises : [ Defect ] . } =
## Handle peers received via gossipsub peer exchange
# TODO: Only consider peers on pubsub topics we subscribe to
let exchangedPeers = peers . filterIt ( it . record . isSome ( ) ) # only peers with populated records
. mapIt ( toRemotePeerInfo ( it . record . get ( ) ) )
debug " connecting to exchanged peers " , src = peer , topic = topic , numPeers = exchangedPeers . len
# asyncSpawn, as we don't want to block here
asyncSpawn node . connectToNodes ( exchangedPeers , " peer exchange " )
peerExchangeHandler = some ( handlePeerExchange )
2022-09-07 15:31:27 +00:00
if conf . relay :
waitFor mountRelay ( node ,
conf . topics . split ( " " ) ,
peerExchangeHandler = peerExchangeHandler )
2020-12-24 08:02:30 +00:00
2021-07-22 09:46:54 +00:00
# Keepalive mounted on all nodes
2022-09-07 15:31:27 +00:00
waitFor mountLibp2pPing ( node )
2020-12-24 08:02:30 +00:00
2021-09-24 17:32:45 +00:00
when defined ( rln ) :
if conf . rlnRelay :
2022-08-30 17:59:02 +00:00
let res = node . mountRlnRelay ( conf )
if res . isErr ( ) :
debug " could not mount WakuRlnRelay "
2022-06-27 19:35:26 +00:00
2021-07-22 09:46:54 +00:00
if conf . swap :
2022-09-07 15:31:27 +00:00
waitFor mountSwap ( node )
2021-07-22 09:46:54 +00:00
# TODO Set swap peer, for now should be same as store peer
2020-09-11 04:16:45 +00:00
2021-07-22 09:46:54 +00:00
# Store setup
if ( conf . storenode ! = " " ) or ( conf . store ) :
2022-09-16 10:55:22 +00:00
waitFor mountStore ( node , mStorage , conf . persistMessages , conf . storeCapacity , conf . sqliteRetentionTime , conf . sqliteStore )
2021-03-26 08:49:51 +00:00
2021-07-22 09:46:54 +00:00
if conf . storenode ! = " " :
setStorePeer ( node , conf . storenode )
2021-03-26 08:49:51 +00:00
2021-07-22 09:46:54 +00:00
# NOTE Must be mounted after relay
if ( conf . lightpushnode ! = " " ) or ( conf . lightpush ) :
2022-09-07 15:31:27 +00:00
waitFor mountLightPush ( node )
2021-07-22 09:46:54 +00:00
if conf . lightpushnode ! = " " :
setLightPushPeer ( node , conf . lightpushnode )
# Filter setup. NOTE Must be mounted after relay
if ( conf . filternode ! = " " ) or ( conf . filter ) :
2022-09-07 15:31:27 +00:00
waitFor mountFilter ( node , filterTimeout = chronos . seconds ( conf . filterTimeout ) )
2021-06-25 21:06:56 +00:00
2021-07-22 09:46:54 +00:00
if conf . filternode ! = " " :
setFilterPeer ( node , conf . filternode )
ok ( true ) # Success
2021-03-26 08:49:51 +00:00
2022-03-17 16:33:17 +00:00
# 5/7 Start node and mounted protocols
proc startNode ( node : WakuNode , conf : WakuNodeConf ,
dynamicBootstrapNodes : seq [ RemotePeerInfo ] = @ [ ] ) : SetupResult [ bool ] =
2021-07-22 09:46:54 +00:00
## Start a configured node and all mounted protocols.
## Resume history, connect to static nodes and start
## keep-alive, if configured.
2022-03-01 14:11:56 +00:00
# Start Waku v2 node
2021-07-22 09:46:54 +00:00
waitFor node . start ( )
2022-03-01 14:11:56 +00:00
2022-03-17 16:33:17 +00:00
# Start discv5 and connect to discovered nodes
2022-03-01 14:11:56 +00:00
if conf . discv5Discovery :
if not waitFor node . startDiscv5 ( ) :
error " could not start Discovery v5 "
2021-11-15 13:29:18 +00:00
2021-07-22 09:46:54 +00:00
# Resume historical messages, this has to be called after the node has been started
if conf . store and conf . persistMessages :
waitFor node . resume ( )
# Connect to configured static nodes
if conf . staticnodes . len > 0 :
2022-01-26 11:02:57 +00:00
waitFor connectToNodes ( node , conf . staticnodes , " static " )
2021-08-12 08:51:38 +00:00
2022-08-24 15:44:43 +00:00
if dynamicBootstrapNodes . len > 0 :
info " Connecting to dynamic bootstrap peers "
waitFor connectToNodes ( node , dynamicBootstrapNodes , " dynamic bootstrap " )
2022-03-17 16:33:17 +00:00
2021-07-22 09:46:54 +00:00
# Start keepalive, if enabled
if conf . keepAlive :
node . startKeepalive ( )
ok ( true ) # Success
2020-11-24 04:53:42 +00:00
2022-03-17 16:33:17 +00:00
# 6/7 Start monitoring tools and external interfaces
2021-07-22 09:46:54 +00:00
proc startExternal ( node : WakuNode , conf : WakuNodeConf ) : SetupResult [ bool ] =
## Start configured external interfaces and monitoring tools
2022-06-28 10:22:59 +00:00
## on a Waku v2 node, including the RPC API, REST API and metrics
2021-07-22 09:46:54 +00:00
## monitoring ports.
2021-05-13 21:21:46 +00:00
2021-07-22 09:46:54 +00:00
if conf . rpc :
2022-07-17 14:25:21 +00:00
startRpcServer ( node , conf . rpcAddress , Port ( conf . rpcPort + conf . portsShift ) , conf )
2022-06-28 10:22:59 +00:00
if conf . rest :
startRestServer ( node , conf . restAddress , Port ( conf . restPort + conf . portsShift ) , conf )
2021-05-13 21:21:46 +00:00
2021-07-22 09:46:54 +00:00
if conf . metricsLogging :
startMetricsLog ( )
if conf . metricsServer :
startMetricsServer ( conf . metricsServerAddress ,
Port ( conf . metricsServerPort + conf . portsShift ) )
ok ( true ) # Success
2022-05-17 15:48:08 +00:00
{. push warning [ ProveInit ] : off . }
let conf = try :
WakuNodeConf . load (
secondarySources = proc ( conf : WakuNodeConf , sources : auto ) =
if conf . configFile . isSome :
sources . addConfigFile ( Toml , conf . configFile . get )
)
except CatchableError as err :
error " Failure while loading the configuration: \n " , err_msg = err . msg
quit 1 # if we don't leave here, the initialization of conf does not work in the success case
{. pop . }
2022-05-17 20:11:07 +00:00
# if called with --version, print the version and quit
if conf . version :
const git_version {. strdefine . } = " n/a "
echo " version / git commit hash: " , git_version
quit ( QuitSuccess )
2021-07-22 09:46:54 +00:00
var
node : WakuNode # This is the node we're going to setup using the conf
2021-05-04 12:11:41 +00:00
2021-07-22 09:46:54 +00:00
##############
# Node setup #
##############
2022-03-01 14:11:56 +00:00
2022-03-17 16:33:17 +00:00
debug " 1/7 Setting up storage "
2022-03-01 14:11:56 +00:00
2021-07-22 09:46:54 +00:00
var
pStorage : WakuPeerStorage
2022-09-13 11:36:04 +00:00
mStorage : SqliteStore
2021-04-29 04:54:31 +00:00
2021-07-22 09:46:54 +00:00
let setupStorageRes = setupStorage ( conf )
if setupStorageRes . isErr :
2022-03-17 16:33:17 +00:00
error " 1/7 Setting up storage failed. Continuing without storage. "
2021-07-22 09:46:54 +00:00
else :
( pStorage , mStorage ) = setupStorageRes . get ( )
2022-03-17 16:33:17 +00:00
debug " 2/7 Retrieve dynamic bootstrap nodes "
var dynamicBootstrapNodes : seq [ RemotePeerInfo ]
let dynamicBootstrapNodesRes = retrieveDynamicBootstrapNodes ( conf )
if dynamicBootstrapNodesRes . isErr :
2022-06-08 20:23:13 +00:00
warn " 2/7 Retrieving dynamic bootstrap nodes failed. Continuing without dynamic bootstrap nodes. "
2022-03-17 16:33:17 +00:00
else :
dynamicBootstrapNodes = dynamicBootstrapNodesRes . get ( )
debug " 3/7 Initializing node "
2021-07-22 09:46:54 +00:00
2022-03-17 16:33:17 +00:00
let initNodeRes = initNode ( conf , pStorage , dynamicBootstrapNodes )
2021-04-29 04:54:31 +00:00
2021-07-22 09:46:54 +00:00
if initNodeRes . isErr :
2022-03-17 16:33:17 +00:00
error " 3/7 Initializing node failed. Quitting. "
2021-07-22 09:46:54 +00:00
quit ( QuitFailure )
else :
node = initNodeRes . get ( )
2022-03-17 16:33:17 +00:00
debug " 4/7 Mounting protocols "
2021-04-24 04:56:37 +00:00
2021-07-22 09:46:54 +00:00
let setupProtocolsRes = setupProtocols ( node , conf , mStorage )
2020-09-01 02:09:54 +00:00
2021-07-22 09:46:54 +00:00
if setupProtocolsRes . isErr :
2022-03-17 16:33:17 +00:00
error " 4/7 Mounting protocols failed. Continuing in current state. "
2020-09-01 02:09:54 +00:00
2022-03-17 16:33:17 +00:00
debug " 5/7 Starting node and mounted protocols "
2021-04-15 08:18:14 +00:00
2022-03-17 16:33:17 +00:00
let startNodeRes = startNode ( node , conf , dynamicBootstrapNodes )
2021-07-22 09:46:54 +00:00
if startNodeRes . isErr :
2022-03-17 16:33:17 +00:00
error " 5/7 Starting node and mounted protocols failed. Continuing in current state. "
2021-07-22 09:46:54 +00:00
2022-03-17 16:33:17 +00:00
debug " 6/7 Starting monitoring and external interfaces "
2021-07-22 09:46:54 +00:00
let startExternalRes = startExternal ( node , conf )
if startExternalRes . isErr :
2022-03-17 16:33:17 +00:00
error " 6/7 Starting monitoring and external interfaces failed. Continuing in current state. "
2021-07-22 09:46:54 +00:00
2022-03-17 16:33:17 +00:00
debug " 7/7 Setting up shutdown hooks "
2021-07-22 09:46:54 +00:00
2022-03-17 16:33:17 +00:00
# 7/7 Setup graceful shutdown hooks
2021-07-22 09:46:54 +00:00
## Setup shutdown hooks for this process.
## Stop node gracefully on shutdown.
2021-04-15 08:18:14 +00:00
# Handle Ctrl-C SIGINT
proc handleCtrlC ( ) {. noconv . } =
when defined ( windows ) :
# workaround for https://github.com/nim-lang/Nim/issues/4057
setupForeignThreadGc ( )
info " Shutting down after receiving SIGINT "
waitFor node . stop ( )
quit ( QuitSuccess )
setControlCHook ( handleCtrlC )
# Handle SIGTERM
when defined ( posix ) :
proc handleSigterm ( signal : cint ) {. noconv . } =
info " Shutting down after receiving SIGTERM "
waitFor node . stop ( )
quit ( QuitSuccess )
2022-09-07 15:31:27 +00:00
c_signal ( ansi_c . SIGTERM , handleSigterm )
2021-06-02 07:53:34 +00:00
2021-07-22 09:46:54 +00:00
debug " Node setup complete "
2020-09-01 02:09:54 +00:00
2021-05-28 18:41:29 +00:00
runForever ( )