2024-03-15 23:08:47 +00:00
|
|
|
import std/[json, strformat, options]
|
2024-03-27 14:08:53 +00:00
|
|
|
import std/sequtils
|
2023-07-07 08:53:00 +00:00
|
|
|
import
|
|
|
|
libp2p/crypto/crypto,
|
|
|
|
libp2p/crypto/secp,
|
|
|
|
stew/shims/net,
|
2023-08-09 17:11:50 +00:00
|
|
|
../../waku/waku_enr/capabilities,
|
2023-07-07 08:53:00 +00:00
|
|
|
../../waku/common/utils/nat,
|
2024-03-26 13:10:25 +00:00
|
|
|
../../waku/factory/external_config,
|
2024-03-27 14:08:53 +00:00
|
|
|
../../waku/waku_core/message/default_values,
|
2023-08-09 17:11:50 +00:00
|
|
|
../../waku/node/waku_node,
|
|
|
|
../../waku/node/config,
|
2024-02-20 20:00:03 +00:00
|
|
|
../events/json_base_event
|
2023-07-07 08:53:00 +00:00
|
|
|
|
2024-03-15 23:08:47 +00:00
|
|
|
proc parsePrivateKey(
|
2024-03-26 13:10:25 +00:00
|
|
|
jsonNode: JsonNode, conf: var WakuNodeConf, errorResp: var string
|
2024-03-15 23:08:47 +00:00
|
|
|
): bool =
|
2024-02-20 20:00:03 +00:00
|
|
|
if not jsonNode.contains("key") or jsonNode["key"].kind == JsonNodeKind.JNull:
|
2024-03-26 13:10:25 +00:00
|
|
|
conf.nodekey = some(PrivateKey.random(Secp256k1, newRng()[]).tryGet())
|
2024-02-20 15:18:03 +00:00
|
|
|
return true
|
2023-07-07 08:53:00 +00:00
|
|
|
|
|
|
|
if jsonNode["key"].kind != JsonNodeKind.JString:
|
2024-02-20 20:00:03 +00:00
|
|
|
errorResp = "The node key should be a string."
|
2023-07-07 08:53:00 +00:00
|
|
|
return false
|
|
|
|
|
|
|
|
let key = jsonNode["key"].getStr()
|
|
|
|
|
|
|
|
try:
|
|
|
|
let skPrivKey = SkPrivateKey.init(crypto.fromHex(key)).tryGet()
|
2024-03-26 13:10:25 +00:00
|
|
|
conf.nodekey = some(crypto.PrivateKey(scheme: Secp256k1, skkey: skPrivKey))
|
2023-07-07 08:53:00 +00:00
|
|
|
except CatchableError:
|
|
|
|
let msg = "Invalid node key: " & getCurrentExceptionMsg()
|
2024-02-20 20:00:03 +00:00
|
|
|
errorResp = msg
|
2023-07-07 08:53:00 +00:00
|
|
|
return false
|
|
|
|
|
|
|
|
return true
|
|
|
|
|
2024-03-15 23:08:47 +00:00
|
|
|
proc parseListenAddr(
|
2024-03-26 13:10:25 +00:00
|
|
|
jsonNode: JsonNode, conf: var WakuNodeConf, errorResp: var string
|
2024-03-15 23:08:47 +00:00
|
|
|
): bool =
|
2024-03-26 13:10:25 +00:00
|
|
|
var listenAddr: IpAddress
|
2023-07-07 08:53:00 +00:00
|
|
|
if not jsonNode.contains("host"):
|
2024-03-26 13:10:25 +00:00
|
|
|
conf.listenAddress = defaultListenAddress()
|
|
|
|
return true
|
2023-07-07 08:53:00 +00:00
|
|
|
|
|
|
|
if jsonNode["host"].kind != JsonNodeKind.JString:
|
2024-02-20 20:00:03 +00:00
|
|
|
errorResp = "The node host should be a string."
|
2023-07-07 08:53:00 +00:00
|
|
|
return false
|
|
|
|
|
|
|
|
let host = jsonNode["host"].getStr()
|
|
|
|
|
|
|
|
try:
|
2023-12-14 06:16:39 +00:00
|
|
|
listenAddr = parseIpAddress(host)
|
2023-07-07 08:53:00 +00:00
|
|
|
except CatchableError:
|
|
|
|
let msg = "Invalid host IP address: " & getCurrentExceptionMsg()
|
2024-02-20 20:00:03 +00:00
|
|
|
errorResp = msg
|
2023-07-07 08:53:00 +00:00
|
|
|
return false
|
|
|
|
|
|
|
|
return true
|
|
|
|
|
2024-03-26 13:10:25 +00:00
|
|
|
proc parsePort(jsonNode: JsonNode, conf: var WakuNodeConf, errorResp: var string): bool =
|
2023-07-07 08:53:00 +00:00
|
|
|
if not jsonNode.contains("port"):
|
2024-03-26 13:10:25 +00:00
|
|
|
conf.tcpPort = Port(60000)
|
|
|
|
return true
|
2023-07-07 08:53:00 +00:00
|
|
|
|
|
|
|
if jsonNode["port"].kind != JsonNodeKind.JInt:
|
2024-02-20 20:00:03 +00:00
|
|
|
errorResp = "The node port should be an integer."
|
2023-07-07 08:53:00 +00:00
|
|
|
return false
|
|
|
|
|
2024-03-26 13:10:25 +00:00
|
|
|
conf.tcpPort = Port(jsonNode["port"].getInt())
|
2023-07-07 08:53:00 +00:00
|
|
|
|
|
|
|
return true
|
|
|
|
|
2024-03-26 13:10:25 +00:00
|
|
|
proc parseRelay(jsonNode: JsonNode, conf: var WakuNodeConf, errorResp: var string): bool =
|
2023-07-07 08:53:00 +00:00
|
|
|
if not jsonNode.contains("relay"):
|
2024-02-20 20:00:03 +00:00
|
|
|
errorResp = "relay attribute is required"
|
2023-07-07 08:53:00 +00:00
|
|
|
return false
|
|
|
|
|
|
|
|
if jsonNode["relay"].kind != JsonNodeKind.JBool:
|
2024-02-20 20:00:03 +00:00
|
|
|
errorResp = "The relay config param should be a boolean"
|
2023-07-07 08:53:00 +00:00
|
|
|
return false
|
|
|
|
|
2024-03-26 13:10:25 +00:00
|
|
|
conf.relay = jsonNode["relay"].getBool()
|
2023-07-07 08:53:00 +00:00
|
|
|
|
|
|
|
return true
|
|
|
|
|
2024-03-27 14:08:53 +00:00
|
|
|
proc parseClusterId(jsonNode: JsonNode, conf: var WakuNodeConf, errorResp: var string): bool =
|
|
|
|
if not jsonNode.contains("relay"):
|
|
|
|
errorResp = "relay attribute is required"
|
|
|
|
return false
|
|
|
|
|
|
|
|
if jsonNode.contains("clusterId"):
|
|
|
|
if jsonNode["clusterId"].kind != JsonNodeKind.JInt:
|
|
|
|
errorResp = "The clusterId config param should be an int"
|
|
|
|
return false
|
|
|
|
else:
|
|
|
|
conf.clusterId = uint32(jsonNode["clusterId"].getInt())
|
|
|
|
|
|
|
|
return true
|
|
|
|
|
2024-03-15 23:08:47 +00:00
|
|
|
proc parseStore(
|
|
|
|
jsonNode: JsonNode,
|
2024-03-26 13:10:25 +00:00
|
|
|
conf: var WakuNodeConf,
|
2024-03-15 23:08:47 +00:00
|
|
|
errorResp: var string,
|
|
|
|
): bool =
|
2023-12-11 07:49:13 +00:00
|
|
|
if not jsonNode.contains("store"):
|
|
|
|
## the store parameter is not required. By default is is disabled
|
2024-03-26 13:10:25 +00:00
|
|
|
conf.store = false
|
2023-12-11 07:49:13 +00:00
|
|
|
return true
|
|
|
|
|
|
|
|
if jsonNode["store"].kind != JsonNodeKind.JBool:
|
2024-02-20 20:00:03 +00:00
|
|
|
errorResp = "The store config param should be a boolean"
|
2023-12-11 07:49:13 +00:00
|
|
|
return false
|
|
|
|
|
2024-03-26 13:10:25 +00:00
|
|
|
conf.store = jsonNode["store"].getBool()
|
2023-12-11 07:49:13 +00:00
|
|
|
|
|
|
|
if jsonNode.contains("storeNode"):
|
|
|
|
if jsonNode["storeNode"].kind != JsonNodeKind.JString:
|
2024-02-20 20:00:03 +00:00
|
|
|
errorResp = "The storeNode config param should be a string"
|
2023-12-11 07:49:13 +00:00
|
|
|
return false
|
|
|
|
|
2024-03-26 13:10:25 +00:00
|
|
|
conf.storeNode = jsonNode["storeNode"].getStr()
|
2023-12-11 07:49:13 +00:00
|
|
|
|
|
|
|
if jsonNode.contains("storeRetentionPolicy"):
|
|
|
|
if jsonNode["storeRetentionPolicy"].kind != JsonNodeKind.JString:
|
2024-02-20 20:00:03 +00:00
|
|
|
errorResp = "The storeRetentionPolicy config param should be a string"
|
2023-12-11 07:49:13 +00:00
|
|
|
return false
|
|
|
|
|
2024-03-26 13:10:25 +00:00
|
|
|
conf.storeMessageRetentionPolicy = jsonNode["storeRetentionPolicy"].getStr()
|
2023-12-11 07:49:13 +00:00
|
|
|
|
|
|
|
if jsonNode.contains("storeDbUrl"):
|
|
|
|
if jsonNode["storeDbUrl"].kind != JsonNodeKind.JString:
|
2024-02-20 20:00:03 +00:00
|
|
|
errorResp = "The storeDbUrl config param should be a string"
|
2023-12-11 07:49:13 +00:00
|
|
|
return false
|
|
|
|
|
2024-03-26 13:10:25 +00:00
|
|
|
conf.storeMessageDbUrl = jsonNode["storeDbUrl"].getStr()
|
2023-12-11 07:49:13 +00:00
|
|
|
|
|
|
|
if jsonNode.contains("storeVacuum"):
|
|
|
|
if jsonNode["storeVacuum"].kind != JsonNodeKind.JBool:
|
2024-02-20 20:00:03 +00:00
|
|
|
errorResp = "The storeVacuum config param should be a bool"
|
2023-12-11 07:49:13 +00:00
|
|
|
return false
|
|
|
|
|
2024-03-26 13:10:25 +00:00
|
|
|
conf.storeMessageDbVacuum = jsonNode["storeVacuum"].getBool()
|
2023-12-11 07:49:13 +00:00
|
|
|
|
|
|
|
if jsonNode.contains("storeDbMigration"):
|
|
|
|
if jsonNode["storeDbMigration"].kind != JsonNodeKind.JBool:
|
2024-02-20 20:00:03 +00:00
|
|
|
errorResp = "The storeDbMigration config param should be a bool"
|
2023-12-11 07:49:13 +00:00
|
|
|
return false
|
|
|
|
|
2024-03-26 13:10:25 +00:00
|
|
|
conf.storeMessageDbMigration = jsonNode["storeDbMigration"].getBool()
|
2023-12-11 07:49:13 +00:00
|
|
|
|
|
|
|
if jsonNode.contains("storeMaxNumDbConnections"):
|
|
|
|
if jsonNode["storeMaxNumDbConnections"].kind != JsonNodeKind.JInt:
|
2024-02-20 20:00:03 +00:00
|
|
|
errorResp = "The storeMaxNumDbConnections config param should be an int"
|
2023-12-11 07:49:13 +00:00
|
|
|
return false
|
|
|
|
|
2024-03-26 13:10:25 +00:00
|
|
|
conf.storeMaxNumDbConnections = jsonNode["storeMaxNumDbConnections"].getInt()
|
2023-12-11 07:49:13 +00:00
|
|
|
|
|
|
|
return true
|
|
|
|
|
2024-03-26 13:10:25 +00:00
|
|
|
proc parseTopics(jsonNode: JsonNode, conf: var WakuNodeConf) =
|
2024-03-26 18:44:55 +00:00
|
|
|
if jsonNode.contains("pubsubTopics"):
|
|
|
|
for topic in jsonNode["pubsubTopics"].items:
|
|
|
|
conf.pubsubTopics.add(topic.getStr())
|
2023-07-07 08:53:00 +00:00
|
|
|
else:
|
2024-03-26 18:44:55 +00:00
|
|
|
conf.pubsubTopics = @["/waku/2/default-waku/proto"]
|
2023-07-07 08:53:00 +00:00
|
|
|
|
2024-03-27 14:08:53 +00:00
|
|
|
proc parseRLNRelay(jsonNode: JsonNode, conf: var WakuNodeConf, errorResp: var string): bool =
|
|
|
|
if not jsonNode.contains("rln-relay"):
|
|
|
|
return true
|
|
|
|
|
|
|
|
let jsonNode = jsonNode["rln-relay"]
|
|
|
|
if not jsonNode.contains("enabled"):
|
|
|
|
errorResp = "rlnRelay.enabled attribute is required"
|
|
|
|
return false
|
|
|
|
|
|
|
|
conf.rlnRelay = jsonNode["enabled"].getBool()
|
|
|
|
conf.rlnRelayCredPath = jsonNode{"cred-password"}.getStr()
|
|
|
|
conf.rlnRelayEthClientAddress = EthRpcUrl.parseCmdArg(jsonNode{"eth-client-address"}.getStr("http://localhost:8540"))
|
|
|
|
conf.rlnRelayEthContractAddress = jsonNode{"eth-contract-address"}.getStr()
|
|
|
|
conf.rlnRelayCredPassword = jsonNode{"cred-password"}.getStr()
|
|
|
|
conf.rlnRelayUserMessageLimit = uint64(jsonNode{"user-message-limit"}.getInt(1))
|
|
|
|
conf.rlnEpochSizeSec = uint64(jsonNode{"epoch-sec"}.getInt(1))
|
|
|
|
conf.rlnRelayCredIndex = some(uint(jsonNode{"membership-index"}.getInt()))
|
|
|
|
conf.rlnRelayDynamic = jsonNode{"dynamic"}.getBool()
|
|
|
|
conf.rlnRelayTreePath = jsonNode{"tree-path"}.getStr()
|
|
|
|
conf.rlnRelayBandwidthThreshold = jsonNode{"bandwidth-threshold"}.getInt()
|
|
|
|
|
|
|
|
return true
|
|
|
|
|
2024-03-15 23:08:47 +00:00
|
|
|
proc parseConfig*(
|
|
|
|
configNodeJson: string,
|
2024-03-26 13:10:25 +00:00
|
|
|
conf: var WakuNodeConf,
|
2024-03-15 23:08:47 +00:00
|
|
|
errorResp: var string,
|
|
|
|
): bool {.raises: [].} =
|
2023-07-07 08:53:00 +00:00
|
|
|
if configNodeJson.len == 0:
|
2024-02-20 20:00:03 +00:00
|
|
|
errorResp = "The configNodeJson is empty"
|
2023-07-07 08:53:00 +00:00
|
|
|
return false
|
|
|
|
|
|
|
|
var jsonNode: JsonNode
|
|
|
|
try:
|
|
|
|
jsonNode = parseJson(configNodeJson)
|
2023-12-14 06:16:39 +00:00
|
|
|
except Exception, IOError, JsonParsingError:
|
2024-02-20 20:00:03 +00:00
|
|
|
errorResp = "Exception: " & getCurrentExceptionMsg()
|
2023-07-07 08:53:00 +00:00
|
|
|
return false
|
|
|
|
|
|
|
|
# key
|
2023-12-14 06:16:39 +00:00
|
|
|
try:
|
2024-03-26 13:10:25 +00:00
|
|
|
if not parsePrivateKey(jsonNode, conf, errorResp):
|
2023-12-14 06:16:39 +00:00
|
|
|
return false
|
|
|
|
except Exception, KeyError:
|
2024-02-20 20:00:03 +00:00
|
|
|
errorResp = "Exception calling parsePrivateKey: " & getCurrentExceptionMsg()
|
2023-07-07 08:53:00 +00:00
|
|
|
return false
|
|
|
|
|
|
|
|
# listenAddr
|
2023-12-14 06:16:39 +00:00
|
|
|
var listenAddr: IpAddress
|
|
|
|
try:
|
|
|
|
listenAddr = parseIpAddress("127.0.0.1")
|
2024-03-26 13:10:25 +00:00
|
|
|
if not parseListenAddr(jsonNode, conf, errorResp):
|
2023-12-14 06:16:39 +00:00
|
|
|
return false
|
|
|
|
except Exception, ValueError:
|
2024-02-20 20:00:03 +00:00
|
|
|
errorResp = "Exception calling parseIpAddress: " & getCurrentExceptionMsg()
|
2023-07-07 08:53:00 +00:00
|
|
|
return false
|
|
|
|
|
|
|
|
# port
|
2023-12-14 06:16:39 +00:00
|
|
|
try:
|
2024-03-26 13:10:25 +00:00
|
|
|
if not parsePort(jsonNode, conf, errorResp):
|
2023-12-14 06:16:39 +00:00
|
|
|
return false
|
|
|
|
except Exception, ValueError:
|
2024-02-20 20:00:03 +00:00
|
|
|
errorResp = "Exception calling parsePort: " & getCurrentExceptionMsg()
|
2023-07-07 08:53:00 +00:00
|
|
|
return false
|
|
|
|
|
|
|
|
# relay
|
2023-12-14 06:16:39 +00:00
|
|
|
try:
|
2024-03-26 13:10:25 +00:00
|
|
|
if not parseRelay(jsonNode, conf, errorResp):
|
2023-12-14 06:16:39 +00:00
|
|
|
return false
|
|
|
|
except Exception, KeyError:
|
2024-02-20 20:00:03 +00:00
|
|
|
errorResp = "Exception calling parseRelay: " & getCurrentExceptionMsg()
|
2023-07-07 08:53:00 +00:00
|
|
|
return false
|
|
|
|
|
2024-03-27 14:08:53 +00:00
|
|
|
# clusterId
|
|
|
|
try:
|
|
|
|
if not parseClusterId(jsonNode, conf, errorResp):
|
|
|
|
return false
|
|
|
|
except Exception, KeyError:
|
|
|
|
errorResp = "Exception calling parseClusterId: " & getCurrentExceptionMsg()
|
|
|
|
return false
|
|
|
|
|
2023-07-07 08:53:00 +00:00
|
|
|
# topics
|
2023-12-14 06:16:39 +00:00
|
|
|
try:
|
2024-03-26 13:10:25 +00:00
|
|
|
parseTopics(jsonNode, conf)
|
2023-12-14 06:16:39 +00:00
|
|
|
except Exception, KeyError:
|
2024-02-20 20:00:03 +00:00
|
|
|
errorResp = "Exception calling parseTopics: " & getCurrentExceptionMsg()
|
2023-12-14 06:16:39 +00:00
|
|
|
return false
|
2023-07-07 08:53:00 +00:00
|
|
|
|
2023-12-11 07:49:13 +00:00
|
|
|
# store
|
2023-12-14 06:16:39 +00:00
|
|
|
try:
|
2024-03-26 13:10:25 +00:00
|
|
|
if not parseStore(jsonNode, conf, errorResp):
|
2023-12-14 06:16:39 +00:00
|
|
|
return false
|
|
|
|
except Exception, KeyError:
|
2024-02-20 20:00:03 +00:00
|
|
|
errorResp = "Exception calling parseStore: " & getCurrentExceptionMsg()
|
2023-12-11 07:49:13 +00:00
|
|
|
return false
|
|
|
|
|
2024-03-27 14:08:53 +00:00
|
|
|
# rln
|
|
|
|
try:
|
|
|
|
if not parseRLNRelay(jsonNode, conf, errorResp):
|
|
|
|
return false
|
|
|
|
except Exception, KeyError:
|
|
|
|
errorResp = "Exception calling parseRLNRelay: " & getCurrentExceptionMsg()
|
|
|
|
return false
|
|
|
|
|
2023-07-07 08:53:00 +00:00
|
|
|
return true
|