2019-11-15 18:56:26 +00:00
|
|
|
package endtoend
|
|
|
|
|
|
|
|
import (
|
|
|
|
"context"
|
|
|
|
"fmt"
|
|
|
|
"os"
|
|
|
|
"path"
|
|
|
|
"testing"
|
|
|
|
"time"
|
|
|
|
|
2019-11-17 16:16:20 +00:00
|
|
|
"github.com/bazelbuild/rules_go/go/tools/bazel"
|
2019-11-15 18:56:26 +00:00
|
|
|
ptypes "github.com/gogo/protobuf/types"
|
2019-11-27 05:08:18 +00:00
|
|
|
eth "github.com/prysmaticlabs/ethereumapis/eth/v1alpha1"
|
2020-01-28 19:16:00 +00:00
|
|
|
ev "github.com/prysmaticlabs/prysm/endtoend/evaluators"
|
2019-11-15 18:56:26 +00:00
|
|
|
"github.com/prysmaticlabs/prysm/shared/params"
|
|
|
|
"google.golang.org/grpc"
|
|
|
|
)
|
|
|
|
|
|
|
|
func runEndToEndTest(t *testing.T, config *end2EndConfig) {
|
|
|
|
tmpPath := bazel.TestTmpDir()
|
|
|
|
config.tmpPath = tmpPath
|
2020-01-07 17:00:51 +00:00
|
|
|
t.Logf("Starting time: %s\n", time.Now().String())
|
|
|
|
t.Logf("Test Path: %s\n\n", tmpPath)
|
2019-11-15 18:56:26 +00:00
|
|
|
|
|
|
|
contractAddr, keystorePath, eth1PID := startEth1(t, tmpPath)
|
|
|
|
config.contractAddr = contractAddr
|
|
|
|
beaconNodes := startBeaconNodes(t, config)
|
2020-01-06 20:50:36 +00:00
|
|
|
valClients := initializeValidators(t, config, keystorePath)
|
2019-11-15 18:56:26 +00:00
|
|
|
processIDs := []int{eth1PID}
|
|
|
|
for _, vv := range valClients {
|
|
|
|
processIDs = append(processIDs, vv.processID)
|
|
|
|
}
|
|
|
|
for _, bb := range beaconNodes {
|
2020-01-28 19:16:00 +00:00
|
|
|
processIDs = append(processIDs, bb.ProcessID)
|
2019-11-15 18:56:26 +00:00
|
|
|
}
|
2020-01-07 17:00:51 +00:00
|
|
|
defer logOutput(t, tmpPath, config)
|
2019-11-15 18:56:26 +00:00
|
|
|
defer killProcesses(t, processIDs)
|
|
|
|
|
|
|
|
if config.numBeaconNodes > 1 {
|
|
|
|
t.Run("all_peers_connect", func(t *testing.T) {
|
2020-01-28 19:16:00 +00:00
|
|
|
if err := ev.PeersConnect(beaconNodes); err != nil {
|
|
|
|
t.Fatalf("Failed to connect to peers: %v", err)
|
2019-11-15 18:56:26 +00:00
|
|
|
}
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
2020-01-07 17:00:51 +00:00
|
|
|
beaconLogFile, err := os.Open(path.Join(tmpPath, fmt.Sprintf(beaconNodeLogFileName, 0)))
|
2019-11-15 18:56:26 +00:00
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
2020-01-25 07:39:56 +00:00
|
|
|
if err := waitForTextInFile(beaconLogFile, "Chain started within the last epoch"); err != nil {
|
|
|
|
t.Fatalf("failed to find chain start in logs, this means the chain did not start: %v", err)
|
2019-11-15 18:56:26 +00:00
|
|
|
}
|
2020-01-06 20:50:36 +00:00
|
|
|
|
|
|
|
// Failing early in case chain doesn't start.
|
|
|
|
if t.Failed() {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2020-01-25 07:39:56 +00:00
|
|
|
conn, err := grpc.Dial("127.0.0.1:4200", grpc.WithInsecure())
|
2019-11-15 18:56:26 +00:00
|
|
|
if err != nil {
|
2019-12-05 19:51:33 +00:00
|
|
|
t.Fatalf("Failed to dial: %v", err)
|
2019-11-15 18:56:26 +00:00
|
|
|
}
|
|
|
|
beaconClient := eth.NewBeaconChainClient(conn)
|
|
|
|
nodeClient := eth.NewNodeClient(conn)
|
|
|
|
|
|
|
|
genesis, err := nodeClient.GetGenesis(context.Background(), &ptypes.Empty{})
|
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
// Small offset so evaluators perform in the middle of an epoch.
|
|
|
|
epochSeconds := params.BeaconConfig().SecondsPerSlot * params.BeaconConfig().SlotsPerEpoch
|
|
|
|
genesisTime := time.Unix(genesis.GenesisTime.Seconds+int64(epochSeconds/2), 0)
|
2020-01-28 19:16:00 +00:00
|
|
|
|
2019-11-15 18:56:26 +00:00
|
|
|
ticker := GetEpochTicker(genesisTime, epochSeconds)
|
2020-01-25 07:39:56 +00:00
|
|
|
for currentEpoch := range ticker.C() {
|
2019-11-15 18:56:26 +00:00
|
|
|
for _, evaluator := range config.evaluators {
|
|
|
|
// Only run if the policy says so.
|
|
|
|
if !evaluator.Policy(currentEpoch) {
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
t.Run(fmt.Sprintf(evaluator.Name, currentEpoch), func(t *testing.T) {
|
|
|
|
if err := evaluator.Evaluation(beaconClient); err != nil {
|
2020-01-25 07:39:56 +00:00
|
|
|
t.Errorf("evaluation failed for epoch %d: %v", currentEpoch, err)
|
2019-11-15 18:56:26 +00:00
|
|
|
}
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
2020-01-25 07:39:56 +00:00
|
|
|
if t.Failed() || currentEpoch >= config.epochsToRun {
|
2020-01-28 19:16:00 +00:00
|
|
|
if err := conn.Close(); err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
2020-01-25 07:39:56 +00:00
|
|
|
ticker.Done()
|
|
|
|
break
|
|
|
|
}
|
2019-11-15 18:56:26 +00:00
|
|
|
}
|
|
|
|
|
2020-01-28 19:16:00 +00:00
|
|
|
syncNodeInfo := startNewBeaconNode(t, config, beaconNodes)
|
|
|
|
beaconNodes = append(beaconNodes, syncNodeInfo)
|
|
|
|
index := len(beaconNodes) - 1
|
|
|
|
|
|
|
|
// Sleep until the next epoch to give time for the newly started node to sync.
|
|
|
|
nextEpochSeconds := (config.epochsToRun+2)*epochSeconds + epochSeconds/2
|
|
|
|
genesisTime.Add(time.Duration(nextEpochSeconds) * time.Second)
|
|
|
|
// Wait until middle of epoch to request to prevent conflicts.
|
|
|
|
time.Sleep(time.Until(genesisTime))
|
|
|
|
|
|
|
|
syncLogFile, err := os.Open(path.Join(tmpPath, fmt.Sprintf(beaconNodeLogFileName, index)))
|
2019-11-15 18:56:26 +00:00
|
|
|
if err != nil {
|
2020-01-28 19:16:00 +00:00
|
|
|
t.Fatal(err)
|
2019-11-15 18:56:26 +00:00
|
|
|
}
|
2020-01-28 19:16:00 +00:00
|
|
|
if err := waitForTextInFile(syncLogFile, "Synced up to"); err != nil {
|
|
|
|
t.Fatalf("Failed to sync: %v", err)
|
2019-11-15 18:56:26 +00:00
|
|
|
}
|
|
|
|
|
2020-01-28 19:16:00 +00:00
|
|
|
t.Run("node_finishes_sync", func(t *testing.T) {
|
|
|
|
if err := ev.FinishedSyncing(syncNodeInfo.RPCPort); err != nil {
|
2020-01-25 07:39:56 +00:00
|
|
|
t.Fatal(err)
|
|
|
|
}
|
2020-01-28 19:16:00 +00:00
|
|
|
})
|
2020-01-07 17:00:51 +00:00
|
|
|
|
2020-01-28 19:16:00 +00:00
|
|
|
t.Run("all_nodes_have_correct_head", func(t *testing.T) {
|
|
|
|
if err := ev.AllChainsHaveSameHead(beaconNodes); err != nil {
|
2020-01-25 07:39:56 +00:00
|
|
|
t.Fatal(err)
|
2019-11-15 18:56:26 +00:00
|
|
|
}
|
2020-01-28 19:16:00 +00:00
|
|
|
})
|
2020-01-07 17:00:51 +00:00
|
|
|
|
2020-01-28 19:16:00 +00:00
|
|
|
defer killProcesses(t, []int{syncNodeInfo.ProcessID})
|
2019-11-15 18:56:26 +00:00
|
|
|
}
|