2022-03-28 21:01:55 +00:00
|
|
|
package backfill
|
|
|
|
|
|
|
|
import (
|
|
|
|
"context"
|
|
|
|
|
|
|
|
"github.com/pkg/errors"
|
2022-08-16 12:20:13 +00:00
|
|
|
"github.com/prysmaticlabs/prysm/v3/beacon-chain/db"
|
|
|
|
"github.com/prysmaticlabs/prysm/v3/consensus-types/blocks"
|
|
|
|
"github.com/prysmaticlabs/prysm/v3/consensus-types/interfaces"
|
2023-01-26 14:40:12 +00:00
|
|
|
"github.com/prysmaticlabs/prysm/v3/consensus-types/primitives"
|
2022-03-28 21:01:55 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
// NewStatus correctly initializes a Status value with the required database value.
|
|
|
|
func NewStatus(store BackfillDB) *Status {
|
|
|
|
return &Status{
|
|
|
|
store: store,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Status provides a way to update and query the status of a backfill process that may be necessary to track when
|
|
|
|
// a node was initialized via checkpoint sync. With checkpoint sync, there will be a gap in node history from genesis
|
|
|
|
// until the checkpoint sync origin block. Status provides the means to update the value keeping track of the lower
|
|
|
|
// end of the missing block range via the Advance() method, to check whether a Slot is missing from the database
|
|
|
|
// via the SlotCovered() method, and to see the current StartGap() and EndGap().
|
|
|
|
type Status struct {
|
2023-01-26 14:40:12 +00:00
|
|
|
start primitives.Slot
|
|
|
|
end primitives.Slot
|
2022-03-28 21:01:55 +00:00
|
|
|
store BackfillDB
|
|
|
|
genesisSync bool
|
|
|
|
}
|
|
|
|
|
|
|
|
// SlotCovered uses StartGap() and EndGap() to determine if the given slot is covered by the current chain history.
|
|
|
|
// If the slot is <= StartGap(), or >= EndGap(), the result is true.
|
|
|
|
// If the slot is between StartGap() and EndGap(), the result is false.
|
2023-01-26 14:40:12 +00:00
|
|
|
func (s *Status) SlotCovered(sl primitives.Slot) bool {
|
2022-03-28 21:01:55 +00:00
|
|
|
// short circuit if the node was synced from genesis
|
|
|
|
if s.genesisSync {
|
|
|
|
return true
|
|
|
|
}
|
|
|
|
if s.StartGap() < sl && sl < s.EndGap() {
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
return true
|
|
|
|
}
|
|
|
|
|
|
|
|
// StartGap returns the slot at the beginning of the range that needs to be backfilled.
|
2023-01-26 14:40:12 +00:00
|
|
|
func (s *Status) StartGap() primitives.Slot {
|
2022-03-28 21:01:55 +00:00
|
|
|
return s.start
|
|
|
|
}
|
|
|
|
|
|
|
|
// EndGap returns the slot at the end of the range that needs to be backfilled.
|
2023-01-26 14:40:12 +00:00
|
|
|
func (s *Status) EndGap() primitives.Slot {
|
2022-03-28 21:01:55 +00:00
|
|
|
return s.end
|
|
|
|
}
|
|
|
|
|
|
|
|
var ErrAdvancePastOrigin = errors.New("cannot advance backfill Status beyond the origin checkpoint slot")
|
|
|
|
|
|
|
|
// Advance advances the backfill position to the given slot & root.
|
|
|
|
// It updates the backfill block root entry in the database,
|
|
|
|
// and also updates the Status value's copy of the backfill position slot.
|
2023-01-26 14:40:12 +00:00
|
|
|
func (s *Status) Advance(ctx context.Context, upTo primitives.Slot, root [32]byte) error {
|
2022-03-28 21:01:55 +00:00
|
|
|
if upTo > s.end {
|
|
|
|
return errors.Wrapf(ErrAdvancePastOrigin, "advance slot=%d, origin slot=%d", upTo, s.end)
|
|
|
|
}
|
|
|
|
s.start = upTo
|
|
|
|
return s.store.SaveBackfillBlockRoot(ctx, root)
|
|
|
|
}
|
|
|
|
|
|
|
|
// Reload queries the database for backfill status, initializing the internal data and validating the database state.
|
|
|
|
func (s *Status) Reload(ctx context.Context) error {
|
|
|
|
cpRoot, err := s.store.OriginCheckpointBlockRoot(ctx)
|
|
|
|
if err != nil {
|
|
|
|
// mark genesis sync and short circuit further lookups
|
|
|
|
if errors.Is(err, db.ErrNotFoundOriginBlockRoot) {
|
|
|
|
s.genesisSync = true
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
cpBlock, err := s.store.Block(ctx, cpRoot)
|
|
|
|
if err != nil {
|
|
|
|
return errors.Wrapf(err, "error retrieving block for origin checkpoint root=%#x", cpRoot)
|
|
|
|
}
|
2022-08-02 15:30:46 +00:00
|
|
|
if err := blocks.BeaconBlockIsNil(cpBlock); err != nil {
|
2022-03-28 21:01:55 +00:00
|
|
|
return err
|
|
|
|
}
|
|
|
|
s.end = cpBlock.Block().Slot()
|
|
|
|
|
|
|
|
_, err = s.store.GenesisBlockRoot(ctx)
|
|
|
|
if err != nil {
|
|
|
|
if errors.Is(err, db.ErrNotFoundGenesisBlockRoot) {
|
|
|
|
return errors.Wrap(err, "genesis block root required for checkpoint sync")
|
|
|
|
}
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
bfRoot, err := s.store.BackfillBlockRoot(ctx)
|
|
|
|
if err != nil {
|
|
|
|
if errors.Is(err, db.ErrNotFoundBackfillBlockRoot) {
|
|
|
|
return errors.Wrap(err, "found origin checkpoint block root, but no backfill block root")
|
|
|
|
}
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
bfBlock, err := s.store.Block(ctx, bfRoot)
|
|
|
|
if err != nil {
|
|
|
|
return errors.Wrapf(err, "error retrieving block for backfill root=%#x", bfRoot)
|
|
|
|
}
|
2022-08-02 15:30:46 +00:00
|
|
|
if err := blocks.BeaconBlockIsNil(bfBlock); err != nil {
|
2022-03-28 21:01:55 +00:00
|
|
|
return err
|
|
|
|
}
|
|
|
|
s.start = bfBlock.Block().Slot()
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// BackfillDB describes the set of DB methods that the Status type needs to function.
|
|
|
|
type BackfillDB interface {
|
|
|
|
SaveBackfillBlockRoot(ctx context.Context, blockRoot [32]byte) error
|
|
|
|
GenesisBlockRoot(ctx context.Context) ([32]byte, error)
|
|
|
|
OriginCheckpointBlockRoot(ctx context.Context) ([32]byte, error)
|
|
|
|
BackfillBlockRoot(ctx context.Context) ([32]byte, error)
|
2022-05-02 18:32:37 +00:00
|
|
|
Block(ctx context.Context, blockRoot [32]byte) (interfaces.SignedBeaconBlock, error)
|
2022-03-28 21:01:55 +00:00
|
|
|
}
|