prysm-pulse/beacon-chain/utils/shuffle.go

46 lines
1.4 KiB
Go
Raw Normal View History

// Package utils defines utility functions for the beacon-chain.
package utils
import (
"errors"
"github.com/ethereum/go-ethereum/common"
"github.com/prysmaticlabs/prysm/beacon-chain/params"
"golang.org/x/crypto/blake2b"
)
// ShuffleIndices returns a list of pseudorandomly sampled
// indices. This is used to use to select attesters and proposers.
func ShuffleIndices(seed common.Hash, validatorList []uint32) ([]uint32, error) {
if len(validatorList) > params.GetConfig().MaxValidators {
return nil, errors.New("Validator count has exceeded MaxValidator Count")
}
hashSeed := blake2b.Sum512(seed[:])
validatorCount := len(validatorList)
// shuffle stops at the second to last index.
for i := 0; i < validatorCount-1; i++ {
// convert every 3 bytes to random number, replace validator index with that number.
for j := 0; j+3 < len(hashSeed); j += 3 {
swapNum := int(hashSeed[j] + hashSeed[j+1] + hashSeed[j+2])
remaining := validatorCount - i
swapPos := swapNum%remaining + i
validatorList[i], validatorList[swapPos] = validatorList[swapPos], validatorList[i]
}
2018-10-03 13:08:12 +00:00
hashSeed = blake2b.Sum512(seed[:])
}
return validatorList, nil
}
// SplitIndices splits a list into n pieces.
func SplitIndices(l []uint32, n int) [][]uint32 {
var divided [][]uint32
for i := 0; i < n; i++ {
start := len(l) * i / n
end := len(l) * (i + 1) / n
divided = append(divided, l[start:end])
}
return divided
}