package stategen import ( "errors" "strconv" "sync" "github.com/prysmaticlabs/prysm/v5/beacon-chain/state" "github.com/prysmaticlabs/prysm/v5/consensus-types/primitives" "k8s.io/client-go/tools/cache" ) var ( // maxCacheSize is 8. That means 8 epochs and roughly an hour // of no finality can be endured. maxCacheSize = uint64(8) errNotSlotRootInfo = errors.New("not slot root info type") errNotRootStateInfo = errors.New("not root state info type") ) // slotRootInfo specifies the slot root info in the epoch boundary state cache. type slotRootInfo struct { slot primitives.Slot blockRoot [32]byte } // slotKeyFn takes the string representation of the slot to be used as key // to retrieve root. func slotKeyFn(obj interface{}) (string, error) { s, ok := obj.(*slotRootInfo) if !ok { return "", errNotSlotRootInfo } return slotToString(s.slot), nil } // rootStateInfo specifies the root state info in the epoch boundary state cache. type rootStateInfo struct { root [32]byte state state.BeaconState } // rootKeyFn takes the string representation of the block root to be used as key // to retrieve epoch boundary state. func rootKeyFn(obj interface{}) (string, error) { s, ok := obj.(*rootStateInfo) if !ok { return "", errNotRootStateInfo } return string(s.root[:]), nil } // epochBoundaryState struct with two queues by looking up beacon state by slot or root. type epochBoundaryState struct { rootStateCache *cache.FIFO slotRootCache *cache.FIFO lock sync.RWMutex } // newBoundaryStateCache creates a new block newBoundaryStateCache for storing and accessing epoch boundary states from // memory. func newBoundaryStateCache() *epochBoundaryState { return &epochBoundaryState{ rootStateCache: cache.NewFIFO(rootKeyFn), slotRootCache: cache.NewFIFO(slotKeyFn), } } // ByBlockRoot satisfies the CachedGetter interface func (e *epochBoundaryState) ByBlockRoot(r [32]byte) (state.BeaconState, error) { rsi, ok, err := e.getByBlockRoot(r) if err != nil { return nil, err } if !ok { return nil, ErrNotInCache } return rsi.state, nil } // get epoch boundary state by its block root. Returns copied state in state info object if exists. Otherwise returns nil. func (e *epochBoundaryState) getByBlockRoot(r [32]byte) (*rootStateInfo, bool, error) { e.lock.RLock() defer e.lock.RUnlock() return e.getByBlockRootLockFree(r) } func (e *epochBoundaryState) getByBlockRootLockFree(r [32]byte) (*rootStateInfo, bool, error) { obj, exists, err := e.rootStateCache.GetByKey(string(r[:])) if err != nil { return nil, false, err } if !exists { return nil, false, nil } s, ok := obj.(*rootStateInfo) if !ok { return nil, false, errNotRootStateInfo } return &rootStateInfo{ root: r, state: s.state.Copy(), }, true, nil } // get epoch boundary state by its slot. Returns copied state in state info object if exists. Otherwise returns nil. func (e *epochBoundaryState) getBySlot(s primitives.Slot) (*rootStateInfo, bool, error) { e.lock.RLock() defer e.lock.RUnlock() obj, exists, err := e.slotRootCache.GetByKey(slotToString(s)) if err != nil { return nil, false, err } if !exists { return nil, false, nil } info, ok := obj.(*slotRootInfo) if !ok { return nil, false, errNotSlotRootInfo } return e.getByBlockRootLockFree(info.blockRoot) } // put adds a state to the epoch boundary state cache. This method also trims the // least recently added state info if the cache size has reached the max cache // size limit. func (e *epochBoundaryState) put(blockRoot [32]byte, s state.BeaconState) error { e.lock.Lock() defer e.lock.Unlock() if err := e.slotRootCache.AddIfNotPresent(&slotRootInfo{ slot: s.Slot(), blockRoot: blockRoot, }); err != nil { return err } if err := e.rootStateCache.AddIfNotPresent(&rootStateInfo{ root: blockRoot, state: s.Copy(), }); err != nil { return err } trim(e.rootStateCache, maxCacheSize) trim(e.slotRootCache, maxCacheSize) return nil } // delete the state from the epoch boundary state cache. func (e *epochBoundaryState) delete(blockRoot [32]byte) error { e.lock.Lock() defer e.lock.Unlock() rInfo, ok, err := e.getByBlockRootLockFree(blockRoot) if err != nil { return err } if !ok { return nil } slotInfo := &slotRootInfo{ slot: rInfo.state.Slot(), blockRoot: blockRoot, } if err = e.slotRootCache.Delete(slotInfo); err != nil { return err } return e.rootStateCache.Delete(&rootStateInfo{ root: blockRoot, }) } // trim the FIFO queue to the maxSize. func trim(queue *cache.FIFO, maxSize uint64) { for s := uint64(len(queue.ListKeys())); s > maxSize; s-- { if _, err := queue.Pop(popProcessNoopFunc); err != nil { // This never returns an error, but we'll handle anyway for sanity. panic(err) } } } // popProcessNoopFunc is a no-op function that never returns an error. func popProcessNoopFunc(_ interface{}) error { return nil } // Converts input uint64 to string. To be used as key for slot to get root. func slotToString(s primitives.Slot) string { return strconv.FormatUint(uint64(s), 10) }