mirror of
https://gitlab.com/pulsechaincom/prysm-pulse.git
synced 2025-01-10 19:51:20 +00:00
0a41b957dc
due to flaky test re-run behavior, logs from a failed test run are overwritten by subsequent retries. This makes it difficult to retrieve logs after the first failed run. It also takes some squinting through output to find the location of the log file in the first place. This flag enables logs to be placed in an arbitrary path. Note that bazel sandboxing generally will force this path to be in the /tmp tree. Co-authored-by: Kasey Kirkham <kasey@users.noreply.github.com>
380 lines
13 KiB
Go
380 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/bazelbuild/rules_go/go/tools/bazel"
|
|
"github.com/ethereum/go-ethereum/common"
|
|
"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
|
|
ContractAddress common.Address
|
|
Ports *ports
|
|
Paths *paths
|
|
}
|
|
|
|
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
|
|
)
|
|
|
|
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
|
|
}
|
|
|
|
TestParams = ¶ms{
|
|
TestPath: filepath.Join(testPath, fmt.Sprintf("shard-%d", testShardIndex)),
|
|
LogPath: logPath,
|
|
TestShardIndex: testShardIndex,
|
|
BeaconNodeCount: beaconNodeCount,
|
|
Ports: testPorts,
|
|
}
|
|
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
|
|
}
|
|
|
|
TestParams = ¶ms{
|
|
TestPath: filepath.Join(testPath, fmt.Sprintf("shard-%d", testShardIndex)),
|
|
LogPath: logPath,
|
|
TestShardIndex: testShardIndex,
|
|
BeaconNodeCount: beaconNodeCount,
|
|
LighthouseBeaconNodeCount: lighthouseNodeCount,
|
|
Ports: testPorts,
|
|
}
|
|
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
|
|
}
|