mirror of
https://gitlab.com/pulsechaincom/prysm-pulse.git
synced 2025-01-09 19:21:19 +00:00
f962d13407
* fix races in committee cache tests * lint * gratuitous defer ClearCache if ClearCache * log warning to avoid failed block processing * gaz * add Clear to cache stubs * fix Clear mistakes * last fake cache fix --------- Co-authored-by: Kasey Kirkham <kasey@users.noreply.github.com>
220 lines
6.9 KiB
Go
220 lines
6.9 KiB
Go
// Package helpers contains helper functions outlined in the Ethereum Beacon Chain spec, such as
|
|
// computing committees, randao, rewards/penalties, and more.
|
|
package helpers
|
|
|
|
import (
|
|
"bytes"
|
|
|
|
"github.com/pkg/errors"
|
|
"github.com/prysmaticlabs/prysm/v4/beacon-chain/cache"
|
|
"github.com/prysmaticlabs/prysm/v4/beacon-chain/core/time"
|
|
"github.com/prysmaticlabs/prysm/v4/beacon-chain/state"
|
|
"github.com/prysmaticlabs/prysm/v4/config/params"
|
|
"github.com/prysmaticlabs/prysm/v4/consensus-types/primitives"
|
|
"github.com/prysmaticlabs/prysm/v4/crypto/hash"
|
|
"github.com/prysmaticlabs/prysm/v4/encoding/bytesutil"
|
|
"github.com/prysmaticlabs/prysm/v4/time/slots"
|
|
log "github.com/sirupsen/logrus"
|
|
)
|
|
|
|
var (
|
|
syncCommitteeCache = cache.NewSyncCommittee()
|
|
)
|
|
|
|
// IsCurrentPeriodSyncCommittee returns true if the input validator index belongs in the current period sync committee
|
|
// along with the sync committee root.
|
|
// 1. Checks if the public key exists in the sync committee cache
|
|
// 2. If 1 fails, checks if the public key exists in the input current sync committee object
|
|
func IsCurrentPeriodSyncCommittee(st state.BeaconState, valIdx primitives.ValidatorIndex) (bool, error) {
|
|
root, err := syncPeriodBoundaryRoot(st)
|
|
if err != nil {
|
|
return false, err
|
|
}
|
|
indices, err := syncCommitteeCache.CurrentPeriodIndexPosition(root, valIdx)
|
|
if err == cache.ErrNonExistingSyncCommitteeKey {
|
|
val, err := st.ValidatorAtIndex(valIdx)
|
|
if err != nil {
|
|
return false, err
|
|
}
|
|
committee, err := st.CurrentSyncCommittee()
|
|
if err != nil {
|
|
return false, err
|
|
}
|
|
|
|
// Fill in the cache on miss.
|
|
go func() {
|
|
if err := syncCommitteeCache.UpdatePositionsInCommittee(root, st); err != nil {
|
|
log.WithError(err).Error("Could not fill sync committee cache on miss")
|
|
}
|
|
}()
|
|
|
|
return len(findSubCommitteeIndices(val.PublicKey, committee.Pubkeys)) > 0, nil
|
|
}
|
|
if err != nil {
|
|
return false, err
|
|
}
|
|
return len(indices) > 0, nil
|
|
}
|
|
|
|
// IsNextPeriodSyncCommittee returns true if the input validator index belongs in the next period sync committee
|
|
// along with the sync period boundary root.
|
|
// 1. Checks if the public key exists in the sync committee cache
|
|
// 2. If 1 fails, checks if the public key exists in the input next sync committee object
|
|
func IsNextPeriodSyncCommittee(
|
|
st state.BeaconState, valIdx primitives.ValidatorIndex,
|
|
) (bool, error) {
|
|
root, err := syncPeriodBoundaryRoot(st)
|
|
if err != nil {
|
|
return false, err
|
|
}
|
|
indices, err := syncCommitteeCache.NextPeriodIndexPosition(root, valIdx)
|
|
if err == cache.ErrNonExistingSyncCommitteeKey {
|
|
val, err := st.ValidatorAtIndex(valIdx)
|
|
if err != nil {
|
|
return false, err
|
|
}
|
|
committee, err := st.NextSyncCommittee()
|
|
if err != nil {
|
|
return false, err
|
|
}
|
|
return len(findSubCommitteeIndices(val.PublicKey, committee.Pubkeys)) > 0, nil
|
|
}
|
|
if err != nil {
|
|
return false, err
|
|
}
|
|
return len(indices) > 0, nil
|
|
}
|
|
|
|
// CurrentPeriodSyncSubcommitteeIndices returns the subcommittee indices of the
|
|
// current period sync committee for input validator.
|
|
func CurrentPeriodSyncSubcommitteeIndices(
|
|
st state.BeaconState, valIdx primitives.ValidatorIndex,
|
|
) ([]primitives.CommitteeIndex, error) {
|
|
root, err := syncPeriodBoundaryRoot(st)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
indices, err := syncCommitteeCache.CurrentPeriodIndexPosition(root, valIdx)
|
|
if err == cache.ErrNonExistingSyncCommitteeKey {
|
|
val, err := st.ValidatorAtIndex(valIdx)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
committee, err := st.CurrentSyncCommittee()
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
// Fill in the cache on miss.
|
|
go func() {
|
|
if err := syncCommitteeCache.UpdatePositionsInCommittee(root, st); err != nil {
|
|
log.WithError(err).Error("Could not fill sync committee cache on miss")
|
|
}
|
|
}()
|
|
|
|
return findSubCommitteeIndices(val.PublicKey, committee.Pubkeys), nil
|
|
}
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
return indices, nil
|
|
}
|
|
|
|
// NextPeriodSyncSubcommitteeIndices returns the subcommittee indices of the next period sync committee for input validator.
|
|
func NextPeriodSyncSubcommitteeIndices(
|
|
st state.BeaconState, valIdx primitives.ValidatorIndex,
|
|
) ([]primitives.CommitteeIndex, error) {
|
|
root, err := syncPeriodBoundaryRoot(st)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
indices, err := syncCommitteeCache.NextPeriodIndexPosition(root, valIdx)
|
|
if err == cache.ErrNonExistingSyncCommitteeKey {
|
|
val, err := st.ValidatorAtIndex(valIdx)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
committee, err := st.NextSyncCommittee()
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
return findSubCommitteeIndices(val.PublicKey, committee.Pubkeys), nil
|
|
}
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
return indices, nil
|
|
}
|
|
|
|
// UpdateSyncCommitteeCache updates sync committee cache.
|
|
// It uses `state`'s latest block header root as key. To avoid misuse, it disallows
|
|
// block header with state root zeroed out.
|
|
func UpdateSyncCommitteeCache(st state.BeaconState) error {
|
|
nextSlot := st.Slot() + 1
|
|
if nextSlot%params.BeaconConfig().SlotsPerEpoch != 0 {
|
|
return errors.New("not at the end of the epoch to update cache")
|
|
}
|
|
if slots.ToEpoch(nextSlot)%params.BeaconConfig().EpochsPerSyncCommitteePeriod != 0 {
|
|
return errors.New("not at sync committee period boundary to update cache")
|
|
}
|
|
|
|
header := st.LatestBlockHeader()
|
|
if bytes.Equal(header.StateRoot, params.BeaconConfig().ZeroHash[:]) {
|
|
return errors.New("zero hash state root can't be used to update cache")
|
|
}
|
|
|
|
prevBlockRoot, err := header.HashTreeRoot()
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
return syncCommitteeCache.UpdatePositionsInCommittee(combineRootAndSlot(prevBlockRoot[:], uint64(header.Slot)), st)
|
|
}
|
|
|
|
// Loop through `pubKeys` for matching `pubKey` and get the indices where it matches.
|
|
func findSubCommitteeIndices(pubKey []byte, pubKeys [][]byte) []primitives.CommitteeIndex {
|
|
var indices []primitives.CommitteeIndex
|
|
for i, k := range pubKeys {
|
|
if bytes.Equal(k, pubKey) {
|
|
indices = append(indices, primitives.CommitteeIndex(i))
|
|
}
|
|
}
|
|
return indices
|
|
}
|
|
|
|
// Retrieve the current sync period boundary root by calculating sync period start epoch
|
|
// and calling `BlockRoot`.
|
|
// It uses the boundary slot - 1 for block root. (Ex: SlotsPerEpoch * EpochsPerSyncCommitteePeriod - 1)
|
|
func syncPeriodBoundaryRoot(st state.ReadOnlyBeaconState) ([32]byte, error) {
|
|
// Can't call `BlockRoot` until the first slot.
|
|
if st.Slot() == params.BeaconConfig().GenesisSlot {
|
|
return params.BeaconConfig().ZeroHash, nil
|
|
}
|
|
|
|
startEpoch, err := slots.SyncCommitteePeriodStartEpoch(time.CurrentEpoch(st))
|
|
if err != nil {
|
|
return [32]byte{}, err
|
|
}
|
|
startEpochSlot, err := slots.EpochStart(startEpoch)
|
|
if err != nil {
|
|
return [32]byte{}, err
|
|
}
|
|
|
|
// Prevent underflow
|
|
if startEpochSlot >= 1 {
|
|
startEpochSlot--
|
|
}
|
|
|
|
root, err := BlockRootAtSlot(st, startEpochSlot)
|
|
if err != nil {
|
|
return [32]byte{}, err
|
|
}
|
|
return combineRootAndSlot(root, uint64(startEpochSlot)), nil
|
|
}
|
|
|
|
func combineRootAndSlot(root []byte, slot uint64) [32]byte {
|
|
slotBytes := bytesutil.Uint64ToBytesLittleEndian(slot)
|
|
keyHash := hash.Hash(append(root, slotBytes...))
|
|
return keyHash
|
|
}
|