2020-03-14 18:39:23 +00:00
|
|
|
package stategen
|
|
|
|
|
|
|
|
import (
|
|
|
|
"context"
|
|
|
|
|
|
|
|
"github.com/pkg/errors"
|
2021-02-16 07:45:34 +00:00
|
|
|
types "github.com/prysmaticlabs/eth2-types"
|
2021-03-08 22:37:33 +00:00
|
|
|
iface "github.com/prysmaticlabs/prysm/beacon-chain/state/interface"
|
2020-03-30 22:10:45 +00:00
|
|
|
pb "github.com/prysmaticlabs/prysm/proto/beacon/p2p/v1"
|
2020-05-08 19:02:48 +00:00
|
|
|
"github.com/prysmaticlabs/prysm/shared/bytesutil"
|
2020-03-26 19:13:45 +00:00
|
|
|
"github.com/prysmaticlabs/prysm/shared/params"
|
2020-03-14 18:39:23 +00:00
|
|
|
"go.opencensus.io/trace"
|
|
|
|
)
|
|
|
|
|
2020-08-27 22:29:59 +00:00
|
|
|
// HasState returns true if the state exists in cache or in DB.
|
|
|
|
func (s *State) HasState(ctx context.Context, blockRoot [32]byte) (bool, error) {
|
2020-10-31 18:38:01 +00:00
|
|
|
has, err := s.HasStateInCache(ctx, blockRoot)
|
2020-08-27 22:29:59 +00:00
|
|
|
if err != nil {
|
|
|
|
return false, err
|
|
|
|
}
|
|
|
|
if has {
|
|
|
|
return true, nil
|
|
|
|
}
|
|
|
|
return s.beaconDB.HasState(ctx, blockRoot), nil
|
|
|
|
}
|
|
|
|
|
2020-10-31 18:38:01 +00:00
|
|
|
// HasStateInCache returns true if the state exists in cache.
|
|
|
|
func (s *State) HasStateInCache(ctx context.Context, blockRoot [32]byte) (bool, error) {
|
2020-12-17 20:40:47 +00:00
|
|
|
if s.hotStateCache.has(blockRoot) {
|
2020-10-31 18:38:01 +00:00
|
|
|
return true, nil
|
|
|
|
}
|
|
|
|
_, has, err := s.epochBoundaryStateCache.getByRoot(blockRoot)
|
|
|
|
if err != nil {
|
|
|
|
return false, err
|
|
|
|
}
|
|
|
|
return has, nil
|
|
|
|
}
|
|
|
|
|
2020-08-27 22:29:59 +00:00
|
|
|
// StateByRoot retrieves the state using input block root.
|
2021-03-08 22:37:33 +00:00
|
|
|
func (s *State) StateByRoot(ctx context.Context, blockRoot [32]byte) (iface.BeaconState, error) {
|
2020-03-14 18:39:23 +00:00
|
|
|
ctx, span := trace.StartSpan(ctx, "stateGen.StateByRoot")
|
|
|
|
defer span.End()
|
|
|
|
|
2020-07-06 17:22:12 +00:00
|
|
|
// Genesis case. If block root is zero hash, short circuit to use genesis cachedState stored in DB.
|
2020-03-26 19:13:45 +00:00
|
|
|
if blockRoot == params.BeaconConfig().ZeroHash {
|
|
|
|
return s.beaconDB.State(ctx, blockRoot)
|
|
|
|
}
|
2020-08-27 22:29:59 +00:00
|
|
|
return s.loadStateByRoot(ctx, blockRoot)
|
2020-03-14 18:39:23 +00:00
|
|
|
}
|
|
|
|
|
2020-05-08 19:02:48 +00:00
|
|
|
// StateByRootInitialSync retrieves the state from the DB for the initial syncing phase.
|
|
|
|
// It assumes initial syncing using a block list rather than a block tree hence the returned
|
|
|
|
// state is not copied.
|
2020-07-13 03:44:06 +00:00
|
|
|
// It invalidates cache for parent root because pre state will get mutated.
|
2020-05-08 19:02:48 +00:00
|
|
|
// Do not use this method for anything other than initial syncing purpose or block tree is applied.
|
2021-03-08 22:37:33 +00:00
|
|
|
func (s *State) StateByRootInitialSync(ctx context.Context, blockRoot [32]byte) (iface.BeaconState, error) {
|
2020-05-08 19:02:48 +00:00
|
|
|
// Genesis case. If block root is zero hash, short circuit to use genesis state stored in DB.
|
|
|
|
if blockRoot == params.BeaconConfig().ZeroHash {
|
|
|
|
return s.beaconDB.State(ctx, blockRoot)
|
|
|
|
}
|
|
|
|
|
2020-07-23 04:25:50 +00:00
|
|
|
// To invalidate cache for parent root because pre state will get mutated.
|
2020-12-17 20:40:47 +00:00
|
|
|
defer s.hotStateCache.delete(blockRoot)
|
2020-07-23 04:25:50 +00:00
|
|
|
|
2020-12-17 20:40:47 +00:00
|
|
|
if s.hotStateCache.has(blockRoot) {
|
|
|
|
return s.hotStateCache.getWithoutCopy(blockRoot), nil
|
2020-05-08 19:02:48 +00:00
|
|
|
}
|
|
|
|
|
2020-07-06 17:22:12 +00:00
|
|
|
cachedInfo, ok, err := s.epochBoundaryStateCache.getByRoot(blockRoot)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
if ok {
|
|
|
|
return cachedInfo.state, nil
|
2020-05-08 19:02:48 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
startState, err := s.lastAncestorState(ctx, blockRoot)
|
|
|
|
if err != nil {
|
|
|
|
return nil, errors.Wrap(err, "could not get ancestor state")
|
|
|
|
}
|
2021-05-24 04:55:42 +00:00
|
|
|
if startState == nil || startState.IsNil() {
|
2020-05-08 19:02:48 +00:00
|
|
|
return nil, errUnknownState
|
|
|
|
}
|
|
|
|
summary, err := s.stateSummary(ctx, blockRoot)
|
|
|
|
if err != nil {
|
|
|
|
return nil, errors.Wrap(err, "could not get state summary")
|
|
|
|
}
|
|
|
|
if startState.Slot() == summary.Slot {
|
|
|
|
return startState, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
blks, err := s.LoadBlocks(ctx, startState.Slot()+1, summary.Slot, bytesutil.ToBytes32(summary.Root))
|
|
|
|
if err != nil {
|
2020-08-27 22:29:59 +00:00
|
|
|
return nil, errors.Wrap(err, "could not load blocks")
|
2020-05-08 19:02:48 +00:00
|
|
|
}
|
|
|
|
startState, err = s.ReplayBlocks(ctx, startState, blks, summary.Slot)
|
|
|
|
if err != nil {
|
2020-08-27 22:29:59 +00:00
|
|
|
return nil, errors.Wrap(err, "could not replay blocks")
|
2020-05-08 19:02:48 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
return startState, nil
|
|
|
|
}
|
|
|
|
|
2020-08-27 22:29:59 +00:00
|
|
|
// StateBySlot retrieves the state using input slot.
|
2021-03-08 22:37:33 +00:00
|
|
|
func (s *State) StateBySlot(ctx context.Context, slot types.Slot) (iface.BeaconState, error) {
|
2020-03-14 18:39:23 +00:00
|
|
|
ctx, span := trace.StartSpan(ctx, "stateGen.StateBySlot")
|
|
|
|
defer span.End()
|
|
|
|
|
2020-08-27 22:29:59 +00:00
|
|
|
return s.loadStateBySlot(ctx, slot)
|
2020-03-14 18:39:23 +00:00
|
|
|
}
|
2020-03-16 19:07:07 +00:00
|
|
|
|
2020-03-30 22:10:45 +00:00
|
|
|
// This returns the state summary object of a given block root, it first checks the cache
|
|
|
|
// then checks the DB. An error is returned if state summary object is nil.
|
|
|
|
func (s *State) stateSummary(ctx context.Context, blockRoot [32]byte) (*pb.StateSummary, error) {
|
|
|
|
var summary *pb.StateSummary
|
|
|
|
var err error
|
2020-12-16 16:56:21 +00:00
|
|
|
|
|
|
|
summary, err = s.beaconDB.StateSummary(ctx, blockRoot)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
2020-03-30 22:10:45 +00:00
|
|
|
}
|
2020-12-16 16:56:21 +00:00
|
|
|
|
2020-03-30 22:10:45 +00:00
|
|
|
if summary == nil {
|
2020-11-30 02:42:45 +00:00
|
|
|
return s.RecoverStateSummary(ctx, blockRoot)
|
2020-03-30 22:10:45 +00:00
|
|
|
}
|
|
|
|
return summary, nil
|
2020-03-16 19:07:07 +00:00
|
|
|
}
|
2020-04-13 16:32:02 +00:00
|
|
|
|
2020-11-30 02:42:45 +00:00
|
|
|
// RecoverStateSummary recovers state summary object of a given block root by using the saved block in DB.
|
|
|
|
func (s *State) RecoverStateSummary(ctx context.Context, blockRoot [32]byte) (*pb.StateSummary, error) {
|
2020-04-13 16:32:02 +00:00
|
|
|
if s.beaconDB.HasBlock(ctx, blockRoot) {
|
|
|
|
b, err := s.beaconDB.Block(ctx, blockRoot)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
2021-05-26 16:19:54 +00:00
|
|
|
summary := &pb.StateSummary{Slot: b.Block().Slot(), Root: blockRoot[:]}
|
2020-04-13 16:32:02 +00:00
|
|
|
if err := s.beaconDB.SaveStateSummary(ctx, summary); err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
return summary, nil
|
|
|
|
}
|
2020-11-30 02:42:45 +00:00
|
|
|
return nil, errors.New("could not find block in DB")
|
2020-04-13 16:32:02 +00:00
|
|
|
}
|
2020-08-27 22:29:59 +00:00
|
|
|
|
|
|
|
// This loads a beacon state from either the cache or DB then replay blocks up the requested block root.
|
2021-03-08 22:37:33 +00:00
|
|
|
func (s *State) loadStateByRoot(ctx context.Context, blockRoot [32]byte) (iface.BeaconState, error) {
|
2020-08-27 22:29:59 +00:00
|
|
|
ctx, span := trace.StartSpan(ctx, "stateGen.loadStateByRoot")
|
|
|
|
defer span.End()
|
|
|
|
|
|
|
|
// First, it checks if the state exists in hot state cache.
|
2020-12-17 20:40:47 +00:00
|
|
|
cachedState := s.hotStateCache.get(blockRoot)
|
2021-05-24 04:55:42 +00:00
|
|
|
if cachedState != nil && !cachedState.IsNil() {
|
2020-08-27 22:29:59 +00:00
|
|
|
return cachedState, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// Second, it checks if the state exits in epoch boundary state cache.
|
|
|
|
cachedInfo, ok, err := s.epochBoundaryStateCache.getByRoot(blockRoot)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
if ok {
|
|
|
|
return cachedInfo.state, nil
|
|
|
|
}
|
|
|
|
|
2020-10-29 16:14:57 +00:00
|
|
|
// Short cut if the cachedState is already in the DB.
|
|
|
|
if s.beaconDB.HasState(ctx, blockRoot) {
|
|
|
|
return s.beaconDB.State(ctx, blockRoot)
|
|
|
|
}
|
|
|
|
|
2020-08-27 22:29:59 +00:00
|
|
|
summary, err := s.stateSummary(ctx, blockRoot)
|
|
|
|
if err != nil {
|
|
|
|
return nil, errors.Wrap(err, "could not get state summary")
|
|
|
|
}
|
|
|
|
targetSlot := summary.Slot
|
|
|
|
|
|
|
|
// Since the requested state is not in caches, start replaying using the last available ancestor state which is
|
|
|
|
// retrieved using input block's parent root.
|
|
|
|
startState, err := s.lastAncestorState(ctx, blockRoot)
|
|
|
|
if err != nil {
|
|
|
|
return nil, errors.Wrap(err, "could not get ancestor state")
|
|
|
|
}
|
2021-05-24 04:55:42 +00:00
|
|
|
if startState == nil || startState.IsNil() {
|
2020-08-27 22:29:59 +00:00
|
|
|
return nil, errUnknownBoundaryState
|
|
|
|
}
|
|
|
|
|
2020-11-19 03:53:24 +00:00
|
|
|
// Return state early if we are retrieving it from our finalized state cache.
|
|
|
|
if startState.Slot() == targetSlot {
|
|
|
|
return startState, nil
|
|
|
|
}
|
|
|
|
|
2020-08-27 22:29:59 +00:00
|
|
|
blks, err := s.LoadBlocks(ctx, startState.Slot()+1, targetSlot, bytesutil.ToBytes32(summary.Root))
|
|
|
|
if err != nil {
|
|
|
|
return nil, errors.Wrap(err, "could not load blocks for hot state using root")
|
|
|
|
}
|
|
|
|
|
|
|
|
replayBlockCount.Observe(float64(len(blks)))
|
|
|
|
|
|
|
|
return s.ReplayBlocks(ctx, startState, blks, targetSlot)
|
|
|
|
}
|
|
|
|
|
|
|
|
// This loads a state by slot.
|
2021-03-08 22:37:33 +00:00
|
|
|
func (s *State) loadStateBySlot(ctx context.Context, slot types.Slot) (iface.BeaconState, error) {
|
2020-08-27 22:29:59 +00:00
|
|
|
ctx, span := trace.StartSpan(ctx, "stateGen.loadStateBySlot")
|
|
|
|
defer span.End()
|
|
|
|
|
|
|
|
// Return genesis state if slot is 0.
|
|
|
|
if slot == 0 {
|
|
|
|
return s.beaconDB.GenesisState(ctx)
|
|
|
|
}
|
|
|
|
|
|
|
|
// Gather the last saved block root and the slot number.
|
|
|
|
lastValidRoot, lastValidSlot, err := s.lastSavedBlock(ctx, slot)
|
|
|
|
if err != nil {
|
|
|
|
return nil, errors.Wrap(err, "could not get last valid block for hot state using slot")
|
|
|
|
}
|
|
|
|
|
2021-01-07 00:41:44 +00:00
|
|
|
replayStartState, err := s.loadStateByRoot(ctx, lastValidRoot)
|
2020-08-27 22:29:59 +00:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
2021-01-07 00:41:44 +00:00
|
|
|
if lastValidSlot < slot {
|
|
|
|
replayStartState, err = processSlotsStateGen(ctx, replayStartState, slot)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
2020-09-23 21:39:16 +00:00
|
|
|
}
|
2021-01-07 00:41:44 +00:00
|
|
|
return replayStartState, nil
|
2020-08-27 22:29:59 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// This returns the highest available ancestor state of the input block root.
|
|
|
|
// It recursively look up block's parent until a corresponding state of the block root
|
|
|
|
// is found in the caches or DB.
|
|
|
|
//
|
|
|
|
// There's three ways to derive block parent state:
|
|
|
|
// 1.) block parent state is the last finalized state
|
|
|
|
// 2.) block parent state is the epoch boundary state and exists in epoch boundary cache.
|
|
|
|
// 3.) block parent state is in DB.
|
2021-03-08 22:37:33 +00:00
|
|
|
func (s *State) lastAncestorState(ctx context.Context, root [32]byte) (iface.BeaconState, error) {
|
2020-08-27 22:29:59 +00:00
|
|
|
ctx, span := trace.StartSpan(ctx, "stateGen.lastAncestorState")
|
|
|
|
defer span.End()
|
|
|
|
|
|
|
|
if s.isFinalizedRoot(root) && s.finalizedState() != nil {
|
|
|
|
return s.finalizedState(), nil
|
|
|
|
}
|
|
|
|
|
|
|
|
b, err := s.beaconDB.Block(ctx, root)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
2021-05-26 16:19:54 +00:00
|
|
|
if b == nil || b.IsNil() {
|
2020-08-27 22:29:59 +00:00
|
|
|
return nil, errUnknownBlock
|
|
|
|
}
|
|
|
|
|
|
|
|
for {
|
|
|
|
if ctx.Err() != nil {
|
|
|
|
return nil, ctx.Err()
|
|
|
|
}
|
|
|
|
// Is the state a genesis state.
|
2021-05-26 16:19:54 +00:00
|
|
|
parentRoot := bytesutil.ToBytes32(b.Block().ParentRoot())
|
2020-08-27 22:29:59 +00:00
|
|
|
if parentRoot == params.BeaconConfig().ZeroHash {
|
|
|
|
return s.beaconDB.GenesisState(ctx)
|
|
|
|
}
|
|
|
|
|
|
|
|
// Does the state exist in the hot state cache.
|
2020-12-17 20:40:47 +00:00
|
|
|
if s.hotStateCache.has(parentRoot) {
|
|
|
|
return s.hotStateCache.get(parentRoot), nil
|
2020-08-27 22:29:59 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// Does the state exist in finalized info cache.
|
|
|
|
if s.isFinalizedRoot(parentRoot) {
|
|
|
|
return s.finalizedState(), nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// Does the state exist in epoch boundary cache.
|
|
|
|
cachedInfo, ok, err := s.epochBoundaryStateCache.getByRoot(parentRoot)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
if ok {
|
|
|
|
return cachedInfo.state, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// Does the state exists in DB.
|
|
|
|
if s.beaconDB.HasState(ctx, parentRoot) {
|
|
|
|
return s.beaconDB.State(ctx, parentRoot)
|
|
|
|
}
|
|
|
|
b, err = s.beaconDB.Block(ctx, parentRoot)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
2021-05-26 16:19:54 +00:00
|
|
|
if b == nil || b.IsNil() {
|
2020-08-27 22:29:59 +00:00
|
|
|
return nil, errUnknownBlock
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|