mirror of
https://gitlab.com/pulsechaincom/prysm-pulse.git
synced 2025-01-18 07:48:46 +00:00
9f1c6b2d18
* fix multiclient e2e * parallelize keystore generation * fix config startup * fix GetAggregateAttestation * regression test * kasey's review * fix test * gaz * fix tests * deep source
393 lines
13 KiB
Go
393 lines
13 KiB
Go
// Package params defines all custom parameter configurations
|
|
// for running end to end tests.
|
|
package params
|
|
|
|
import (
|
|
"errors"
|
|
"fmt"
|
|
"net"
|
|
"net/url"
|
|
"os"
|
|
"path"
|
|
"path/filepath"
|
|
"strconv"
|
|
"testing"
|
|
"time"
|
|
|
|
"github.com/ethereum/go-ethereum/core/types"
|
|
|
|
"github.com/bazelbuild/rules_go/go/tools/bazel"
|
|
"github.com/prysmaticlabs/prysm/v3/io/file"
|
|
)
|
|
|
|
// params struct defines the parameters needed for running E2E tests to properly handle test sharding.
|
|
type params struct {
|
|
TestPath string
|
|
LogPath string
|
|
TestShardIndex int
|
|
BeaconNodeCount int
|
|
LighthouseBeaconNodeCount int
|
|
Ports *ports
|
|
Paths *paths
|
|
Eth1GenesisBlock *types.Block
|
|
StartTime time.Time
|
|
CLGenesisTime uint64
|
|
Eth1GenesisTime uint64
|
|
}
|
|
|
|
type ports struct {
|
|
BootNodePort int
|
|
BootNodeMetricsPort int
|
|
Eth1Port int
|
|
Eth1RPCPort int
|
|
Eth1AuthRPCPort int
|
|
Eth1WSPort int
|
|
Eth1ProxyPort int
|
|
PrysmBeaconNodeRPCPort int
|
|
PrysmBeaconNodeUDPPort int
|
|
PrysmBeaconNodeTCPPort int
|
|
PrysmBeaconNodeGatewayPort int
|
|
PrysmBeaconNodeMetricsPort int
|
|
PrysmBeaconNodePprofPort int
|
|
LighthouseBeaconNodeP2PPort int
|
|
LighthouseBeaconNodeHTTPPort int
|
|
LighthouseBeaconNodeMetricsPort int
|
|
ValidatorMetricsPort int
|
|
ValidatorGatewayPort int
|
|
JaegerTracingPort int
|
|
}
|
|
|
|
type paths struct{}
|
|
|
|
// Eth1StaticFile abstracts the location of the eth1 static file folder in the e2e directory, so that
|
|
// a relative path can be used.
|
|
// The relative path is specified as a variadic slice of path parts, in the same way as path.Join.
|
|
func (*paths) Eth1StaticFile(rel ...string) string {
|
|
parts := append([]string{Eth1StaticFilesPath}, rel...)
|
|
return path.Join(parts...)
|
|
}
|
|
|
|
// Eth1Runfile returns the full path to a file in the eth1 static directory, within bazel's run context.
|
|
// The relative path is specified as a variadic slice of path parts, in the same style as path.Join.
|
|
func (p *paths) Eth1Runfile(rel ...string) (string, error) {
|
|
return bazel.Runfile(p.Eth1StaticFile(rel...))
|
|
}
|
|
|
|
// MinerKeyPath returns the full path to the file containing the miner's cryptographic keys.
|
|
func (p *paths) MinerKeyPath() (string, error) {
|
|
return p.Eth1Runfile(minerKeyFilename)
|
|
}
|
|
|
|
// TestParams is the globally accessible var for getting config elements.
|
|
var TestParams *params
|
|
|
|
// Logfile gives the full path to a file in the bazel test environment log directory.
|
|
// The relative path is specified as a variadic slice of path parts, in the same style as path.Join.
|
|
func (p *params) Logfile(rel ...string) string {
|
|
return path.Join(append([]string{p.LogPath}, rel...)...)
|
|
}
|
|
|
|
// Eth1RPCURL gives the full url to use to connect to the given eth1 client's RPC endpoint.
|
|
// The `index` param corresponds to the `index` field of the `eth1.Node` e2e component.
|
|
// These are are off by one compared to corresponding beacon nodes, because the miner is assigned index 0.
|
|
// eg instance the index of the EL instance associated with beacon node index `0` would typically be `1`.
|
|
func (p *params) Eth1RPCURL(index int) *url.URL {
|
|
return &url.URL{
|
|
Scheme: baseELScheme,
|
|
Host: net.JoinHostPort(baseELHost, fmt.Sprintf("%d", p.Ports.Eth1RPCPort+index)),
|
|
}
|
|
}
|
|
|
|
// BootNodeLogFileName is the file name used for the beacon chain node logs.
|
|
var BootNodeLogFileName = "bootnode.log"
|
|
|
|
// TracingRequestSinkFileName is the file name for writing raw trace requests.
|
|
var TracingRequestSinkFileName = "tracing-http-requests.log.gz"
|
|
|
|
// BeaconNodeLogFileName is the file name used for the beacon chain node logs.
|
|
var BeaconNodeLogFileName = "beacon-%d.log"
|
|
|
|
// ValidatorLogFileName is the file name used for the validator client logs.
|
|
var ValidatorLogFileName = "vals-%d.log"
|
|
|
|
// StandardBeaconCount is a global constant for the count of beacon nodes of standard E2E tests.
|
|
var StandardBeaconCount = 2
|
|
|
|
// StandardLighthouseNodeCount is a global constant for the count of lighthouse beacon nodes of standard E2E tests.
|
|
var StandardLighthouseNodeCount = 2
|
|
|
|
// DepositCount is the number of deposits the E2E runner should make to evaluate post-genesis deposit processing.
|
|
var DepositCount = uint64(64)
|
|
|
|
// NumOfExecEngineTxs is the number of transaction sent to the execution engine.
|
|
var NumOfExecEngineTxs = uint64(200)
|
|
|
|
// ExpectedExecEngineTxsThreshold is the portion of execution engine transactions we expect to find in blocks.
|
|
var ExpectedExecEngineTxsThreshold = 0.5
|
|
|
|
// Base port values.
|
|
const (
|
|
portSpan = 50
|
|
|
|
BootNodePort = 2150
|
|
BootNodeMetricsPort = BootNodePort + portSpan
|
|
|
|
Eth1Port = 3150
|
|
Eth1RPCPort = Eth1Port + portSpan
|
|
Eth1WSPort = Eth1Port + 2*portSpan
|
|
Eth1AuthRPCPort = Eth1Port + 3*portSpan
|
|
Eth1ProxyPort = Eth1Port + 4*portSpan
|
|
|
|
PrysmBeaconNodeRPCPort = 4150
|
|
PrysmBeaconNodeUDPPort = PrysmBeaconNodeRPCPort + portSpan
|
|
PrysmBeaconNodeTCPPort = PrysmBeaconNodeRPCPort + 2*portSpan
|
|
PrysmBeaconNodeGatewayPort = PrysmBeaconNodeRPCPort + 3*portSpan
|
|
PrysmBeaconNodeMetricsPort = PrysmBeaconNodeRPCPort + 4*portSpan
|
|
PrysmBeaconNodePprofPort = PrysmBeaconNodeRPCPort + 5*portSpan
|
|
|
|
LighthouseBeaconNodeP2PPort = 5150
|
|
LighthouseBeaconNodeHTTPPort = LighthouseBeaconNodeP2PPort + portSpan
|
|
LighthouseBeaconNodeMetricsPort = LighthouseBeaconNodeP2PPort + 2*portSpan
|
|
|
|
ValidatorGatewayPort = 6150
|
|
ValidatorMetricsPort = ValidatorGatewayPort + portSpan
|
|
|
|
JaegerTracingPort = 9150
|
|
|
|
StartupBufferSecs = 5
|
|
StartupBufferSecsMulticlient = 10
|
|
)
|
|
|
|
func logDir() string {
|
|
wTime := func(p string) string {
|
|
return path.Join(p, time.Now().Format("20060102/150405"))
|
|
}
|
|
path, ok := os.LookupEnv("E2E_LOG_PATH")
|
|
if ok {
|
|
return wTime(path)
|
|
}
|
|
path, _ = os.LookupEnv("TEST_UNDECLARED_OUTPUTS_DIR")
|
|
return wTime(path)
|
|
}
|
|
|
|
// Init initializes the E2E config, properly handling test sharding.
|
|
func Init(t *testing.T, beaconNodeCount int) error {
|
|
d := logDir()
|
|
if d == "" {
|
|
return errors.New("unable to determine log directory, no value for E2E_LOG_PATH or TEST_UNDECLARED_OUTPUTS_DIR")
|
|
}
|
|
logPath := path.Join(d, t.Name())
|
|
if err := file.MkdirAll(logPath); err != nil {
|
|
return err
|
|
}
|
|
testPath := bazel.TestTmpDir()
|
|
testTotalShardsStr, ok := os.LookupEnv("TEST_TOTAL_SHARDS")
|
|
if !ok {
|
|
testTotalShardsStr = "1"
|
|
}
|
|
testTotalShards, err := strconv.Atoi(testTotalShardsStr)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
testShardIndexStr, ok := os.LookupEnv("TEST_SHARD_INDEX")
|
|
if !ok {
|
|
testShardIndexStr = "0"
|
|
}
|
|
testShardIndex, err := strconv.Atoi(testShardIndexStr)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
var existingRegistrations []int
|
|
testPorts := &ports{}
|
|
err = initializeStandardPorts(testTotalShards, testShardIndex, testPorts, &existingRegistrations)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
genTime := uint64(time.Now().Unix()) + StartupBufferSecs
|
|
TestParams = ¶ms{
|
|
TestPath: filepath.Join(testPath, fmt.Sprintf("shard-%d", testShardIndex)),
|
|
LogPath: logPath,
|
|
TestShardIndex: testShardIndex,
|
|
BeaconNodeCount: beaconNodeCount,
|
|
Ports: testPorts,
|
|
CLGenesisTime: genTime,
|
|
Eth1GenesisTime: genTime,
|
|
}
|
|
return nil
|
|
}
|
|
|
|
// InitMultiClient initializes the multiclient E2E config, properly handling test sharding.
|
|
func InitMultiClient(t *testing.T, beaconNodeCount int, lighthouseNodeCount int) error {
|
|
testPath := bazel.TestTmpDir()
|
|
logPath, ok := os.LookupEnv("TEST_UNDECLARED_OUTPUTS_DIR")
|
|
if !ok {
|
|
return errors.New("expected TEST_UNDECLARED_OUTPUTS_DIR to be defined")
|
|
}
|
|
logPath = path.Join(logPath, t.Name())
|
|
if err := file.MkdirAll(logPath); err != nil {
|
|
return err
|
|
}
|
|
testTotalShardsStr, ok := os.LookupEnv("TEST_TOTAL_SHARDS")
|
|
if !ok {
|
|
testTotalShardsStr = "1"
|
|
}
|
|
testTotalShards, err := strconv.Atoi(testTotalShardsStr)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
testShardIndexStr, ok := os.LookupEnv("TEST_SHARD_INDEX")
|
|
if !ok {
|
|
testShardIndexStr = "0"
|
|
}
|
|
testShardIndex, err := strconv.Atoi(testShardIndexStr)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
var existingRegistrations []int
|
|
testPorts := &ports{}
|
|
err = initializeStandardPorts(testTotalShards, testShardIndex, testPorts, &existingRegistrations)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
err = initializeMulticlientPorts(testTotalShards, testShardIndex, testPorts, &existingRegistrations)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
genTime := uint64(time.Now().Unix()) + StartupBufferSecsMulticlient
|
|
TestParams = ¶ms{
|
|
TestPath: filepath.Join(testPath, fmt.Sprintf("shard-%d", testShardIndex)),
|
|
LogPath: logPath,
|
|
TestShardIndex: testShardIndex,
|
|
BeaconNodeCount: beaconNodeCount,
|
|
LighthouseBeaconNodeCount: lighthouseNodeCount,
|
|
Ports: testPorts,
|
|
CLGenesisTime: genTime,
|
|
Eth1GenesisTime: genTime,
|
|
}
|
|
return nil
|
|
}
|
|
|
|
// port returns a safe port number based on the seed and shard data.
|
|
func port(seed, shardCount, shardIndex int, existingRegistrations *[]int) (int, error) {
|
|
portToRegister := seed + portSpan/shardCount*shardIndex
|
|
for _, p := range *existingRegistrations {
|
|
if portToRegister >= p && portToRegister <= p+(portSpan/shardCount)-1 {
|
|
return 0, fmt.Errorf("port %d overlaps with already registered port %d", seed, p)
|
|
}
|
|
}
|
|
*existingRegistrations = append(*existingRegistrations, portToRegister)
|
|
|
|
// Calculation example: 3 shards, seed 2000, port span 50.
|
|
// Shard 0: 2000 + (50 / 3 * 0) = 2000 (we can safely use ports 2000-2015)
|
|
// Shard 1: 2000 + (50 / 3 * 1) = 2016 (we can safely use ports 2016-2031)
|
|
// Shard 2: 2000 + (50 / 3 * 2) = 2032 (we can safely use ports 2032-2047, and in reality 2032-2049)
|
|
return portToRegister, nil
|
|
}
|
|
|
|
func initializeStandardPorts(shardCount, shardIndex int, ports *ports, existingRegistrations *[]int) error {
|
|
bootnodePort, err := port(BootNodePort, shardCount, shardIndex, existingRegistrations)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
bootnodeMetricsPort, err := port(BootNodeMetricsPort, shardCount, shardIndex, existingRegistrations)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
eth1Port, err := port(Eth1Port, shardCount, shardIndex, existingRegistrations)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
eth1RPCPort, err := port(Eth1RPCPort, shardCount, shardIndex, existingRegistrations)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
eth1WSPort, err := port(Eth1WSPort, shardCount, shardIndex, existingRegistrations)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
eth1AuthPort, err := port(Eth1AuthRPCPort, shardCount, shardIndex, existingRegistrations)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
eth1ProxyPort, err := port(Eth1ProxyPort, shardCount, shardIndex, existingRegistrations)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
beaconNodeRPCPort, err := port(PrysmBeaconNodeRPCPort, shardCount, shardIndex, existingRegistrations)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
beaconNodeUDPPort, err := port(PrysmBeaconNodeUDPPort, shardCount, shardIndex, existingRegistrations)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
beaconNodeTCPPort, err := port(PrysmBeaconNodeTCPPort, shardCount, shardIndex, existingRegistrations)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
beaconNodeGatewayPort, err := port(PrysmBeaconNodeGatewayPort, shardCount, shardIndex, existingRegistrations)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
beaconNodeMetricsPort, err := port(PrysmBeaconNodeMetricsPort, shardCount, shardIndex, existingRegistrations)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
beaconNodePprofPort, err := port(PrysmBeaconNodePprofPort, shardCount, shardIndex, existingRegistrations)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
validatorGatewayPort, err := port(ValidatorGatewayPort, shardCount, shardIndex, existingRegistrations)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
validatorMetricsPort, err := port(ValidatorMetricsPort, shardCount, shardIndex, existingRegistrations)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
jaegerTracingPort, err := port(JaegerTracingPort, shardCount, shardIndex, existingRegistrations)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
ports.BootNodePort = bootnodePort
|
|
ports.BootNodeMetricsPort = bootnodeMetricsPort
|
|
ports.Eth1Port = eth1Port
|
|
ports.Eth1RPCPort = eth1RPCPort
|
|
ports.Eth1AuthRPCPort = eth1AuthPort
|
|
ports.Eth1WSPort = eth1WSPort
|
|
ports.Eth1ProxyPort = eth1ProxyPort
|
|
ports.PrysmBeaconNodeRPCPort = beaconNodeRPCPort
|
|
ports.PrysmBeaconNodeUDPPort = beaconNodeUDPPort
|
|
ports.PrysmBeaconNodeTCPPort = beaconNodeTCPPort
|
|
ports.PrysmBeaconNodeGatewayPort = beaconNodeGatewayPort
|
|
ports.PrysmBeaconNodeMetricsPort = beaconNodeMetricsPort
|
|
ports.PrysmBeaconNodePprofPort = beaconNodePprofPort
|
|
ports.ValidatorMetricsPort = validatorMetricsPort
|
|
ports.ValidatorGatewayPort = validatorGatewayPort
|
|
ports.JaegerTracingPort = jaegerTracingPort
|
|
return nil
|
|
}
|
|
|
|
func initializeMulticlientPorts(shardCount, shardIndex int, ports *ports, existingRegistrations *[]int) error {
|
|
lighthouseBeaconNodeP2PPort, err := port(LighthouseBeaconNodeP2PPort, shardCount, shardIndex, existingRegistrations)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
lighthouseBeaconNodeHTTPPort, err := port(LighthouseBeaconNodeHTTPPort, shardCount, shardIndex, existingRegistrations)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
lighthouseBeaconNodeMetricsPort, err := port(LighthouseBeaconNodeMetricsPort, shardCount, shardIndex, existingRegistrations)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
ports.LighthouseBeaconNodeP2PPort = lighthouseBeaconNodeP2PPort
|
|
ports.LighthouseBeaconNodeHTTPPort = lighthouseBeaconNodeHTTPPort
|
|
ports.LighthouseBeaconNodeMetricsPort = lighthouseBeaconNodeMetricsPort
|
|
return nil
|
|
}
|