2020-08-26 14:20:04 +02:00
|
|
|
import
|
2020-09-01 04:09:54 +02:00
|
|
|
confutils, chronicles, chronos, stew/byteutils, stew/shims/net as stewNet,
|
2020-08-27 04:44:52 +02:00
|
|
|
eth/[keys, p2p],
|
2020-08-26 14:20:04 +02:00
|
|
|
../../waku/protocol/v1/waku_protocol,
|
|
|
|
../../waku/node/v1/waku_helpers,
|
2020-09-01 04:09:54 +02:00
|
|
|
../../waku/node/common,
|
2020-08-26 14:20:04 +02:00
|
|
|
./config_example
|
|
|
|
|
|
|
|
## This is a simple Waku v1 example to show the Waku v1 API usage.
|
|
|
|
|
|
|
|
const clientId = "Waku example v1"
|
|
|
|
|
|
|
|
let
|
|
|
|
# Load the cli configuration from `config_example.nim`.
|
|
|
|
config = WakuNodeConf.load()
|
|
|
|
# Seed the rng.
|
|
|
|
rng = keys.newRng()
|
|
|
|
# Set up the address according to NAT information.
|
2020-09-01 04:09:54 +02:00
|
|
|
(ipExt, tcpPortExt, udpPortExt) = setupNat(config.nat, clientId,
|
|
|
|
Port(config.tcpPort + 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.tcpPort + config.portsShift),
|
|
|
|
udpPort: Port(config.udpPort + config.portsShift))
|
|
|
|
else:
|
|
|
|
Address(ip: ipExt.get(),
|
|
|
|
tcpPort: Port(config.tcpPort + config.portsShift),
|
|
|
|
udpPort: Port(config.udpPort + config.portsShift))
|
2020-08-26 14:20:04 +02:00
|
|
|
|
|
|
|
# Create Ethereum Node
|
|
|
|
var node = newEthereumNode(config.nodekey, # Node identifier
|
|
|
|
address, # Address reachable for incoming requests
|
|
|
|
1, # Network Id, only applicable for ETH protocol
|
|
|
|
nil, # Database, not required for Waku
|
|
|
|
clientId, # Client id string
|
|
|
|
addAllCapabilities = false, # Disable default all RLPx capabilities
|
|
|
|
rng = rng)
|
|
|
|
|
|
|
|
node.addCapability Waku # Enable only the Waku protocol.
|
|
|
|
|
|
|
|
# Set up the Waku configuration.
|
|
|
|
let wakuConfig = WakuConfig(powRequirement: 0.002,
|
|
|
|
bloom: some(fullBloom()), # Full bloom filter
|
|
|
|
isLightNode: false, # Full node
|
|
|
|
maxMsgSize: waku_protocol.defaultMaxMsgSize,
|
|
|
|
topics: none(seq[waku_protocol.Topic]) # empty topic interest
|
|
|
|
)
|
|
|
|
node.configureWaku(wakuConfig)
|
|
|
|
|
|
|
|
# Optionally direct connect to a set of nodes.
|
|
|
|
if config.staticnodes.len > 0:
|
|
|
|
connectToNodes(node, config.staticnodes)
|
|
|
|
|
|
|
|
# Connect to the network, which will make the node start listening and/or
|
|
|
|
# connect to bootnodes, and/or start discovery.
|
|
|
|
# This will block until first connection is made, which in this case can only
|
|
|
|
# happen if we directly connect to nodes (step above) or if an incoming
|
2020-08-27 04:44:52 +02:00
|
|
|
# connection occurs, which is why we use a callback to exit on errors instead of
|
|
|
|
# using `await`.
|
2020-08-26 14:20:04 +02:00
|
|
|
# TODO: This looks a bit awkward and the API should perhaps be altered here.
|
2020-08-27 04:44:52 +02:00
|
|
|
let connectedFut = node.connectToNetwork(@[],
|
2020-08-26 14:20:04 +02:00
|
|
|
true, # Enable listening
|
|
|
|
false # Disable discovery (only discovery v4 is currently supported)
|
|
|
|
)
|
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-08-26 14:20:04 +02:00
|
|
|
|
|
|
|
# Using a hardcoded symmetric key for encryption of the payload for the sake of
|
|
|
|
# simplicity.
|
|
|
|
var symKey: SymKey
|
|
|
|
symKey[31] = 1
|
|
|
|
# Asymmetric keypair to sign the payload.
|
|
|
|
let signKeyPair = KeyPair.random(rng[])
|
|
|
|
|
|
|
|
# Code to be executed on receival of a message on filter.
|
|
|
|
proc handler(msg: ReceivedMessage) =
|
|
|
|
if msg.decoded.src.isSome():
|
|
|
|
echo "Received message from ", $msg.decoded.src.get(), ": ",
|
|
|
|
string.fromBytes(msg.decoded.payload)
|
|
|
|
|
|
|
|
# Create and subscribe filter with above handler.
|
|
|
|
let
|
|
|
|
topic = [byte 0, 0, 0, 0]
|
|
|
|
filter = initFilter(symKey = some(symKey), topics = @[topic])
|
|
|
|
discard node.subscribeFilter(filter, handler)
|
|
|
|
|
|
|
|
# Repeat the posting of a message every 5 seconds.
|
|
|
|
proc repeatMessage(udata: pointer) {.gcsafe.} =
|
|
|
|
{.gcsafe.}:
|
|
|
|
# Post a waku message on the network, encrypted with provided symmetric key,
|
|
|
|
# signed with asymmetric key, on topic and with ttl of 30 seconds.
|
|
|
|
let posted = node.postMessage(
|
|
|
|
symKey = some(symKey), src = some(signKeyPair.seckey),
|
|
|
|
ttl = 30, topic = topic, payload = @[byte 0x48, 0x65, 0x6C, 0x6C, 0x6F])
|
|
|
|
|
|
|
|
if posted: echo "Posted message as ", $signKeyPair.pubkey
|
|
|
|
else: echo "Posting message failed."
|
|
|
|
|
|
|
|
discard setTimer(Moment.fromNow(5.seconds), repeatMessage)
|
|
|
|
discard setTimer(Moment.fromNow(5.seconds), repeatMessage)
|
|
|
|
|
|
|
|
runForever()
|