mirror of
https://gitlab.com/pulsechaincom/prysm-pulse.git
synced 2025-01-05 09:14:28 +00:00
364 lines
12 KiB
Go
364 lines
12 KiB
Go
// Package components defines utilities to spin up actual
|
|
// beacon node and validator processes as needed by end to end tests.
|
|
package components
|
|
|
|
import (
|
|
"context"
|
|
"fmt"
|
|
"os"
|
|
"os/exec"
|
|
"path"
|
|
"strconv"
|
|
"strings"
|
|
"syscall"
|
|
|
|
"github.com/bazelbuild/rules_go/go/tools/bazel"
|
|
"github.com/pkg/errors"
|
|
"github.com/prysmaticlabs/prysm/v5/beacon-chain/state"
|
|
cmdshared "github.com/prysmaticlabs/prysm/v5/cmd"
|
|
"github.com/prysmaticlabs/prysm/v5/cmd/beacon-chain/flags"
|
|
"github.com/prysmaticlabs/prysm/v5/cmd/beacon-chain/sync/genesis"
|
|
"github.com/prysmaticlabs/prysm/v5/config/features"
|
|
"github.com/prysmaticlabs/prysm/v5/config/params"
|
|
"github.com/prysmaticlabs/prysm/v5/io/file"
|
|
"github.com/prysmaticlabs/prysm/v5/runtime/interop"
|
|
"github.com/prysmaticlabs/prysm/v5/testing/endtoend/helpers"
|
|
e2e "github.com/prysmaticlabs/prysm/v5/testing/endtoend/params"
|
|
e2etypes "github.com/prysmaticlabs/prysm/v5/testing/endtoend/types"
|
|
)
|
|
|
|
var _ e2etypes.ComponentRunner = (*BeaconNode)(nil)
|
|
var _ e2etypes.ComponentRunner = (*BeaconNodeSet)(nil)
|
|
var _ e2etypes.MultipleComponentRunners = (*BeaconNodeSet)(nil)
|
|
var _ e2etypes.BeaconNodeSet = (*BeaconNodeSet)(nil)
|
|
|
|
// BeaconNodeSet represents set of beacon nodes.
|
|
type BeaconNodeSet struct {
|
|
e2etypes.ComponentRunner
|
|
config *e2etypes.E2EConfig
|
|
nodes []e2etypes.ComponentRunner
|
|
enr string
|
|
ids []string
|
|
started chan struct{}
|
|
}
|
|
|
|
// SetENR assigns ENR to the set of beacon nodes.
|
|
func (s *BeaconNodeSet) SetENR(enr string) {
|
|
s.enr = enr
|
|
}
|
|
|
|
// NewBeaconNodes creates and returns a set of beacon nodes.
|
|
func NewBeaconNodes(config *e2etypes.E2EConfig) *BeaconNodeSet {
|
|
return &BeaconNodeSet{
|
|
config: config,
|
|
started: make(chan struct{}, 1),
|
|
}
|
|
}
|
|
|
|
// Start starts all the beacon nodes in set.
|
|
func (s *BeaconNodeSet) Start(ctx context.Context) error {
|
|
if s.enr == "" {
|
|
return errors.New("empty ENR")
|
|
}
|
|
|
|
// Create beacon nodes.
|
|
nodes := make([]e2etypes.ComponentRunner, e2e.TestParams.BeaconNodeCount)
|
|
for i := 0; i < e2e.TestParams.BeaconNodeCount; i++ {
|
|
nodes[i] = NewBeaconNode(s.config, i, s.enr)
|
|
}
|
|
s.nodes = nodes
|
|
|
|
// Wait for all nodes to finish their job (blocking).
|
|
// Once nodes are ready passed in handler function will be called.
|
|
return helpers.WaitOnNodes(ctx, nodes, func() {
|
|
if s.config.UseFixedPeerIDs {
|
|
for i := 0; i < len(nodes); i++ {
|
|
s.ids = append(s.ids, nodes[i].(*BeaconNode).peerID)
|
|
}
|
|
s.config.PeerIDs = s.ids
|
|
}
|
|
// All nodes started, close channel, so that all services waiting on a set, can proceed.
|
|
close(s.started)
|
|
})
|
|
}
|
|
|
|
// Started checks whether beacon node set is started and all nodes are ready to be queried.
|
|
func (s *BeaconNodeSet) Started() <-chan struct{} {
|
|
return s.started
|
|
}
|
|
|
|
// Pause pauses the component and its underlying process.
|
|
func (s *BeaconNodeSet) Pause() error {
|
|
for _, n := range s.nodes {
|
|
if err := n.Pause(); err != nil {
|
|
return err
|
|
}
|
|
}
|
|
return nil
|
|
}
|
|
|
|
// Resume resumes the component and its underlying process.
|
|
func (s *BeaconNodeSet) Resume() error {
|
|
for _, n := range s.nodes {
|
|
if err := n.Resume(); err != nil {
|
|
return err
|
|
}
|
|
}
|
|
return nil
|
|
}
|
|
|
|
// Stop stops the component and its underlying process.
|
|
func (s *BeaconNodeSet) Stop() error {
|
|
for _, n := range s.nodes {
|
|
if err := n.Stop(); err != nil {
|
|
return err
|
|
}
|
|
}
|
|
return nil
|
|
}
|
|
|
|
// PauseAtIndex pauses the component and its underlying process at the desired index.
|
|
func (s *BeaconNodeSet) PauseAtIndex(i int) error {
|
|
if i >= len(s.nodes) {
|
|
return errors.Errorf("provided index exceeds slice size: %d >= %d", i, len(s.nodes))
|
|
}
|
|
return s.nodes[i].Pause()
|
|
}
|
|
|
|
// ResumeAtIndex resumes the component and its underlying process at the desired index.
|
|
func (s *BeaconNodeSet) ResumeAtIndex(i int) error {
|
|
if i >= len(s.nodes) {
|
|
return errors.Errorf("provided index exceeds slice size: %d >= %d", i, len(s.nodes))
|
|
}
|
|
return s.nodes[i].Resume()
|
|
}
|
|
|
|
// StopAtIndex stops the component and its underlying process at the desired index.
|
|
func (s *BeaconNodeSet) StopAtIndex(i int) error {
|
|
if i >= len(s.nodes) {
|
|
return errors.Errorf("provided index exceeds slice size: %d >= %d", i, len(s.nodes))
|
|
}
|
|
return s.nodes[i].Stop()
|
|
}
|
|
|
|
// ComponentAtIndex returns the component at the provided index.
|
|
func (s *BeaconNodeSet) ComponentAtIndex(i int) (e2etypes.ComponentRunner, error) {
|
|
if i >= len(s.nodes) {
|
|
return nil, errors.Errorf("provided index exceeds slice size: %d >= %d", i, len(s.nodes))
|
|
}
|
|
return s.nodes[i], nil
|
|
}
|
|
|
|
// BeaconNode represents beacon node.
|
|
type BeaconNode struct {
|
|
e2etypes.ComponentRunner
|
|
config *e2etypes.E2EConfig
|
|
started chan struct{}
|
|
index int
|
|
enr string
|
|
peerID string
|
|
cmd *exec.Cmd
|
|
}
|
|
|
|
// NewBeaconNode creates and returns a beacon node.
|
|
func NewBeaconNode(config *e2etypes.E2EConfig, index int, enr string) *BeaconNode {
|
|
return &BeaconNode{
|
|
config: config,
|
|
index: index,
|
|
enr: enr,
|
|
started: make(chan struct{}, 1),
|
|
}
|
|
}
|
|
|
|
func (node *BeaconNode) saveGenesis(ctx context.Context) (string, error) {
|
|
// The deposit contract starts with an empty trie, we use the BeaconState to "pre-mine" the validator registry,
|
|
g, err := generateGenesis(ctx)
|
|
if err != nil {
|
|
return "", err
|
|
}
|
|
|
|
root, err := g.HashTreeRoot(ctx)
|
|
if err != nil {
|
|
return "", err
|
|
}
|
|
lbhr, err := g.LatestBlockHeader().HashTreeRoot()
|
|
if err != nil {
|
|
return "", err
|
|
}
|
|
log.WithField("forkVersion", g.Fork().CurrentVersion).
|
|
WithField("latestBlockHeaderRoot", fmt.Sprintf("%#x", lbhr)).
|
|
WithField("stateRoot", fmt.Sprintf("%#x", root)).
|
|
Infof("BeaconState info")
|
|
|
|
genesisBytes, err := g.MarshalSSZ()
|
|
if err != nil {
|
|
return "", err
|
|
}
|
|
genesisDir := path.Join(e2e.TestParams.TestPath, fmt.Sprintf("genesis/%d", node.index))
|
|
if err := file.MkdirAll(genesisDir); err != nil {
|
|
return "", err
|
|
}
|
|
genesisPath := path.Join(genesisDir, "genesis.ssz")
|
|
return genesisPath, file.WriteFile(genesisPath, genesisBytes)
|
|
}
|
|
|
|
func (node *BeaconNode) saveConfig() (string, error) {
|
|
cfg := params.BeaconConfig().Copy()
|
|
cfgBytes := params.ConfigToYaml(cfg)
|
|
cfgDir := path.Join(e2e.TestParams.TestPath, fmt.Sprintf("config/%d", node.index))
|
|
if err := file.MkdirAll(cfgDir); err != nil {
|
|
return "", err
|
|
}
|
|
cfgPath := path.Join(cfgDir, "beacon-config.yaml")
|
|
return cfgPath, file.WriteFile(cfgPath, cfgBytes)
|
|
}
|
|
|
|
// Start starts a fresh beacon node, connecting to all passed in beacon nodes.
|
|
func (node *BeaconNode) Start(ctx context.Context) error {
|
|
binaryPath, found := bazel.FindBinary("cmd/beacon-chain", "beacon-chain")
|
|
if !found {
|
|
log.Info(binaryPath)
|
|
return errors.New("beacon chain binary not found")
|
|
}
|
|
|
|
config, index, enr := node.config, node.index, node.enr
|
|
stdOutFile, err := helpers.DeleteAndCreateFile(e2e.TestParams.LogPath, fmt.Sprintf(e2e.BeaconNodeLogFileName, index))
|
|
if err != nil {
|
|
return err
|
|
}
|
|
expectedNumOfPeers := e2e.TestParams.BeaconNodeCount + e2e.TestParams.LighthouseBeaconNodeCount - 1
|
|
if node.config.TestSync {
|
|
expectedNumOfPeers += 1
|
|
}
|
|
if node.config.TestCheckpointSync {
|
|
expectedNumOfPeers += 1
|
|
}
|
|
jwtPath := path.Join(e2e.TestParams.TestPath, "eth1data/"+strconv.Itoa(node.index)+"/")
|
|
if index == 0 {
|
|
jwtPath = path.Join(e2e.TestParams.TestPath, "eth1data/miner/")
|
|
}
|
|
jwtPath = path.Join(jwtPath, "geth/jwtsecret")
|
|
|
|
genesisPath, err := node.saveGenesis(ctx)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
cfgPath, err := node.saveConfig()
|
|
if err != nil {
|
|
return err
|
|
}
|
|
args := []string{
|
|
fmt.Sprintf("--%s=%s", genesis.StatePath.Name, genesisPath),
|
|
fmt.Sprintf("--%s=%s/eth2-beacon-node-%d", cmdshared.DataDirFlag.Name, e2e.TestParams.TestPath, index),
|
|
fmt.Sprintf("--%s=%s", cmdshared.LogFileName.Name, stdOutFile.Name()),
|
|
fmt.Sprintf("--%s=%s", flags.DepositContractFlag.Name, params.BeaconConfig().DepositContractAddress),
|
|
fmt.Sprintf("--%s=%d", flags.RPCPort.Name, e2e.TestParams.Ports.PrysmBeaconNodeRPCPort+index),
|
|
fmt.Sprintf("--%s=http://127.0.0.1:%d", flags.ExecutionEngineEndpoint.Name, e2e.TestParams.Ports.Eth1ProxyPort+index),
|
|
fmt.Sprintf("--%s=%s", flags.ExecutionJWTSecretFlag.Name, jwtPath),
|
|
fmt.Sprintf("--%s=%d", flags.MinSyncPeers.Name, 1),
|
|
fmt.Sprintf("--%s=%d", cmdshared.P2PUDPPort.Name, e2e.TestParams.Ports.PrysmBeaconNodeUDPPort+index),
|
|
fmt.Sprintf("--%s=%d", cmdshared.P2PTCPPort.Name, e2e.TestParams.Ports.PrysmBeaconNodeTCPPort+index),
|
|
fmt.Sprintf("--%s=%d", cmdshared.P2PMaxPeers.Name, expectedNumOfPeers),
|
|
fmt.Sprintf("--%s=%d", flags.MonitoringPortFlag.Name, e2e.TestParams.Ports.PrysmBeaconNodeMetricsPort+index),
|
|
fmt.Sprintf("--%s=%d", flags.GRPCGatewayPort.Name, e2e.TestParams.Ports.PrysmBeaconNodeGatewayPort+index),
|
|
fmt.Sprintf("--%s=%d", flags.ContractDeploymentBlock.Name, 0),
|
|
fmt.Sprintf("--%s=%d", flags.MinPeersPerSubnet.Name, 0),
|
|
fmt.Sprintf("--%s=%d", cmdshared.RPCMaxPageSizeFlag.Name, params.BeaconConfig().MinGenesisActiveValidatorCount),
|
|
fmt.Sprintf("--%s=%s", cmdshared.BootstrapNode.Name, enr),
|
|
fmt.Sprintf("--%s=%s", cmdshared.VerbosityFlag.Name, "debug"),
|
|
fmt.Sprintf("--%s=%d", flags.BlockBatchLimitBurstFactor.Name, 8),
|
|
fmt.Sprintf("--%s=%d", flags.BlobBatchLimitBurstFactor.Name, 16),
|
|
fmt.Sprintf("--%s=%d", flags.BlobBatchLimit.Name, 256),
|
|
fmt.Sprintf("--%s=%s", cmdshared.ChainConfigFileFlag.Name, cfgPath),
|
|
"--" + cmdshared.ValidatorMonitorIndicesFlag.Name + "=1",
|
|
"--" + cmdshared.ValidatorMonitorIndicesFlag.Name + "=2",
|
|
"--" + cmdshared.ForceClearDB.Name,
|
|
"--" + flags.EnableDebugRPCEndpoints.Name,
|
|
}
|
|
if config.UsePprof {
|
|
args = append(args, "--pprof", fmt.Sprintf("--pprofport=%d", e2e.TestParams.Ports.PrysmBeaconNodePprofPort+index))
|
|
}
|
|
// Only add in the feature flags if we either aren't performing a control test
|
|
// on our features or the beacon index is a multiplier of 2 (idea is to split nodes
|
|
// equally down the line with one group having feature flags and the other without
|
|
// feature flags; this is to allow A-B testing on new features)
|
|
if !config.TestFeature || index != 1 {
|
|
args = append(args, features.E2EBeaconChainFlags...)
|
|
}
|
|
if config.UseBuilder {
|
|
args = append(args, fmt.Sprintf("--%s=%s:%d", flags.MevRelayEndpoint.Name, "http://127.0.0.1", e2e.TestParams.Ports.Eth1ProxyPort+index))
|
|
}
|
|
args = append(args, config.BeaconFlags...)
|
|
|
|
cmd := exec.CommandContext(ctx, binaryPath, args...) // #nosec G204 -- Safe
|
|
// Write stderr to log files.
|
|
stderr, err := os.Create(path.Join(e2e.TestParams.LogPath, fmt.Sprintf("beacon_node_%d_stderr.log", index)))
|
|
if err != nil {
|
|
return err
|
|
}
|
|
defer func() {
|
|
if err := stderr.Close(); err != nil {
|
|
log.WithError(err).Error("Failed to close stderr file")
|
|
}
|
|
}()
|
|
cmd.Stderr = stderr
|
|
log.Infof("Starting beacon chain %d with flags: %s", index, strings.Join(args, " "))
|
|
if err = cmd.Start(); err != nil {
|
|
return fmt.Errorf("failed to start beacon node: %w", err)
|
|
}
|
|
|
|
if err = helpers.WaitForTextInFile(stdOutFile, "gRPC server listening on port"); err != nil {
|
|
return fmt.Errorf("could not find multiaddr for node %d, this means the node had issues starting: %w", index, err)
|
|
}
|
|
|
|
if config.UseFixedPeerIDs {
|
|
peerId, err := helpers.FindFollowingTextInFile(stdOutFile, "Running node with peer id of ")
|
|
if err != nil {
|
|
return fmt.Errorf("could not find peer id: %w", err)
|
|
}
|
|
node.peerID = peerId
|
|
}
|
|
|
|
// Mark node as ready.
|
|
close(node.started)
|
|
|
|
node.cmd = cmd
|
|
return cmd.Wait()
|
|
}
|
|
|
|
// Started checks whether beacon node is started and ready to be queried.
|
|
func (node *BeaconNode) Started() <-chan struct{} {
|
|
return node.started
|
|
}
|
|
|
|
// Pause pauses the component and its underlying process.
|
|
func (node *BeaconNode) Pause() error {
|
|
return node.cmd.Process.Signal(syscall.SIGSTOP)
|
|
}
|
|
|
|
// Resume resumes the component and its underlying process.
|
|
func (node *BeaconNode) Resume() error {
|
|
return node.cmd.Process.Signal(syscall.SIGCONT)
|
|
}
|
|
|
|
// Stop stops the component and its underlying process.
|
|
func (node *BeaconNode) Stop() error {
|
|
return node.cmd.Process.Kill()
|
|
}
|
|
|
|
func (node *BeaconNode) UnderlyingProcess() *os.Process {
|
|
return node.cmd.Process
|
|
}
|
|
|
|
func generateGenesis(ctx context.Context) (state.BeaconState, error) {
|
|
if e2e.TestParams.Eth1GenesisBlock == nil {
|
|
return nil, errors.New("Cannot construct bellatrix block, e2e.TestParams.Eth1GenesisBlock == nil")
|
|
}
|
|
gb := e2e.TestParams.Eth1GenesisBlock
|
|
t := e2e.TestParams.CLGenesisTime
|
|
pcreds := e2e.TestParams.NumberOfExecutionCreds
|
|
nvals := params.BeaconConfig().MinGenesisActiveValidatorCount
|
|
version := e2etypes.GenesisFork()
|
|
return interop.NewPreminedGenesis(ctx, t, nvals, pcreds, version, gb)
|
|
}
|