mirror of
https://gitlab.com/pulsechaincom/prysm-pulse.git
synced 2025-01-08 02:31:19 +00:00
f3b49d4eaf
* Revert "Modify the algorithm of `updateFinalizedBlockRoots` (#13486)"
This reverts commit 32fb183392
.
* migration to fix index corruption from pr 13486
* bail as soon as we see 10 epochs without the bug
---------
Co-authored-by: Kasey Kirkham <kasey@users.noreply.github.com>
88 lines
2.9 KiB
Go
88 lines
2.9 KiB
Go
package kv
|
|
|
|
import (
|
|
"bytes"
|
|
"context"
|
|
"fmt"
|
|
|
|
"github.com/pkg/errors"
|
|
"github.com/prysmaticlabs/prysm/v5/config/params"
|
|
"github.com/prysmaticlabs/prysm/v5/consensus-types/primitives"
|
|
ethpb "github.com/prysmaticlabs/prysm/v5/proto/prysm/v1alpha1"
|
|
bolt "go.etcd.io/bbolt"
|
|
)
|
|
|
|
var migrationFinalizedParent = []byte("parent_bug_32fb183")
|
|
|
|
func migrateFinalizedParent(ctx context.Context, db *bolt.DB) error {
|
|
if updateErr := db.Update(func(tx *bolt.Tx) error {
|
|
mb := tx.Bucket(migrationsBucket)
|
|
if b := mb.Get(migrationFinalizedParent); bytes.Equal(b, migrationCompleted) {
|
|
return nil // Migration already completed.
|
|
}
|
|
|
|
bkt := tx.Bucket(finalizedBlockRootsIndexBucket)
|
|
if bkt == nil {
|
|
return fmt.Errorf("unable to read %s bucket for migration", finalizedBlockRootsIndexBucket)
|
|
}
|
|
bb := tx.Bucket(blocksBucket)
|
|
if bb == nil {
|
|
return fmt.Errorf("unable to read %s bucket for migration", blocksBucket)
|
|
}
|
|
|
|
c := bkt.Cursor()
|
|
var slotsWithoutBug primitives.Slot
|
|
maxBugSearch := params.BeaconConfig().SlotsPerEpoch * 10
|
|
for k, v := c.Last(); k != nil; k, v = c.Prev() {
|
|
// check if context is cancelled in between
|
|
if ctx.Err() != nil {
|
|
return ctx.Err()
|
|
}
|
|
|
|
idxEntry := ðpb.FinalizedBlockRootContainer{}
|
|
if err := decode(ctx, v, idxEntry); err != nil {
|
|
return errors.Wrapf(err, "unable to decode finalized block root container for root=%#x", k)
|
|
}
|
|
// Not one of the corrupt values
|
|
if !bytes.Equal(idxEntry.ParentRoot, k) {
|
|
slotsWithoutBug += 1
|
|
if slotsWithoutBug > maxBugSearch {
|
|
break
|
|
}
|
|
continue
|
|
}
|
|
slotsWithoutBug = 0
|
|
log.WithField("root", fmt.Sprintf("%#x", k)).Debug("found index entry with incorrect parent root")
|
|
|
|
// Look up full block to get the correct parent root.
|
|
encBlk := bb.Get(k)
|
|
if encBlk == nil {
|
|
return errors.Wrapf(ErrNotFound, "could not find block for corrupt finalized index entry %#x", k)
|
|
}
|
|
blk, err := unmarshalBlock(ctx, encBlk)
|
|
if err != nil {
|
|
return errors.Wrapf(err, "unable to decode block for root=%#x", k)
|
|
}
|
|
// Replace parent root in the index with the correct value and write it back.
|
|
pr := blk.Block().ParentRoot()
|
|
idxEntry.ParentRoot = pr[:]
|
|
idxEnc, err := encode(ctx, idxEntry)
|
|
if err != nil {
|
|
return errors.Wrapf(err, "failed to encode finalized index entry for root=%#x", k)
|
|
}
|
|
if err := bkt.Put(k, idxEnc); err != nil {
|
|
return errors.Wrapf(err, "failed to update finalized index entry for root=%#x", k)
|
|
}
|
|
log.WithField("root", fmt.Sprintf("%#x", k)).
|
|
WithField("parentRoot", fmt.Sprintf("%#x", idxEntry.ParentRoot)).
|
|
Debug("updated corrupt index entry with correct parent")
|
|
}
|
|
// Mark migration complete.
|
|
return mb.Put(migrationFinalizedParent, migrationCompleted)
|
|
}); updateErr != nil {
|
|
log.WithError(updateErr).Errorf("could not run finalized parent root index repair migration")
|
|
return updateErr
|
|
}
|
|
return nil
|
|
}
|