mirror of
https://gitlab.com/pulsechaincom/prysm-pulse.git
synced 2025-01-10 11:41:21 +00:00
eb713d1177
* change parameters to main config * add more changes * change to accepted format * fix changes in config * gaz * fix test * fix test again
225 lines
8.4 KiB
Go
225 lines
8.4 KiB
Go
package helpers
|
|
|
|
import (
|
|
"encoding/binary"
|
|
"errors"
|
|
"fmt"
|
|
"time"
|
|
|
|
"github.com/prysmaticlabs/prysm/v4/config/params"
|
|
"github.com/prysmaticlabs/prysm/v4/consensus-types/primitives"
|
|
"github.com/prysmaticlabs/prysm/v4/crypto/hash"
|
|
ethpb "github.com/prysmaticlabs/prysm/v4/proto/prysm/v1alpha1"
|
|
prysmTime "github.com/prysmaticlabs/prysm/v4/time"
|
|
"github.com/prysmaticlabs/prysm/v4/time/slots"
|
|
log "github.com/sirupsen/logrus"
|
|
)
|
|
|
|
var (
|
|
ErrTooLate = errors.New("attestation is too late")
|
|
)
|
|
|
|
// ValidateNilAttestation checks if any composite field of input attestation is nil.
|
|
// Access to these nil fields will result in run time panic,
|
|
// it is recommended to run these checks as first line of defense.
|
|
func ValidateNilAttestation(attestation *ethpb.Attestation) error {
|
|
if attestation == nil {
|
|
return errors.New("attestation can't be nil")
|
|
}
|
|
if attestation.Data == nil {
|
|
return errors.New("attestation's data can't be nil")
|
|
}
|
|
if attestation.Data.Source == nil {
|
|
return errors.New("attestation's source can't be nil")
|
|
}
|
|
if attestation.Data.Target == nil {
|
|
return errors.New("attestation's target can't be nil")
|
|
}
|
|
if attestation.AggregationBits == nil {
|
|
return errors.New("attestation's bitfield can't be nil")
|
|
}
|
|
return nil
|
|
}
|
|
|
|
// ValidateSlotTargetEpoch checks if attestation data's epoch matches target checkpoint's epoch.
|
|
// It is recommended to run `ValidateNilAttestation` first to ensure `data.Target` can't be nil.
|
|
func ValidateSlotTargetEpoch(data *ethpb.AttestationData) error {
|
|
if slots.ToEpoch(data.Slot) != data.Target.Epoch {
|
|
return fmt.Errorf("slot %d does not match target epoch %d", data.Slot, data.Target.Epoch)
|
|
}
|
|
return nil
|
|
}
|
|
|
|
// IsAggregator returns true if the signature is from the input validator. The committee
|
|
// count is provided as an argument rather than imported implementation from spec. Having
|
|
// committee count as an argument allows cheaper computation at run time.
|
|
//
|
|
// Spec pseudocode definition:
|
|
//
|
|
// def is_aggregator(state: BeaconState, slot: Slot, index: CommitteeIndex, slot_signature: BLSSignature) -> bool:
|
|
// committee = get_beacon_committee(state, slot, index)
|
|
// modulo = max(1, len(committee) // TARGET_AGGREGATORS_PER_COMMITTEE)
|
|
// return bytes_to_uint64(hash(slot_signature)[0:8]) % modulo == 0
|
|
func IsAggregator(committeeCount uint64, slotSig []byte) (bool, error) {
|
|
modulo := uint64(1)
|
|
if committeeCount/params.BeaconConfig().TargetAggregatorsPerCommittee > 1 {
|
|
modulo = committeeCount / params.BeaconConfig().TargetAggregatorsPerCommittee
|
|
}
|
|
|
|
b := hash.Hash(slotSig)
|
|
return binary.LittleEndian.Uint64(b[:8])%modulo == 0, nil
|
|
}
|
|
|
|
// IsAggregated returns true if the attestation is an aggregated attestation,
|
|
// false otherwise.
|
|
func IsAggregated(attestation *ethpb.Attestation) bool {
|
|
return attestation.AggregationBits.Count() > 1
|
|
}
|
|
|
|
// ComputeSubnetForAttestation returns the subnet for which the provided attestation will be broadcasted to.
|
|
// This differs from the spec definition by instead passing in the active validators indices in the attestation's
|
|
// given epoch.
|
|
//
|
|
// Spec pseudocode definition:
|
|
// def compute_subnet_for_attestation(committees_per_slot: uint64, slot: Slot, committee_index: CommitteeIndex) -> uint64:
|
|
//
|
|
// """
|
|
// Compute the correct subnet for an attestation for Phase 0.
|
|
// Note, this mimics expected future behavior where attestations will be mapped to their shard subnet.
|
|
// """
|
|
// slots_since_epoch_start = uint64(slot % SLOTS_PER_EPOCH)
|
|
// committees_since_epoch_start = committees_per_slot * slots_since_epoch_start
|
|
//
|
|
// return uint64((committees_since_epoch_start + committee_index) % ATTESTATION_SUBNET_COUNT)
|
|
func ComputeSubnetForAttestation(activeValCount uint64, att *ethpb.Attestation) uint64 {
|
|
return ComputeSubnetFromCommitteeAndSlot(activeValCount, att.Data.CommitteeIndex, att.Data.Slot)
|
|
}
|
|
|
|
// ComputeSubnetFromCommitteeAndSlot is a flattened version of ComputeSubnetForAttestation where we only pass in
|
|
// the relevant fields from the attestation as function arguments.
|
|
//
|
|
// Spec pseudocode definition:
|
|
// def compute_subnet_for_attestation(committees_per_slot: uint64, slot: Slot, committee_index: CommitteeIndex) -> uint64:
|
|
//
|
|
// """
|
|
// Compute the correct subnet for an attestation for Phase 0.
|
|
// Note, this mimics expected future behavior where attestations will be mapped to their shard subnet.
|
|
// """
|
|
// slots_since_epoch_start = uint64(slot % SLOTS_PER_EPOCH)
|
|
// committees_since_epoch_start = committees_per_slot * slots_since_epoch_start
|
|
//
|
|
// return uint64((committees_since_epoch_start + committee_index) % ATTESTATION_SUBNET_COUNT)
|
|
func ComputeSubnetFromCommitteeAndSlot(activeValCount uint64, comIdx primitives.CommitteeIndex, attSlot primitives.Slot) uint64 {
|
|
slotSinceStart := slots.SinceEpochStarts(attSlot)
|
|
comCount := SlotCommitteeCount(activeValCount)
|
|
commsSinceStart := uint64(slotSinceStart.Mul(comCount))
|
|
computedSubnet := (commsSinceStart + uint64(comIdx)) % params.BeaconConfig().AttestationSubnetCount
|
|
return computedSubnet
|
|
}
|
|
|
|
// ValidateAttestationTime Validates that the incoming attestation is in the desired time range.
|
|
// An attestation is valid only if received within the last ATTESTATION_PROPAGATION_SLOT_RANGE
|
|
// slots.
|
|
//
|
|
// Example:
|
|
//
|
|
// ATTESTATION_PROPAGATION_SLOT_RANGE = 5
|
|
// clockDisparity = 24 seconds
|
|
// current_slot = 100
|
|
// invalid_attestation_slot = 92
|
|
// invalid_attestation_slot = 103
|
|
// valid_attestation_slot = 98
|
|
// valid_attestation_slot = 101
|
|
//
|
|
// In the attestation must be within the range of 95 to 102 in the example above.
|
|
func ValidateAttestationTime(attSlot primitives.Slot, genesisTime time.Time, clockDisparity time.Duration) error {
|
|
if err := slots.ValidateClock(attSlot, uint64(genesisTime.Unix())); err != nil {
|
|
return err
|
|
}
|
|
attTime, err := slots.ToTime(uint64(genesisTime.Unix()), attSlot)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
currentSlot := slots.Since(genesisTime)
|
|
|
|
// When receiving an attestation, it can be from the future.
|
|
// so the upper bounds is set to now + clockDisparity(SECONDS_PER_SLOT * 2).
|
|
// But when sending an attestation, it should not be in future slot.
|
|
// so the upper bounds is set to now + clockDisparity(MAXIMUM_GOSSIP_CLOCK_DISPARITY).
|
|
upperBounds := prysmTime.Now().Add(clockDisparity)
|
|
|
|
// An attestation cannot be older than the current slot - attestation propagation slot range
|
|
// with a minor tolerance for peer clock disparity.
|
|
lowerBoundsSlot := primitives.Slot(0)
|
|
if currentSlot > params.BeaconConfig().AttestationPropagationSlotRange {
|
|
lowerBoundsSlot = currentSlot - params.BeaconConfig().AttestationPropagationSlotRange
|
|
}
|
|
lowerTime, err := slots.ToTime(uint64(genesisTime.Unix()), lowerBoundsSlot)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
lowerBounds := lowerTime.Add(-clockDisparity)
|
|
|
|
// Verify attestation slot within the time range.
|
|
attError := fmt.Errorf(
|
|
"attestation slot %d not within attestation propagation range of %d to %d (current slot)",
|
|
attSlot,
|
|
lowerBoundsSlot,
|
|
currentSlot,
|
|
)
|
|
if attTime.After(upperBounds) {
|
|
attReceivedTooEarlyCount.Inc()
|
|
return attError
|
|
}
|
|
|
|
attEpoch := slots.ToEpoch(attSlot)
|
|
if attEpoch < params.BeaconConfig().DenebForkEpoch {
|
|
if attTime.Before(lowerBounds) {
|
|
attReceivedTooLateCount.Inc()
|
|
return errors.Join(ErrTooLate, attError)
|
|
}
|
|
return nil
|
|
}
|
|
|
|
// EIP-7045: Starting in Deneb, allow any attestations from the current or previous epoch.
|
|
|
|
currentEpoch := slots.ToEpoch(currentSlot)
|
|
prevEpoch, err := currentEpoch.SafeSub(1)
|
|
if err != nil {
|
|
log.WithError(err).Debug("Ignoring underflow for a deneb attestation inclusion check in epoch 0")
|
|
prevEpoch = 0
|
|
}
|
|
attSlotEpoch := slots.ToEpoch(attSlot)
|
|
if attSlotEpoch != currentEpoch && attSlotEpoch != prevEpoch {
|
|
attError = fmt.Errorf(
|
|
"attestation epoch %d not within current epoch %d or previous epoch %d",
|
|
attSlot/params.BeaconConfig().SlotsPerEpoch,
|
|
currentEpoch,
|
|
prevEpoch,
|
|
)
|
|
return errors.Join(ErrTooLate, attError)
|
|
}
|
|
|
|
return nil
|
|
}
|
|
|
|
// VerifyCheckpointEpoch is within current epoch and previous epoch
|
|
// with respect to current time. Returns true if it's within, false if it's not.
|
|
func VerifyCheckpointEpoch(c *ethpb.Checkpoint, genesis time.Time) bool {
|
|
now := uint64(prysmTime.Now().Unix())
|
|
genesisTime := uint64(genesis.Unix())
|
|
currentSlot := primitives.Slot((now - genesisTime) / params.BeaconConfig().SecondsPerSlot)
|
|
currentEpoch := slots.ToEpoch(currentSlot)
|
|
|
|
var prevEpoch primitives.Epoch
|
|
if currentEpoch > 1 {
|
|
prevEpoch = currentEpoch - 1
|
|
}
|
|
|
|
if c.Epoch != prevEpoch && c.Epoch != currentEpoch {
|
|
return false
|
|
}
|
|
|
|
return true
|
|
}
|