package stategen import ( "context" "fmt" "math" "github.com/prysmaticlabs/prysm/v5/beacon-chain/core/helpers" "github.com/prysmaticlabs/prysm/v5/beacon-chain/state" "github.com/prysmaticlabs/prysm/v5/config/params" "github.com/prysmaticlabs/prysm/v5/encoding/bytesutil" ethpb "github.com/prysmaticlabs/prysm/v5/proto/prysm/v1alpha1" "github.com/prysmaticlabs/prysm/v5/time/slots" "github.com/sirupsen/logrus" "go.opencensus.io/trace" ) // SaveState saves the state in the cache and/or DB. func (s *State) SaveState(ctx context.Context, blockRoot [32]byte, st state.BeaconState) error { ctx, span := trace.StartSpan(ctx, "stateGen.SaveState") defer span.End() return s.saveStateByRoot(ctx, blockRoot, st) } // ForceCheckpoint initiates a cold state save of the given block root's state. This method does not update the // "last archived state" but simply saves the specified state from the root argument into the DB. // // The name "Checkpoint" isn't referring to checkpoint in the sense of our consensus type, but checkpoint for our historical states. func (s *State) ForceCheckpoint(ctx context.Context, blockRoot []byte) error { ctx, span := trace.StartSpan(ctx, "stateGen.ForceCheckpoint") defer span.End() root32 := bytesutil.ToBytes32(blockRoot) // Before the first finalized checkpoint, the finalized root is zero hash. // Return early if there hasn't been a finalized checkpoint. if root32 == params.BeaconConfig().ZeroHash { return nil } fs, err := s.loadStateByRoot(ctx, root32) if err != nil { return err } return s.beaconDB.SaveState(ctx, fs, root32) } // This saves a post beacon state. On the epoch boundary, // it saves a full state. On an intermediate slot, it saves a back pointer to the // nearest epoch boundary state. func (s *State) saveStateByRoot(ctx context.Context, blockRoot [32]byte, st state.BeaconState) error { ctx, span := trace.StartSpan(ctx, "stateGen.saveStateByRoot") defer span.End() // Duration can't be 0 to prevent panic for division. duration := uint64(math.Max(float64(s.saveHotStateDB.duration), 1)) s.saveHotStateDB.lock.Lock() if s.saveHotStateDB.enabled && st.Slot().Mod(duration) == 0 { if err := s.beaconDB.SaveState(ctx, st, blockRoot); err != nil { s.saveHotStateDB.lock.Unlock() return err } s.saveHotStateDB.blockRootsOfSavedStates = append(s.saveHotStateDB.blockRootsOfSavedStates, blockRoot) log.WithFields(logrus.Fields{ "slot": st.Slot(), "totalHotStateSavedInDB": len(s.saveHotStateDB.blockRootsOfSavedStates), }).Info("Saving hot state to DB") } s.saveHotStateDB.lock.Unlock() // If the hot state is already in cache, one can be sure the state was processed and in the DB. if s.hotStateCache.has(blockRoot) { return nil } // Only on an epoch boundary slot, save epoch boundary state in epoch boundary root state cache. if slots.IsEpochStart(st.Slot()) { if err := s.epochBoundaryStateCache.put(blockRoot, st); err != nil { return err } } else { // Always check that the correct epoch boundary states have been saved // for the current epoch. epochStart, err := slots.EpochStart(slots.ToEpoch(st.Slot())) if err != nil { return err } bRoot, err := helpers.BlockRootAtSlot(st, epochStart) if err != nil { return err } _, ok, err := s.epochBoundaryStateCache.getByBlockRoot([32]byte(bRoot)) if err != nil { return err } // We would only recover the boundary states under this condition: // // 1) Would indicate that the epoch boundary was skipped due to a missed slot, we // then recover by saving the state at that particular slot here. if !ok { // Only recover the state if it is in our hot state cache, otherwise we // simply skip this step. if s.hotStateCache.has([32]byte(bRoot)) { log.WithFields(logrus.Fields{ "slot": epochStart, "root": fmt.Sprintf("%#x", bRoot), }).Debug("Recovering state for epoch boundary cache") hState := s.hotStateCache.get([32]byte(bRoot)) if err := s.epochBoundaryStateCache.put([32]byte(bRoot), hState); err != nil { return err } } } } if err := s.beaconDB.SaveStateSummary(ctx, ðpb.StateSummary{ Slot: st.Slot(), Root: blockRoot[:], }); err != nil { return err } // Store the copied state in the hot state cache. s.hotStateCache.put(blockRoot, st) return nil } // EnableSaveHotStateToDB enters the mode that saves hot beacon state to the DB. // This usually gets triggered when there's long duration since finality. func (s *State) EnableSaveHotStateToDB(_ context.Context) { s.saveHotStateDB.lock.Lock() defer s.saveHotStateDB.lock.Unlock() if s.saveHotStateDB.enabled { return } s.saveHotStateDB.enabled = true log.WithFields(logrus.Fields{ "enabled": s.saveHotStateDB.enabled, "slotsInterval": s.saveHotStateDB.duration, }).Warn("Entering mode to save hot states in DB") } // DisableSaveHotStateToDB exits the mode that saves beacon state to DB for the hot states. // This usually gets triggered once there's finality after long duration since finality. func (s *State) DisableSaveHotStateToDB(ctx context.Context) error { s.saveHotStateDB.lock.Lock() defer s.saveHotStateDB.lock.Unlock() if !s.saveHotStateDB.enabled { return nil } log.WithFields(logrus.Fields{ "enabled": s.saveHotStateDB.enabled, "deletedHotStates": len(s.saveHotStateDB.blockRootsOfSavedStates), }).Warn("Exiting mode to save hot states in DB") // Delete previous saved states in DB as we are turning this mode off. s.saveHotStateDB.enabled = false if err := s.beaconDB.DeleteStates(ctx, s.saveHotStateDB.blockRootsOfSavedStates); err != nil { return err } s.saveHotStateDB.blockRootsOfSavedStates = nil return nil }