mirror of
https://gitlab.com/pulsechaincom/prysm-pulse.git
synced 2025-01-15 06:28:20 +00:00
1fa864cb1a
* adding splitRoots, refactor to use it * use splitRoots & work in roots only the most common use case for this method is to get a list of candidate roots and check if they are canonical. there isn't a great reason to look up all the non-canonical blocks, because forkchoice checks based on the root only, so just return roots and defer the responsibility of resolving those to full blocks. * update comment * clean up shadowing * more clear non-error return * add test case for single root in index slot * fmt Co-authored-by: Kasey Kirkham <kasey@users.noreply.github.com> Co-authored-by: prylabs-bulldozer[bot] <58059840+prylabs-bulldozer[bot]@users.noreply.github.com>
196 lines
7.1 KiB
Go
196 lines
7.1 KiB
Go
package stategen
|
|
|
|
import (
|
|
"context"
|
|
"fmt"
|
|
|
|
"github.com/pkg/errors"
|
|
"github.com/prysmaticlabs/prysm/beacon-chain/db"
|
|
"github.com/prysmaticlabs/prysm/beacon-chain/state"
|
|
"github.com/prysmaticlabs/prysm/config/params"
|
|
"github.com/prysmaticlabs/prysm/consensus-types/interfaces"
|
|
types "github.com/prysmaticlabs/prysm/consensus-types/primitives"
|
|
"github.com/prysmaticlabs/prysm/consensus-types/wrapper"
|
|
"github.com/prysmaticlabs/prysm/encoding/bytesutil"
|
|
"go.opencensus.io/trace"
|
|
)
|
|
|
|
func WithCache(c CachedGetter) CanonicalHistoryOption {
|
|
return func(h *CanonicalHistory) {
|
|
h.cache = c
|
|
}
|
|
}
|
|
|
|
type CanonicalHistoryOption func(*CanonicalHistory)
|
|
|
|
func NewCanonicalHistory(h HistoryAccessor, cc CanonicalChecker, cs CurrentSlotter, opts ...CanonicalHistoryOption) *CanonicalHistory {
|
|
ch := &CanonicalHistory{
|
|
h: h,
|
|
cc: cc,
|
|
cs: cs,
|
|
}
|
|
for _, o := range opts {
|
|
o(ch)
|
|
}
|
|
return ch
|
|
}
|
|
|
|
type CanonicalHistory struct {
|
|
h HistoryAccessor
|
|
cc CanonicalChecker
|
|
cs CurrentSlotter
|
|
cache CachedGetter
|
|
}
|
|
|
|
func (c *CanonicalHistory) ReplayerForSlot(target types.Slot) Replayer {
|
|
return &stateReplayer{chainer: c, method: forSlot, target: target}
|
|
}
|
|
|
|
func (c *CanonicalHistory) BlockRootForSlot(ctx context.Context, target types.Slot) ([32]byte, error) {
|
|
if currentSlot := c.cs.CurrentSlot(); target > currentSlot {
|
|
return [32]byte{}, errors.Wrap(ErrFutureSlotRequested, fmt.Sprintf("requested=%d, current=%d", target, currentSlot))
|
|
}
|
|
|
|
slotAbove := target + 1
|
|
// don't bother searching for candidate roots when we know the target slot is genesis
|
|
for slotAbove > 1 {
|
|
if ctx.Err() != nil {
|
|
return [32]byte{}, errors.Wrap(ctx.Err(), "context canceled during canonicalBlockForSlot")
|
|
}
|
|
slot, roots, err := c.h.HighestRootsBelowSlot(ctx, slotAbove)
|
|
if err != nil {
|
|
return [32]byte{}, errors.Wrap(err, fmt.Sprintf("error finding highest block w/ slot < %d", slotAbove))
|
|
}
|
|
if len(roots) == 0 {
|
|
return [32]byte{}, errors.Wrap(ErrNoBlocksBelowSlot, fmt.Sprintf("slot=%d", slotAbove))
|
|
}
|
|
r, err := c.bestForSlot(ctx, roots)
|
|
if err == nil {
|
|
// we found a valid, canonical block!
|
|
return r, nil
|
|
}
|
|
|
|
// we found a block, but it wasn't considered canonical - keep looking
|
|
if errors.Is(err, ErrNoCanonicalBlockForSlot) {
|
|
// break once we've seen slot 0 (and prevent underflow)
|
|
if slot == params.BeaconConfig().GenesisSlot {
|
|
break
|
|
}
|
|
slotAbove = slot
|
|
continue
|
|
}
|
|
return [32]byte{}, err
|
|
}
|
|
|
|
return c.h.GenesisBlockRoot(ctx)
|
|
}
|
|
|
|
// bestForSlot encapsulates several messy realities of the underlying db code, looping through multiple blocks,
|
|
// performing null/validity checks, and using CanonicalChecker to only pick canonical blocks.
|
|
func (c *CanonicalHistory) bestForSlot(ctx context.Context, roots [][32]byte) ([32]byte, error) {
|
|
for _, root := range roots {
|
|
canon, err := c.cc.IsCanonical(ctx, root)
|
|
if err != nil {
|
|
return [32]byte{}, errors.Wrap(err, "replayer could not check if block is canonical")
|
|
}
|
|
if canon {
|
|
return root, nil
|
|
}
|
|
}
|
|
return [32]byte{}, errors.Wrap(ErrNoCanonicalBlockForSlot, "no good block for slot")
|
|
}
|
|
|
|
// ChainForSlot creates a value that satisfies the Replayer interface via db queries
|
|
// and the stategen transition helper methods. This implementation uses the following algorithm:
|
|
// - find the highest canonical block <= the target slot
|
|
// - starting with this block, recursively search backwards for a stored state, and accumulate intervening blocks
|
|
func (c *CanonicalHistory) chainForSlot(ctx context.Context, target types.Slot) (state.BeaconState, []interfaces.SignedBeaconBlock, error) {
|
|
ctx, span := trace.StartSpan(ctx, "canonicalChainer.chainForSlot")
|
|
defer span.End()
|
|
r, err := c.BlockRootForSlot(ctx, target)
|
|
if err != nil {
|
|
return nil, nil, errors.Wrapf(err, "no canonical block root found below slot=%d", target)
|
|
}
|
|
b, err := c.h.Block(ctx, r)
|
|
if err != nil {
|
|
return nil, nil, errors.Wrapf(err, "unable to retrieve canonical block for slot, root=%#x", r)
|
|
}
|
|
s, descendants, err := c.ancestorChain(ctx, b)
|
|
if err != nil {
|
|
return nil, nil, errors.Wrap(err, "failed to query for ancestor and descendant blocks")
|
|
}
|
|
|
|
return s, descendants, nil
|
|
}
|
|
|
|
func (c *CanonicalHistory) getState(ctx context.Context, blockRoot [32]byte) (state.BeaconState, error) {
|
|
if c.cache != nil {
|
|
st, err := c.cache.ByBlockRoot(blockRoot)
|
|
if err == nil {
|
|
return st, nil
|
|
}
|
|
if !errors.Is(err, ErrNotInCache) {
|
|
return nil, errors.Wrap(err, "error reading from state cache during state replay")
|
|
}
|
|
}
|
|
return c.h.StateOrError(ctx, blockRoot)
|
|
}
|
|
|
|
// ancestorChain works backwards through the chain lineage, accumulating blocks and checking for a saved state.
|
|
// If it finds a saved state that the tail block was descended from, it returns this state and
|
|
// all blocks in the lineage, including the tail block. Blocks are returned in ascending order.
|
|
// Note that this function assumes that the tail is a canonical block, and therefore assumes that
|
|
// all ancestors are also canonical.
|
|
func (c *CanonicalHistory) ancestorChain(ctx context.Context, tail interfaces.SignedBeaconBlock) (state.BeaconState, []interfaces.SignedBeaconBlock, error) {
|
|
ctx, span := trace.StartSpan(ctx, "canonicalChainer.ancestorChain")
|
|
defer span.End()
|
|
chain := make([]interfaces.SignedBeaconBlock, 0)
|
|
for {
|
|
if err := ctx.Err(); err != nil {
|
|
msg := fmt.Sprintf("context canceled while finding ancestors of block at slot %d", tail.Block().Slot())
|
|
return nil, nil, errors.Wrap(err, msg)
|
|
}
|
|
b := tail.Block()
|
|
// compute hash_tree_root of current block and try to look up the corresponding state
|
|
root, err := b.HashTreeRoot()
|
|
if err != nil {
|
|
msg := fmt.Sprintf("could not compute htr for descendant block at slot=%d", b.Slot())
|
|
return nil, nil, errors.Wrap(err, msg)
|
|
}
|
|
st, err := c.getState(ctx, root)
|
|
// err == nil, we've got a real state - the job is done!
|
|
// Note: in cases where there are skipped slots we could find a state that is a descendant
|
|
// of the block we are searching for. We don't want to return a future block, so in this case
|
|
// we keep working backwards.
|
|
if err == nil && st.Slot() == b.Slot() {
|
|
// we found the state by the root of the head, meaning it has already been applied.
|
|
// we only want to return the blocks descended from it.
|
|
reverseChain(chain)
|
|
return st, chain, nil
|
|
}
|
|
// ErrNotFoundState errors are fine, but other errors mean something is wrong with the db
|
|
if err != nil && !errors.Is(err, db.ErrNotFoundState) {
|
|
return nil, nil, errors.Wrap(err, fmt.Sprintf("error querying database for state w/ block root = %#x", root))
|
|
}
|
|
parent, err := c.h.Block(ctx, bytesutil.ToBytes32(b.ParentRoot()))
|
|
if err != nil {
|
|
msg := fmt.Sprintf("db error when retrieving parent of block at slot=%d by root=%#x", b.Slot(), b.ParentRoot())
|
|
return nil, nil, errors.Wrap(err, msg)
|
|
}
|
|
if wrapper.BeaconBlockIsNil(parent) != nil {
|
|
msg := fmt.Sprintf("unable to retrieve parent of block at slot=%d by root=%#x", b.Slot(), b.ParentRoot())
|
|
return nil, nil, errors.Wrap(db.ErrNotFound, msg)
|
|
}
|
|
chain = append(chain, tail)
|
|
tail = parent
|
|
}
|
|
}
|
|
|
|
func reverseChain(c []interfaces.SignedBeaconBlock) {
|
|
last := len(c) - 1
|
|
swaps := (last + 1) / 2
|
|
for i := 0; i < swaps; i++ {
|
|
c[i], c[last-i] = c[last-i], c[i]
|
|
}
|
|
}
|