erigon-pulse/cmd/devnet/node/node.go
Leonard Chinonso 599fc24aed
Devnet Test for EIP1559 (#6938)
**Current Problem**

When X transactions are created, with M transactions out of them having
`gasFeeCap` greater than the current `baseFeePerGas` for the block, and
N transactions having `gasFeeCap` lower, all X transactions get added to
the yellow pool (baseFee) in the txpool and they never get mined.
However when X transactions all having `gasFeeCap` higher than the
`baseFeePerGas` are created, they all get added to the green pool
(pending) in the txpool and thus get mined.

This phenomenon can be inspected in the
`signEIP1559TxsLowerAndHigherThanBaseFee2` function where the number of
transactions that should have `gasFeeCap` higher than and lower than the
current `baseFeePerGas` is specified in its parameter - `func
signEIP1559TxsLowerAndHigherThanBaseFee2(amountLower, amountHigher
int...`.
When `amountLower` is set as `0`, all transactions created from
`amountHigher` will be mined because they have `gasFeeCap` >
`baseFeePerGas`. When `amountLower` has a value > `0`, all the
transactions created (including transactions created with
`amountHigher`) will go to the yellow pool (baseFee) and thus do not get
mined.
2023-03-02 10:25:11 +00:00

154 lines
5.0 KiB
Go

package node
import (
"fmt"
"os"
"sync"
"time"
"github.com/urfave/cli/v2"
"github.com/ledgerwatch/erigon-lib/common/dbg"
"github.com/ledgerwatch/erigon/cmd/devnet/devnetutils"
"github.com/ledgerwatch/erigon/cmd/devnet/models"
"github.com/ledgerwatch/erigon/cmd/devnet/requests"
"github.com/ledgerwatch/erigon/params"
erigonapp "github.com/ledgerwatch/erigon/turbo/app"
erigoncli "github.com/ledgerwatch/erigon/turbo/cli"
"github.com/ledgerwatch/erigon/turbo/node"
"github.com/ledgerwatch/log/v3"
)
// Holds the number id of each node on the network, the first node is node 0
var nodeNumber int
// Start starts the process for two erigon nodes running on the dev chain
func Start(wg *sync.WaitGroup) {
// add one goroutine to the wait-list
wg.Add(1)
// start the first node
go StartNode(wg, miningNodeArgs())
// sleep for a while to allow first node to start
time.Sleep(time.Second * 10)
// get the enode of the first node
enode, err := getEnode()
if err != nil {
// TODO: Log the error, it means node did not start well
fmt.Printf("error starting the node: %s\n", err)
}
// add one goroutine to the wait-list
wg.Add(1)
// start the second node, connect it to the mining node with the enode
go StartNode(wg, nonMiningNodeArgs(2, enode))
}
// StartNode starts an erigon node on the dev chain
func StartNode(wg *sync.WaitGroup, args []string) {
fmt.Printf("\nRunning node %d with flags ==> %v\n", nodeNumber, args)
// catch any errors and avoid panics if an error occurs
defer func() {
panicResult := recover()
if panicResult == nil {
wg.Done()
return
}
log.Error("catch panic", "err", panicResult, "stack", dbg.Stack())
wg.Done()
os.Exit(1)
}()
app := erigonapp.MakeApp(runNode, erigoncli.DefaultFlags)
nodeNumber++ // increment the number of nodes on the network
if err := app.Run(args); err != nil {
_, printErr := fmt.Fprintln(os.Stderr, err)
if printErr != nil {
log.Warn("Error writing app run error to stderr", "err", printErr)
}
wg.Done()
os.Exit(1)
}
}
// runNode configures, creates and serves an erigon node
func runNode(ctx *cli.Context) error {
logger := log.New()
// Initializing the node and providing the current git commit there
logger.Info("Build info", "git_branch", params.GitBranch, "git_tag", params.GitTag, "git_commit", params.GitCommit)
nodeCfg := node.NewNodConfigUrfave(ctx)
ethCfg := node.NewEthConfigUrfave(ctx, nodeCfg)
ethNode, err := node.New(nodeCfg, ethCfg, logger)
if err != nil {
log.Error("Devnet startup", "err", err)
return err
}
err = ethNode.Serve()
if err != nil {
log.Error("error while serving Devnet node", "err", err)
}
return err
}
// miningNodeArgs returns custom args for starting a mining node
func miningNodeArgs() []string {
dataDir, _ := models.ParameterFromArgument(models.DataDirArg, models.DataDirParam+fmt.Sprintf("%d", nodeNumber))
chainType, _ := models.ParameterFromArgument(models.ChainArg, models.ChainParam)
devPeriod, _ := models.ParameterFromArgument(models.DevPeriodArg, models.DevPeriodParam)
privateApiAddr, _ := models.ParameterFromArgument(models.PrivateApiAddrArg, models.PrivateApiParamMine)
httpApi, _ := models.ParameterFromArgument(models.HttpApiArg, models.HttpApiParam)
ws := models.WSArg
consoleVerbosity, _ := models.ParameterFromArgument(models.ConsoleVerbosityArg, models.ConsoleVerbosityParam)
logDir, _ := models.ParameterFromArgument(models.LogDirArg, models.LogDirParam+"/node_1")
return []string{models.BuildDirArg, dataDir, chainType, privateApiAddr, models.Mine, httpApi, ws, devPeriod, consoleVerbosity, logDir}
}
// nonMiningNodeArgs returns custom args for starting a non-mining node
func nonMiningNodeArgs(nodeNumber int, enode string) []string {
dataDir, _ := models.ParameterFromArgument(models.DataDirArg, models.DataDirParam+fmt.Sprintf("%d", nodeNumber))
chainType, _ := models.ParameterFromArgument(models.ChainArg, models.ChainParam)
privateApiAddr, _ := models.ParameterFromArgument(models.PrivateApiAddrArg, models.PrivateApiParamNoMine)
staticPeers, _ := models.ParameterFromArgument(models.StaticPeersArg, enode)
consoleVerbosity, _ := models.ParameterFromArgument(models.ConsoleVerbosityArg, models.ConsoleVerbosityParam)
logDir, _ := models.ParameterFromArgument(models.LogDirArg, models.LogDirParam+"/node_2")
torrentPort, _ := models.ParameterFromArgument(models.TorrentPortArg, models.TorrentPortParam)
return []string{models.BuildDirArg, dataDir, chainType, privateApiAddr, staticPeers, models.NoDiscover, consoleVerbosity, logDir, torrentPort}
}
// getEnode returns the enode of the mining node
func getEnode() (string, error) {
nodeInfo, err := requests.AdminNodeInfo(0)
if err != nil {
return "", err
}
enode, err := devnetutils.UniqueIDFromEnode(nodeInfo.Enode)
if err != nil {
return "", err
}
return enode, nil
}
// QuitOnSignal stops the node goroutines after all checks have been made on the devnet
func QuitOnSignal(wg *sync.WaitGroup) {
models.QuitNodeChan = make(chan bool)
go func() {
for <-models.QuitNodeChan {
wg.Done()
wg.Done()
}
}()
}