2021-04-13 14:54:06 -04:00
|
|
|
package waku
|
2021-03-18 19:21:45 -04:00
|
|
|
|
|
|
|
import (
|
|
|
|
"context"
|
2021-10-04 18:38:27 -04:00
|
|
|
"crypto/ecdsa"
|
2021-04-18 20:03:16 -04:00
|
|
|
"database/sql"
|
2022-07-25 08:24:42 -04:00
|
|
|
"encoding/json"
|
2023-05-08 11:19:44 -04:00
|
|
|
"errors"
|
2021-03-18 19:21:45 -04:00
|
|
|
"fmt"
|
|
|
|
"net"
|
|
|
|
"os"
|
|
|
|
"os/signal"
|
2022-12-07 23:04:56 +05:30
|
|
|
"sync"
|
2021-03-18 19:21:45 -04:00
|
|
|
"syscall"
|
2021-06-24 09:02:53 -04:00
|
|
|
"time"
|
2021-03-18 19:21:45 -04:00
|
|
|
|
2023-05-24 11:34:35 -04:00
|
|
|
rcmgr "github.com/libp2p/go-libp2p/p2p/host/resource-manager"
|
|
|
|
"github.com/pbnjay/memory"
|
|
|
|
|
2022-11-25 17:24:34 -04:00
|
|
|
wmetrics "github.com/waku-org/go-waku/waku/v2/metrics"
|
2023-06-05 10:39:38 -04:00
|
|
|
"github.com/waku-org/go-waku/waku/v2/peers"
|
2023-03-09 11:48:25 -04:00
|
|
|
"github.com/waku-org/go-waku/waku/v2/rendezvous"
|
2022-11-25 17:24:34 -04:00
|
|
|
|
2022-07-25 08:24:42 -04:00
|
|
|
"github.com/ethereum/go-ethereum/accounts/keystore"
|
2021-03-23 10:46:16 -04:00
|
|
|
"github.com/ethereum/go-ethereum/crypto"
|
2021-09-30 19:03:19 -04:00
|
|
|
"github.com/ethereum/go-ethereum/p2p/enode"
|
2021-04-13 14:54:06 -04:00
|
|
|
dssql "github.com/ipfs/go-ds-sql"
|
2022-02-21 11:23:52 -04:00
|
|
|
"github.com/urfave/cli/v2"
|
2022-01-18 14:17:06 -04:00
|
|
|
"go.uber.org/zap"
|
2023-02-01 19:35:31 -04:00
|
|
|
"go.uber.org/zap/zapcore"
|
2021-10-03 17:45:07 -04:00
|
|
|
|
2021-04-13 14:54:06 -04:00
|
|
|
"github.com/libp2p/go-libp2p"
|
2021-10-14 22:15:02 -04:00
|
|
|
"github.com/libp2p/go-libp2p/config"
|
2022-10-19 15:39:32 -04:00
|
|
|
"github.com/libp2p/go-libp2p/core/peer"
|
2023-06-05 10:39:38 -04:00
|
|
|
"github.com/libp2p/go-libp2p/core/peerstore"
|
2023-02-16 12:17:52 -04:00
|
|
|
"github.com/libp2p/go-libp2p/core/protocol"
|
2022-08-25 16:36:04 -04:00
|
|
|
"github.com/libp2p/go-libp2p/p2p/transport/tcp"
|
2021-10-04 18:38:27 -04:00
|
|
|
|
2021-10-08 09:50:56 -04:00
|
|
|
pubsub "github.com/libp2p/go-libp2p-pubsub"
|
2022-10-19 15:39:32 -04:00
|
|
|
"github.com/libp2p/go-libp2p/p2p/host/peerstore/pstoreds"
|
2022-08-25 16:36:04 -04:00
|
|
|
ws "github.com/libp2p/go-libp2p/p2p/transport/websocket"
|
2021-04-22 09:07:22 -04:00
|
|
|
"github.com/multiformats/go-multiaddr"
|
2022-11-09 15:53:01 -04:00
|
|
|
"github.com/waku-org/go-waku/logging"
|
|
|
|
"github.com/waku-org/go-waku/waku/metrics"
|
|
|
|
"github.com/waku-org/go-waku/waku/persistence"
|
|
|
|
"github.com/waku-org/go-waku/waku/persistence/sqlite"
|
|
|
|
"github.com/waku-org/go-waku/waku/v2/dnsdisc"
|
|
|
|
"github.com/waku-org/go-waku/waku/v2/node"
|
|
|
|
"github.com/waku-org/go-waku/waku/v2/protocol/filter"
|
2023-04-11 10:38:16 -04:00
|
|
|
"github.com/waku-org/go-waku/waku/v2/protocol/legacy_filter"
|
2022-11-09 15:53:01 -04:00
|
|
|
"github.com/waku-org/go-waku/waku/v2/protocol/lightpush"
|
|
|
|
"github.com/waku-org/go-waku/waku/v2/protocol/peer_exchange"
|
|
|
|
"github.com/waku-org/go-waku/waku/v2/protocol/relay"
|
|
|
|
"github.com/waku-org/go-waku/waku/v2/protocol/store"
|
|
|
|
"github.com/waku-org/go-waku/waku/v2/rest"
|
|
|
|
"github.com/waku-org/go-waku/waku/v2/rpc"
|
|
|
|
"github.com/waku-org/go-waku/waku/v2/utils"
|
2021-03-18 19:21:45 -04:00
|
|
|
)
|
|
|
|
|
2021-10-03 17:45:07 -04:00
|
|
|
func failOnErr(err error, msg string) {
|
2021-04-13 14:54:06 -04:00
|
|
|
if err != nil {
|
2022-01-18 14:17:06 -04:00
|
|
|
utils.Logger().Fatal(msg, zap.Error(err))
|
2021-04-13 14:54:06 -04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-03-09 11:48:25 -04:00
|
|
|
func requiresDB(options Options) bool {
|
2023-03-09 18:42:50 -04:00
|
|
|
return options.Store.Enable || options.Rendezvous.Server
|
2023-03-09 11:48:25 -04:00
|
|
|
}
|
|
|
|
|
2023-05-24 11:34:35 -04:00
|
|
|
func scalePerc(value float64) float64 {
|
|
|
|
if value > 100 {
|
|
|
|
return 100
|
|
|
|
}
|
|
|
|
|
|
|
|
if value < 0.1 {
|
|
|
|
return 0.1
|
|
|
|
}
|
|
|
|
|
|
|
|
return value
|
|
|
|
}
|
|
|
|
|
2022-08-25 20:18:03 -04:00
|
|
|
const dialTimeout = 7 * time.Second
|
|
|
|
|
2021-10-09 14:18:53 -04:00
|
|
|
// Execute starts a go-waku node with settings determined by the Options parameter
|
2021-10-03 17:45:07 -04:00
|
|
|
func Execute(options Options) {
|
2021-10-04 18:38:27 -04:00
|
|
|
if options.GenerateKey {
|
2022-07-25 08:24:42 -04:00
|
|
|
if err := writePrivateKeyToFile(options.KeyFile, []byte(options.KeyPasswd), options.Overwrite); err != nil {
|
2021-10-04 18:38:27 -04:00
|
|
|
failOnErr(err, "nodekey error")
|
|
|
|
}
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2021-10-14 22:15:02 -04:00
|
|
|
hostAddr, err := net.ResolveTCPAddr("tcp", fmt.Sprintf("%s:%d", options.Address, options.Port))
|
|
|
|
failOnErr(err, "invalid host address")
|
2021-03-18 19:21:45 -04:00
|
|
|
|
2021-10-04 18:38:27 -04:00
|
|
|
prvKey, err := getPrivKey(options)
|
|
|
|
failOnErr(err, "nodekey error")
|
2021-04-13 14:54:06 -04:00
|
|
|
|
2022-05-27 15:55:35 -04:00
|
|
|
p2pPrvKey := utils.EcdsaPrivKeyToSecp256k1PrivKey(prvKey)
|
|
|
|
id, err := peer.IDFromPublicKey(p2pPrvKey.GetPublic())
|
2022-05-27 09:25:06 -04:00
|
|
|
failOnErr(err, "deriving peer ID from private key")
|
|
|
|
logger := utils.Logger().With(logging.HostID("node", id))
|
|
|
|
|
2021-10-03 17:45:07 -04:00
|
|
|
var db *sql.DB
|
2023-01-04 13:58:14 -04:00
|
|
|
var migrationFn func(*sql.DB) error
|
2023-03-09 11:48:25 -04:00
|
|
|
if requiresDB(options) {
|
2023-02-10 16:17:23 -04:00
|
|
|
db, migrationFn, err = ExtractDBAndMigration(options.Store.DatabaseURL)
|
2023-01-04 13:58:14 -04:00
|
|
|
failOnErr(err, "Could not connect to DB")
|
2021-10-03 17:45:07 -04:00
|
|
|
}
|
2021-04-22 09:07:22 -04:00
|
|
|
|
2021-10-03 17:45:07 -04:00
|
|
|
ctx := context.Background()
|
2021-04-18 20:03:16 -04:00
|
|
|
|
2021-10-03 17:45:07 -04:00
|
|
|
var metricsServer *metrics.Server
|
|
|
|
if options.Metrics.Enable {
|
2022-05-30 11:55:30 -04:00
|
|
|
metricsServer = metrics.NewMetricsServer(options.Metrics.Address, options.Metrics.Port, logger)
|
2021-10-03 17:45:07 -04:00
|
|
|
go metricsServer.Start()
|
2022-12-09 23:08:18 -04:00
|
|
|
wmetrics.RecordVersion(ctx, node.Version, node.GitCommit)
|
2021-10-03 17:45:07 -04:00
|
|
|
}
|
2021-04-18 20:03:16 -04:00
|
|
|
|
2023-02-01 19:35:31 -04:00
|
|
|
lvl, err := zapcore.ParseLevel(options.LogLevel)
|
|
|
|
if err != nil {
|
|
|
|
failOnErr(err, "log level error")
|
|
|
|
}
|
|
|
|
|
2021-10-03 17:45:07 -04:00
|
|
|
nodeOpts := []node.WakuNodeOption{
|
2022-05-27 09:25:06 -04:00
|
|
|
node.WithLogger(logger),
|
2023-02-01 19:35:31 -04:00
|
|
|
node.WithLogLevel(lvl),
|
2021-10-03 17:45:07 -04:00
|
|
|
node.WithPrivateKey(prvKey),
|
2021-11-17 12:19:42 -04:00
|
|
|
node.WithHostAddress(hostAddr),
|
2022-08-15 13:13:45 -04:00
|
|
|
node.WithKeepAlive(options.KeepAlive),
|
2021-10-03 17:45:07 -04:00
|
|
|
}
|
2023-02-08 12:02:06 -04:00
|
|
|
if len(options.AdvertiseAddresses) != 0 {
|
|
|
|
nodeOpts = append(nodeOpts, node.WithAdvertiseAddresses(options.AdvertiseAddresses...))
|
2022-03-21 20:48:46 -04:00
|
|
|
}
|
|
|
|
|
2023-05-08 11:19:44 -04:00
|
|
|
if options.ExtIP != "" {
|
|
|
|
ip := net.ParseIP(options.ExtIP)
|
|
|
|
if ip == nil {
|
|
|
|
failOnErr(errors.New("invalid IP address"), "could not set external IP address")
|
|
|
|
}
|
|
|
|
nodeOpts = append(nodeOpts, node.WithExternalIP(ip))
|
|
|
|
}
|
|
|
|
|
2023-05-15 11:49:51 -04:00
|
|
|
if options.Dns4DomainName != "" {
|
|
|
|
nodeOpts = append(nodeOpts, node.WithDns4Domain(options.Dns4DomainName))
|
|
|
|
}
|
|
|
|
|
2022-03-22 09:12:58 -04:00
|
|
|
libp2pOpts := node.DefaultLibP2POptions
|
2023-05-24 11:34:35 -04:00
|
|
|
|
|
|
|
memPerc := scalePerc(options.ResourceScalingMemoryPercent)
|
|
|
|
fdPerc := scalePerc(options.ResourceScalingFDPercent)
|
|
|
|
limits := rcmgr.DefaultLimits // Default memory limit: 1/8th of total memory, minimum 128MB, maximum 1GB
|
|
|
|
scaledLimits := limits.Scale(int64(float64(memory.TotalMemory())*memPerc/100), int(float64(getNumFDs())*fdPerc/100))
|
|
|
|
resourceManager, err := rcmgr.NewResourceManager(rcmgr.NewFixedLimiter(scaledLimits))
|
|
|
|
failOnErr(err, "setting resource limits")
|
|
|
|
|
|
|
|
libp2pOpts = append(libp2pOpts, libp2p.ResourceManager(resourceManager))
|
|
|
|
libp2p.SetDefaultServiceLimits(&limits)
|
|
|
|
|
2023-02-08 12:02:06 -04:00
|
|
|
if len(options.AdvertiseAddresses) == 0 {
|
2022-03-22 09:12:58 -04:00
|
|
|
libp2pOpts = append(libp2pOpts, libp2p.NATPortMap()) // Attempt to open ports using uPNP for NATed hosts.)
|
|
|
|
}
|
|
|
|
|
2023-05-12 17:52:42 -04:00
|
|
|
// Node can be a circuit relay server
|
|
|
|
if options.CircuitRelay {
|
|
|
|
libp2pOpts = append(libp2pOpts, libp2p.EnableRelayService())
|
|
|
|
}
|
|
|
|
|
2022-10-26 09:28:28 -04:00
|
|
|
if options.UserAgent != "" {
|
|
|
|
libp2pOpts = append(libp2pOpts, libp2p.UserAgent(options.UserAgent))
|
|
|
|
}
|
|
|
|
|
2022-03-22 09:12:58 -04:00
|
|
|
if options.Websocket.Enable {
|
2022-08-25 16:36:04 -04:00
|
|
|
nodeOpts = append(nodeOpts, node.WithWebsockets(options.Websocket.Address, options.Websocket.WSPort))
|
2022-03-22 09:12:58 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
if options.Websocket.Secure {
|
2022-08-25 16:36:04 -04:00
|
|
|
nodeOpts = append(nodeOpts, node.WithSecureWebsockets(options.Websocket.Address, options.Websocket.WSSPort, options.Websocket.CertPath, options.Websocket.KeyPath))
|
2021-10-03 17:45:07 -04:00
|
|
|
}
|
2021-04-18 20:03:16 -04:00
|
|
|
|
2021-10-14 14:17:01 -04:00
|
|
|
if options.ShowAddresses {
|
2021-10-14 22:15:02 -04:00
|
|
|
printListeningAddresses(ctx, nodeOpts, options)
|
2021-10-14 14:17:01 -04:00
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2022-11-25 16:54:11 -04:00
|
|
|
if options.Store.Enable && options.PersistPeers {
|
|
|
|
// Create persistent peerstore
|
|
|
|
queries, err := sqlite.NewQueries("peerstore", db)
|
|
|
|
failOnErr(err, "Peerstore")
|
2021-10-01 13:43:03 -04:00
|
|
|
|
2022-11-25 16:54:11 -04:00
|
|
|
datastore := dssql.NewDatastore(db, queries)
|
|
|
|
opts := pstoreds.DefaultOpts()
|
|
|
|
peerStore, err := pstoreds.NewPeerstore(ctx, datastore, opts)
|
|
|
|
failOnErr(err, "Peerstore")
|
2021-10-01 13:43:03 -04:00
|
|
|
|
2023-06-05 10:39:38 -04:00
|
|
|
nodeOpts = append(nodeOpts, node.WithPeerStore(peerStore))
|
2021-10-03 17:45:07 -04:00
|
|
|
}
|
2021-03-18 19:21:45 -04:00
|
|
|
|
2021-10-03 17:45:07 -04:00
|
|
|
nodeOpts = append(nodeOpts, node.WithLibP2POptions(libp2pOpts...))
|
2022-12-08 23:08:04 -04:00
|
|
|
nodeOpts = append(nodeOpts, node.WithNTP())
|
2021-10-01 13:49:50 -04:00
|
|
|
|
2022-02-21 11:23:52 -04:00
|
|
|
if options.Relay.Enable {
|
2021-10-03 17:45:07 -04:00
|
|
|
var wakurelayopts []pubsub.Option
|
|
|
|
wakurelayopts = append(wakurelayopts, pubsub.WithPeerExchange(options.Relay.PeerExchange))
|
2021-12-06 09:43:00 +01:00
|
|
|
nodeOpts = append(nodeOpts, node.WithWakuRelayAndMinPeers(options.Relay.MinRelayPeersToPublish, wakurelayopts...))
|
2021-10-03 17:45:07 -04:00
|
|
|
}
|
2021-06-10 15:59:51 +03:00
|
|
|
|
2023-04-13 15:28:46 -04:00
|
|
|
nodeOpts = append(nodeOpts, node.WithWakuFilterLightNode())
|
|
|
|
|
2021-10-03 17:45:07 -04:00
|
|
|
if options.Filter.Enable {
|
2023-04-13 15:28:46 -04:00
|
|
|
nodeOpts = append(nodeOpts, node.WithWakuFilterFullNode(filter.WithTimeout(options.Filter.Timeout)))
|
|
|
|
}
|
|
|
|
|
|
|
|
if options.Filter.UseV1 {
|
|
|
|
nodeOpts = append(nodeOpts, node.WithLegacyWakuFilter(!options.Filter.DisableFullNode, legacy_filter.WithTimeout(options.Filter.Timeout)))
|
2021-10-03 17:45:07 -04:00
|
|
|
}
|
2021-06-28 10:14:28 -04:00
|
|
|
|
2023-03-09 11:48:25 -04:00
|
|
|
var dbStore *persistence.DBStore
|
|
|
|
if requiresDB(options) {
|
|
|
|
dbStore, err = persistence.NewDBStore(logger,
|
2023-01-04 13:58:14 -04:00
|
|
|
persistence.WithDB(db),
|
2023-03-09 11:48:25 -04:00
|
|
|
persistence.WithMigrations(migrationFn), // TODO: refactor migrations out of DBStore, or merge DBStore with rendezvous DB
|
2023-01-04 13:58:14 -04:00
|
|
|
persistence.WithRetentionPolicy(options.Store.RetentionMaxMessages, options.Store.RetentionTime),
|
|
|
|
)
|
2022-09-12 15:58:15 -04:00
|
|
|
failOnErr(err, "DBStore")
|
|
|
|
nodeOpts = append(nodeOpts, node.WithMessageProvider(dbStore))
|
2021-10-03 17:45:07 -04:00
|
|
|
}
|
2021-09-30 12:01:53 -04:00
|
|
|
|
2023-03-09 11:48:25 -04:00
|
|
|
if options.Store.Enable {
|
|
|
|
nodeOpts = append(nodeOpts, node.WithWakuStore(options.Store.ResumeNodes...))
|
|
|
|
nodeOpts = append(nodeOpts, node.WithMessageProvider(dbStore))
|
|
|
|
}
|
|
|
|
|
2021-10-03 17:45:07 -04:00
|
|
|
if options.LightPush.Enable {
|
|
|
|
nodeOpts = append(nodeOpts, node.WithLightPush())
|
|
|
|
}
|
2021-04-12 13:59:41 -04:00
|
|
|
|
2022-05-05 15:17:35 -04:00
|
|
|
var discoveredNodes []dnsdisc.DiscoveredNode
|
|
|
|
if options.DNSDiscovery.Enable {
|
2022-09-14 15:19:11 -04:00
|
|
|
if len(options.DNSDiscovery.URLs.Value()) != 0 {
|
|
|
|
for _, url := range options.DNSDiscovery.URLs.Value() {
|
|
|
|
logger.Info("attempting DNS discovery with ", zap.String("URL", url))
|
|
|
|
nodes, err := dnsdisc.RetrieveNodes(ctx, url, dnsdisc.WithNameserver(options.DNSDiscovery.Nameserver))
|
|
|
|
if err != nil {
|
|
|
|
logger.Warn("dns discovery error ", zap.Error(err))
|
|
|
|
} else {
|
2023-05-10 10:13:10 -04:00
|
|
|
var discPeerInfo []peer.AddrInfo
|
2022-09-14 15:19:11 -04:00
|
|
|
for _, n := range nodes {
|
2023-05-10 10:13:10 -04:00
|
|
|
discPeerInfo = append(discPeerInfo, n.PeerInfo)
|
2022-09-14 15:19:11 -04:00
|
|
|
}
|
2023-05-10 10:13:10 -04:00
|
|
|
logger.Info("found dns entries ", zap.Any("nodes", discPeerInfo))
|
2023-05-04 10:57:35 +05:30
|
|
|
discoveredNodes = append(discoveredNodes, nodes...)
|
2022-08-25 20:18:03 -04:00
|
|
|
}
|
2022-05-05 15:17:35 -04:00
|
|
|
}
|
|
|
|
} else {
|
2022-05-27 09:25:06 -04:00
|
|
|
logger.Fatal("DNS discovery URL is required")
|
2022-05-05 15:17:35 -04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-11-17 12:19:42 -04:00
|
|
|
if options.DiscV5.Enable {
|
|
|
|
var bootnodes []*enode.Node
|
2022-02-21 11:23:52 -04:00
|
|
|
for _, addr := range options.DiscV5.Nodes.Value() {
|
2021-11-17 12:19:42 -04:00
|
|
|
bootnode, err := enode.Parse(enode.ValidSchemes, addr)
|
|
|
|
if err != nil {
|
2022-05-27 09:25:06 -04:00
|
|
|
logger.Fatal("parsing ENR", zap.Error(err))
|
2021-11-17 12:19:42 -04:00
|
|
|
}
|
|
|
|
bootnodes = append(bootnodes, bootnode)
|
|
|
|
}
|
2022-05-05 15:17:35 -04:00
|
|
|
|
|
|
|
for _, n := range discoveredNodes {
|
|
|
|
if n.ENR != nil {
|
|
|
|
bootnodes = append(bootnodes, n.ENR)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-01-13 19:58:22 -04:00
|
|
|
nodeOpts = append(nodeOpts, node.WithDiscoveryV5(options.DiscV5.Port, bootnodes, options.DiscV5.AutoUpdate))
|
2021-11-17 12:19:42 -04:00
|
|
|
}
|
|
|
|
|
2022-10-23 09:13:43 -04:00
|
|
|
if options.PeerExchange.Enable {
|
|
|
|
nodeOpts = append(nodeOpts, node.WithPeerExchange())
|
|
|
|
}
|
|
|
|
|
2023-03-09 11:48:25 -04:00
|
|
|
if options.Rendezvous.Enable {
|
2023-03-09 18:42:50 -04:00
|
|
|
nodeOpts = append(nodeOpts, node.WithRendezvous(options.Rendezvous.Nodes))
|
|
|
|
}
|
|
|
|
|
|
|
|
if options.Rendezvous.Server {
|
2023-03-09 11:48:25 -04:00
|
|
|
rdb := rendezvous.NewDB(ctx, db, logger)
|
|
|
|
nodeOpts = append(nodeOpts, node.WithRendezvousServer(rdb))
|
|
|
|
}
|
|
|
|
|
2022-09-11 17:08:58 -04:00
|
|
|
checkForRLN(logger, options, &nodeOpts)
|
2022-07-05 17:28:34 -04:00
|
|
|
|
2023-01-06 18:37:57 -04:00
|
|
|
wakuNode, err := node.New(nodeOpts...)
|
2021-04-15 17:23:07 -04:00
|
|
|
|
2023-02-01 19:35:31 -04:00
|
|
|
utils.Logger().Info("Version details ", zap.String("version", node.Version), zap.String("commit", node.GitCommit))
|
|
|
|
|
2021-10-03 17:45:07 -04:00
|
|
|
failOnErr(err, "Wakunode")
|
2021-09-30 12:01:53 -04:00
|
|
|
|
2023-04-13 15:28:46 -04:00
|
|
|
if options.Filter.UseV1 {
|
2023-06-05 10:39:38 -04:00
|
|
|
addStaticPeers(wakuNode, options.Filter.NodesV1, legacy_filter.FilterID_v20beta1)
|
2023-02-07 18:28:46 -04:00
|
|
|
}
|
2021-10-04 22:13:54 -04:00
|
|
|
|
2023-01-06 18:37:57 -04:00
|
|
|
if err = wakuNode.Start(ctx); err != nil {
|
2022-05-27 09:25:06 -04:00
|
|
|
logger.Fatal("starting waku node", zap.Error(err))
|
2021-10-06 11:42:57 -04:00
|
|
|
}
|
2021-10-04 22:13:54 -04:00
|
|
|
|
2023-06-05 10:39:38 -04:00
|
|
|
for _, d := range discoveredNodes {
|
|
|
|
wakuNode.Host().Peerstore().AddAddrs(d.PeerID, d.PeerInfo.Addrs, peerstore.PermanentAddrTTL)
|
|
|
|
}
|
|
|
|
|
|
|
|
addStaticPeers(wakuNode, options.Store.Nodes, store.StoreID_v20beta4)
|
|
|
|
addStaticPeers(wakuNode, options.LightPush.Nodes, lightpush.LightPushID_v20beta1)
|
|
|
|
addStaticPeers(wakuNode, options.Rendezvous.Nodes, rendezvous.RendezvousID)
|
|
|
|
addStaticPeers(wakuNode, options.Filter.Nodes, filter.FilterSubscribeID_v20beta1)
|
2023-06-01 08:26:03 -04:00
|
|
|
|
2021-11-17 12:19:42 -04:00
|
|
|
if options.DiscV5.Enable {
|
2022-12-10 11:38:18 -04:00
|
|
|
if err = wakuNode.DiscV5().Start(ctx); err != nil {
|
2022-05-27 09:25:06 -04:00
|
|
|
logger.Fatal("starting discovery v5", zap.Error(err))
|
2021-11-17 12:19:42 -04:00
|
|
|
}
|
2022-12-03 09:49:20 -04:00
|
|
|
}
|
2022-10-23 09:13:43 -04:00
|
|
|
|
2022-12-03 09:49:20 -04:00
|
|
|
// retrieve and connect to peer exchange peers
|
|
|
|
if options.PeerExchange.Enable && options.PeerExchange.Node != nil {
|
|
|
|
logger.Info("retrieving peer info via peer exchange protocol")
|
2022-10-23 09:13:43 -04:00
|
|
|
|
2023-06-05 10:39:38 -04:00
|
|
|
peerId, err := wakuNode.AddPeer(*options.PeerExchange.Node, peers.Static, peer_exchange.PeerExchangeID_v20alpha1)
|
2022-12-03 09:49:20 -04:00
|
|
|
if err != nil {
|
|
|
|
logger.Error("adding peer exchange peer", logging.MultiAddrs("node", *options.PeerExchange.Node), zap.Error(err))
|
|
|
|
} else {
|
2023-06-05 10:39:38 -04:00
|
|
|
desiredOutDegree := wakuNode.Relay().Params().D
|
2022-11-25 16:54:11 -04:00
|
|
|
if err = wakuNode.PeerExchange().Request(ctx, desiredOutDegree, peer_exchange.WithPeer(peerId)); err != nil {
|
2022-12-03 09:49:20 -04:00
|
|
|
logger.Error("requesting peers via peer exchange", zap.Error(err))
|
2022-10-23 09:13:43 -04:00
|
|
|
}
|
|
|
|
}
|
2021-11-17 12:19:42 -04:00
|
|
|
}
|
|
|
|
|
2022-02-21 11:23:52 -04:00
|
|
|
if len(options.Relay.Topics.Value()) == 0 {
|
|
|
|
options.Relay.Topics = *cli.NewStringSlice(relay.DefaultWakuTopic)
|
2021-10-03 17:45:07 -04:00
|
|
|
}
|
2021-09-30 12:01:53 -04:00
|
|
|
|
2022-02-21 11:23:52 -04:00
|
|
|
if options.Relay.Enable {
|
|
|
|
for _, nodeTopic := range options.Relay.Topics.Value() {
|
2022-04-25 23:31:26 +04:00
|
|
|
nodeTopic := nodeTopic
|
2022-02-18 13:49:11 -04:00
|
|
|
sub, err := wakuNode.Relay().SubscribeToTopic(ctx, nodeTopic)
|
2021-10-30 10:29:34 -04:00
|
|
|
failOnErr(err, "Error subscring to topic")
|
2023-05-05 15:19:15 +05:30
|
|
|
sub.Unsubscribe()
|
2021-10-30 10:29:34 -04:00
|
|
|
}
|
2023-05-02 11:10:45 -04:00
|
|
|
|
|
|
|
for _, protectedTopic := range options.Relay.ProtectedTopics {
|
2023-05-22 17:03:40 -04:00
|
|
|
err := wakuNode.Relay().AddSignedTopicValidator(protectedTopic.Topic, protectedTopic.PublicKey)
|
2023-05-02 11:10:45 -04:00
|
|
|
failOnErr(err, "Error adding signed topic validator")
|
|
|
|
}
|
2021-10-03 17:45:07 -04:00
|
|
|
}
|
2021-03-18 19:21:45 -04:00
|
|
|
|
2022-08-15 13:13:45 -04:00
|
|
|
for _, n := range options.StaticNodes {
|
2022-08-25 20:18:03 -04:00
|
|
|
go func(ctx context.Context, node multiaddr.Multiaddr) {
|
|
|
|
ctx, cancel := context.WithTimeout(ctx, dialTimeout)
|
|
|
|
defer cancel()
|
2022-08-15 13:13:45 -04:00
|
|
|
err = wakuNode.DialPeerWithMultiAddress(ctx, node)
|
2021-10-03 17:45:07 -04:00
|
|
|
if err != nil {
|
2022-05-27 09:25:06 -04:00
|
|
|
logger.Error("dialing peer", zap.Error(err))
|
2021-03-18 19:21:45 -04:00
|
|
|
}
|
2022-08-25 20:18:03 -04:00
|
|
|
}(ctx, n)
|
2021-10-03 17:45:07 -04:00
|
|
|
}
|
2021-03-18 19:21:45 -04:00
|
|
|
|
2022-05-05 15:17:35 -04:00
|
|
|
if len(discoveredNodes) != 0 {
|
|
|
|
for _, n := range discoveredNodes {
|
2023-05-10 10:13:10 -04:00
|
|
|
go func(ctx context.Context, info peer.AddrInfo) {
|
|
|
|
ctx, cancel := context.WithTimeout(ctx, dialTimeout)
|
|
|
|
defer cancel()
|
|
|
|
err = wakuNode.DialPeerWithInfo(ctx, info)
|
|
|
|
if err != nil {
|
|
|
|
logger.Error("dialing peer", logging.HostID("peer", info.ID), zap.Error(err))
|
|
|
|
}
|
|
|
|
}(ctx, n.PeerInfo)
|
|
|
|
|
2021-09-30 19:03:19 -04:00
|
|
|
}
|
2021-10-03 17:45:07 -04:00
|
|
|
}
|
2021-09-30 19:03:19 -04:00
|
|
|
|
2021-11-02 10:54:34 +01:00
|
|
|
var rpcServer *rpc.WakuRpc
|
|
|
|
if options.RPCServer.Enable {
|
2022-12-10 20:08:25 -04:00
|
|
|
rpcServer = rpc.NewWakuRpc(wakuNode, options.RPCServer.Address, options.RPCServer.Port, options.RPCServer.Admin, options.RPCServer.Private, options.PProf, options.RPCServer.RelayCacheCapacity, logger)
|
2021-11-19 15:17:38 +01:00
|
|
|
rpcServer.Start()
|
2021-11-02 10:54:34 +01:00
|
|
|
}
|
|
|
|
|
2022-12-07 23:04:56 +05:30
|
|
|
var wg sync.WaitGroup
|
|
|
|
|
2022-07-24 16:51:42 -04:00
|
|
|
var restServer *rest.WakuRest
|
|
|
|
if options.RESTServer.Enable {
|
2022-12-07 23:04:56 +05:30
|
|
|
wg.Add(1)
|
2022-12-10 20:08:25 -04:00
|
|
|
restServer = rest.NewWakuRest(wakuNode, options.RESTServer.Address, options.RESTServer.Port, options.RESTServer.Admin, options.RESTServer.Private, options.PProf, options.RESTServer.RelayCacheCapacity, logger)
|
2022-12-10 12:21:22 -04:00
|
|
|
restServer.Start(ctx, &wg)
|
2022-07-24 16:51:42 -04:00
|
|
|
}
|
|
|
|
|
2022-12-07 23:04:56 +05:30
|
|
|
wg.Wait()
|
2022-05-30 14:48:22 -04:00
|
|
|
logger.Info("Node setup complete")
|
2022-05-26 08:23:10 -04:00
|
|
|
|
2021-10-03 17:45:07 -04:00
|
|
|
// Wait for a SIGINT or SIGTERM signal
|
|
|
|
ch := make(chan os.Signal, 1)
|
|
|
|
signal.Notify(ch, syscall.SIGINT, syscall.SIGTERM)
|
|
|
|
<-ch
|
2022-05-27 09:25:06 -04:00
|
|
|
logger.Info("Received signal, shutting down...")
|
2021-04-18 20:03:16 -04:00
|
|
|
|
2021-10-03 17:45:07 -04:00
|
|
|
// shut the node down
|
|
|
|
wakuNode.Stop()
|
2021-06-28 09:20:23 -04:00
|
|
|
|
2021-11-02 10:54:34 +01:00
|
|
|
if options.RPCServer.Enable {
|
|
|
|
err := rpcServer.Stop(ctx)
|
|
|
|
failOnErr(err, "RPCClose")
|
|
|
|
}
|
|
|
|
|
2022-07-24 16:51:42 -04:00
|
|
|
if options.RESTServer.Enable {
|
|
|
|
err := restServer.Stop(ctx)
|
|
|
|
failOnErr(err, "RESTClose")
|
|
|
|
}
|
|
|
|
|
2021-10-03 17:45:07 -04:00
|
|
|
if options.Metrics.Enable {
|
2021-10-18 14:43:17 +02:00
|
|
|
err = metricsServer.Stop(ctx)
|
|
|
|
failOnErr(err, "MetricsClose")
|
2021-10-03 17:45:07 -04:00
|
|
|
}
|
2021-03-18 19:21:45 -04:00
|
|
|
|
2022-11-25 16:54:11 -04:00
|
|
|
if options.Store.Enable {
|
2021-10-03 17:45:07 -04:00
|
|
|
err = db.Close()
|
|
|
|
failOnErr(err, "DBClose")
|
|
|
|
}
|
2021-03-18 19:21:45 -04:00
|
|
|
}
|
|
|
|
|
2023-06-05 10:39:38 -04:00
|
|
|
func addStaticPeers(wakuNode *node.WakuNode, addresses []multiaddr.Multiaddr, protocols ...protocol.ID) {
|
2022-08-15 13:13:45 -04:00
|
|
|
for _, addr := range addresses {
|
2023-06-05 10:39:38 -04:00
|
|
|
_, err := wakuNode.AddPeer(addr, peers.Static, protocols...)
|
2021-10-03 17:45:07 -04:00
|
|
|
failOnErr(err, "error adding peer")
|
2021-09-30 12:01:53 -04:00
|
|
|
}
|
|
|
|
}
|
2021-10-04 18:38:27 -04:00
|
|
|
|
2022-07-25 08:24:42 -04:00
|
|
|
func loadPrivateKeyFromFile(path string, passwd string) (*ecdsa.PrivateKey, error) {
|
2023-02-16 12:17:52 -04:00
|
|
|
src, err := os.ReadFile(path)
|
2021-10-04 18:38:27 -04:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
2022-07-25 08:24:42 -04:00
|
|
|
var encryptedK keystore.CryptoJSON
|
|
|
|
err = json.Unmarshal(src, &encryptedK)
|
2021-10-04 18:38:27 -04:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
2022-07-25 08:24:42 -04:00
|
|
|
pKey, err := keystore.DecryptDataV3(encryptedK, passwd)
|
2022-05-27 15:55:35 -04:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
2021-10-04 18:38:27 -04:00
|
|
|
|
2022-07-25 08:24:42 -04:00
|
|
|
return crypto.ToECDSA(pKey)
|
2021-10-04 18:38:27 -04:00
|
|
|
}
|
|
|
|
|
2022-08-08 20:02:08 -04:00
|
|
|
func checkForFileExistence(path string, overwrite bool) error {
|
2021-10-04 18:38:27 -04:00
|
|
|
_, err := os.Stat(path)
|
|
|
|
|
2021-12-08 14:21:30 +00:00
|
|
|
if err == nil && !overwrite {
|
2021-10-04 18:38:27 -04:00
|
|
|
return fmt.Errorf("%s already exists. Use --overwrite to overwrite the file", path)
|
|
|
|
}
|
|
|
|
|
|
|
|
if err := os.Remove(path); err != nil && !os.IsNotExist(err) {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
2021-12-08 14:21:30 +00:00
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func generatePrivateKey() ([]byte, error) {
|
2021-10-04 18:38:27 -04:00
|
|
|
key, err := crypto.GenerateKey()
|
|
|
|
if err != nil {
|
2021-12-08 14:21:30 +00:00
|
|
|
return nil, err
|
2021-10-04 18:38:27 -04:00
|
|
|
}
|
|
|
|
|
2022-07-25 08:24:42 -04:00
|
|
|
return key.D.Bytes(), nil
|
2021-12-08 14:21:30 +00:00
|
|
|
}
|
|
|
|
|
2022-07-25 08:24:42 -04:00
|
|
|
func writePrivateKeyToFile(path string, passwd []byte, overwrite bool) error {
|
2022-08-08 20:02:08 -04:00
|
|
|
if err := checkForFileExistence(path, overwrite); err != nil {
|
2021-12-08 14:21:30 +00:00
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
2022-07-25 08:24:42 -04:00
|
|
|
key, err := generatePrivateKey()
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
encryptedK, err := keystore.EncryptDataV3(key, passwd, keystore.StandardScryptN, keystore.StandardScryptP)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
output, err := json.Marshal(encryptedK)
|
2021-12-08 14:21:30 +00:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
2023-02-16 12:17:52 -04:00
|
|
|
return os.WriteFile(path, output, 0600)
|
2021-10-04 18:38:27 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
func getPrivKey(options Options) (*ecdsa.PrivateKey, error) {
|
|
|
|
var prvKey *ecdsa.PrivateKey
|
2023-04-02 16:23:37 +05:30
|
|
|
// get private key from nodeKey or keyFile
|
2022-08-15 13:13:45 -04:00
|
|
|
if options.NodeKey != nil {
|
|
|
|
prvKey = options.NodeKey
|
2021-10-04 18:38:27 -04:00
|
|
|
} else {
|
2023-04-02 16:23:37 +05:30
|
|
|
if _, err := os.Stat(options.KeyFile); err == nil {
|
|
|
|
if prvKey, err = loadPrivateKeyFromFile(options.KeyFile, options.KeyPasswd); err != nil {
|
|
|
|
return nil, fmt.Errorf("could not read keyfile: %w", err)
|
2021-10-04 18:38:27 -04:00
|
|
|
}
|
|
|
|
} else {
|
2023-04-02 16:23:37 +05:30
|
|
|
if os.IsNotExist(err) {
|
|
|
|
if prvKey, err = crypto.GenerateKey(); err != nil {
|
|
|
|
return nil, fmt.Errorf("error generating key: %w", err)
|
2021-10-04 18:38:27 -04:00
|
|
|
}
|
|
|
|
} else {
|
2023-04-02 16:23:37 +05:30
|
|
|
return nil, fmt.Errorf("could not read keyfile: %w", err)
|
2021-10-04 18:38:27 -04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return prvKey, nil
|
|
|
|
}
|
2021-10-14 14:17:01 -04:00
|
|
|
|
2021-10-14 22:15:02 -04:00
|
|
|
func printListeningAddresses(ctx context.Context, nodeOpts []node.WakuNodeOption, options Options) {
|
2021-10-14 14:17:01 -04:00
|
|
|
params := new(node.WakuNodeParameters)
|
|
|
|
for _, opt := range nodeOpts {
|
|
|
|
err := opt(params)
|
|
|
|
if err != nil {
|
|
|
|
panic(err)
|
|
|
|
}
|
|
|
|
}
|
2021-10-14 22:15:02 -04:00
|
|
|
|
|
|
|
var libp2pOpts []config.Option
|
2021-11-17 12:19:42 -04:00
|
|
|
libp2pOpts = append(libp2pOpts,
|
|
|
|
params.Identity(),
|
|
|
|
libp2p.ListenAddrs(params.MultiAddresses()...),
|
|
|
|
)
|
2021-10-14 22:15:02 -04:00
|
|
|
|
2022-08-25 16:36:04 -04:00
|
|
|
if options.Websocket.Secure {
|
|
|
|
transports := libp2p.ChainOptions(
|
|
|
|
libp2p.Transport(tcp.NewTCPTransport),
|
|
|
|
libp2p.Transport(ws.New, ws.WithTLSConfig(params.TLSConfig())),
|
|
|
|
)
|
|
|
|
libp2pOpts = append(libp2pOpts, transports)
|
|
|
|
}
|
|
|
|
|
2021-11-17 12:19:42 -04:00
|
|
|
addrFactory := params.AddressFactory()
|
|
|
|
if addrFactory != nil {
|
|
|
|
libp2pOpts = append(libp2pOpts, libp2p.AddrsFactory(addrFactory))
|
2021-10-14 22:15:02 -04:00
|
|
|
}
|
|
|
|
|
2022-03-22 09:12:58 -04:00
|
|
|
h, err := libp2p.New(libp2pOpts...)
|
2021-10-14 14:17:01 -04:00
|
|
|
if err != nil {
|
|
|
|
panic(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
hostInfo, _ := multiaddr.NewMultiaddr(fmt.Sprintf("/p2p/%s", h.ID().Pretty()))
|
2021-10-14 22:15:02 -04:00
|
|
|
|
2021-10-14 14:17:01 -04:00
|
|
|
for _, addr := range h.Addrs() {
|
|
|
|
fmt.Println(addr.Encapsulate(hostInfo))
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|