package args import ( "fmt" "math/big" "net" "path/filepath" "strconv" "github.com/ledgerwatch/erigon-lib/chain/networkname" "github.com/ledgerwatch/erigon/cmd/devnet/accounts" "github.com/ledgerwatch/erigon/cmd/devnet/requests" ) type Node struct { requests.RequestGenerator `arg:"-"` Name string `arg:"-"` BuildDir string `arg:"positional" default:"./build/bin/devnet" json:"builddir"` DataDir string `arg:"--datadir" default:"./dev" json:"datadir"` Chain string `arg:"--chain" default:"dev" json:"chain"` Port int `arg:"--port" json:"port,omitempty"` AllowedPorts string `arg:"--p2p.allowed-ports" json:"p2p.allowed-ports,omitempty"` NAT string `arg:"--nat" default:"none" json:"nat"` ConsoleVerbosity string `arg:"--log.console.verbosity" default:"0" json:"log.console.verbosity"` DirVerbosity string `arg:"--log.dir.verbosity" json:"log.dir.verbosity,omitempty"` LogDirPath string `arg:"--log.dir.path" json:"log.dir.path,omitempty"` LogDirPrefix string `arg:"--log.dir.prefix" json:"log.dir.prefix,omitempty"` P2PProtocol string `arg:"--p2p.protocol" default:"68" json:"p2p.protocol"` Snapshots bool `arg:"--snapshots" flag:"" default:"false" json:"snapshots,omitempty"` Downloader string `arg:"--no-downloader" default:"true" json:"no-downloader"` WS string `arg:"--ws" flag:"" default:"true" json:"ws"` PrivateApiAddr string `arg:"--private.api.addr" default:"localhost:9090" json:"private.api.addr"` HttpPort int `arg:"--http.port" default:"8545" json:"http.port"` HttpVHosts string `arg:"--http.vhosts" json:"http.vhosts"` HttpCorsDomain string `arg:"--http.corsdomain" json:"http.corsdomain"` AuthRpcPort int `arg:"--authrpc.port" default:"8551" json:"authrpc.port"` AuthRpcVHosts string `arg:"--authrpc.vhosts" json:"authrpc.vhosts"` WSPort int `arg:"-" default:"8546" json:"-"` // flag not defined GRPCPort int `arg:"-" default:"8547" json:"-"` // flag not defined TCPPort int `arg:"-" default:"8548" json:"-"` // flag not defined Metrics bool `arg:"--metrics" flag:"" default:"false" json:"metrics"` MetricsPort int `arg:"--metrics.port" json:"metrics.port,omitempty"` MetricsAddr string `arg:"--metrics.addr" json:"metrics.addr,omitempty"` StaticPeers string `arg:"--staticpeers" json:"staticpeers,omitempty"` WithoutHeimdall bool `arg:"--bor.withoutheimdall" flag:"" default:"false" json:"bor.withoutheimdall,omitempty"` HeimdallGRpc string `arg:"--bor.heimdallgRPC" json:"bor.heimdallgRPC,omitempty"` VMDebug bool `arg:"--vmdebug" flag:"" default:"false" json:"dmdebug"` } func (node *Node) configure(base Node, nodeNumber int) error { if len(node.Name) == 0 { node.Name = fmt.Sprintf("%s-%d", base.Chain, nodeNumber) } node.DataDir = filepath.Join(base.DataDir, node.Name) node.LogDirPath = filepath.Join(base.DataDir, "logs") node.LogDirPrefix = node.Name node.Chain = base.Chain node.StaticPeers = base.StaticPeers node.Metrics = base.Metrics node.MetricsPort = base.MetricsPort node.MetricsAddr = base.MetricsAddr node.Snapshots = base.Snapshots var err error node.PrivateApiAddr, _, err = portFromBase(base.PrivateApiAddr, nodeNumber, 1) if err != nil { return err } apiPort := base.HttpPort + (nodeNumber * 5) node.HttpPort = apiPort node.WSPort = apiPort + 1 node.GRPCPort = apiPort + 2 node.TCPPort = apiPort + 3 node.AuthRpcPort = apiPort + 4 node.Port = base.Port + nodeNumber return nil } func (node Node) ChainID() *big.Int { return &big.Int{} } type BlockProducer struct { Node Mine bool `arg:"--mine" flag:"true"` Etherbase string `arg:"--miner.etherbase"` DevPeriod int `arg:"--dev.period"` BorPeriod int `arg:"--bor.period"` BorMinBlockSize int `arg:"--bor.minblocksize"` HttpApi string `arg:"--http.api" default:"admin,eth,erigon,web3,net,debug,trace,txpool,parity,ots"` AccountSlots int `arg:"--txpool.accountslots" default:"16"` account *accounts.Account } func (m BlockProducer) Configure(baseNode Node, nodeNumber int) (int, interface{}, error) { err := m.configure(baseNode, nodeNumber) if err != nil { return -1, nil, err } switch m.Chain { case networkname.DevChainName: if m.DevPeriod == 0 { m.DevPeriod = 30 } m.account = accounts.NewAccount(m.Name() + "-etherbase") case networkname.BorDevnetChainName: m.account = accounts.NewAccount(m.Name() + "-etherbase") if len(m.HttpApi) == 0 { m.HttpApi = "admin,eth,erigon,web3,net,debug,trace,txpool,parity,ots,bor" } } if m.account != nil { m.Etherbase = m.account.Address.Hex() } return m.HttpPort, m, nil } func (n BlockProducer) Name() string { return n.Node.Name } func (n BlockProducer) Account() *accounts.Account { return n.account } func (n BlockProducer) IsBlockProducer() bool { return true } type NonBlockProducer struct { Node HttpApi string `arg:"--http.api" default:"admin,eth,debug,net,trace,web3,erigon,txpool" json:"http.api"` TorrentPort string `arg:"--torrent.port" default:"42070" json:"torrent.port"` NoDiscover string `arg:"--nodiscover" flag:"" default:"true" json:"nodiscover"` } func (n NonBlockProducer) Configure(baseNode Node, nodeNumber int) (int, interface{}, error) { err := n.configure(baseNode, nodeNumber) if err != nil { return -1, nil, err } return n.HttpPort, n, nil } func (n NonBlockProducer) Name() string { return n.Node.Name } func (n NonBlockProducer) IsBlockProducer() bool { return false } func (n NonBlockProducer) Account() *accounts.Account { return nil } func portFromBase(baseAddr string, increment int, portCount int) (string, int, error) { apiHost, apiPort, err := net.SplitHostPort(baseAddr) if err != nil { return "", -1, err } portNo, err := strconv.Atoi(apiPort) if err != nil { return "", -1, err } portNo += (increment * portCount) return fmt.Sprintf("%s:%d", apiHost, portNo), portNo, nil }