2020-03-17 19:25:17 +00:00
|
|
|
package stateutil_test
|
|
|
|
|
|
|
|
import (
|
|
|
|
"testing"
|
|
|
|
|
2022-08-16 12:20:13 +00:00
|
|
|
"github.com/prysmaticlabs/prysm/v3/beacon-chain/state"
|
|
|
|
"github.com/prysmaticlabs/prysm/v3/beacon-chain/state/stateutil"
|
|
|
|
"github.com/prysmaticlabs/prysm/v3/config/features"
|
|
|
|
"github.com/prysmaticlabs/prysm/v3/config/params"
|
|
|
|
types "github.com/prysmaticlabs/prysm/v3/consensus-types/primitives"
|
|
|
|
"github.com/prysmaticlabs/prysm/v3/crypto/hash"
|
|
|
|
"github.com/prysmaticlabs/prysm/v3/encoding/bytesutil"
|
|
|
|
ethpb "github.com/prysmaticlabs/prysm/v3/proto/prysm/v1alpha1"
|
|
|
|
"github.com/prysmaticlabs/prysm/v3/testing/assert"
|
|
|
|
"github.com/prysmaticlabs/prysm/v3/testing/require"
|
|
|
|
"github.com/prysmaticlabs/prysm/v3/testing/util"
|
2020-03-17 19:25:17 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
func TestReturnTrieLayer_OK(t *testing.T) {
|
2021-09-23 18:53:46 +00:00
|
|
|
newState, _ := util.DeterministicGenesisState(t, 32)
|
2022-02-22 09:27:51 +00:00
|
|
|
root, err := stateutil.RootsArrayHashTreeRoot(newState.BlockRoots(), uint64(params.BeaconConfig().SlotsPerHistoricalRoot))
|
2020-07-18 07:56:48 +00:00
|
|
|
require.NoError(t, err)
|
2022-02-28 13:56:12 +00:00
|
|
|
roots := retrieveBlockRoots(newState)
|
2021-11-02 14:31:16 +00:00
|
|
|
layers, err := stateutil.ReturnTrieLayer(roots, uint64(len(roots)))
|
|
|
|
assert.NoError(t, err)
|
2020-03-17 19:25:17 +00:00
|
|
|
newRoot := *layers[len(layers)-1][0]
|
2020-07-18 07:56:48 +00:00
|
|
|
assert.Equal(t, root, newRoot)
|
2022-02-28 13:56:12 +00:00
|
|
|
|
|
|
|
flags := &features.Flags{}
|
|
|
|
flags.EnableVectorizedHTR = true
|
|
|
|
reset := features.InitWithReset(flags)
|
|
|
|
defer reset()
|
|
|
|
|
|
|
|
layers, err = stateutil.ReturnTrieLayer(roots, uint64(len(roots)))
|
|
|
|
assert.NoError(t, err)
|
|
|
|
lastRoot := *layers[len(layers)-1][0]
|
|
|
|
assert.Equal(t, root, lastRoot)
|
|
|
|
}
|
|
|
|
|
|
|
|
func BenchmarkReturnTrieLayer_NormalAlgorithm(b *testing.B) {
|
|
|
|
newState, _ := util.DeterministicGenesisState(b, 32)
|
|
|
|
root, err := stateutil.RootsArrayHashTreeRoot(newState.BlockRoots(), uint64(params.BeaconConfig().SlotsPerHistoricalRoot))
|
|
|
|
require.NoError(b, err)
|
|
|
|
roots := retrieveBlockRoots(newState)
|
|
|
|
|
|
|
|
b.ResetTimer()
|
|
|
|
for i := 0; i < b.N; i++ {
|
|
|
|
layers, err := stateutil.ReturnTrieLayer(roots, uint64(len(roots)))
|
|
|
|
assert.NoError(b, err)
|
|
|
|
newRoot := *layers[len(layers)-1][0]
|
|
|
|
assert.Equal(b, root, newRoot)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func BenchmarkReturnTrieLayer_VectorizedAlgorithm(b *testing.B) {
|
|
|
|
flags := &features.Flags{}
|
|
|
|
flags.EnableVectorizedHTR = true
|
|
|
|
reset := features.InitWithReset(flags)
|
|
|
|
defer reset()
|
|
|
|
|
|
|
|
newState, _ := util.DeterministicGenesisState(b, 32)
|
|
|
|
root, err := stateutil.RootsArrayHashTreeRoot(newState.BlockRoots(), uint64(params.BeaconConfig().SlotsPerHistoricalRoot))
|
|
|
|
require.NoError(b, err)
|
|
|
|
roots := retrieveBlockRoots(newState)
|
|
|
|
|
|
|
|
b.ResetTimer()
|
|
|
|
for i := 0; i < b.N; i++ {
|
|
|
|
layers, err := stateutil.ReturnTrieLayer(roots, uint64(len(roots)))
|
|
|
|
assert.NoError(b, err)
|
|
|
|
newRoot := *layers[len(layers)-1][0]
|
|
|
|
assert.Equal(b, root, newRoot)
|
|
|
|
}
|
2020-03-17 19:25:17 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
func TestReturnTrieLayerVariable_OK(t *testing.T) {
|
2021-09-23 18:53:46 +00:00
|
|
|
newState, _ := util.DeterministicGenesisState(t, 32)
|
2021-11-29 16:30:17 +00:00
|
|
|
root, err := stateutil.ValidatorRegistryRoot(newState.Validators())
|
2020-07-18 07:56:48 +00:00
|
|
|
require.NoError(t, err)
|
2021-09-15 22:55:11 +00:00
|
|
|
hasher := hash.CustomSHA256Hasher()
|
2020-03-17 19:25:17 +00:00
|
|
|
validators := newState.Validators()
|
|
|
|
roots := make([][32]byte, 0, len(validators))
|
|
|
|
for _, val := range validators {
|
2021-03-18 23:29:06 +00:00
|
|
|
rt, err := stateutil.ValidatorRootWithHasher(hasher, val)
|
2020-07-18 07:56:48 +00:00
|
|
|
require.NoError(t, err)
|
2020-03-17 19:25:17 +00:00
|
|
|
roots = append(roots, rt)
|
|
|
|
}
|
|
|
|
layers := stateutil.ReturnTrieLayerVariable(roots, params.BeaconConfig().ValidatorRegistryLimit)
|
|
|
|
newRoot := *layers[len(layers)-1][0]
|
|
|
|
newRoot, err = stateutil.AddInMixin(newRoot, uint64(len(validators)))
|
2020-07-18 07:56:48 +00:00
|
|
|
require.NoError(t, err)
|
|
|
|
assert.Equal(t, root, newRoot)
|
2022-02-28 13:56:12 +00:00
|
|
|
|
|
|
|
flags := &features.Flags{}
|
|
|
|
flags.EnableVectorizedHTR = true
|
|
|
|
reset := features.InitWithReset(flags)
|
|
|
|
defer reset()
|
|
|
|
|
|
|
|
layers = stateutil.ReturnTrieLayerVariable(roots, params.BeaconConfig().ValidatorRegistryLimit)
|
|
|
|
lastRoot := *layers[len(layers)-1][0]
|
|
|
|
lastRoot, err = stateutil.AddInMixin(lastRoot, uint64(len(validators)))
|
|
|
|
require.NoError(t, err)
|
|
|
|
assert.Equal(t, root, lastRoot)
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
func BenchmarkReturnTrieLayerVariable_NormalAlgorithm(b *testing.B) {
|
|
|
|
newState, _ := util.DeterministicGenesisState(b, 16000)
|
|
|
|
root, err := stateutil.ValidatorRegistryRoot(newState.Validators())
|
|
|
|
require.NoError(b, err)
|
|
|
|
hasher := hash.CustomSHA256Hasher()
|
|
|
|
validators := newState.Validators()
|
|
|
|
roots := make([][32]byte, 0, len(validators))
|
|
|
|
for _, val := range validators {
|
|
|
|
rt, err := stateutil.ValidatorRootWithHasher(hasher, val)
|
|
|
|
require.NoError(b, err)
|
|
|
|
roots = append(roots, rt)
|
|
|
|
}
|
|
|
|
b.ResetTimer()
|
|
|
|
for i := 0; i < b.N; i++ {
|
|
|
|
layers := stateutil.ReturnTrieLayerVariable(roots, params.BeaconConfig().ValidatorRegistryLimit)
|
|
|
|
newRoot := *layers[len(layers)-1][0]
|
|
|
|
newRoot, err = stateutil.AddInMixin(newRoot, uint64(len(validators)))
|
|
|
|
require.NoError(b, err)
|
|
|
|
assert.Equal(b, root, newRoot)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func BenchmarkReturnTrieLayerVariable_VectorizedAlgorithm(b *testing.B) {
|
|
|
|
flags := &features.Flags{}
|
|
|
|
flags.EnableVectorizedHTR = true
|
|
|
|
reset := features.InitWithReset(flags)
|
|
|
|
defer reset()
|
|
|
|
|
|
|
|
newState, _ := util.DeterministicGenesisState(b, 16000)
|
|
|
|
root, err := stateutil.ValidatorRegistryRoot(newState.Validators())
|
|
|
|
require.NoError(b, err)
|
|
|
|
hasher := hash.CustomSHA256Hasher()
|
|
|
|
validators := newState.Validators()
|
|
|
|
roots := make([][32]byte, 0, len(validators))
|
|
|
|
for _, val := range validators {
|
|
|
|
rt, err := stateutil.ValidatorRootWithHasher(hasher, val)
|
|
|
|
require.NoError(b, err)
|
|
|
|
roots = append(roots, rt)
|
|
|
|
}
|
|
|
|
b.ResetTimer()
|
|
|
|
for i := 0; i < b.N; i++ {
|
|
|
|
layers := stateutil.ReturnTrieLayerVariable(roots, params.BeaconConfig().ValidatorRegistryLimit)
|
|
|
|
newRoot := *layers[len(layers)-1][0]
|
|
|
|
newRoot, err = stateutil.AddInMixin(newRoot, uint64(len(validators)))
|
|
|
|
require.NoError(b, err)
|
|
|
|
assert.Equal(b, root, newRoot)
|
|
|
|
}
|
2020-03-17 19:25:17 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
func TestRecomputeFromLayer_FixedSizedArray(t *testing.T) {
|
2021-09-23 18:53:46 +00:00
|
|
|
newState, _ := util.DeterministicGenesisState(t, 32)
|
2022-02-28 13:56:12 +00:00
|
|
|
roots := retrieveBlockRoots(newState)
|
|
|
|
|
2021-11-02 14:31:16 +00:00
|
|
|
layers, err := stateutil.ReturnTrieLayer(roots, uint64(len(roots)))
|
|
|
|
require.NoError(t, err)
|
2020-03-17 19:25:17 +00:00
|
|
|
|
|
|
|
changedIdx := []uint64{24, 41}
|
|
|
|
changedRoots := [][32]byte{{'A', 'B', 'C'}, {'D', 'E', 'F'}}
|
2020-07-18 07:56:48 +00:00
|
|
|
require.NoError(t, newState.UpdateBlockRootAtIndex(changedIdx[0], changedRoots[0]))
|
|
|
|
require.NoError(t, newState.UpdateBlockRootAtIndex(changedIdx[1], changedRoots[1]))
|
2020-03-17 19:25:17 +00:00
|
|
|
|
2022-02-22 09:27:51 +00:00
|
|
|
expectedRoot, err := stateutil.RootsArrayHashTreeRoot(newState.BlockRoots(), uint64(params.BeaconConfig().SlotsPerHistoricalRoot))
|
2020-07-18 07:56:48 +00:00
|
|
|
require.NoError(t, err)
|
2020-03-17 19:25:17 +00:00
|
|
|
root, _, err := stateutil.RecomputeFromLayer(changedRoots, changedIdx, layers)
|
2020-07-18 07:56:48 +00:00
|
|
|
require.NoError(t, err)
|
|
|
|
assert.Equal(t, expectedRoot, root)
|
2020-03-17 19:25:17 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
func TestRecomputeFromLayer_VariableSizedArray(t *testing.T) {
|
2021-09-23 18:53:46 +00:00
|
|
|
newState, _ := util.DeterministicGenesisState(t, 32)
|
2020-03-17 19:25:17 +00:00
|
|
|
validators := newState.Validators()
|
2021-09-15 22:55:11 +00:00
|
|
|
hasher := hash.CustomSHA256Hasher()
|
2020-03-17 19:25:17 +00:00
|
|
|
roots := make([][32]byte, 0, len(validators))
|
|
|
|
for _, val := range validators {
|
2021-03-18 23:29:06 +00:00
|
|
|
rt, err := stateutil.ValidatorRootWithHasher(hasher, val)
|
2020-07-18 07:56:48 +00:00
|
|
|
require.NoError(t, err)
|
2020-03-17 19:25:17 +00:00
|
|
|
roots = append(roots, rt)
|
|
|
|
}
|
|
|
|
layers := stateutil.ReturnTrieLayerVariable(roots, params.BeaconConfig().ValidatorRegistryLimit)
|
|
|
|
|
|
|
|
changedIdx := []uint64{2, 29}
|
|
|
|
val1, err := newState.ValidatorAtIndex(10)
|
2020-07-18 07:56:48 +00:00
|
|
|
require.NoError(t, err)
|
2020-03-17 19:25:17 +00:00
|
|
|
val2, err := newState.ValidatorAtIndex(11)
|
2020-07-18 07:56:48 +00:00
|
|
|
require.NoError(t, err)
|
2020-03-17 19:25:17 +00:00
|
|
|
val1.Slashed = true
|
|
|
|
val1.ExitEpoch = 20
|
|
|
|
|
|
|
|
val2.Slashed = true
|
|
|
|
val2.ExitEpoch = 40
|
|
|
|
|
|
|
|
changedVals := []*ethpb.Validator{val1, val2}
|
2021-02-23 00:14:50 +00:00
|
|
|
require.NoError(t, newState.UpdateValidatorAtIndex(types.ValidatorIndex(changedIdx[0]), changedVals[0]))
|
|
|
|
require.NoError(t, newState.UpdateValidatorAtIndex(types.ValidatorIndex(changedIdx[1]), changedVals[1]))
|
2020-03-17 19:25:17 +00:00
|
|
|
|
2021-11-29 16:30:17 +00:00
|
|
|
expectedRoot, err := stateutil.ValidatorRegistryRoot(newState.Validators())
|
2020-07-18 07:56:48 +00:00
|
|
|
require.NoError(t, err)
|
2020-03-17 19:25:17 +00:00
|
|
|
roots = make([][32]byte, 0, len(changedVals))
|
|
|
|
for _, val := range changedVals {
|
2021-03-18 23:29:06 +00:00
|
|
|
rt, err := stateutil.ValidatorRootWithHasher(hasher, val)
|
2020-07-18 07:56:48 +00:00
|
|
|
require.NoError(t, err)
|
2020-03-17 19:25:17 +00:00
|
|
|
roots = append(roots, rt)
|
|
|
|
}
|
|
|
|
root, _, err := stateutil.RecomputeFromLayerVariable(roots, changedIdx, layers)
|
2020-07-18 07:56:48 +00:00
|
|
|
require.NoError(t, err)
|
2020-03-17 19:25:17 +00:00
|
|
|
root, err = stateutil.AddInMixin(root, uint64(len(validators)))
|
2020-07-18 07:56:48 +00:00
|
|
|
require.NoError(t, err)
|
|
|
|
assert.Equal(t, expectedRoot, root)
|
2020-03-17 19:25:17 +00:00
|
|
|
}
|
2021-10-12 10:36:57 +00:00
|
|
|
|
|
|
|
func TestMerkleizeTrieLeaves_BadHashLayer(t *testing.T) {
|
|
|
|
hashLayer := make([][32]byte, 12)
|
|
|
|
layers := make([][][32]byte, 20)
|
|
|
|
_, _, err := stateutil.MerkleizeTrieLeaves(layers, hashLayer, func(bytes []byte) [32]byte {
|
|
|
|
return [32]byte{}
|
|
|
|
})
|
|
|
|
assert.ErrorContains(t, "hash layer is a non power of 2", err)
|
|
|
|
}
|
2022-02-28 13:56:12 +00:00
|
|
|
|
|
|
|
func retrieveBlockRoots(b state.BeaconState) [][32]byte {
|
|
|
|
blockRts := b.BlockRoots()
|
|
|
|
roots := make([][32]byte, 0, len(blockRts))
|
|
|
|
for _, rt := range blockRts {
|
|
|
|
roots = append(roots, bytesutil.ToBytes32(rt))
|
|
|
|
}
|
|
|
|
return roots
|
|
|
|
}
|