package altair import ( "context" "fmt" "time" "github.com/pkg/errors" "github.com/prysmaticlabs/prysm/beacon-chain/core/helpers" coreTime "github.com/prysmaticlabs/prysm/beacon-chain/core/time" "github.com/prysmaticlabs/prysm/beacon-chain/state" fieldparams "github.com/prysmaticlabs/prysm/config/fieldparams" "github.com/prysmaticlabs/prysm/config/params" types "github.com/prysmaticlabs/prysm/consensus-types/primitives" "github.com/prysmaticlabs/prysm/crypto/bls" "github.com/prysmaticlabs/prysm/crypto/hash" "github.com/prysmaticlabs/prysm/encoding/bytesutil" "github.com/prysmaticlabs/prysm/math" ethpb "github.com/prysmaticlabs/prysm/proto/prysm/v1alpha1" "github.com/prysmaticlabs/prysm/time/slots" ) const maxRandomByte = uint64(1<<8 - 1) // ValidateNilSyncContribution validates the following fields are not nil: // -the contribution and proof itself // -the message within contribution and proof // -the contribution within contribution and proof // -the aggregation bits within contribution func ValidateNilSyncContribution(s *ethpb.SignedContributionAndProof) error { if s == nil { return errors.New("signed message can't be nil") } if s.Message == nil { return errors.New("signed contribution's message can't be nil") } if s.Message.Contribution == nil { return errors.New("inner contribution can't be nil") } if s.Message.Contribution.AggregationBits == nil { return errors.New("contribution's bitfield can't be nil") } return nil } // NextSyncCommittee returns the next sync committee for a given state. // // Spec code: // def get_next_sync_committee(state: BeaconState) -> SyncCommittee: // """ // Return the next sync committee, with possible pubkey duplicates. // """ // indices = get_next_sync_committee_indices(state) // pubkeys = [state.validators[index].pubkey for index in indices] // aggregate_pubkey = bls.AggregatePKs(pubkeys) // return SyncCommittee(pubkeys=pubkeys, aggregate_pubkey=aggregate_pubkey) func NextSyncCommittee(ctx context.Context, s state.BeaconStateAltair) (*ethpb.SyncCommittee, error) { indices, err := NextSyncCommitteeIndices(ctx, s) if err != nil { return nil, err } pubkeys := make([][]byte, len(indices)) for i, index := range indices { p := s.PubkeyAtIndex(index) pubkeys[i] = p[:] } aggregated, err := bls.AggregatePublicKeys(pubkeys) if err != nil { return nil, err } return ðpb.SyncCommittee{ Pubkeys: pubkeys, AggregatePubkey: aggregated.Marshal(), }, nil } // NextSyncCommitteeIndices returns the next sync committee indices for a given state. // // Spec code: // def get_next_sync_committee_indices(state: BeaconState) -> Sequence[ValidatorIndex]: // """ // Return the sync committee indices, with possible duplicates, for the next sync committee. // """ // epoch = Epoch(get_current_epoch(state) + 1) // // MAX_RANDOM_BYTE = 2**8 - 1 // active_validator_indices = get_active_validator_indices(state, epoch) // active_validator_count = uint64(len(active_validator_indices)) // seed = get_seed(state, epoch, DOMAIN_SYNC_COMMITTEE) // i = 0 // sync_committee_indices: List[ValidatorIndex] = [] // while len(sync_committee_indices) < SYNC_COMMITTEE_SIZE: // shuffled_index = compute_shuffled_index(uint64(i % active_validator_count), active_validator_count, seed) // candidate_index = active_validator_indices[shuffled_index] // random_byte = hash(seed + uint_to_bytes(uint64(i // 32)))[i % 32] // effective_balance = state.validators[candidate_index].effective_balance // if effective_balance * MAX_RANDOM_BYTE >= MAX_EFFECTIVE_BALANCE * random_byte: // sync_committee_indices.append(candidate_index) // i += 1 // return sync_committee_indices func NextSyncCommitteeIndices(ctx context.Context, s state.BeaconStateAltair) ([]types.ValidatorIndex, error) { epoch := coreTime.NextEpoch(s) indices, err := helpers.ActiveValidatorIndices(ctx, s, epoch) if err != nil { return nil, err } seed, err := helpers.Seed(s, epoch, params.BeaconConfig().DomainSyncCommittee) if err != nil { return nil, err } count := uint64(len(indices)) cfg := params.BeaconConfig() syncCommitteeSize := cfg.SyncCommitteeSize cIndices := make([]types.ValidatorIndex, 0, syncCommitteeSize) hashFunc := hash.CustomSHA256Hasher() for i := types.ValidatorIndex(0); uint64(len(cIndices)) < params.BeaconConfig().SyncCommitteeSize; i++ { if ctx.Err() != nil { return nil, ctx.Err() } sIndex, err := helpers.ComputeShuffledIndex(i.Mod(count), count, seed, true) if err != nil { return nil, err } b := append(seed[:], bytesutil.Bytes8(uint64(i.Div(32)))...) randomByte := hashFunc(b)[i%32] cIndex := indices[sIndex] v, err := s.ValidatorAtIndexReadOnly(cIndex) if err != nil { return nil, err } effectiveBal := v.EffectiveBalance() if effectiveBal*maxRandomByte >= cfg.MaxEffectiveBalance*uint64(randomByte) { cIndices = append(cIndices, cIndex) } } return cIndices, nil } // SyncSubCommitteePubkeys returns the pubkeys participating in a sync subcommittee. // // def get_sync_subcommittee_pubkeys(state: BeaconState, subcommittee_index: uint64) -> Sequence[BLSPubkey]: // # Committees assigned to `slot` sign for `slot - 1` // # This creates the exceptional logic below when transitioning between sync committee periods // next_slot_epoch = compute_epoch_at_slot(Slot(state.slot + 1)) // if compute_sync_committee_period(get_current_epoch(state)) == compute_sync_committee_period(next_slot_epoch): // sync_committee = state.current_sync_committee // else: // sync_committee = state.next_sync_committee // // # Return pubkeys for the subcommittee index // sync_subcommittee_size = SYNC_COMMITTEE_SIZE // SYNC_COMMITTEE_SUBNET_COUNT // i = subcommittee_index * sync_subcommittee_size // return sync_committee.pubkeys[i:i + sync_subcommittee_size] func SyncSubCommitteePubkeys(syncCommittee *ethpb.SyncCommittee, subComIdx types.CommitteeIndex) ([][]byte, error) { cfg := params.BeaconConfig() subCommSize := cfg.SyncCommitteeSize / cfg.SyncCommitteeSubnetCount i := uint64(subComIdx) * subCommSize endOfSubCom := i + subCommSize pubkeyLen := uint64(len(syncCommittee.Pubkeys)) if endOfSubCom > pubkeyLen { return nil, errors.Errorf("end index is larger than array length: %d > %d", endOfSubCom, pubkeyLen) } return syncCommittee.Pubkeys[i:endOfSubCom], nil } // IsSyncCommitteeAggregator checks whether the provided signature is for a valid // aggregator. // // def is_sync_committee_aggregator(signature: BLSSignature) -> bool: // modulo = max(1, SYNC_COMMITTEE_SIZE // SYNC_COMMITTEE_SUBNET_COUNT // TARGET_AGGREGATORS_PER_SYNC_SUBCOMMITTEE) // return bytes_to_uint64(hash(signature)[0:8]) % modulo == 0 func IsSyncCommitteeAggregator(sig []byte) (bool, error) { if len(sig) != fieldparams.BLSSignatureLength { return false, errors.New("incorrect sig length") } cfg := params.BeaconConfig() modulo := math.Max(1, cfg.SyncCommitteeSize/cfg.SyncCommitteeSubnetCount/cfg.TargetAggregatorsPerSyncSubcommittee) hashedSig := hash.Hash(sig) return bytesutil.FromBytes8(hashedSig[:8])%modulo == 0, nil } // ValidateSyncMessageTime validates sync message to ensure that the provided slot is valid. func ValidateSyncMessageTime(slot types.Slot, genesisTime time.Time, clockDisparity time.Duration) error { if err := slots.ValidateClock(slot, uint64(genesisTime.Unix())); err != nil { return err } messageTime, err := slots.ToTime(uint64(genesisTime.Unix()), slot) if err != nil { return err } currentSlot := slots.Since(genesisTime) slotStartTime, err := slots.ToTime(uint64(genesisTime.Unix()), currentSlot) if err != nil { return err } lowestSlotBound := slotStartTime.Add(-clockDisparity) currentLowerBound := time.Now().Add(-clockDisparity) // In the event the Slot's start time, is before the // current allowable bound, we set the slot's start // time as the bound. if slotStartTime.Before(currentLowerBound) { lowestSlotBound = slotStartTime } lowerBound := lowestSlotBound upperBound := time.Now().Add(clockDisparity) // Verify sync message slot is within the time range. if messageTime.Before(lowerBound) || messageTime.After(upperBound) { return fmt.Errorf( "sync message slot %d not within allowable range of %d to %d (current slot)", slot, uint64(lowerBound.Unix()-genesisTime.Unix())/params.BeaconConfig().SecondsPerSlot, uint64(upperBound.Unix()-genesisTime.Unix())/params.BeaconConfig().SecondsPerSlot, ) } return nil }