package forkchoice import ( "fmt" "github.com/ledgerwatch/erigon/cl/cltypes" "github.com/ledgerwatch/erigon/cmd/erigon-cl/core/transition" "github.com/ledgerwatch/erigon/cmd/erigon-cl/forkchoice/fork_graph" "github.com/ledgerwatch/log/v3" ) func (f *ForkChoiceStore) OnBlock(block *cltypes.SignedBeaconBlock) error { f.mu.Lock() defer f.mu.Unlock() blockRoot, err := block.Block.HashSSZ() if err != nil { return err } // Check that block is later than the finalized epoch slot (optimization to reduce calls to get_ancestor) finalizedSlot := f.computeStartSlotAtEpoch(f.finalizedCheckpoint.Epoch) if block.Block.Slot <= finalizedSlot { return fmt.Errorf("block is too late compared to finalized") } config := f.forkGraph.Config() status, err := f.forkGraph.AddChainSegment(block) if status != fork_graph.Success { if status != fork_graph.PreValidated { log.Debug("Could not replay block", "slot", block.Block.Slot, "code", status, "reason", err) } return err } lastProcessedState := f.forkGraph.LastState() // Add proposer score boost if the block is timely timeIntoSlot := (f.time - f.forkGraph.GenesisTime()) % lastProcessedState.BeaconConfig().SecondsPerSlot isBeforeAttestingInterval := timeIntoSlot < config.SecondsPerSlot/config.IntervalsPerSlot if f.Slot() == block.Block.Slot && isBeforeAttestingInterval { f.proposerBoostRoot = blockRoot } // Update checkpoints f.updateCheckpoints(lastProcessedState.CurrentJustifiedCheckpoint().Copy(), lastProcessedState.FinalizedCheckpoint().Copy()) // Eagerly compute unrealized justification and finality lastProcessedState.StartCollectingReverseChangeSet() if err := transition.ProcessJustificationBitsAndFinality(lastProcessedState); err != nil { lastProcessedState.RevertWithChangeset(lastProcessedState.StopCollectingReverseChangeSet()) return err } // Add justied checkpoint copiedCheckpoint := *lastProcessedState.CurrentJustifiedCheckpoint() f.unrealizedJustifications.Add(blockRoot, &copiedCheckpoint) f.updateUnrealizedCheckpoints(lastProcessedState.CurrentJustifiedCheckpoint().Copy(), lastProcessedState.FinalizedCheckpoint().Copy()) // If the block is from a prior epoch, apply the realized values blockEpoch := f.computeEpochAtSlot(block.Block.Slot) currentEpoch := f.computeEpochAtSlot(f.Slot()) if blockEpoch < currentEpoch { f.updateCheckpoints(lastProcessedState.CurrentJustifiedCheckpoint().Copy(), lastProcessedState.FinalizedCheckpoint().Copy()) } // Lastly revert the changes to the state. lastProcessedState.RevertWithChangeset(lastProcessedState.StopCollectingReverseChangeSet()) return nil }