2020-05-19 13:00:03 +08:00
|
|
|
import
|
2021-09-15 10:01:09 +02:00
|
|
|
strformat, os, osproc, net, strformat, chronicles, confutils, json,
|
2020-05-19 13:00:03 +08:00
|
|
|
libp2p/multiaddress,
|
|
|
|
libp2p/crypto/crypto,
|
2020-05-20 13:34:43 +08:00
|
|
|
libp2p/crypto/secp,
|
2020-05-19 13:00:03 +08:00
|
|
|
libp2p/peerinfo
|
|
|
|
|
2020-05-20 13:34:43 +08:00
|
|
|
# Fix ambiguous call error
|
|
|
|
import strutils except fromHex
|
|
|
|
|
2020-05-20 14:21:17 +08:00
|
|
|
const
|
2021-09-15 10:01:09 +02:00
|
|
|
defaults ="--log-level:TRACE --metrics-logging --metrics-server --rpc"
|
2020-07-13 12:08:03 +02:00
|
|
|
wakuNodeBin = "build" / "wakunode2"
|
2020-06-15 12:06:41 +08:00
|
|
|
metricsDir = "metrics"
|
2020-05-21 12:28:57 +08:00
|
|
|
portOffset = 2
|
2020-05-20 14:21:17 +08:00
|
|
|
|
2020-05-19 13:00:03 +08:00
|
|
|
type
|
|
|
|
NodeInfo* = object
|
|
|
|
cmd: string
|
2020-05-21 12:28:57 +08:00
|
|
|
master: bool
|
2020-05-19 13:00:03 +08:00
|
|
|
address: string
|
2020-05-20 14:21:17 +08:00
|
|
|
shift: int
|
2020-05-19 13:00:03 +08:00
|
|
|
label: string
|
|
|
|
|
2020-05-27 13:37:27 +08:00
|
|
|
Topology = enum
|
|
|
|
Star,
|
|
|
|
FullMesh
|
2021-09-15 10:01:09 +02:00
|
|
|
|
|
|
|
WakuNetworkConf* = object
|
|
|
|
topology* {.
|
|
|
|
desc: "Set the network topology."
|
|
|
|
defaultValue: FullMesh
|
|
|
|
name: "topology" .}: Topology
|
|
|
|
|
|
|
|
amount* {.
|
|
|
|
desc: "Amount of relay nodes to be started."
|
|
|
|
defaultValue: 16
|
|
|
|
name: "amount" .}: int
|
2020-05-19 13:00:03 +08:00
|
|
|
|
2020-05-27 13:37:27 +08:00
|
|
|
# NOTE: Don't distinguish between node types here a la full node, light node etc
|
2020-05-21 12:28:57 +08:00
|
|
|
proc initNodeCmd(shift: int, staticNodes: seq[string] = @[], master = false, label: string): NodeInfo =
|
2020-05-19 13:00:03 +08:00
|
|
|
let
|
2020-07-13 12:08:03 +02:00
|
|
|
rng = crypto.newRng()
|
|
|
|
key = SkPrivateKey.random(rng[])
|
2020-05-20 13:34:43 +08:00
|
|
|
hkey = key.getBytes().toHex()
|
2020-05-26 11:55:53 +08:00
|
|
|
rkey = SkPrivateKey.init(fromHex(hkey))[] #assumes ok
|
2020-05-20 13:34:43 +08:00
|
|
|
privKey = PrivateKey(scheme: Secp256k1, skkey: rkey)
|
|
|
|
#privKey = PrivateKey.random(Secp256k1)
|
2021-11-04 15:46:38 +01:00
|
|
|
pubkey = privKey.getPublicKey()[] #assumes ok
|
2020-05-26 11:55:53 +08:00
|
|
|
keys = KeyPair(seckey: privKey, pubkey: pubkey)
|
2021-11-04 15:46:38 +01:00
|
|
|
peerInfo = PeerInfo.new(privKey)
|
2020-05-28 11:28:44 +08:00
|
|
|
port = 60000 + shift
|
|
|
|
#DefaultAddr = "/ip4/127.0.0.1/tcp/55505"
|
|
|
|
address = "/ip4/127.0.0.1/tcp/" & $port
|
2020-06-01 11:15:37 +08:00
|
|
|
hostAddress = MultiAddress.init(address).tryGet()
|
2020-05-19 13:00:03 +08:00
|
|
|
|
2020-05-28 11:40:41 +08:00
|
|
|
info "Address", address
|
2020-05-28 11:28:44 +08:00
|
|
|
# TODO: Need to port shift
|
2020-05-19 13:00:03 +08:00
|
|
|
peerInfo.addrs.add(hostAddress)
|
2020-09-16 12:23:10 +08:00
|
|
|
let id = $peerInfo.peerId
|
2020-05-20 14:21:17 +08:00
|
|
|
|
|
|
|
info "PeerInfo", id = id, addrs = peerInfo.addrs
|
|
|
|
let listenStr = $peerInfo.addrs[0] & "/p2p/" & id
|
|
|
|
|
|
|
|
result.cmd = wakuNodeBin & " " & defaults & " "
|
|
|
|
result.cmd &= "--nodekey:" & hkey & " "
|
|
|
|
result.cmd &= "--ports-shift:" & $shift & " "
|
|
|
|
if staticNodes.len > 0:
|
|
|
|
for staticNode in staticNodes:
|
|
|
|
result.cmd &= "--staticnode:" & staticNode & " "
|
|
|
|
result.shift = shift
|
|
|
|
result.label = label
|
2020-05-21 12:28:57 +08:00
|
|
|
result.master = master
|
2020-05-20 14:21:17 +08:00
|
|
|
result.address = listenStr
|
|
|
|
|
|
|
|
info "Node command created.", cmd=result.cmd, address = result.address
|
2020-05-19 13:00:03 +08:00
|
|
|
|
2020-05-21 12:36:30 +08:00
|
|
|
proc starNetwork(amount: int): seq[NodeInfo] =
|
|
|
|
let masterNode = initNodeCmd(portOffset, master = true, label = "master node")
|
|
|
|
result.add(masterNode)
|
|
|
|
for i in 1..<amount:
|
|
|
|
result.add(initNodeCmd(portOffset + i, @[masterNode.address], label = "full node"))
|
2020-05-19 13:00:03 +08:00
|
|
|
|
2020-05-27 13:37:27 +08:00
|
|
|
proc fullMeshNetwork(amount: int): seq[NodeInfo] =
|
|
|
|
debug "amount", amount
|
|
|
|
for i in 0..<amount:
|
|
|
|
var staticnodes: seq[string]
|
|
|
|
for item in result:
|
|
|
|
staticnodes.add(item.address)
|
2020-05-28 10:58:37 +08:00
|
|
|
result.add(initNodeCmd(portOffset + i, staticnodes, label = "full node"))
|
2020-05-27 13:37:27 +08:00
|
|
|
|
2020-06-15 12:06:41 +08:00
|
|
|
proc generatePrometheusConfig(nodes: seq[NodeInfo], outputFile: string) =
|
|
|
|
var config = """
|
|
|
|
global:
|
|
|
|
scrape_interval: 1s
|
|
|
|
|
|
|
|
scrape_configs:
|
|
|
|
- job_name: "wakusim"
|
|
|
|
static_configs:"""
|
|
|
|
var count = 0
|
|
|
|
for node in nodes:
|
|
|
|
let port = 8008 + node.shift
|
|
|
|
config &= &"""
|
|
|
|
|
|
|
|
- targets: ['127.0.0.1:{port}']
|
|
|
|
labels:
|
|
|
|
node: '{count}'"""
|
|
|
|
count += 1
|
|
|
|
|
|
|
|
var (path, file) = splitPath(outputFile)
|
|
|
|
createDir(path)
|
|
|
|
writeFile(outputFile, config)
|
|
|
|
|
|
|
|
proc proccessGrafanaDashboard(nodes: int, inputFile: string,
|
|
|
|
outputFile: string) =
|
|
|
|
# from https://github.com/status-im/nim-beacon-chain/blob/master/tests/simulation/process_dashboard.nim
|
|
|
|
var
|
|
|
|
inputData = parseFile(inputFile)
|
|
|
|
panels = inputData["panels"].copy()
|
|
|
|
numPanels = len(panels)
|
|
|
|
gridHeight = 0
|
|
|
|
outputData = inputData
|
|
|
|
|
|
|
|
for panel in panels:
|
|
|
|
if panel["gridPos"]["x"].getInt() == 0:
|
|
|
|
gridHeight += panel["gridPos"]["h"].getInt()
|
|
|
|
|
|
|
|
outputData["panels"] = %* []
|
|
|
|
for nodeNum in 0 .. (nodes - 1):
|
|
|
|
var
|
|
|
|
nodePanels = panels.copy()
|
|
|
|
panelIndex = 0
|
|
|
|
for panel in nodePanels.mitems:
|
|
|
|
panel["title"] = %* replace(panel["title"].getStr(), "#0", "#" & $nodeNum)
|
|
|
|
panel["id"] = %* (panelIndex + (nodeNum * numPanels))
|
|
|
|
panel["gridPos"]["y"] = %* (panel["gridPos"]["y"].getInt() + (nodeNum * gridHeight))
|
|
|
|
var targets = panel["targets"]
|
|
|
|
for target in targets.mitems:
|
|
|
|
target["expr"] = %* replace(target["expr"].getStr(), "{node=\"0\"}", "{node=\"" & $nodeNum & "\"}")
|
|
|
|
outputData["panels"].add(panel)
|
|
|
|
panelIndex.inc()
|
|
|
|
|
|
|
|
outputData["uid"] = %* (outputData["uid"].getStr() & "a")
|
|
|
|
outputData["title"] = %* (outputData["title"].getStr() & " (all nodes)")
|
|
|
|
writeFile(outputFile, pretty(outputData))
|
|
|
|
|
2020-05-21 12:36:30 +08:00
|
|
|
when isMainModule:
|
2021-09-15 10:01:09 +02:00
|
|
|
let conf = WakuNetworkConf.load()
|
|
|
|
|
2020-05-21 12:36:30 +08:00
|
|
|
# TODO: WakuNetworkConf
|
|
|
|
var nodes: seq[NodeInfo]
|
2021-09-15 10:01:09 +02:00
|
|
|
let topology = conf.topology
|
2020-06-15 11:06:08 +08:00
|
|
|
|
|
|
|
# Scenario xx2 14
|
2021-09-15 10:01:09 +02:00
|
|
|
let amount = conf.amount
|
2020-05-27 13:37:27 +08:00
|
|
|
|
|
|
|
case topology:
|
|
|
|
of Star:
|
|
|
|
nodes = starNetwork(amount)
|
|
|
|
of FullMesh:
|
|
|
|
nodes = fullMeshNetwork(amount)
|
|
|
|
|
2021-09-15 10:01:09 +02:00
|
|
|
# var staticnodes: seq[string]
|
|
|
|
# for i in 0..<amount:
|
|
|
|
# # TODO: could also select nodes randomly
|
|
|
|
# staticnodes.add(nodes[i].address)
|
2020-05-28 11:28:44 +08:00
|
|
|
|
2020-06-15 11:06:08 +08:00
|
|
|
# Scenario xx1 - 16 full nodes, one app topic, full mesh, gossip
|
|
|
|
|
|
|
|
# Scenario xx2 - 14 full nodes, two edge nodes, one app topic, full mesh, gossip
|
2020-06-02 11:18:55 +08:00
|
|
|
# NOTE: Only connecting to one node here
|
2020-06-15 11:06:08 +08:00
|
|
|
#var nodesubseta: seq[string]
|
|
|
|
#var nodesubsetb: seq[string]
|
|
|
|
#nodesubseta.add(staticnodes[0])
|
|
|
|
#nodesubsetb.add(staticnodes[amount-1])
|
|
|
|
## XXX: Let's turn them into normal nodes
|
|
|
|
#nodes.add(initNodeCmd(0, nodesubseta, label = "edge node (A)"))
|
|
|
|
#nodes.add(initNodeCmd(1, nodesubsetb, label = "edge node (B)"))
|
2020-05-21 12:36:30 +08:00
|
|
|
|
|
|
|
var commandStr = "multitail -s 2 -M 0 -x \"Waku Simulation\""
|
|
|
|
var count = 0
|
|
|
|
var sleepDuration = 0
|
|
|
|
for node in nodes:
|
2020-05-28 11:28:44 +08:00
|
|
|
if topology in {Star}: #DiscoveryBased
|
|
|
|
sleepDuration = if node.master: 0
|
|
|
|
else: 1
|
2020-05-21 12:36:30 +08:00
|
|
|
commandStr &= &" -cT ansi -t 'node #{count} {node.label}' -l 'sleep {sleepDuration}; {node.cmd}; echo [node execution completed]; while true; do sleep 100; done'"
|
2020-05-28 11:28:44 +08:00
|
|
|
if topology == FullMesh:
|
|
|
|
sleepDuration += 1
|
|
|
|
count += 1
|
2020-05-21 12:36:30 +08:00
|
|
|
|
2020-06-15 12:06:41 +08:00
|
|
|
generatePrometheusConfig(nodes, metricsDir / "prometheus" / "prometheus.yml")
|
|
|
|
proccessGrafanaDashboard(nodes.len,
|
|
|
|
metricsDir / "waku-grafana-dashboard.json",
|
|
|
|
metricsDir / "waku-sim-all-nodes-grafana-dashboard.json")
|
|
|
|
|
|
|
|
|
2020-05-21 12:36:30 +08:00
|
|
|
let errorCode = execCmd(commandStr)
|
|
|
|
if errorCode != 0:
|
|
|
|
error "launch command failed", command=commandStr
|