// 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/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/db" "github.com/prysmaticlabs/prysm/beacon-chain/utils" pb "github.com/prysmaticlabs/prysm/proto/beacon/p2p/v1" "github.com/prysmaticlabs/prysm/shared/bls" "github.com/prysmaticlabs/prysm/shared/hashutil" "github.com/prysmaticlabs/prysm/shared/params" "github.com/prysmaticlabs/prysm/shared/sliceutil" "github.com/prysmaticlabs/prysm/shared/trieutil" 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 state *pb.BeaconState prevBlockRoots [][32]byte inMemoryBlocks []*pb.BeaconBlock depositTrie *trieutil.DepositTrie } // SimulatedObjects is a container to hold the // required primitives for generation of a beacon // block. type SimulatedObjects struct { simDeposit *StateTestDeposit simProposerSlashing *StateTestProposerSlashing simAttesterSlashing *StateTestAttesterSlashing simValidatorExit *StateTestValidatorExit } // 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 := db.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, inMemoryBlocks: make([]*pb.BeaconBlock, 0), }, nil } // SetupBackend sets up the simulated backend with simulated deposits, and initializes the // state and genesis block. func (sb *SimulatedBackend) SetupBackend(numOfDeposits uint64) ([]*bls.SecretKey, error) { initialDeposits, privKeys, err := generateInitialSimulatedDeposits(numOfDeposits) if err != nil { return nil, fmt.Errorf("could not simulate initial validator deposits: %v", err) } if err := sb.setupBeaconStateAndGenesisBlock(initialDeposits); err != nil { return nil, fmt.Errorf("could not set up beacon state and initialize genesis block %v", err) } sb.depositTrie = trieutil.NewDepositTrie() return privKeys, nil } // DB returns the underlying db instance in the simulated // backend. func (sb *SimulatedBackend) DB() *db.BeaconDB { return sb.beaconDB } // GenerateBlockAndAdvanceChain generates a simulated block and runs that block though // state transition. func (sb *SimulatedBackend) GenerateBlockAndAdvanceChain(objects *SimulatedObjects, privKeys []*bls.SecretKey) error { prevBlockRoot := sb.prevBlockRoots[len(sb.prevBlockRoots)-1] // We generate a new block to pass into the state transition. newBlock, newBlockRoot, err := generateSimulatedBlock( sb.state, prevBlockRoot, sb.depositTrie, objects, privKeys, ) if err != nil { return fmt.Errorf("could not generate simulated beacon block %v", err) } latestRoot := sb.depositTrie.Root() sb.state.LatestEth1Data = &pb.Eth1Data{ DepositRootHash32: latestRoot[:], BlockHash32: []byte{}, } newState, err := state.ExecuteStateTransition( sb.state, newBlock, prevBlockRoot, false, /* no sig verify */ ) if err != nil { return fmt.Errorf("could not execute state transition: %v", err) } sb.state = newState sb.prevBlockRoots = append(sb.prevBlockRoots, newBlockRoot) sb.inMemoryBlocks = append(sb.inMemoryBlocks, newBlock) return nil } // GenerateNilBlockAndAdvanceChain would trigger a state transition with a nil block. func (sb *SimulatedBackend) GenerateNilBlockAndAdvanceChain() error { prevBlockRoot := sb.prevBlockRoots[len(sb.prevBlockRoots)-1] newState, err := state.ExecuteStateTransition( sb.state, nil, prevBlockRoot, false, /* no sig verify */ ) if err != nil { return fmt.Errorf("could not execute state transition: %v", err) } sb.state = newState return nil } // Shutdown closes the db associated with the simulated backend. func (sb *SimulatedBackend) Shutdown() error { return sb.beaconDB.Close() } // State is a getter to return the current beacon state // of the backend. func (sb *SimulatedBackend) State() *pb.BeaconState { return sb.state } // InMemoryBlocks returns the blocks that have been processed by the simulated // backend. func (sb *SimulatedBackend) InMemoryBlocks() []*pb.BeaconBlock { return sb.inMemoryBlocks } // 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 db.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.SlotsPerEpoch = testCase.Config.CycleLength c.TargetCommitteeSize = testCase.Config.MinCommitteeSize params.OverrideBeaconConfig(c) // Then, we create the validators based on the custom test config. validators := make([]*pb.Validator, testCase.Config.ValidatorCount) for i := uint64(0); i < testCase.Config.ValidatorCount; i++ { validators[i] = &pb.Validator{ ExitEpoch: params.BeaconConfig().ActivationExitDelay, Pubkey: []byte{}, } } // 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 db.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 db.TeardownDB(sb.beaconDB) setTestConfig(testCase) privKeys, err := sb.initializeStateTest(testCase) if err != nil { return fmt.Errorf("could not initialize state test %v", err) } averageTimesPerTransition := []time.Duration{} startSlot := params.BeaconConfig().GenesisSlot for i := startSlot; i < startSlot+testCase.Config.NumSlots; i++ { // If the slot is marked as skipped in the configuration options, // we simply run the state transition with a nil block argument. if sliceutil.IsInUint64(i, testCase.Config.SkipSlots) { if err := sb.GenerateNilBlockAndAdvanceChain(); err != nil { return fmt.Errorf("could not advance the chain with a nil block %v", err) } continue } simulatedObjects := sb.generateSimulatedObjects(testCase, i) startTime := time.Now() if err := sb.GenerateBlockAndAdvanceChain(simulatedObjects, privKeys); err != nil { return fmt.Errorf("could not generate the block and advance the chain %v", err) } endTime := time.Now() averageTimesPerTransition = append(averageTimesPerTransition, endTime.Sub(startTime)) } log.Infof( "with %d initial deposits, each state transition took average time = %v", testCase.Config.DepositsForChainStart, averageDuration(averageTimesPerTransition), ) if err := sb.compareTestCase(testCase); err != nil { return err } return nil } // initializeStateTest sets up the environment by generating all the required objects in order // to proceed with the state test. func (sb *SimulatedBackend) initializeStateTest(testCase *StateTestCase) ([]*bls.SecretKey, error) { initialDeposits, privKeys, err := generateInitialSimulatedDeposits(testCase.Config.DepositsForChainStart) if err != nil { return nil, fmt.Errorf("could not simulate initial validator deposits: %v", err) } if err := sb.setupBeaconStateAndGenesisBlock(initialDeposits); err != nil { return nil, fmt.Errorf("could not set up beacon state and initialize genesis block %v", err) } sb.depositTrie = trieutil.NewDepositTrie() return privKeys, nil } // setupBeaconStateAndGenesisBlock creates the initial beacon state and genesis block in order to // proceed with the test. func (sb *SimulatedBackend) setupBeaconStateAndGenesisBlock(initialDeposits []*pb.Deposit) error { var err error genesisTime := time.Date(2018, 9, 0, 0, 0, 0, 0, time.UTC).Unix() sb.state, err = state.GenesisBeaconState(initialDeposits, uint64(genesisTime), nil) if err != nil { return fmt.Errorf("could not initialize simulated beacon state: %v", err) } // We do not expect hashing initial beacon state and genesis block to // fail, so we can safely ignore the error below. // #nosec G104 stateRoot, err := hashutil.HashProto(sb.state) if err != nil { return fmt.Errorf("could not tree hash state: %v", err) } genesisBlock := b.NewGenesisBlock(stateRoot[:]) genesisBlockRoot, err := hashutil.HashBeaconBlock(genesisBlock) if err != nil { return fmt.Errorf("could not tree hash genesis block: %v", err) } // We now keep track of generated blocks for each state transition in // a slice. sb.prevBlockRoots = [][32]byte{genesisBlockRoot} sb.inMemoryBlocks = append(sb.inMemoryBlocks, genesisBlock) return nil } // generateSimulatedObjects generates the simulated objects depending on the testcase and current slot. func (sb *SimulatedBackend) generateSimulatedObjects(testCase *StateTestCase, slotNumber uint64) *SimulatedObjects { // 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 == slotNumber { simulatedDeposit = deposit break } } var simulatedProposerSlashing *StateTestProposerSlashing for _, pSlashing := range testCase.Config.ProposerSlashings { if pSlashing.Slot == slotNumber { simulatedProposerSlashing = pSlashing break } } var simulatedAttesterSlashing *StateTestAttesterSlashing for _, cSlashing := range testCase.Config.AttesterSlashings { if cSlashing.Slot == slotNumber { simulatedAttesterSlashing = cSlashing break } } var simulatedValidatorExit *StateTestValidatorExit for _, exit := range testCase.Config.ValidatorExits { if exit.Epoch == slotNumber/params.BeaconConfig().SlotsPerEpoch { simulatedValidatorExit = exit break } } return &SimulatedObjects{ simDeposit: simulatedDeposit, simProposerSlashing: simulatedProposerSlashing, simAttesterSlashing: simulatedAttesterSlashing, simValidatorExit: simulatedValidatorExit, } } // compareTestCase compares the state in the simulated backend against the values in inputted test case. If // there are any discrepancies it returns an error. func (sb *SimulatedBackend) compareTestCase(testCase *StateTestCase) error { if sb.state.Slot != testCase.Results.Slot { return fmt.Errorf( "incorrect state slot after %d state transitions without blocks, wanted %d, received %d", testCase.Config.NumSlots, sb.state.Slot, testCase.Results.Slot, ) } if len(sb.state.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(sb.state.ValidatorRegistry), ) } for _, slashed := range testCase.Results.SlashedValidators { if sb.state.ValidatorRegistry[slashed].SlashedEpoch == params.BeaconConfig().FarFutureEpoch { return fmt.Errorf( "expected validator at index %d to have been slashed", slashed, ) } } for _, exited := range testCase.Results.ExitedValidators { if sb.state.ValidatorRegistry[exited].StatusFlags != pb.Validator_INITIATED_EXIT { return fmt.Errorf( "expected validator at index %d to have exited", exited, ) } } return nil } func setTestConfig(testCase *StateTestCase) { // We setup the initial configuration for running state // transition tests below. c := params.BeaconConfig() c.SlotsPerEpoch = testCase.Config.SlotsPerEpoch c.DepositsForChainStart = testCase.Config.DepositsForChainStart params.OverrideBeaconConfig(c) } func averageDuration(times []time.Duration) time.Duration { sum := int64(0) for _, t := range times { sum += t.Nanoseconds() } return time.Duration(sum / int64(len(times))) }