mirror of
https://gitlab.com/pulsechaincom/prysm-pulse.git
synced 2025-01-05 09:14:28 +00:00
5a66807989
* First take at updating everything to v5 * Patch gRPC gateway to use prysm v5 Fix patch * Update go ssz --------- Co-authored-by: Preston Van Loon <pvanloon@offchainlabs.com>
180 lines
6.5 KiB
Go
180 lines
6.5 KiB
Go
package stategen
|
|
|
|
import (
|
|
"context"
|
|
"fmt"
|
|
"time"
|
|
|
|
"github.com/pkg/errors"
|
|
"github.com/prysmaticlabs/prysm/v5/beacon-chain/state"
|
|
"github.com/prysmaticlabs/prysm/v5/consensus-types/interfaces"
|
|
"github.com/prysmaticlabs/prysm/v5/consensus-types/primitives"
|
|
"github.com/sirupsen/logrus"
|
|
"go.opencensus.io/trace"
|
|
)
|
|
|
|
var ErrFutureSlotRequested = errors.New("cannot replay to future slots")
|
|
var ErrNoCanonicalBlockForSlot = errors.New("none of the blocks found in the db slot index are canonical")
|
|
var ErrNoBlocksBelowSlot = errors.New("no blocks found in db below slot")
|
|
var ErrReplayTargetSlotExceeded = errors.New("desired replay slot is less than state's slot")
|
|
|
|
type retrievalMethod int
|
|
|
|
const (
|
|
forSlot retrievalMethod = iota
|
|
)
|
|
|
|
// HistoryAccessor describes the minimum set of database methods needed to support the ReplayerBuilder.
|
|
type HistoryAccessor interface {
|
|
HighestRootsBelowSlot(ctx context.Context, slot primitives.Slot) (primitives.Slot, [][32]byte, error)
|
|
GenesisBlockRoot(ctx context.Context) ([32]byte, error)
|
|
Block(ctx context.Context, blockRoot [32]byte) (interfaces.ReadOnlySignedBeaconBlock, error)
|
|
StateOrError(ctx context.Context, blockRoot [32]byte) (state.BeaconState, error)
|
|
}
|
|
|
|
// CanonicalChecker determines whether the given block root is canonical.
|
|
// In practice this should be satisfied by a type that uses the fork choice store.
|
|
type CanonicalChecker interface {
|
|
IsCanonical(ctx context.Context, blockRoot [32]byte) (bool, error)
|
|
}
|
|
|
|
// CurrentSlotter provides the current Slot.
|
|
type CurrentSlotter interface {
|
|
CurrentSlot() primitives.Slot
|
|
}
|
|
|
|
// Replayer encapsulates database query and replay logic. It can be constructed via a ReplayerBuilder.
|
|
type Replayer interface {
|
|
// ReplayBlocks replays the blocks the Replayer knows about based on Builder params
|
|
ReplayBlocks(ctx context.Context) (state.BeaconState, error)
|
|
// ReplayToSlot invokes ReplayBlocks under the hood,
|
|
// but then also runs process_slots to advance the state past the root or slot used in the builder.
|
|
// For example, if you wanted the state to be at the target slot, but only integrating blocks up to
|
|
// slot-1, you could request Builder.ReplayerForSlot(slot-1).ReplayToSlot(slot)
|
|
ReplayToSlot(ctx context.Context, target primitives.Slot) (state.BeaconState, error)
|
|
}
|
|
|
|
var _ Replayer = &stateReplayer{}
|
|
|
|
// chainer is responsible for supplying the chain components necessary to rebuild a state,
|
|
// namely a starting BeaconState and all available blocks from the starting state up to and including the target slot
|
|
type chainer interface {
|
|
chainForSlot(ctx context.Context, target primitives.Slot) (state.BeaconState, []interfaces.ReadOnlySignedBeaconBlock, error)
|
|
}
|
|
|
|
type stateReplayer struct {
|
|
target primitives.Slot
|
|
method retrievalMethod
|
|
chainer chainer
|
|
}
|
|
|
|
// ReplayBlocks applies all the blocks that were accumulated when building the Replayer.
|
|
// This method relies on the correctness of the code that constructed the Replayer data.
|
|
func (rs *stateReplayer) ReplayBlocks(ctx context.Context) (state.BeaconState, error) {
|
|
ctx, span := trace.StartSpan(ctx, "stateGen.stateReplayer.ReplayBlocks")
|
|
defer span.End()
|
|
|
|
var s state.BeaconState
|
|
var descendants []interfaces.ReadOnlySignedBeaconBlock
|
|
var err error
|
|
switch rs.method {
|
|
case forSlot:
|
|
s, descendants, err = rs.chainer.chainForSlot(ctx, rs.target)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
default:
|
|
return nil, errors.New("Replayer initialized using unknown state retrieval method")
|
|
}
|
|
|
|
start := time.Now()
|
|
diff, err := rs.target.SafeSubSlot(s.Slot())
|
|
if err != nil {
|
|
msg := fmt.Sprintf("error subtracting state.slot %d from replay target slot %d", s.Slot(), rs.target)
|
|
return nil, errors.Wrap(err, msg)
|
|
}
|
|
if diff == 0 {
|
|
return s, nil
|
|
}
|
|
|
|
log.WithFields(logrus.Fields{
|
|
"startSlot": s.Slot(),
|
|
"endSlot": rs.target,
|
|
"diff": diff,
|
|
}).Debug("Replaying canonical blocks from most recent state")
|
|
|
|
for _, b := range descendants {
|
|
if ctx.Err() != nil {
|
|
return nil, ctx.Err()
|
|
}
|
|
s, err = executeStateTransitionStateGen(ctx, s, b)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
}
|
|
if rs.target > s.Slot() {
|
|
s, err = ReplayProcessSlots(ctx, s, rs.target)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
}
|
|
|
|
duration := time.Since(start)
|
|
log.WithFields(logrus.Fields{
|
|
"duration": duration,
|
|
}).Debug("Finished calling process_blocks on all blocks in ReplayBlocks")
|
|
replayBlocksSummary.Observe(float64(duration.Milliseconds()))
|
|
return s, nil
|
|
}
|
|
|
|
// ReplayToSlot invokes ReplayBlocks under the hood,
|
|
// but then also runs process_slots to advance the state past the root or slot used in the builder.
|
|
// for example, if you wanted the state to be at the target slot, but only integrating blocks up to
|
|
// slot-1, you could request Builder.ReplayerForSlot(slot-1).ReplayToSlot(slot)
|
|
func (rs *stateReplayer) ReplayToSlot(ctx context.Context, replayTo primitives.Slot) (state.BeaconState, error) {
|
|
ctx, span := trace.StartSpan(ctx, "stateGen.stateReplayer.ReplayToSlot")
|
|
defer span.End()
|
|
|
|
s, err := rs.ReplayBlocks(ctx)
|
|
if err != nil {
|
|
return nil, errors.Wrap(err, "failed to ReplayBlocks")
|
|
}
|
|
if replayTo < s.Slot() {
|
|
return nil, errors.Wrapf(ErrReplayTargetSlotExceeded, "slot desired=%d, state.slot=%d", replayTo, s.Slot())
|
|
}
|
|
if replayTo == s.Slot() {
|
|
return s, nil
|
|
}
|
|
|
|
start := time.Now()
|
|
log.WithFields(logrus.Fields{
|
|
"startSlot": s.Slot(),
|
|
"endSlot": replayTo,
|
|
"diff": replayTo - s.Slot(),
|
|
}).Debug("calling process_slots on remaining slots")
|
|
|
|
// err will be handled after the bookend log
|
|
s, err = ReplayProcessSlots(ctx, s, replayTo)
|
|
if err != nil {
|
|
return nil, errors.Wrap(err, fmt.Sprintf("ReplayToSlot failed to seek to slot %d after applying blocks", replayTo))
|
|
}
|
|
duration := time.Since(start)
|
|
log.WithFields(logrus.Fields{
|
|
"duration": duration,
|
|
}).Debug("time spent in process_slots")
|
|
replayToSlotSummary.Observe(float64(duration.Milliseconds()))
|
|
|
|
return s, nil
|
|
}
|
|
|
|
// ReplayerBuilder creates a Replayer that can be used to obtain a state at a specified slot or root
|
|
// (only ForSlot implemented so far).
|
|
// See documentation on Replayer for more on how to use this to obtain pre/post-block states
|
|
type ReplayerBuilder interface {
|
|
// ReplayerForSlot creates a builder that will create a state that includes blocks up to and including the requested slot
|
|
// The resulting Replayer will always yield a state with .Slot=target; if there are skipped blocks
|
|
// between the highest canonical block in the db and the target, the replayer will fast-forward past the intervening
|
|
// slots via process_slots.
|
|
ReplayerForSlot(target primitives.Slot) Replayer
|
|
}
|