package helpers import ( "math" "github.com/pkg/errors" stateTrie "github.com/prysmaticlabs/prysm/beacon-chain/state" "github.com/prysmaticlabs/prysm/shared/params" ) // BlockRootAtSlot returns the block root stored in the BeaconState for a recent slot. // It returns an error if the requested block root is not within the slot range. // // Spec pseudocode definition: // def get_block_root_at_slot(state: BeaconState, slot: Slot) -> Hash: // """ // Return the block root at a recent ``slot``. // """ // assert slot < state.slot <= slot + SLOTS_PER_HISTORICAL_ROOT // return state.block_roots[slot % SLOTS_PER_HISTORICAL_ROOT] func BlockRootAtSlot(state *stateTrie.BeaconState, slot uint64) ([]byte, error) { if math.MaxUint64-slot < params.BeaconConfig().SlotsPerHistoricalRoot { return []byte{}, errors.New("slot overflows uint64") } if slot >= state.Slot() || state.Slot() > slot+params.BeaconConfig().SlotsPerHistoricalRoot { return []byte{}, errors.Errorf("slot %d out of bounds", slot) } return state.BlockRootAtIndex(slot % params.BeaconConfig().SlotsPerHistoricalRoot) } // StateRootAtSlot returns the cached state root at that particular slot. If no state // root has been cached it will return a zero-hash. func StateRootAtSlot(state *stateTrie.BeaconState, slot uint64) ([]byte, error) { if slot >= state.Slot() || state.Slot() > slot+params.BeaconConfig().SlotsPerHistoricalRoot { return []byte{}, errors.Errorf("slot %d out of bounds", slot) } return state.StateRootAtIndex(slot % params.BeaconConfig().SlotsPerHistoricalRoot) } // BlockRoot returns the block root stored in the BeaconState for epoch start slot. // // Spec pseudocode definition: // def get_block_root(state: BeaconState, epoch: Epoch) -> Hash: // """ // Return the block root at the start of a recent ``epoch``. // """ // return get_block_root_at_slot(state, compute_start_slot_at_epoch(epoch)) func BlockRoot(state *stateTrie.BeaconState, epoch uint64) ([]byte, error) { s, err := StartSlot(epoch) if err != nil { return nil, err } return BlockRootAtSlot(state, s) }