2018-11-22 11:18:06 -08:00
|
|
|
package validators
|
2018-08-23 22:09:59 -06:00
|
|
|
|
|
|
|
import (
|
2018-11-24 12:57:07 -06:00
|
|
|
"fmt"
|
|
|
|
|
2018-08-23 22:09:59 -06:00
|
|
|
"github.com/prysmaticlabs/prysm/beacon-chain/utils"
|
2018-08-25 11:59:46 -07:00
|
|
|
pb "github.com/prysmaticlabs/prysm/proto/beacon/p2p/v1"
|
2018-11-18 08:39:35 -08:00
|
|
|
"github.com/prysmaticlabs/prysm/shared/params"
|
2018-08-23 22:09:59 -06:00
|
|
|
)
|
|
|
|
|
2018-12-03 19:05:22 -08:00
|
|
|
// ShuffleValidatorRegistryToCommittees shuffles validator indices and splits them by slot and shard.
|
|
|
|
func ShuffleValidatorRegistryToCommittees(
|
2018-11-22 11:18:06 -08:00
|
|
|
seed [32]byte,
|
|
|
|
validators []*pb.ValidatorRecord,
|
|
|
|
crosslinkStartShard uint64,
|
|
|
|
) ([]*pb.ShardAndCommitteeArray, error) {
|
|
|
|
indices := ActiveValidatorIndices(validators)
|
2018-09-01 12:01:53 -04:00
|
|
|
// split the shuffled list for slot.
|
2018-12-03 19:05:22 -08:00
|
|
|
shuffledValidatorRegistry, err := utils.ShuffleIndices(seed, indices)
|
2018-08-23 22:09:59 -06:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
2018-12-03 19:05:22 -08:00
|
|
|
return splitBySlotShard(shuffledValidatorRegistry, crosslinkStartShard), nil
|
2018-08-30 22:55:52 -04:00
|
|
|
}
|
|
|
|
|
2018-10-15 21:17:07 +08:00
|
|
|
// InitialShardAndCommitteesForSlots initialises the committees for shards by shuffling the validators
|
|
|
|
// and assigning them to specific shards.
|
|
|
|
func InitialShardAndCommitteesForSlots(validators []*pb.ValidatorRecord) ([]*pb.ShardAndCommitteeArray, error) {
|
2018-11-22 11:18:06 -08:00
|
|
|
seed := [32]byte{}
|
2018-12-03 19:05:22 -08:00
|
|
|
committees, err := ShuffleValidatorRegistryToCommittees(seed, validators, 1)
|
2018-10-15 21:17:07 +08:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
2018-11-13 07:26:47 -08:00
|
|
|
// Initialize with 3 cycles of the same committees.
|
2018-11-18 08:39:35 -08:00
|
|
|
initialCommittees := make([]*pb.ShardAndCommitteeArray, 0, 3*params.BeaconConfig().CycleLength)
|
2018-11-13 07:26:47 -08:00
|
|
|
initialCommittees = append(initialCommittees, committees...)
|
2018-10-24 01:07:43 +09:00
|
|
|
initialCommittees = append(initialCommittees, committees...)
|
|
|
|
initialCommittees = append(initialCommittees, committees...)
|
|
|
|
return initialCommittees, nil
|
2018-10-15 21:17:07 +08:00
|
|
|
}
|
|
|
|
|
2018-11-24 12:57:07 -06:00
|
|
|
// AttesterIndices returns the validator indices that acted as attesters
|
|
|
|
// for a particular attestation.
|
|
|
|
func AttesterIndices(
|
|
|
|
shardCommittees *pb.ShardAndCommitteeArray,
|
|
|
|
attestation *pb.AggregatedAttestation,
|
|
|
|
) ([]uint32, error) {
|
|
|
|
shardCommitteesArray := shardCommittees.ArrayShardAndCommittee
|
|
|
|
for _, shardCommittee := range shardCommitteesArray {
|
|
|
|
if attestation.Shard == shardCommittee.Shard {
|
|
|
|
return shardCommittee.Committee, nil
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return nil, fmt.Errorf("unable to find committee for shard %d", attestation.Shard)
|
|
|
|
}
|
|
|
|
|
2018-08-30 22:55:52 -04:00
|
|
|
// splitBySlotShard splits the validator list into evenly sized committees and assigns each
|
|
|
|
// committee to a slot and a shard. If the validator set is large, multiple committees are assigned
|
2018-10-13 14:05:19 +09:00
|
|
|
// to a single slot and shard. See getCommitteesPerSlot for more details.
|
2018-12-03 19:05:22 -08:00
|
|
|
func splitBySlotShard(shuffledValidatorRegistry []uint32, crosslinkStartShard uint64) []*pb.ShardAndCommitteeArray {
|
|
|
|
committeesPerSlot := getCommitteesPerSlot(uint64(len(shuffledValidatorRegistry)))
|
2018-08-30 22:55:52 -04:00
|
|
|
committeBySlotAndShard := []*pb.ShardAndCommitteeArray{}
|
|
|
|
|
|
|
|
// split the validator indices by slot.
|
2018-12-03 19:05:22 -08:00
|
|
|
validatorsBySlot := utils.SplitIndices(shuffledValidatorRegistry, params.BeaconConfig().CycleLength)
|
2018-08-30 22:55:52 -04:00
|
|
|
for i, validatorsForSlot := range validatorsBySlot {
|
|
|
|
shardCommittees := []*pb.ShardAndCommittee{}
|
|
|
|
validatorsByShard := utils.SplitIndices(validatorsForSlot, committeesPerSlot)
|
2018-10-24 01:37:17 +09:00
|
|
|
shardStart := crosslinkStartShard + uint64(i)*committeesPerSlot
|
2018-08-23 22:09:59 -06:00
|
|
|
|
2018-08-30 22:55:52 -04:00
|
|
|
for j, validatorsForShard := range validatorsByShard {
|
2018-11-18 08:39:35 -08:00
|
|
|
shardID := (shardStart + uint64(j)) % params.BeaconConfig().ShardCount
|
2018-08-30 22:55:52 -04:00
|
|
|
shardCommittees = append(shardCommittees, &pb.ShardAndCommittee{
|
2018-10-24 01:37:17 +09:00
|
|
|
Shard: shardID,
|
2018-08-30 22:55:52 -04:00
|
|
|
Committee: validatorsForShard,
|
2018-08-23 22:09:59 -06:00
|
|
|
})
|
|
|
|
}
|
2018-08-30 22:55:52 -04:00
|
|
|
|
|
|
|
committeBySlotAndShard = append(committeBySlotAndShard, &pb.ShardAndCommitteeArray{
|
|
|
|
ArrayShardAndCommittee: shardCommittees,
|
|
|
|
})
|
|
|
|
}
|
|
|
|
return committeBySlotAndShard
|
|
|
|
}
|
|
|
|
|
2018-12-03 19:05:22 -08:00
|
|
|
// getCommitteesPerSlot calculates the parameters for ShuffleValidatorRegistryToCommittees.
|
2018-10-13 14:05:19 +09:00
|
|
|
// The minimum value for committeesPerSlot is 1.
|
|
|
|
// Otherwise, the value for committeesPerSlot is the smaller of
|
2018-12-03 19:05:22 -08:00
|
|
|
// numActiveValidatorRegistry / CycleLength / (MinCommitteeSize*2) + 1 or
|
2018-10-13 14:05:19 +09:00
|
|
|
// ShardCount / CycleLength.
|
2018-12-03 19:05:22 -08:00
|
|
|
func getCommitteesPerSlot(numActiveValidatorRegistry uint64) uint64 {
|
2018-11-18 08:39:35 -08:00
|
|
|
cycleLength := params.BeaconConfig().CycleLength
|
2018-12-03 19:05:22 -08:00
|
|
|
boundOnValidatorRegistry := numActiveValidatorRegistry/cycleLength/(params.BeaconConfig().TargetCommitteeSize*2) + 1
|
2018-11-18 08:39:35 -08:00
|
|
|
boundOnShardCount := params.BeaconConfig().ShardCount / cycleLength
|
2018-10-13 14:05:19 +09:00
|
|
|
// Ensure that comitteesPerSlot is at least 1.
|
|
|
|
if boundOnShardCount == 0 {
|
|
|
|
return 1
|
2018-12-03 19:05:22 -08:00
|
|
|
} else if boundOnValidatorRegistry > boundOnShardCount {
|
2018-10-10 19:27:56 -04:00
|
|
|
return boundOnShardCount
|
2018-08-30 22:55:52 -04:00
|
|
|
}
|
2018-12-03 19:05:22 -08:00
|
|
|
return boundOnValidatorRegistry
|
2018-08-23 22:09:59 -06:00
|
|
|
}
|