2020-02-08 02:05:43 +00:00
|
|
|
package blockchain
|
|
|
|
|
|
|
|
import (
|
2020-07-13 03:44:06 +00:00
|
|
|
"bytes"
|
2020-02-08 02:05:43 +00:00
|
|
|
"context"
|
2020-05-06 19:15:31 +00:00
|
|
|
"fmt"
|
2020-02-08 02:05:43 +00:00
|
|
|
|
|
|
|
"github.com/pkg/errors"
|
2022-08-16 12:20:13 +00:00
|
|
|
"github.com/prysmaticlabs/prysm/v3/beacon-chain/core/feed"
|
|
|
|
statefeed "github.com/prysmaticlabs/prysm/v3/beacon-chain/core/feed/state"
|
|
|
|
"github.com/prysmaticlabs/prysm/v3/beacon-chain/core/helpers"
|
|
|
|
"github.com/prysmaticlabs/prysm/v3/beacon-chain/forkchoice"
|
|
|
|
"github.com/prysmaticlabs/prysm/v3/beacon-chain/state"
|
|
|
|
fieldparams "github.com/prysmaticlabs/prysm/v3/config/fieldparams"
|
|
|
|
"github.com/prysmaticlabs/prysm/v3/config/params"
|
|
|
|
"github.com/prysmaticlabs/prysm/v3/consensus-types/blocks"
|
|
|
|
"github.com/prysmaticlabs/prysm/v3/consensus-types/interfaces"
|
|
|
|
types "github.com/prysmaticlabs/prysm/v3/consensus-types/primitives"
|
|
|
|
"github.com/prysmaticlabs/prysm/v3/encoding/bytesutil"
|
2022-08-25 23:59:08 +00:00
|
|
|
"github.com/prysmaticlabs/prysm/v3/math"
|
2022-08-16 12:20:13 +00:00
|
|
|
ethpbv1 "github.com/prysmaticlabs/prysm/v3/proto/eth/v1"
|
|
|
|
"github.com/prysmaticlabs/prysm/v3/time/slots"
|
2020-05-06 19:15:31 +00:00
|
|
|
"github.com/sirupsen/logrus"
|
2020-02-08 02:05:43 +00:00
|
|
|
"go.opencensus.io/trace"
|
|
|
|
)
|
|
|
|
|
2022-04-01 00:08:01 +00:00
|
|
|
// UpdateAndSaveHeadWithBalances updates the beacon state head after getting justified balanced from cache.
|
|
|
|
// This function is only used in spec-tests, it does save the head after updating it.
|
|
|
|
func (s *Service) UpdateAndSaveHeadWithBalances(ctx context.Context) error {
|
2022-06-25 03:57:52 +00:00
|
|
|
jp := s.CurrentJustifiedCheckpt()
|
2022-05-22 18:37:01 +00:00
|
|
|
|
|
|
|
balances, err := s.justifiedBalances.get(ctx, bytesutil.ToBytes32(jp.Root))
|
2022-01-31 19:03:48 +00:00
|
|
|
if err != nil {
|
2022-05-22 18:37:01 +00:00
|
|
|
msg := fmt.Sprintf("could not read balances for state w/ justified checkpoint %#x", jp.Root)
|
2022-01-31 19:03:48 +00:00
|
|
|
return errors.Wrap(err, msg)
|
|
|
|
}
|
2022-06-16 18:21:40 +00:00
|
|
|
headRoot, err := s.cfg.ForkChoiceStore.Head(ctx, balances)
|
2022-04-01 00:08:01 +00:00
|
|
|
if err != nil {
|
|
|
|
return errors.Wrap(err, "could not update head")
|
|
|
|
}
|
2022-05-15 02:23:01 +00:00
|
|
|
headBlock, err := s.getBlock(ctx, headRoot)
|
2022-04-06 11:05:53 +00:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
2022-04-06 23:36:52 +00:00
|
|
|
headState, err := s.cfg.StateGen.StateByRoot(ctx, headRoot)
|
|
|
|
if err != nil {
|
|
|
|
return errors.Wrap(err, "could not retrieve head state in DB")
|
|
|
|
}
|
|
|
|
return s.saveHead(ctx, headRoot, headBlock, headState)
|
2022-01-31 19:03:48 +00:00
|
|
|
}
|
|
|
|
|
2020-02-15 18:57:49 +00:00
|
|
|
// This defines the current chain service's view of head.
|
|
|
|
type head struct {
|
2022-05-02 18:32:37 +00:00
|
|
|
slot types.Slot // current head slot.
|
|
|
|
root [32]byte // current head root.
|
|
|
|
block interfaces.SignedBeaconBlock // current head block.
|
|
|
|
state state.BeaconState // current head state.
|
2020-02-15 18:57:49 +00:00
|
|
|
}
|
|
|
|
|
2020-02-08 02:05:43 +00:00
|
|
|
// This saves head info to the local service cache, it also saves the
|
|
|
|
// new head root to the DB.
|
2022-05-29 19:32:42 +00:00
|
|
|
func (s *Service) saveHead(ctx context.Context, newHeadRoot [32]byte, headBlock interfaces.SignedBeaconBlock, headState state.BeaconState) error {
|
2020-07-09 23:50:48 +00:00
|
|
|
ctx, span := trace.StartSpan(ctx, "blockChain.saveHead")
|
2020-02-08 02:05:43 +00:00
|
|
|
defer span.End()
|
|
|
|
|
|
|
|
// Do nothing if head hasn't changed.
|
2022-06-25 03:57:52 +00:00
|
|
|
var oldHeadRoot [32]byte
|
|
|
|
s.headLock.RLock()
|
|
|
|
if s.head == nil {
|
|
|
|
oldHeadRoot = s.originBlockRoot
|
|
|
|
} else {
|
|
|
|
oldHeadRoot = s.head.root
|
2020-10-06 10:57:18 +00:00
|
|
|
}
|
2022-06-25 03:57:52 +00:00
|
|
|
s.headLock.RUnlock()
|
|
|
|
if newHeadRoot == oldHeadRoot {
|
2020-02-08 02:05:43 +00:00
|
|
|
return nil
|
|
|
|
}
|
2022-08-02 15:30:46 +00:00
|
|
|
if err := blocks.BeaconBlockIsNil(headBlock); err != nil {
|
2022-04-06 11:05:53 +00:00
|
|
|
return err
|
|
|
|
}
|
2022-04-06 23:36:52 +00:00
|
|
|
if headState == nil || headState.IsNil() {
|
|
|
|
return errors.New("cannot save nil head state")
|
|
|
|
}
|
2020-02-08 02:05:43 +00:00
|
|
|
|
2020-02-09 16:12:06 +00:00
|
|
|
// If the head state is not available, just return nil.
|
|
|
|
// There's nothing to cache
|
2022-05-29 19:32:42 +00:00
|
|
|
if !s.cfg.BeaconDB.HasStateSummary(ctx, newHeadRoot) {
|
2020-06-23 00:19:33 +00:00
|
|
|
return nil
|
2020-02-09 16:12:06 +00:00
|
|
|
}
|
|
|
|
|
2022-05-27 14:24:43 +00:00
|
|
|
s.headLock.RLock()
|
2022-08-02 15:30:46 +00:00
|
|
|
oldHeadBlock, err := s.headBlock()
|
|
|
|
if err != nil {
|
|
|
|
return errors.Wrap(err, "could not get old head block")
|
|
|
|
}
|
|
|
|
oldStateRoot := oldHeadBlock.Block().StateRoot()
|
2022-05-27 14:24:43 +00:00
|
|
|
s.headLock.RUnlock()
|
|
|
|
headSlot := s.HeadSlot()
|
|
|
|
newHeadSlot := headBlock.Block().Slot()
|
2022-04-06 11:05:53 +00:00
|
|
|
newStateRoot := headBlock.Block().StateRoot()
|
2022-08-18 16:53:17 +00:00
|
|
|
|
|
|
|
// A chain re-org occurred, so we fire an event notifying the rest of the services.
|
2022-09-06 14:30:16 +00:00
|
|
|
if headBlock.Block().ParentRoot() != oldHeadRoot {
|
2022-08-25 23:59:08 +00:00
|
|
|
commonRoot, forkSlot, err := s.ForkChoicer().CommonAncestor(ctx, oldHeadRoot, newHeadRoot)
|
2022-08-18 16:53:17 +00:00
|
|
|
if err != nil {
|
|
|
|
log.WithError(err).Error("Could not find common ancestor root")
|
|
|
|
commonRoot = params.BeaconConfig().ZeroHash
|
|
|
|
}
|
2020-05-06 19:15:31 +00:00
|
|
|
log.WithFields(logrus.Fields{
|
2022-08-18 16:53:17 +00:00
|
|
|
"newSlot": fmt.Sprintf("%d", newHeadSlot),
|
|
|
|
"newRoot": fmt.Sprintf("%#x", newHeadRoot),
|
|
|
|
"oldSlot": fmt.Sprintf("%d", headSlot),
|
|
|
|
"oldRoot": fmt.Sprintf("%#x", oldHeadRoot),
|
|
|
|
"commonAncestorRoot": fmt.Sprintf("%#x", commonRoot),
|
2022-08-25 23:59:08 +00:00
|
|
|
"distance": headSlot + newHeadSlot - 2*forkSlot,
|
|
|
|
"depth": math.Max(uint64(headSlot-forkSlot), uint64(newHeadSlot-forkSlot)),
|
2022-08-18 16:53:17 +00:00
|
|
|
}).Info("Chain reorg occurred")
|
2022-03-25 16:31:50 +00:00
|
|
|
isOptimistic, err := s.IsOptimistic(ctx)
|
|
|
|
if err != nil {
|
|
|
|
return errors.Wrap(err, "could not check if node is optimistically synced")
|
|
|
|
}
|
2021-03-17 18:36:56 +00:00
|
|
|
s.cfg.StateNotifier.StateFeed().Send(&feed.Event{
|
2020-05-06 19:15:31 +00:00
|
|
|
Type: statefeed.Reorg,
|
2021-06-10 21:13:15 +00:00
|
|
|
Data: ðpbv1.EventChainReorg{
|
2022-03-25 16:31:50 +00:00
|
|
|
Slot: newHeadSlot,
|
2022-08-25 23:59:08 +00:00
|
|
|
Depth: math.Max(uint64(headSlot-forkSlot), uint64(newHeadSlot-forkSlot)),
|
2022-03-25 16:31:50 +00:00
|
|
|
OldHeadBlock: oldHeadRoot[:],
|
2022-05-29 19:32:42 +00:00
|
|
|
NewHeadBlock: newHeadRoot[:],
|
2022-09-06 14:30:16 +00:00
|
|
|
OldHeadState: oldStateRoot[:],
|
|
|
|
NewHeadState: newStateRoot[:],
|
2022-03-25 16:31:50 +00:00
|
|
|
Epoch: slots.ToEpoch(newHeadSlot),
|
|
|
|
ExecutionOptimistic: isOptimistic,
|
2020-05-06 19:15:31 +00:00
|
|
|
},
|
|
|
|
})
|
2020-05-12 05:48:16 +00:00
|
|
|
|
2022-06-25 03:57:52 +00:00
|
|
|
if err := s.saveOrphanedAtts(ctx, oldHeadRoot, newHeadRoot); err != nil {
|
2021-08-24 03:05:08 +00:00
|
|
|
return err
|
|
|
|
}
|
2020-05-12 05:48:16 +00:00
|
|
|
reorgCount.Inc()
|
2020-05-06 19:15:31 +00:00
|
|
|
}
|
|
|
|
|
2020-02-08 02:05:43 +00:00
|
|
|
// Cache the new head info.
|
2022-08-02 15:30:46 +00:00
|
|
|
if err := s.setHead(newHeadRoot, headBlock, headState); err != nil {
|
|
|
|
return errors.Wrap(err, "could not set head")
|
|
|
|
}
|
2020-02-08 02:05:43 +00:00
|
|
|
|
|
|
|
// Save the new head root to DB.
|
2022-05-29 19:32:42 +00:00
|
|
|
if err := s.cfg.BeaconDB.SaveHeadBlockRoot(ctx, newHeadRoot); err != nil {
|
2020-02-08 02:05:43 +00:00
|
|
|
return errors.Wrap(err, "could not save head root in DB")
|
|
|
|
}
|
|
|
|
|
2021-06-10 21:13:15 +00:00
|
|
|
// Forward an event capturing a new chain head over a common event feed
|
|
|
|
// done in a goroutine to avoid blocking the critical runtime main routine.
|
|
|
|
go func() {
|
2022-09-06 14:30:16 +00:00
|
|
|
if err := s.notifyNewHeadEvent(ctx, newHeadSlot, headState, newStateRoot[:], newHeadRoot[:]); err != nil {
|
2021-06-10 21:13:15 +00:00
|
|
|
log.WithError(err).Error("Could not notify event feed of new chain head")
|
|
|
|
}
|
|
|
|
}()
|
|
|
|
|
2020-02-08 02:05:43 +00:00
|
|
|
return nil
|
|
|
|
}
|
2020-02-15 18:57:49 +00:00
|
|
|
|
|
|
|
// This gets called to update canonical root mapping. It does not save head block
|
2020-06-23 21:11:20 +00:00
|
|
|
// root in DB. With the inception of initial-sync-cache-state flag, it uses finalized
|
2020-02-15 18:57:49 +00:00
|
|
|
// check point as anchors to resume sync therefore head is no longer needed to be saved on per slot basis.
|
2022-05-02 18:32:37 +00:00
|
|
|
func (s *Service) saveHeadNoDB(ctx context.Context, b interfaces.SignedBeaconBlock, r [32]byte, hs state.BeaconState) error {
|
2022-08-02 15:30:46 +00:00
|
|
|
if err := blocks.BeaconBlockIsNil(b); err != nil {
|
2021-02-15 15:11:25 +00:00
|
|
|
return err
|
|
|
|
}
|
2020-07-13 03:44:06 +00:00
|
|
|
cachedHeadRoot, err := s.HeadRoot(ctx)
|
|
|
|
if err != nil {
|
|
|
|
return errors.Wrap(err, "could not get head root from cache")
|
|
|
|
}
|
|
|
|
if bytes.Equal(r[:], cachedHeadRoot) {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2022-08-02 15:30:46 +00:00
|
|
|
bCp, err := b.Copy()
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
if err := s.setHeadInitialSync(r, bCp, hs); err != nil {
|
|
|
|
return errors.Wrap(err, "could not set head")
|
|
|
|
}
|
2020-02-15 18:57:49 +00:00
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// This sets head view object which is used to track the head slot, root, block and state.
|
2022-08-02 15:30:46 +00:00
|
|
|
func (s *Service) setHead(root [32]byte, block interfaces.SignedBeaconBlock, state state.BeaconState) error {
|
2020-02-15 18:57:49 +00:00
|
|
|
s.headLock.Lock()
|
|
|
|
defer s.headLock.Unlock()
|
|
|
|
|
|
|
|
// This does a full copy of the block and state.
|
2022-08-02 15:30:46 +00:00
|
|
|
bCp, err := block.Copy()
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
2020-02-15 18:57:49 +00:00
|
|
|
s.head = &head{
|
2021-05-26 16:19:54 +00:00
|
|
|
slot: block.Block().Slot(),
|
2020-02-15 18:57:49 +00:00
|
|
|
root: root,
|
2022-08-02 15:30:46 +00:00
|
|
|
block: bCp,
|
2020-02-15 18:57:49 +00:00
|
|
|
state: state.Copy(),
|
|
|
|
}
|
2022-08-02 15:30:46 +00:00
|
|
|
return nil
|
2020-02-15 18:57:49 +00:00
|
|
|
}
|
|
|
|
|
2020-05-06 05:04:02 +00:00
|
|
|
// This sets head view object which is used to track the head slot, root, block and state. The method
|
|
|
|
// assumes that state being passed into the method will not be modified by any other alternate
|
|
|
|
// caller which holds the state's reference.
|
2022-08-02 15:30:46 +00:00
|
|
|
func (s *Service) setHeadInitialSync(root [32]byte, block interfaces.SignedBeaconBlock, state state.BeaconState) error {
|
2020-05-06 05:04:02 +00:00
|
|
|
s.headLock.Lock()
|
|
|
|
defer s.headLock.Unlock()
|
|
|
|
|
|
|
|
// This does a full copy of the block only.
|
2022-08-02 15:30:46 +00:00
|
|
|
bCp, err := block.Copy()
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
2020-05-06 05:04:02 +00:00
|
|
|
s.head = &head{
|
2021-05-26 16:19:54 +00:00
|
|
|
slot: block.Block().Slot(),
|
2020-05-06 05:04:02 +00:00
|
|
|
root: root,
|
2022-08-02 15:30:46 +00:00
|
|
|
block: bCp,
|
2020-05-06 05:04:02 +00:00
|
|
|
state: state,
|
|
|
|
}
|
2022-08-02 15:30:46 +00:00
|
|
|
return nil
|
2020-05-06 05:04:02 +00:00
|
|
|
}
|
|
|
|
|
2020-02-15 18:57:49 +00:00
|
|
|
// This returns the head slot.
|
2020-10-06 10:57:18 +00:00
|
|
|
// This is a lock free version.
|
2021-02-16 07:45:34 +00:00
|
|
|
func (s *Service) headSlot() types.Slot {
|
2020-02-15 18:57:49 +00:00
|
|
|
return s.head.slot
|
|
|
|
}
|
|
|
|
|
|
|
|
// This returns the head root.
|
|
|
|
// It does a full copy on head root for immutability.
|
2020-10-06 10:57:18 +00:00
|
|
|
// This is a lock free version.
|
2020-02-15 18:57:49 +00:00
|
|
|
func (s *Service) headRoot() [32]byte {
|
|
|
|
if s.head == nil {
|
|
|
|
return params.BeaconConfig().ZeroHash
|
|
|
|
}
|
|
|
|
|
|
|
|
return s.head.root
|
|
|
|
}
|
|
|
|
|
|
|
|
// This returns the head block.
|
|
|
|
// It does a full copy on head block for immutability.
|
2020-10-06 10:57:18 +00:00
|
|
|
// This is a lock free version.
|
2022-08-02 15:30:46 +00:00
|
|
|
func (s *Service) headBlock() (interfaces.SignedBeaconBlock, error) {
|
2021-05-26 16:19:54 +00:00
|
|
|
return s.head.block.Copy()
|
2020-02-15 18:57:49 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// This returns the head state.
|
|
|
|
// It does a full copy on head state for immutability.
|
2020-10-06 10:57:18 +00:00
|
|
|
// This is a lock free version.
|
2021-07-23 16:11:21 +00:00
|
|
|
func (s *Service) headState(ctx context.Context) state.BeaconState {
|
2022-07-14 17:00:33 +00:00
|
|
|
ctx, span := trace.StartSpan(ctx, "blockChain.headState")
|
2020-06-26 22:52:11 +00:00
|
|
|
defer span.End()
|
|
|
|
|
2020-02-15 18:57:49 +00:00
|
|
|
return s.head.state.Copy()
|
|
|
|
}
|
|
|
|
|
2022-02-14 13:34:38 +00:00
|
|
|
// This returns the genesis validators root of the head state.
|
2020-10-06 10:57:18 +00:00
|
|
|
// This is a lock free version.
|
2022-02-14 13:34:38 +00:00
|
|
|
func (s *Service) headGenesisValidatorsRoot() [32]byte {
|
|
|
|
return bytesutil.ToBytes32(s.head.state.GenesisValidatorsRoot())
|
2020-04-23 03:18:29 +00:00
|
|
|
}
|
|
|
|
|
2021-09-17 02:03:18 +00:00
|
|
|
// This returns the validator referenced by the provided index in
|
|
|
|
// the head state.
|
|
|
|
// This is a lock free version.
|
|
|
|
func (s *Service) headValidatorAtIndex(index types.ValidatorIndex) (state.ReadOnlyValidator, error) {
|
|
|
|
return s.head.state.ValidatorAtIndexReadOnly(index)
|
|
|
|
}
|
|
|
|
|
2022-01-20 02:37:27 +00:00
|
|
|
// This returns the validator index referenced by the provided pubkey in
|
|
|
|
// the head state.
|
|
|
|
// This is a lock free version.
|
|
|
|
func (s *Service) headValidatorIndexAtPubkey(pubKey [fieldparams.BLSPubkeyLength]byte) (types.ValidatorIndex, bool) {
|
|
|
|
return s.head.state.ValidatorIndexByPubkey(pubKey)
|
|
|
|
}
|
|
|
|
|
2020-10-03 20:53:05 +00:00
|
|
|
// Returns true if head state exists.
|
|
|
|
// This is the lock free version.
|
|
|
|
func (s *Service) hasHeadState() bool {
|
2020-02-15 18:57:49 +00:00
|
|
|
return s.head != nil && s.head.state != nil
|
|
|
|
}
|
2020-05-20 19:16:56 +00:00
|
|
|
|
2021-06-10 21:13:15 +00:00
|
|
|
// Notifies a common event feed of a new chain head event. Called right after a new
|
|
|
|
// chain head is determined, set, and saved to disk.
|
|
|
|
func (s *Service) notifyNewHeadEvent(
|
2022-03-25 16:31:50 +00:00
|
|
|
ctx context.Context,
|
2021-06-10 21:13:15 +00:00
|
|
|
newHeadSlot types.Slot,
|
2021-07-23 16:11:21 +00:00
|
|
|
newHeadState state.BeaconState,
|
2021-06-10 21:13:15 +00:00
|
|
|
newHeadStateRoot,
|
|
|
|
newHeadRoot []byte,
|
|
|
|
) error {
|
2021-12-09 22:23:00 +00:00
|
|
|
previousDutyDependentRoot := s.originBlockRoot[:]
|
|
|
|
currentDutyDependentRoot := s.originBlockRoot[:]
|
2021-06-10 21:13:15 +00:00
|
|
|
|
|
|
|
var previousDutyEpoch types.Epoch
|
2021-10-01 20:17:57 +00:00
|
|
|
currentDutyEpoch := slots.ToEpoch(newHeadSlot)
|
2021-06-10 21:13:15 +00:00
|
|
|
if currentDutyEpoch > 0 {
|
|
|
|
previousDutyEpoch = currentDutyEpoch.Sub(1)
|
|
|
|
}
|
2021-10-01 20:17:57 +00:00
|
|
|
currentDutySlot, err := slots.EpochStart(currentDutyEpoch)
|
2021-06-10 21:13:15 +00:00
|
|
|
if err != nil {
|
|
|
|
return errors.Wrap(err, "could not get duty slot")
|
|
|
|
}
|
2021-10-01 20:17:57 +00:00
|
|
|
previousDutySlot, err := slots.EpochStart(previousDutyEpoch)
|
2021-06-10 21:13:15 +00:00
|
|
|
if err != nil {
|
|
|
|
return errors.Wrap(err, "could not get duty slot")
|
|
|
|
}
|
|
|
|
if currentDutySlot > 0 {
|
|
|
|
currentDutyDependentRoot, err = helpers.BlockRootAtSlot(newHeadState, currentDutySlot-1)
|
|
|
|
if err != nil {
|
|
|
|
return errors.Wrap(err, "could not get duty dependent root")
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if previousDutySlot > 0 {
|
|
|
|
previousDutyDependentRoot, err = helpers.BlockRootAtSlot(newHeadState, previousDutySlot-1)
|
|
|
|
if err != nil {
|
|
|
|
return errors.Wrap(err, "could not get duty dependent root")
|
|
|
|
}
|
|
|
|
}
|
2022-03-25 16:31:50 +00:00
|
|
|
isOptimistic, err := s.IsOptimistic(ctx)
|
|
|
|
if err != nil {
|
|
|
|
return errors.Wrap(err, "could not check if node is optimistically synced")
|
|
|
|
}
|
2021-06-10 21:13:15 +00:00
|
|
|
s.cfg.StateNotifier.StateFeed().Send(&feed.Event{
|
|
|
|
Type: statefeed.NewHead,
|
|
|
|
Data: ðpbv1.EventHead{
|
|
|
|
Slot: newHeadSlot,
|
|
|
|
Block: newHeadRoot,
|
|
|
|
State: newHeadStateRoot,
|
2021-10-01 20:17:57 +00:00
|
|
|
EpochTransition: slots.IsEpochStart(newHeadSlot),
|
2021-06-10 21:13:15 +00:00
|
|
|
PreviousDutyDependentRoot: previousDutyDependentRoot,
|
|
|
|
CurrentDutyDependentRoot: currentDutyDependentRoot,
|
2022-03-25 16:31:50 +00:00
|
|
|
ExecutionOptimistic: isOptimistic,
|
2021-06-10 21:13:15 +00:00
|
|
|
},
|
|
|
|
})
|
|
|
|
return nil
|
|
|
|
}
|
2021-08-24 03:05:08 +00:00
|
|
|
|
2022-05-29 19:32:42 +00:00
|
|
|
// This saves the attestations between `orphanedRoot` and the common ancestor root that is derived using `newHeadRoot`.
|
|
|
|
// It also filters out the attestations that is one epoch older as a defense so invalid attestations don't flow into the attestation pool.
|
|
|
|
func (s *Service) saveOrphanedAtts(ctx context.Context, orphanedRoot [32]byte, newHeadRoot [32]byte) error {
|
2022-08-25 23:59:08 +00:00
|
|
|
commonAncestorRoot, _, err := s.ForkChoicer().CommonAncestor(ctx, newHeadRoot, orphanedRoot)
|
2022-05-29 19:32:42 +00:00
|
|
|
switch {
|
2022-06-24 18:07:31 +00:00
|
|
|
// Exit early if there's no common ancestor and root doesn't exist, there would be nothing to save.
|
2022-05-29 19:32:42 +00:00
|
|
|
case errors.Is(err, forkchoice.ErrUnknownCommonAncestor):
|
|
|
|
return nil
|
|
|
|
case err != nil:
|
2021-08-24 03:05:08 +00:00
|
|
|
return err
|
|
|
|
}
|
2022-05-29 19:32:42 +00:00
|
|
|
for orphanedRoot != commonAncestorRoot {
|
|
|
|
if ctx.Err() != nil {
|
|
|
|
return ctx.Err()
|
|
|
|
}
|
2021-08-24 03:05:08 +00:00
|
|
|
|
2022-05-29 19:32:42 +00:00
|
|
|
orphanedBlk, err := s.getBlock(ctx, orphanedRoot)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
// If the block is an epoch older, break out of the loop since we can't include atts anyway.
|
|
|
|
// This prevents stuck within this for loop longer than necessary.
|
|
|
|
if orphanedBlk.Block().Slot()+params.BeaconConfig().SlotsPerEpoch <= s.CurrentSlot() {
|
|
|
|
break
|
2021-08-24 03:05:08 +00:00
|
|
|
}
|
2022-05-29 19:32:42 +00:00
|
|
|
for _, a := range orphanedBlk.Block().Body().Attestations() {
|
|
|
|
// if the attestation is one epoch older, it wouldn't been useful to save it.
|
|
|
|
if a.Data.Slot+params.BeaconConfig().SlotsPerEpoch < s.CurrentSlot() {
|
|
|
|
continue
|
2021-08-24 03:05:08 +00:00
|
|
|
}
|
2022-05-29 19:32:42 +00:00
|
|
|
if helpers.IsAggregated(a) {
|
|
|
|
if err := s.cfg.AttPool.SaveAggregatedAttestation(a); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
if err := s.cfg.AttPool.SaveUnaggregatedAttestation(a); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
2021-08-24 03:05:08 +00:00
|
|
|
}
|
2022-05-29 19:32:42 +00:00
|
|
|
saveOrphanedAttCount.Inc()
|
2021-08-24 03:05:08 +00:00
|
|
|
}
|
2022-09-06 14:30:16 +00:00
|
|
|
parentRoot := orphanedBlk.Block().ParentRoot()
|
|
|
|
orphanedRoot = bytesutil.ToBytes32(parentRoot[:])
|
2021-08-24 03:05:08 +00:00
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|