2020-12-21 11:45:07 +00:00
|
|
|
import
|
2021-02-04 08:51:22 +00:00
|
|
|
std/[strutils, tables],
|
2020-12-21 11:45:07 +00:00
|
|
|
chronos, confutils, chronicles, chronicles/topics_registry, metrics,
|
2021-03-18 10:24:42 +00:00
|
|
|
stew/endians2,
|
2020-12-21 11:45:07 +00:00
|
|
|
stew/shims/net as stewNet, json_rpc/rpcserver,
|
|
|
|
# Waku v1 imports
|
|
|
|
eth/[keys, p2p], eth/common/utils,
|
|
|
|
eth/p2p/[enode, whispernodes],
|
|
|
|
../v1/protocol/waku_protocol,
|
|
|
|
./utils/nat,
|
|
|
|
../v1/node/rpc/wakusim,
|
2021-02-04 08:51:22 +00:00
|
|
|
../v1/node/rpc/waku,
|
|
|
|
../v1/node/rpc/key_storage,
|
2020-12-21 11:45:07 +00:00
|
|
|
../v1/node/waku_helpers,
|
|
|
|
# Waku v2 imports
|
|
|
|
libp2p/crypto/crypto,
|
2021-02-04 08:51:22 +00:00
|
|
|
../v2/protocol/waku_filter/waku_filter_types,
|
2020-12-21 11:45:07 +00:00
|
|
|
../v2/node/wakunode2,
|
2021-02-04 08:51:22 +00:00
|
|
|
../v2/node/jsonrpc/[debug_api,
|
|
|
|
filter_api,
|
|
|
|
relay_api,
|
|
|
|
store_api],
|
2020-12-21 11:45:07 +00:00
|
|
|
# Common cli config
|
|
|
|
./config_bridge
|
|
|
|
|
2021-03-18 10:24:42 +00:00
|
|
|
const
|
|
|
|
clientIdV1 = "nim-waku v1 node"
|
|
|
|
defaultBridgeTopic = "/waku/2/default-bridge/proto"
|
|
|
|
defaultTTL = 5'u32
|
|
|
|
|
|
|
|
proc toWakuMessage(env: Envelope): WakuMessage =
|
|
|
|
# Translate a Waku v1 envelope to a Waku v2 message
|
|
|
|
WakuMessage(payload: env.data,
|
|
|
|
contentTopic: ContentTopic(uint32.fromBytes(env.topic)),
|
|
|
|
version: 1)
|
|
|
|
|
|
|
|
proc toWakuV2(env: Envelope, nodev2: WakuNode) {.async.} =
|
|
|
|
await nodev2.publish(defaultBridgeTopic, env.toWakuMessage())
|
|
|
|
|
|
|
|
proc toWakuV1(msg: WakuMessage, nodev1: EthereumNode) {.gcsafe.} =
|
|
|
|
discard nodev1.postMessage(ttl = defaultTTL,
|
|
|
|
topic = msg.contentTopic.toBytes(),
|
|
|
|
payload = msg.payload)
|
2020-12-21 11:45:07 +00:00
|
|
|
|
|
|
|
proc startWakuV1(config: WakuNodeConf, rng: ref BrHmacDrbgContext):
|
|
|
|
EthereumNode =
|
|
|
|
let
|
|
|
|
(ipExt, _, _) = setupNat(config.nat, clientIdV1,
|
|
|
|
Port(config.devp2pTcpPort + config.portsShift),
|
|
|
|
Port(config.udpPort + config.portsShift))
|
|
|
|
# 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.devp2pTcpPort + config.portsShift),
|
|
|
|
udpPort: Port(config.udpPort + config.portsShift))
|
|
|
|
else:
|
|
|
|
Address(ip: ipExt.get(),
|
|
|
|
tcpPort: Port(config.devp2pTcpPort + config.portsShift),
|
|
|
|
udpPort: Port(config.udpPort + config.portsShift))
|
|
|
|
|
|
|
|
# Set-up node
|
2021-03-17 09:20:33 +00:00
|
|
|
var node = newEthereumNode(config.nodekeyv1, address, NetworkId(1), nil, clientIdV1,
|
2020-12-21 11:45:07 +00:00
|
|
|
addAllCapabilities = false, rng = rng)
|
|
|
|
node.addCapability Waku # Always enable Waku protocol
|
|
|
|
# Set up the Waku configuration.
|
|
|
|
# This node is being set up as a bridge so it gets configured as a node with
|
|
|
|
# a full bloom filter so that it will receive and forward all messages.
|
|
|
|
# TODO: What is the PoW setting now?
|
|
|
|
let wakuConfig = WakuConfig(powRequirement: config.wakuPow,
|
|
|
|
bloom: some(fullBloom()), isLightNode: false,
|
|
|
|
maxMsgSize: waku_protocol.defaultMaxMsgSize,
|
|
|
|
topics: none(seq[waku_protocol.Topic]))
|
|
|
|
node.configureWaku(wakuConfig)
|
|
|
|
|
|
|
|
# Optionally direct connect with a set of nodes
|
|
|
|
if config.staticnodesv1.len > 0: connectToNodes(node, config.staticnodesv1)
|
|
|
|
elif config.fleetv1 == prod: connectToNodes(node, WhisperNodes)
|
|
|
|
elif config.fleetv1 == staging: connectToNodes(node, WhisperNodesStaging)
|
|
|
|
elif config.fleetv1 == test: connectToNodes(node, WhisperNodesTest)
|
|
|
|
|
|
|
|
let connectedFut = node.connectToNetwork(@[],
|
|
|
|
true, # Always enable listening
|
|
|
|
false # Disable discovery (only discovery v4 is currently supported)
|
|
|
|
)
|
|
|
|
connectedFut.callback = proc(data: pointer) {.gcsafe.} =
|
|
|
|
{.gcsafe.}:
|
|
|
|
if connectedFut.failed:
|
|
|
|
fatal "connectToNetwork failed", msg = connectedFut.readError.msg
|
|
|
|
quit(1)
|
|
|
|
|
|
|
|
return node
|
|
|
|
|
|
|
|
proc startWakuV2(config: WakuNodeConf): Future[WakuNode] {.async.} =
|
|
|
|
let
|
|
|
|
(extIp, extTcpPort, _) = setupNat(config.nat, clientId,
|
|
|
|
Port(uint16(config.libp2pTcpPort) + config.portsShift),
|
|
|
|
Port(uint16(config.udpPort) + config.portsShift))
|
|
|
|
node = WakuNode.init(config.nodeKeyv2, config.listenAddress,
|
|
|
|
Port(uint16(config.libp2pTcpPort) + config.portsShift), extIp, extTcpPort)
|
|
|
|
|
|
|
|
await node.start()
|
|
|
|
|
|
|
|
if config.store:
|
|
|
|
mountStore(node)
|
|
|
|
|
|
|
|
if config.filter:
|
|
|
|
mountFilter(node)
|
|
|
|
|
|
|
|
if config.relay:
|
2021-02-02 11:46:38 +00:00
|
|
|
mountRelay(node, config.topics.split(" "))
|
2020-12-21 11:45:07 +00:00
|
|
|
|
|
|
|
if config.staticnodesv2.len > 0:
|
|
|
|
waitFor connectToNodes(node, config.staticnodesv2)
|
|
|
|
|
|
|
|
if config.storenode != "":
|
|
|
|
setStorePeer(node, config.storenode)
|
|
|
|
|
|
|
|
if config.filternode != "":
|
|
|
|
setFilterPeer(node, config.filternode)
|
|
|
|
|
|
|
|
return node
|
|
|
|
|
|
|
|
when isMainModule:
|
2021-02-04 08:51:22 +00:00
|
|
|
proc startV2Rpc(node: WakuNode, rpcServer: RpcHttpServer, conf: WakuNodeConf) =
|
|
|
|
installDebugApiHandlers(node, rpcServer)
|
|
|
|
|
|
|
|
# Install enabled API handlers:
|
|
|
|
if conf.relay:
|
|
|
|
let topicCache = newTable[string, seq[WakuMessage]]()
|
|
|
|
installRelayApiHandlers(node, rpcServer, topicCache)
|
|
|
|
|
|
|
|
if conf.filter:
|
|
|
|
let messageCache = newTable[ContentTopic, seq[WakuMessage]]()
|
|
|
|
installFilterApiHandlers(node, rpcServer, messageCache)
|
|
|
|
|
|
|
|
if conf.store:
|
|
|
|
installStoreApiHandlers(node, rpcServer)
|
|
|
|
|
|
|
|
rpcServer.start()
|
|
|
|
|
2020-12-21 11:45:07 +00:00
|
|
|
let
|
|
|
|
rng = keys.newRng()
|
|
|
|
let conf = WakuNodeConf.load()
|
|
|
|
|
|
|
|
if conf.logLevel != LogLevel.NONE:
|
|
|
|
setLogLevel(conf.logLevel)
|
2021-03-18 10:24:42 +00:00
|
|
|
|
|
|
|
var
|
|
|
|
nodev1 {.threadvar.}: EthereumNode
|
|
|
|
nodev2 {.threadvar.}: WakuNode
|
|
|
|
|
|
|
|
nodev1 = startWakuV1(conf, rng)
|
|
|
|
nodev2 = waitFor startWakuV2(conf)
|
|
|
|
|
|
|
|
|
|
|
|
# Handle messages on Waku v1 and bridge to Waku v2
|
|
|
|
proc handleEnvReceived(envelope: Envelope) {.gcsafe.} =
|
|
|
|
debug "Bridging envelope from V1 to V2", envelope=envelope
|
|
|
|
waitFor envelope.toWakuV2(nodev2)
|
|
|
|
|
|
|
|
nodev1.registerEnvReceivedHandler(handleEnvReceived)
|
|
|
|
|
|
|
|
# Handle messages on Waku v2 and bridge to Waku v1
|
|
|
|
proc relayHandler(pubsubTopic: string, data: seq[byte]) {.async, gcsafe.} =
|
|
|
|
let msg = WakuMessage.init(data)
|
|
|
|
if msg.isOk():
|
|
|
|
debug "Bridging message from V2 to V1", msg=msg[]
|
|
|
|
msg[].toWakuV1(nodev1)
|
|
|
|
|
|
|
|
nodev2.subscribe(defaultBridgeTopic, relayHandler)
|
2020-12-21 11:45:07 +00:00
|
|
|
|
|
|
|
|
|
|
|
if conf.rpc:
|
|
|
|
let ta = initTAddress(conf.rpcAddress,
|
|
|
|
Port(conf.rpcPort + conf.portsShift))
|
|
|
|
var rpcServer = newRpcHttpServer([ta])
|
|
|
|
# Waku v1 RPC
|
2021-02-04 08:51:22 +00:00
|
|
|
let keys = newKeyStorage()
|
|
|
|
setupWakuRPC(nodev1, keys, rpcServer, rng)
|
2020-12-21 11:45:07 +00:00
|
|
|
setupWakuSimRPC(nodev1, rpcServer)
|
|
|
|
# Waku v2 rpc
|
2021-02-04 08:51:22 +00:00
|
|
|
startV2Rpc(nodev2, rpcServer, conf)
|
2020-12-21 11:45:07 +00:00
|
|
|
|
|
|
|
rpcServer.start()
|
|
|
|
|
|
|
|
when defined(insecure):
|
|
|
|
if conf.metricsServer:
|
|
|
|
let
|
|
|
|
address = conf.metricsServerAddress
|
|
|
|
port = conf.metricsServerPort + conf.portsShift
|
|
|
|
info "Starting metrics HTTP server", address, port
|
|
|
|
metrics.startHttpServer($address, Port(port))
|
|
|
|
|
|
|
|
runForever()
|