mirror of
https://gitlab.com/pulsechaincom/prysm-pulse.git
synced 2025-01-05 17:22:18 +00:00
297 lines
10 KiB
Go
297 lines
10 KiB
Go
// Package backend contains utilities for simulating an entire
|
|
// ETH 2.0 beacon chain for e2e tests and benchmarking
|
|
// purposes.
|
|
package backend
|
|
|
|
import (
|
|
"context"
|
|
"fmt"
|
|
"reflect"
|
|
"time"
|
|
|
|
"github.com/ethereum/go-ethereum/common"
|
|
"github.com/gogo/protobuf/proto"
|
|
"github.com/prysmaticlabs/prysm/beacon-chain/blockchain"
|
|
b "github.com/prysmaticlabs/prysm/beacon-chain/core/blocks"
|
|
"github.com/prysmaticlabs/prysm/beacon-chain/core/state"
|
|
"github.com/prysmaticlabs/prysm/beacon-chain/core/validators"
|
|
"github.com/prysmaticlabs/prysm/beacon-chain/db"
|
|
"github.com/prysmaticlabs/prysm/beacon-chain/utils"
|
|
pb "github.com/prysmaticlabs/prysm/proto/beacon/p2p/v1"
|
|
"github.com/prysmaticlabs/prysm/shared/hashutil"
|
|
"github.com/prysmaticlabs/prysm/shared/params"
|
|
"github.com/prysmaticlabs/prysm/shared/slices"
|
|
"github.com/prysmaticlabs/prysm/shared/trie"
|
|
log "github.com/sirupsen/logrus"
|
|
)
|
|
|
|
// SimulatedBackend allowing for a programmatic advancement
|
|
// of an in-memory beacon chain for client test runs
|
|
// and other e2e use cases.
|
|
type SimulatedBackend struct {
|
|
chainService *blockchain.ChainService
|
|
beaconDB *db.BeaconDB
|
|
}
|
|
|
|
// NewSimulatedBackend creates an instance by initializing a chain service
|
|
// utilizing a mockDB which will act according to test run parameters specified
|
|
// in the common ETH 2.0 client test YAML format.
|
|
func NewSimulatedBackend() (*SimulatedBackend, error) {
|
|
db, err := setupDB()
|
|
if err != nil {
|
|
return nil, fmt.Errorf("could not setup simulated backend db: %v", err)
|
|
}
|
|
cs, err := blockchain.NewChainService(context.Background(), &blockchain.Config{
|
|
BeaconDB: db,
|
|
IncomingBlockBuf: 0,
|
|
EnablePOWChain: false,
|
|
})
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
return &SimulatedBackend{
|
|
chainService: cs,
|
|
beaconDB: db,
|
|
}, nil
|
|
}
|
|
|
|
// RunForkChoiceTest uses a parsed set of chaintests from a YAML file
|
|
// according to the ETH 2.0 client chain test specification and runs them
|
|
// against the simulated backend.
|
|
func (sb *SimulatedBackend) RunForkChoiceTest(testCase *ForkChoiceTestCase) error {
|
|
defer teardownDB(sb.beaconDB)
|
|
// Utilize the config parameters in the test case to setup
|
|
// the DB and set global config parameters accordingly.
|
|
// Config parameters include: ValidatorCount, ShardCount,
|
|
// CycleLength, MinCommitteeSize, and more based on the YAML
|
|
// test language specification.
|
|
c := params.BeaconConfig()
|
|
c.ShardCount = testCase.Config.ShardCount
|
|
c.EpochLength = testCase.Config.CycleLength
|
|
c.TargetCommitteeSize = testCase.Config.MinCommitteeSize
|
|
params.OverrideBeaconConfig(c)
|
|
|
|
// Then, we create the validators based on the custom test config.
|
|
randaoPreCommit := [32]byte{}
|
|
randaoReveal := hashutil.Hash(randaoPreCommit[:])
|
|
validators := make([]*pb.ValidatorRecord, testCase.Config.ValidatorCount)
|
|
for i := uint64(0); i < testCase.Config.ValidatorCount; i++ {
|
|
validators[i] = &pb.ValidatorRecord{
|
|
ExitSlot: params.BeaconConfig().EntryExitDelay,
|
|
Pubkey: []byte{},
|
|
RandaoCommitmentHash32: randaoReveal[:],
|
|
}
|
|
}
|
|
// TODO(#718): Next step is to update and save the blocks specified
|
|
// in the case case into the DB.
|
|
//
|
|
// Then, we call the updateHead routine and confirm the
|
|
// chain's head is the expected result from the test case.
|
|
return nil
|
|
}
|
|
|
|
// RunShuffleTest uses validator set specified from a YAML file, runs the validator shuffle
|
|
// algorithm, then compare the output with the expected output from the YAML file.
|
|
func (sb *SimulatedBackend) RunShuffleTest(testCase *ShuffleTestCase) error {
|
|
defer teardownDB(sb.beaconDB)
|
|
seed := common.BytesToHash([]byte(testCase.Seed))
|
|
output, err := utils.ShuffleIndices(seed, testCase.Input)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
if !reflect.DeepEqual(output, testCase.Output) {
|
|
return fmt.Errorf("shuffle result error: expected %v, actual %v", testCase.Output, output)
|
|
}
|
|
return nil
|
|
}
|
|
|
|
// RunStateTransitionTest advances a beacon chain state transition an N amount of
|
|
// slots from a genesis state, with a block being processed at every iteration
|
|
// of the state transition function.
|
|
func (sb *SimulatedBackend) RunStateTransitionTest(testCase *StateTestCase) error {
|
|
defer teardownDB(sb.beaconDB)
|
|
// We setup the initial configuration for running state
|
|
// transition tests below.
|
|
c := params.BeaconConfig()
|
|
c.EpochLength = testCase.Config.EpochLength
|
|
c.DepositsForChainStart = testCase.Config.DepositsForChainStart
|
|
params.OverrideBeaconConfig(c)
|
|
|
|
// We create a list of randao hash onions for the given number of slots
|
|
// the simulation will attempt.
|
|
hashOnions := generateSimulatedRandaoHashOnions(testCase.Config.NumSlots)
|
|
|
|
// We then generate initial validator deposits for initializing the
|
|
// beacon state based where every validator will use the last layer in the randao
|
|
// onions list as the commitment in the deposit instance.
|
|
lastRandaoLayer := hashOnions[len(hashOnions)-1]
|
|
initialDeposits, err := generateInitialSimulatedDeposits(lastRandaoLayer)
|
|
if err != nil {
|
|
return fmt.Errorf("could not simulate initial validator deposits: %v", err)
|
|
}
|
|
|
|
genesisTime := params.BeaconConfig().GenesisTime.Unix()
|
|
beaconState, err := state.InitialBeaconState(initialDeposits, uint64(genesisTime), nil)
|
|
if err != nil {
|
|
return fmt.Errorf("could not initialize simulated beacon state")
|
|
}
|
|
|
|
// We do not expect hashing initial beacon state and genesis block to
|
|
// fail, so we can safely ignore the error below.
|
|
// #nosec G104
|
|
encodedState, _ := proto.Marshal(beaconState)
|
|
stateRoot := hashutil.Hash(encodedState)
|
|
genesisBlock := b.NewGenesisBlock(stateRoot[:])
|
|
// #nosec G104
|
|
encodedGenesisBlock, _ := proto.Marshal(genesisBlock)
|
|
genesisBlockRoot := hashutil.Hash(encodedGenesisBlock)
|
|
|
|
// We now keep track of generated blocks for each state transition in
|
|
// a slice.
|
|
prevBlockRoots := [][32]byte{genesisBlockRoot}
|
|
|
|
// We keep track of the randao layers peeled for each proposer index in a map.
|
|
layersPeeledForProposer := make(map[uint32]int, len(beaconState.ValidatorRegistry))
|
|
for idx := range beaconState.ValidatorRegistry {
|
|
layersPeeledForProposer[uint32(idx)] = 0
|
|
}
|
|
|
|
depositsTrie := trie.NewDepositTrie()
|
|
averageTimesPerTransition := []time.Duration{}
|
|
for i := uint64(0); i < testCase.Config.NumSlots; i++ {
|
|
prevBlockRoot := prevBlockRoots[len(prevBlockRoots)-1]
|
|
|
|
committeeArray, err := validators.CrosslinkCommitteesAtSlot(beaconState, i)
|
|
if err != nil {
|
|
return fmt.Errorf("could not get crosslink committee: %v", err)
|
|
}
|
|
firstCommittee := committeeArray[0].Committee
|
|
proposerIndex := firstCommittee[i%uint64(len(firstCommittee))]
|
|
|
|
// If the slot is marked as skipped in the configuration options,
|
|
// we simply run the state transition with a nil block argument.
|
|
if slices.IsInUint64(i, testCase.Config.SkipSlots) {
|
|
newState, err := state.ExecuteStateTransition(beaconState, nil, prevBlockRoot)
|
|
if err != nil {
|
|
return fmt.Errorf("could not execute state transition: %v", err)
|
|
}
|
|
beaconState = newState
|
|
layersPeeledForProposer[proposerIndex]++
|
|
continue
|
|
}
|
|
|
|
// If the slot is not skipped, we check if we are simulating a deposit at the current slot.
|
|
var simulatedDeposit *StateTestDeposit
|
|
for _, deposit := range testCase.Config.Deposits {
|
|
if deposit.Slot == i {
|
|
simulatedDeposit = deposit
|
|
break
|
|
}
|
|
}
|
|
var simulatedProposerSlashing *StateTestProposerSlashing
|
|
for _, pSlashing := range testCase.Config.ProposerSlashings {
|
|
if pSlashing.Slot == i {
|
|
simulatedProposerSlashing = pSlashing
|
|
break
|
|
}
|
|
}
|
|
var simulatedCasperSlashing *StateTestCasperSlashing
|
|
for _, cSlashing := range testCase.Config.CasperSlashings {
|
|
if cSlashing.Slot == i {
|
|
simulatedCasperSlashing = cSlashing
|
|
break
|
|
}
|
|
}
|
|
var simulatedValidatorExit *StateTestValidatorExit
|
|
for _, exit := range testCase.Config.ValidatorExits {
|
|
if exit.Slot == i {
|
|
simulatedValidatorExit = exit
|
|
break
|
|
}
|
|
}
|
|
|
|
layersPeeled := layersPeeledForProposer[proposerIndex]
|
|
blockRandaoReveal := determineSimulatedBlockRandaoReveal(layersPeeled, hashOnions)
|
|
|
|
// We generate a new block to pass into the state transition.
|
|
newBlock, newBlockRoot, err := generateSimulatedBlock(
|
|
beaconState,
|
|
prevBlockRoot,
|
|
blockRandaoReveal,
|
|
lastRandaoLayer,
|
|
simulatedDeposit,
|
|
depositsTrie,
|
|
simulatedProposerSlashing,
|
|
simulatedCasperSlashing,
|
|
simulatedValidatorExit,
|
|
)
|
|
if err != nil {
|
|
return fmt.Errorf("could not generate simulated beacon block %v", err)
|
|
}
|
|
latestRoot := depositsTrie.Root()
|
|
beaconState.LatestDepositRootHash32 = latestRoot[:]
|
|
|
|
startTime := time.Now()
|
|
newState, err := state.ExecuteStateTransition(beaconState, newBlock, prevBlockRoot)
|
|
if err != nil {
|
|
return fmt.Errorf("could not execute state transition: %v", err)
|
|
}
|
|
endTime := time.Now()
|
|
averageTimesPerTransition = append(averageTimesPerTransition, endTime.Sub(startTime))
|
|
|
|
// We then keep track of information about the state after the
|
|
// state transition was applied.
|
|
beaconState = newState
|
|
prevBlockRoots = append(prevBlockRoots, newBlockRoot)
|
|
layersPeeledForProposer[proposerIndex]++
|
|
}
|
|
|
|
log.Infof(
|
|
"with %d initial deposits, each state transition took average time = %v",
|
|
testCase.Config.DepositsForChainStart,
|
|
averageDuration(averageTimesPerTransition),
|
|
)
|
|
|
|
if beaconState.Slot != testCase.Results.Slot {
|
|
return fmt.Errorf(
|
|
"incorrect state slot after %d state transitions without blocks, wanted %d, received %d",
|
|
testCase.Config.NumSlots,
|
|
testCase.Config.NumSlots,
|
|
testCase.Results.Slot,
|
|
)
|
|
}
|
|
if len(beaconState.ValidatorRegistry) != testCase.Results.NumValidators {
|
|
return fmt.Errorf(
|
|
"incorrect num validators after %d state transitions without blocks, wanted %d, received %d",
|
|
testCase.Config.NumSlots,
|
|
testCase.Results.NumValidators,
|
|
len(beaconState.ValidatorRegistry),
|
|
)
|
|
}
|
|
for _, penalized := range testCase.Results.PenalizedValidators {
|
|
if beaconState.ValidatorRegistry[penalized].PenalizedSlot == params.BeaconConfig().FarFutureSlot {
|
|
return fmt.Errorf(
|
|
"expected validator at index %d to have been penalized",
|
|
penalized,
|
|
)
|
|
}
|
|
}
|
|
for _, exited := range testCase.Results.ExitedValidators {
|
|
if beaconState.ValidatorRegistry[exited].StatusFlags != pb.ValidatorRecord_INITIATED_EXIT {
|
|
return fmt.Errorf(
|
|
"expected validator at index %d to have exited",
|
|
exited,
|
|
)
|
|
}
|
|
}
|
|
return nil
|
|
}
|
|
|
|
func averageDuration(times []time.Duration) time.Duration {
|
|
sum := int64(0)
|
|
for _, t := range times {
|
|
sum += t.Nanoseconds()
|
|
}
|
|
return time.Duration(sum / int64(len(times)))
|
|
}
|