2021-07-19 16:14:37 +08:00
|
|
|
{.push raises: [Defect].}
|
|
|
|
|
2020-04-28 14:04:27 +08:00
|
|
|
import
|
2021-06-10 16:18:41 +02:00
|
|
|
confutils, chronos, json_rpc/rpcserver,
|
|
|
|
metrics, metrics/chronicles_support, metrics/chronos_httpserver,
|
2020-09-01 04:09:54 +02:00
|
|
|
stew/shims/net as stewNet,
|
2020-08-27 04:44:52 +02:00
|
|
|
eth/[keys, p2p], eth/common/utils,
|
2020-04-28 14:04:27 +08:00
|
|
|
eth/p2p/[discovery, enode, peer_pool, bootnodes, whispernodes],
|
2021-07-19 16:14:37 +08:00
|
|
|
../../whisper/whisper_protocol,
|
2020-11-17 17:34:53 +08:00
|
|
|
../protocol/[waku_protocol, waku_bridge],
|
|
|
|
../../common/utils/nat,
|
2020-08-26 14:20:04 +02:00
|
|
|
./rpc/[waku, wakusim, key_storage], ./waku_helpers, ./config
|
2020-04-28 14:04:27 +08:00
|
|
|
|
|
|
|
const clientId = "Nimbus waku node"
|
|
|
|
|
2021-07-19 16:14:37 +08:00
|
|
|
proc run(config: WakuNodeConf, rng: ref BrHmacDrbgContext)
|
|
|
|
{.raises: [Defect, ValueError, RpcBindError, CatchableError, Exception]} =
|
2021-10-12 13:43:01 +02:00
|
|
|
## `udpPort` is only supplied to satisfy underlying APIs but is not
|
|
|
|
## actually a supported transport.
|
|
|
|
let udpPort = config.tcpPort
|
2020-04-28 14:04:27 +08:00
|
|
|
let
|
2021-10-12 13:43:01 +02:00
|
|
|
(ipExt, tcpPortExt, _) = setupNat(config.nat, clientId,
|
2020-09-01 04:09:54 +02:00
|
|
|
Port(config.tcpPort + config.portsShift),
|
2021-10-12 13:43:01 +02:00
|
|
|
Port(udpPort + config.portsShift))
|
2020-09-01 04:09:54 +02:00
|
|
|
# TODO: EthereumNode should have a better split of binding address and
|
|
|
|
# external address. Also, can't have different ports as it stands now.
|
|
|
|
address = if ipExt.isNone():
|
|
|
|
Address(ip: parseIpAddress("0.0.0.0"),
|
|
|
|
tcpPort: Port(config.tcpPort + config.portsShift),
|
2021-10-12 13:43:01 +02:00
|
|
|
udpPort: Port(udpPort + config.portsShift))
|
2020-09-01 04:09:54 +02:00
|
|
|
else:
|
|
|
|
Address(ip: ipExt.get(),
|
|
|
|
tcpPort: Port(config.tcpPort + config.portsShift),
|
2021-10-12 13:43:01 +02:00
|
|
|
udpPort: Port(udpPort + config.portsShift))
|
2020-04-28 14:04:27 +08:00
|
|
|
|
|
|
|
# Set-up node
|
2021-03-17 10:52:27 +02:00
|
|
|
var node = newEthereumNode(config.nodekey, address, NetworkId(1), nil, clientId,
|
2020-08-26 14:20:04 +02:00
|
|
|
addAllCapabilities = false, rng = rng)
|
2020-04-28 14:04:27 +08:00
|
|
|
if not config.bootnodeOnly:
|
|
|
|
node.addCapability Waku # Always enable Waku protocol
|
|
|
|
var topicInterest: Option[seq[waku_protocol.Topic]]
|
|
|
|
var bloom: Option[Bloom]
|
|
|
|
if config.wakuTopicInterest:
|
|
|
|
var topics: seq[waku_protocol.Topic]
|
|
|
|
topicInterest = some(topics)
|
|
|
|
else:
|
|
|
|
bloom = some(fullBloom())
|
|
|
|
let wakuConfig = WakuConfig(powRequirement: config.wakuPow,
|
|
|
|
bloom: bloom,
|
|
|
|
isLightNode: config.lightNode,
|
|
|
|
maxMsgSize: waku_protocol.defaultMaxMsgSize,
|
|
|
|
topics: topicInterest)
|
|
|
|
node.configureWaku(wakuConfig)
|
|
|
|
if config.whisper or config.whisperBridge:
|
|
|
|
node.addCapability Whisper
|
|
|
|
node.protocolState(Whisper).config.powRequirement = 0.002
|
|
|
|
if config.whisperBridge:
|
|
|
|
node.shareMessageQueue()
|
|
|
|
|
|
|
|
# TODO: Status fleet bootnodes are discv5? That will not work.
|
|
|
|
let bootnodes = if config.bootnodes.len > 0: setBootNodes(config.bootnodes)
|
|
|
|
elif config.fleet == prod: setBootNodes(StatusBootNodes)
|
|
|
|
elif config.fleet == staging: setBootNodes(StatusBootNodesStaging)
|
|
|
|
elif config.fleet == test : setBootNodes(StatusBootNodesTest)
|
|
|
|
else: @[]
|
|
|
|
|
2020-08-27 04:44:52 +02:00
|
|
|
let connectedFut = node.connectToNetwork(bootnodes, not config.noListen,
|
2020-04-28 14:04:27 +08:00
|
|
|
config.discovery)
|
2020-08-27 04:44:52 +02:00
|
|
|
connectedFut.callback = proc(data: pointer) {.gcsafe.} =
|
|
|
|
{.gcsafe.}:
|
|
|
|
if connectedFut.failed:
|
|
|
|
fatal "connectToNetwork failed", msg = connectedFut.readError.msg
|
|
|
|
quit(1)
|
2020-04-28 14:04:27 +08:00
|
|
|
|
|
|
|
if not config.bootnodeOnly:
|
|
|
|
# Optionally direct connect with a set of nodes
|
|
|
|
if config.staticnodes.len > 0: connectToNodes(node, config.staticnodes)
|
|
|
|
elif config.fleet == prod: connectToNodes(node, WhisperNodes)
|
|
|
|
elif config.fleet == staging: connectToNodes(node, WhisperNodesStaging)
|
|
|
|
elif config.fleet == test: connectToNodes(node, WhisperNodesTest)
|
|
|
|
|
|
|
|
if config.rpc:
|
|
|
|
let ta = initTAddress(config.rpcAddress,
|
|
|
|
Port(config.rpcPort + config.portsShift))
|
|
|
|
var rpcServer = newRpcHttpServer([ta])
|
|
|
|
let keys = newKeyStorage()
|
2020-07-13 12:08:03 +02:00
|
|
|
setupWakuRPC(node, keys, rpcServer, rng)
|
2020-04-28 14:04:27 +08:00
|
|
|
setupWakuSimRPC(node, rpcServer)
|
|
|
|
rpcServer.start()
|
|
|
|
|
2020-06-18 05:16:23 +02:00
|
|
|
|
2020-07-02 23:52:54 +02:00
|
|
|
if config.logAccounting:
|
2021-03-26 10:52:04 +01:00
|
|
|
# https://github.com/nim-lang/Nim/issues/17369
|
|
|
|
var logPeerAccounting: proc(udata: pointer) {.gcsafe, raises: [Defect].}
|
|
|
|
logPeerAccounting = proc(udata: pointer) =
|
2020-07-02 23:52:54 +02:00
|
|
|
{.gcsafe.}:
|
|
|
|
for peer in node.peerPool.peers:
|
|
|
|
let
|
|
|
|
sent = peer.state(Waku).accounting.sent
|
|
|
|
received = peer.state(Waku).accounting.received
|
|
|
|
id = peer.network.toEnode
|
|
|
|
info "Peer accounting", id, sent, received
|
|
|
|
peer.state(Waku).accounting = Accounting(sent: 0, received: 0)
|
|
|
|
|
|
|
|
discard setTimer(Moment.fromNow(2.seconds), logPeerAccounting)
|
|
|
|
discard setTimer(Moment.fromNow(2.seconds), logPeerAccounting)
|
2020-06-18 05:16:23 +02:00
|
|
|
|
2021-06-10 16:18:41 +02:00
|
|
|
if config.metricsServer:
|
|
|
|
let
|
|
|
|
address = config.metricsServerAddress
|
|
|
|
port = config.metricsServerPort + config.portsShift
|
|
|
|
info "Starting metrics HTTP server", address, port
|
|
|
|
startMetricsHttpServer($address, Port(port))
|
2020-04-28 14:04:27 +08:00
|
|
|
|
|
|
|
if config.logMetrics:
|
2021-03-26 10:52:04 +01:00
|
|
|
# https://github.com/nim-lang/Nim/issues/17369
|
|
|
|
var logMetrics: proc(udata: pointer) {.gcsafe, raises: [Defect].}
|
|
|
|
logMetrics = proc(udata: pointer) =
|
2020-04-28 14:04:27 +08:00
|
|
|
{.gcsafe.}:
|
|
|
|
let
|
2020-06-09 13:44:39 +02:00
|
|
|
connectedPeers = connected_peers
|
|
|
|
validEnvelopes = waku_protocol.envelopes_valid
|
|
|
|
droppedEnvelopes = waku_protocol.envelopes_dropped
|
|
|
|
|
|
|
|
info "Node metrics", connectedPeers, validEnvelopes, droppedEnvelopes
|
2020-07-02 23:52:54 +02:00
|
|
|
discard setTimer(Moment.fromNow(2.seconds), logMetrics)
|
|
|
|
discard setTimer(Moment.fromNow(2.seconds), logMetrics)
|
2020-04-28 14:04:27 +08:00
|
|
|
|
|
|
|
runForever()
|
|
|
|
|
2021-07-19 16:14:37 +08:00
|
|
|
{.pop.} # @TODO confutils.nim(775, 17) Error: can raise an unlisted exception: ref IOError
|
2020-04-28 14:04:27 +08:00
|
|
|
when isMainModule:
|
2020-07-13 12:08:03 +02:00
|
|
|
let
|
|
|
|
rng = keys.newRng()
|
|
|
|
conf = WakuNodeConf.load()
|
2020-06-03 20:31:18 +02:00
|
|
|
|
|
|
|
if conf.logLevel != LogLevel.NONE:
|
|
|
|
setLogLevel(conf.logLevel)
|
|
|
|
|
|
|
|
case conf.cmd
|
|
|
|
of genNodekey:
|
2020-07-13 12:08:03 +02:00
|
|
|
echo PrivateKey.random(rng[])
|
2020-06-03 20:31:18 +02:00
|
|
|
of noCommand:
|
2020-07-13 12:08:03 +02:00
|
|
|
run(conf, rng)
|