import confutils, config, strutils, chronos, json_rpc/rpcserver, metrics, metrics/chronicles_support, eth/[keys, p2p, async_utils], eth/common/utils, eth/net/nat, eth/p2p/[discovery, enode, peer_pool, bootnodes, whispernodes], eth/p2p/rlpx_protocols/whisper_protocol, ../../protocol/v1/[waku_protocol, waku_bridge], ./rpc/[waku, wakusim, key_storage] const clientId = "Nimbus waku node" let globalListeningAddr = parseIpAddress("0.0.0.0") proc setBootNodes(nodes: openArray[string]): seq[ENode] = result = newSeqOfCap[ENode](nodes.len) for nodeId in nodes: # TODO: something more user friendly than an expect result.add(ENode.fromString(nodeId).expect("correct node")) proc connectToNodes(node: EthereumNode, nodes: openArray[string]) = for nodeId in nodes: # TODO: something more user friendly than an assert let whisperENode = ENode.fromString(nodeId).expect("correct node") traceAsyncErrors node.peerPool.connectToNode(newNode(whisperENode)) proc setupNat(conf: WakuNodeConf): tuple[ip: IpAddress, tcpPort: Port, udpPort: Port] = # defaults result.ip = globalListeningAddr result.tcpPort = Port(conf.tcpPort + conf.portsShift) result.udpPort = Port(conf.udpPort + conf.portsShift) var nat: NatStrategy case conf.nat.toLowerAscii(): of "any": nat = NatAny of "none": nat = NatNone of "upnp": nat = NatUpnp of "pmp": nat = NatPmp else: if conf.nat.startsWith("extip:") and isIpAddress(conf.nat[6..^1]): # any required port redirection is assumed to be done by hand result.ip = parseIpAddress(conf.nat[6..^1]) nat = NatNone else: error "not a valid NAT mechanism, nor a valid IP address", value = conf.nat quit(QuitFailure) if nat != NatNone: let extIP = getExternalIP(nat) if extIP.isSome: result.ip = extIP.get() let extPorts = redirectPorts(tcpPort = result.tcpPort, udpPort = result.udpPort, description = clientId) if extPorts.isSome: (result.tcpPort, result.udpPort) = extPorts.get() proc run(config: WakuNodeConf, rng: ref BrHmacDrbgContext) = let (ip, tcpPort, udpPort) = setupNat(config) address = Address(ip: ip, tcpPort: tcpPort, udpPort: udpPort) # Set-up node var node = newEthereumNode(config.nodekey, address, 1, nil, clientId, addAllCapabilities = false) 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: @[] traceAsyncErrors node.connectToNetwork(bootnodes, not config.noListen, config.discovery) 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() setupWakuRPC(node, keys, rpcServer, rng) setupWakuSimRPC(node, rpcServer) rpcServer.start() if config.logAccounting: proc logPeerAccounting(udata: pointer) {.closure, gcsafe.} = {.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) when defined(insecure): if config.metricsServer: let address = config.metricsServerAddress port = config.metricsServerPort + config.portsShift info "Starting metrics HTTP server", address, port metrics.startHttpServer($address, Port(port)) if config.logMetrics: proc logMetrics(udata: pointer) {.closure, gcsafe.} = {.gcsafe.}: let connectedPeers = connected_peers validEnvelopes = waku_protocol.envelopes_valid droppedEnvelopes = waku_protocol.envelopes_dropped info "Node metrics", connectedPeers, validEnvelopes, droppedEnvelopes discard setTimer(Moment.fromNow(2.seconds), logMetrics) discard setTimer(Moment.fromNow(2.seconds), logMetrics) runForever() when isMainModule: let rng = keys.newRng() conf = WakuNodeConf.load() if conf.logLevel != LogLevel.NONE: setLogLevel(conf.logLevel) case conf.cmd of genNodekey: echo PrivateKey.random(rng[]) of noCommand: run(conf, rng)