prysm-pulse/beacon-chain/core/helpers/shuffle_test.go
terence tsao 3edfa8cb88
Use Custom Type ValidatorIndex Across Prysm (#8478)
* Use ValidtorIndex across Prysm. Build ok

* First take at fixing tests

* Clean up e2e, fuzz... etc

* Fix new lines

* Update beacon-chain/cache/proposer_indices_test.go

Co-authored-by: Raul Jordan <raul@prysmaticlabs.com>

* Update beacon-chain/core/helpers/rewards_penalties.go

Co-authored-by: Raul Jordan <raul@prysmaticlabs.com>

* Update beacon-chain/core/helpers/shuffle.go

Co-authored-by: Raul Jordan <raul@prysmaticlabs.com>

* Update validator/graffiti/parse_graffiti_test.go

Co-authored-by: Raul Jordan <raul@prysmaticlabs.com>

* Raul's feedback

* Fix downcast int -> uint64

* Victor's feedback

Co-authored-by: Raul Jordan <raul@prysmaticlabs.com>
Co-authored-by: prylabs-bulldozer[bot] <58059840+prylabs-bulldozer[bot]@users.noreply.github.com>
2021-02-23 00:14:50 +00:00

164 lines
5.2 KiB
Go

package helpers
import (
"fmt"
"reflect"
"testing"
types "github.com/prysmaticlabs/eth2-types"
"github.com/prysmaticlabs/prysm/shared/params"
"github.com/prysmaticlabs/prysm/shared/sliceutil"
"github.com/prysmaticlabs/prysm/shared/testutil/assert"
"github.com/prysmaticlabs/prysm/shared/testutil/require"
)
func TestShuffleList_InvalidValidatorCount(t *testing.T) {
maxShuffleListSize = 20
list := make([]types.ValidatorIndex, 21)
if _, err := ShuffleList(list, [32]byte{123, 125}); err == nil {
t.Error("Shuffle should have failed when validator count exceeds ModuloBias")
maxShuffleListSize = 1 << 40
}
maxShuffleListSize = 1 << 40
}
func TestShuffleList_OK(t *testing.T) {
var list1 []types.ValidatorIndex
seed1 := [32]byte{1, 128, 12}
seed2 := [32]byte{2, 128, 12}
for i := 0; i < 10; i++ {
list1 = append(list1, types.ValidatorIndex(i))
}
list2 := make([]types.ValidatorIndex, len(list1))
copy(list2, list1)
list1, err := ShuffleList(list1, seed1)
assert.NoError(t, err, "Shuffle failed with")
list2, err = ShuffleList(list2, seed2)
assert.NoError(t, err, "Shuffle failed with")
if reflect.DeepEqual(list1, list2) {
t.Errorf("2 shuffled lists shouldn't be equal")
}
assert.DeepEqual(t, []types.ValidatorIndex{0, 7, 8, 6, 3, 9, 4, 5, 2, 1}, list1, "List 1 was incorrectly shuffled got")
assert.DeepEqual(t, []types.ValidatorIndex{0, 5, 2, 1, 6, 8, 7, 3, 4, 9}, list2, "List 2 was incorrectly shuffled got")
}
func TestSplitIndices_OK(t *testing.T) {
var l []uint64
numValidators := uint64(64000)
for i := uint64(0); i < numValidators; i++ {
l = append(l, i)
}
split := SplitIndices(l, uint64(params.BeaconConfig().SlotsPerEpoch))
assert.Equal(t, uint64(params.BeaconConfig().SlotsPerEpoch), uint64(len(split)), "Split list failed due to incorrect length")
for _, s := range split {
assert.Equal(t, numValidators/uint64(params.BeaconConfig().SlotsPerEpoch), uint64(len(s)), "Split list failed due to incorrect length")
}
}
func TestShuffleList_Vs_ShuffleIndex(t *testing.T) {
var list []types.ValidatorIndex
listSize := uint64(1000)
seed := [32]byte{123, 42}
for i := types.ValidatorIndex(0); uint64(i) < listSize; i++ {
list = append(list, i)
}
shuffledListByIndex := make([]types.ValidatorIndex, listSize)
for i := types.ValidatorIndex(0); uint64(i) < listSize; i++ {
si, err := ShuffledIndex(i, listSize, seed)
assert.NoError(t, err)
shuffledListByIndex[si] = i
}
shuffledList, err := ShuffleList(list, seed)
require.NoError(t, err, "Shuffled list error")
assert.DeepEqual(t, shuffledListByIndex, shuffledList, "Shuffled lists ar not equal")
}
func BenchmarkShuffledIndex(b *testing.B) {
listSizes := []uint64{4000000, 40000, 400}
seed := [32]byte{123, 42}
for _, listSize := range listSizes {
b.Run(fmt.Sprintf("ShuffledIndex_%d", listSize), func(ib *testing.B) {
for i := uint64(0); i < uint64(ib.N); i++ {
_, err := ShuffledIndex(types.ValidatorIndex(i%listSize), listSize, seed)
assert.NoError(b, err)
}
})
}
}
func BenchmarkIndexComparison(b *testing.B) {
listSizes := []uint64{400000, 40000, 400}
seed := [32]byte{123, 42}
for _, listSize := range listSizes {
b.Run(fmt.Sprintf("Indexwise_ShuffleList_%d", listSize), func(ib *testing.B) {
for i := 0; i < ib.N; i++ {
// Simulate a list-shuffle by running shuffle-index listSize times.
for j := types.ValidatorIndex(0); uint64(j) < listSize; j++ {
_, err := ShuffledIndex(j, listSize, seed)
assert.NoError(b, err)
}
}
})
}
}
func BenchmarkShuffleList(b *testing.B) {
listSizes := []uint64{400000, 40000, 400}
seed := [32]byte{123, 42}
for _, listSize := range listSizes {
testIndices := make([]types.ValidatorIndex, listSize)
for i := uint64(0); i < listSize; i++ {
testIndices[i] = types.ValidatorIndex(i)
}
b.Run(fmt.Sprintf("ShuffleList_%d", listSize), func(ib *testing.B) {
for i := 0; i < ib.N; i++ {
_, err := ShuffleList(testIndices, seed)
assert.NoError(b, err)
}
})
}
}
func TestShuffledIndex(t *testing.T) {
var list []types.ValidatorIndex
listSize := uint64(399)
for i := types.ValidatorIndex(0); uint64(i) < listSize; i++ {
list = append(list, i)
}
shuffledList := make([]types.ValidatorIndex, listSize)
unshuffledlist := make([]types.ValidatorIndex, listSize)
seed := [32]byte{123, 42}
for i := types.ValidatorIndex(0); uint64(i) < listSize; i++ {
si, err := ShuffledIndex(i, listSize, seed)
assert.NoError(t, err)
shuffledList[si] = i
}
for i := types.ValidatorIndex(0); uint64(i) < listSize; i++ {
ui, err := UnShuffledIndex(i, listSize, seed)
assert.NoError(t, err)
unshuffledlist[ui] = shuffledList[i]
}
assert.DeepEqual(t, list, unshuffledlist)
}
func TestSplitIndicesAndOffset_OK(t *testing.T) {
var l []uint64
validators := uint64(64000)
for i := uint64(0); i < validators; i++ {
l = append(l, i)
}
chunks := uint64(6)
split := SplitIndices(l, chunks)
for i := uint64(0); i < chunks; i++ {
if !reflect.DeepEqual(split[i], l[sliceutil.SplitOffset(uint64(len(l)), chunks, i):sliceutil.SplitOffset(uint64(len(l)), chunks, i+1)]) {
t.Errorf("Want: %v got: %v", l[sliceutil.SplitOffset(uint64(len(l)), chunks, i):sliceutil.SplitOffset(uint64(len(l)), chunks, i+1)], split[i])
break
}
}
}