2019-02-20 05:07:28 +00:00
|
|
|
package state_test
|
2018-12-23 22:51:04 +00:00
|
|
|
|
|
|
|
import (
|
|
|
|
"bytes"
|
2019-02-26 03:42:31 +00:00
|
|
|
"encoding/binary"
|
2018-12-23 22:51:04 +00:00
|
|
|
"reflect"
|
2018-12-31 07:40:14 +00:00
|
|
|
"strconv"
|
2018-12-23 22:51:04 +00:00
|
|
|
"testing"
|
2019-01-05 05:39:34 +00:00
|
|
|
"time"
|
2018-12-23 22:51:04 +00:00
|
|
|
|
2019-02-26 03:42:31 +00:00
|
|
|
"github.com/prysmaticlabs/prysm/shared/hashutil"
|
|
|
|
|
2019-02-08 17:52:29 +00:00
|
|
|
"github.com/prysmaticlabs/prysm/beacon-chain/core/helpers"
|
2019-02-23 06:06:20 +00:00
|
|
|
"github.com/prysmaticlabs/prysm/beacon-chain/core/state"
|
2018-12-23 22:51:04 +00:00
|
|
|
pb "github.com/prysmaticlabs/prysm/proto/beacon/p2p/v1"
|
|
|
|
"github.com/prysmaticlabs/prysm/shared/params"
|
|
|
|
)
|
|
|
|
|
2019-02-22 15:11:26 +00:00
|
|
|
func TestGenesisBeaconState_OK(t *testing.T) {
|
2019-02-18 16:52:16 +00:00
|
|
|
if params.BeaconConfig().SlotsPerEpoch != 64 {
|
|
|
|
t.Errorf("SlotsPerEpoch should be 64 for these tests to pass")
|
2018-12-31 07:40:14 +00:00
|
|
|
}
|
|
|
|
|
2019-02-10 22:09:35 +00:00
|
|
|
if params.BeaconConfig().GenesisSlot != 1<<63 {
|
|
|
|
t.Error("GenesisSlot should be 2^63 for these tests to pass")
|
2018-12-31 07:40:14 +00:00
|
|
|
}
|
2019-02-18 16:52:16 +00:00
|
|
|
genesisEpochNumber := params.BeaconConfig().GenesisEpoch
|
2018-12-31 07:40:14 +00:00
|
|
|
|
2019-01-06 04:36:05 +00:00
|
|
|
if params.BeaconConfig().GenesisForkVersion != 0 {
|
2019-02-18 16:52:16 +00:00
|
|
|
t.Error("GenesisSlot( should be 0 for these tests to pass")
|
2018-12-31 07:40:14 +00:00
|
|
|
}
|
2019-02-18 16:52:16 +00:00
|
|
|
genesisForkVersion := params.BeaconConfig().GenesisForkVersion
|
2018-12-31 07:40:14 +00:00
|
|
|
|
|
|
|
if params.BeaconConfig().ZeroHash != [32]byte{} {
|
|
|
|
t.Error("ZeroHash should be all 0s for these tests to pass")
|
|
|
|
}
|
|
|
|
|
|
|
|
if params.BeaconConfig().LatestRandaoMixesLength != 8192 {
|
|
|
|
t.Error("LatestRandaoMixesLength should be 8192 for these tests to pass")
|
|
|
|
}
|
|
|
|
latestRandaoMixesLength := int(params.BeaconConfig().LatestRandaoMixesLength)
|
|
|
|
|
|
|
|
if params.BeaconConfig().ShardCount != 1024 {
|
|
|
|
t.Error("ShardCount should be 1024 for these tests to pass")
|
|
|
|
}
|
|
|
|
shardCount := int(params.BeaconConfig().ShardCount)
|
|
|
|
|
|
|
|
if params.BeaconConfig().LatestBlockRootsLength != 8192 {
|
|
|
|
t.Error("LatestBlockRootsLength should be 8192 for these tests to pass")
|
|
|
|
}
|
|
|
|
|
|
|
|
if params.BeaconConfig().DepositsForChainStart != 16384 {
|
|
|
|
t.Error("DepositsForChainStart should be 16384 for these tests to pass")
|
|
|
|
}
|
|
|
|
depositsForChainStart := int(params.BeaconConfig().DepositsForChainStart)
|
|
|
|
|
2019-02-18 16:52:16 +00:00
|
|
|
if params.BeaconConfig().LatestSlashedExitLength != 8192 {
|
|
|
|
t.Error("LatestSlashedExitLength should be 8192 for these tests to pass")
|
2019-01-05 06:39:07 +00:00
|
|
|
}
|
2019-02-18 16:52:16 +00:00
|
|
|
latestSlashedExitLength := int(params.BeaconConfig().LatestSlashedExitLength)
|
2019-01-05 06:39:07 +00:00
|
|
|
|
2018-12-31 07:40:14 +00:00
|
|
|
genesisTime := uint64(99999)
|
|
|
|
processedPowReceiptRoot := []byte{'A', 'B', 'C'}
|
2019-02-09 13:09:09 +00:00
|
|
|
maxDeposit := params.BeaconConfig().MaxDepositAmount
|
2018-12-31 07:40:14 +00:00
|
|
|
var deposits []*pb.Deposit
|
|
|
|
for i := 0; i < depositsForChainStart; i++ {
|
2019-02-19 18:05:34 +00:00
|
|
|
depositData, err := helpers.EncodeDepositData(
|
2019-01-05 05:39:34 +00:00
|
|
|
&pb.DepositInput{
|
2019-02-09 20:35:51 +00:00
|
|
|
Pubkey: []byte(strconv.Itoa(i)),
|
|
|
|
ProofOfPossession: []byte{'B'},
|
|
|
|
WithdrawalCredentialsHash32: []byte{'C'},
|
2019-01-05 05:39:34 +00:00
|
|
|
},
|
|
|
|
maxDeposit,
|
|
|
|
time.Now().Unix(),
|
|
|
|
)
|
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("Could not encode deposit data: %v", err)
|
|
|
|
}
|
|
|
|
deposits = append(deposits, &pb.Deposit{
|
|
|
|
MerkleBranchHash32S: [][]byte{{1}, {2}, {3}},
|
|
|
|
MerkleTreeIndex: 0,
|
|
|
|
DepositData: depositData,
|
|
|
|
})
|
2018-12-31 07:40:14 +00:00
|
|
|
}
|
|
|
|
|
2019-02-20 05:07:28 +00:00
|
|
|
newState, err := state.GenesisBeaconState(
|
2018-12-31 07:40:14 +00:00
|
|
|
deposits,
|
|
|
|
genesisTime,
|
2019-03-02 23:38:22 +00:00
|
|
|
&pb.Eth1Data{
|
|
|
|
DepositRootHash32: processedPowReceiptRoot,
|
|
|
|
BlockHash32: []byte{},
|
|
|
|
})
|
2018-12-31 07:40:14 +00:00
|
|
|
if err != nil {
|
2019-02-18 16:52:16 +00:00
|
|
|
t.Fatalf("could not execute GenesisBeaconState: %v", err)
|
2018-12-31 07:40:14 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// Misc fields checks.
|
2019-02-20 05:07:28 +00:00
|
|
|
if newState.Slot != params.BeaconConfig().GenesisSlot {
|
2018-12-31 07:40:14 +00:00
|
|
|
t.Error("Slot was not correctly initialized")
|
|
|
|
}
|
2019-02-20 05:07:28 +00:00
|
|
|
if newState.GenesisTime != genesisTime {
|
2018-12-31 07:40:14 +00:00
|
|
|
t.Error("GenesisTime was not correctly initialized")
|
|
|
|
}
|
2019-02-20 05:07:28 +00:00
|
|
|
if !reflect.DeepEqual(*newState.Fork, pb.Fork{
|
2019-02-18 16:52:16 +00:00
|
|
|
PreviousVersion: genesisForkVersion,
|
|
|
|
CurrentVersion: genesisForkVersion,
|
|
|
|
Epoch: genesisEpochNumber,
|
2018-12-31 07:40:14 +00:00
|
|
|
}) {
|
2019-01-20 18:26:33 +00:00
|
|
|
t.Error("Fork was not correctly initialized")
|
2018-12-31 07:40:14 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// Validator registry fields checks.
|
2019-02-26 00:37:55 +00:00
|
|
|
if newState.ValidatorRegistryUpdateEpoch != params.BeaconConfig().GenesisEpoch {
|
2019-01-20 18:26:33 +00:00
|
|
|
t.Error("ValidatorRegistryUpdateSlot was not correctly initialized")
|
2018-12-31 07:40:14 +00:00
|
|
|
}
|
2019-02-20 05:07:28 +00:00
|
|
|
if len(newState.ValidatorRegistry) != depositsForChainStart {
|
2018-12-31 07:40:14 +00:00
|
|
|
t.Error("ValidatorRegistry was not correctly initialized")
|
|
|
|
}
|
2019-02-20 05:07:28 +00:00
|
|
|
if len(newState.ValidatorBalances) != depositsForChainStart {
|
2018-12-31 07:40:14 +00:00
|
|
|
t.Error("ValidatorBalances was not correctly initialized")
|
|
|
|
}
|
|
|
|
|
|
|
|
// Randomness and committees fields checks.
|
2019-02-20 05:07:28 +00:00
|
|
|
if len(newState.LatestRandaoMixes) != latestRandaoMixesLength {
|
2019-02-19 20:24:00 +00:00
|
|
|
t.Error("Length of LatestRandaoMixes was not correctly initialized")
|
2018-12-31 07:40:14 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// Finality fields checks.
|
2019-02-20 05:07:28 +00:00
|
|
|
if newState.PreviousJustifiedEpoch != genesisEpochNumber {
|
2019-02-13 16:42:33 +00:00
|
|
|
t.Error("PreviousJustifiedEpoch was not correctly initialized")
|
2018-12-31 07:40:14 +00:00
|
|
|
}
|
2019-02-20 05:07:28 +00:00
|
|
|
if newState.JustifiedEpoch != genesisEpochNumber {
|
2019-02-13 16:42:33 +00:00
|
|
|
t.Error("JustifiedEpoch was not correctly initialized")
|
2018-12-31 07:40:14 +00:00
|
|
|
}
|
2019-02-20 05:07:28 +00:00
|
|
|
if newState.FinalizedEpoch != genesisEpochNumber {
|
2018-12-31 07:40:14 +00:00
|
|
|
t.Error("FinalizedSlot was not correctly initialized")
|
|
|
|
}
|
2019-02-20 05:07:28 +00:00
|
|
|
if newState.JustificationBitfield != 0 {
|
2018-12-31 07:40:14 +00:00
|
|
|
t.Error("JustificationBitfield was not correctly initialized")
|
|
|
|
}
|
|
|
|
|
|
|
|
// Recent state checks.
|
2019-02-20 05:07:28 +00:00
|
|
|
if len(newState.LatestCrosslinks) != shardCount {
|
2018-12-31 07:40:14 +00:00
|
|
|
t.Error("Length of LatestCrosslinks was not correctly initialized")
|
|
|
|
}
|
2019-02-20 05:07:28 +00:00
|
|
|
if !reflect.DeepEqual(newState.LatestSlashedBalances,
|
2019-02-18 16:52:16 +00:00
|
|
|
make([]uint64, latestSlashedExitLength)) {
|
|
|
|
t.Error("LatestSlashedBalances was not correctly initialized")
|
2018-12-31 07:40:14 +00:00
|
|
|
}
|
2019-02-20 05:07:28 +00:00
|
|
|
if !reflect.DeepEqual(newState.LatestAttestations, []*pb.PendingAttestation{}) {
|
2018-12-31 07:40:14 +00:00
|
|
|
t.Error("LatestAttestations was not correctly initialized")
|
|
|
|
}
|
2019-02-20 05:07:28 +00:00
|
|
|
if !reflect.DeepEqual(newState.BatchedBlockRootHash32S, [][]byte{}) {
|
2018-12-31 07:40:14 +00:00
|
|
|
t.Error("BatchedBlockRootHash32S was not correctly initialized")
|
|
|
|
}
|
2019-02-20 05:07:28 +00:00
|
|
|
activeValidators := helpers.ActiveValidatorIndices(newState.ValidatorRegistry, params.BeaconConfig().GenesisEpoch)
|
2019-02-26 03:42:31 +00:00
|
|
|
indicesBytes := []byte{}
|
|
|
|
for _, val := range activeValidators {
|
|
|
|
buf := make([]byte, 8)
|
|
|
|
binary.LittleEndian.PutUint64(buf, val)
|
|
|
|
indicesBytes = append(indicesBytes, buf...)
|
2019-02-08 17:52:29 +00:00
|
|
|
}
|
2019-02-26 03:42:31 +00:00
|
|
|
genesisActiveIndexRoot := hashutil.Hash(indicesBytes)
|
2019-02-20 05:07:28 +00:00
|
|
|
if !bytes.Equal(newState.LatestIndexRootHash32S[0], genesisActiveIndexRoot[:]) {
|
2019-02-08 17:52:29 +00:00
|
|
|
t.Errorf(
|
|
|
|
"Expected index roots to be the tree hash root of active validator indices, received %#x",
|
2019-02-20 05:07:28 +00:00
|
|
|
newState.LatestIndexRootHash32S[0],
|
2019-02-08 17:52:29 +00:00
|
|
|
)
|
|
|
|
}
|
2019-02-20 05:07:28 +00:00
|
|
|
seed, err := helpers.GenerateSeed(newState, params.BeaconConfig().GenesisEpoch)
|
2019-02-08 17:52:29 +00:00
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("Could not generate initial seed: %v", err)
|
|
|
|
}
|
2019-02-20 05:07:28 +00:00
|
|
|
if !bytes.Equal(seed[:], newState.CurrentShufflingSeedHash32) {
|
|
|
|
t.Errorf("Expected current epoch seed to be %#x, received %#x", seed[:], newState.CurrentShufflingSeedHash32)
|
2019-02-08 17:52:29 +00:00
|
|
|
}
|
2018-12-31 07:40:14 +00:00
|
|
|
|
2019-01-09 04:42:42 +00:00
|
|
|
// deposit root checks.
|
2019-02-20 05:07:28 +00:00
|
|
|
if !bytes.Equal(newState.LatestEth1Data.DepositRootHash32, processedPowReceiptRoot) {
|
2019-02-02 21:24:42 +00:00
|
|
|
t.Error("LatestEth1Data DepositRootHash32 was not correctly initialized")
|
2018-12-31 07:40:14 +00:00
|
|
|
}
|
2019-02-20 05:07:28 +00:00
|
|
|
if !reflect.DeepEqual(newState.Eth1DataVotes, []*pb.Eth1DataVote{}) {
|
2019-02-02 21:24:42 +00:00
|
|
|
t.Error("Eth1DataVotes was not correctly initialized")
|
2018-12-31 07:40:14 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-12-23 22:51:04 +00:00
|
|
|
func TestGenesisState_HashEquality(t *testing.T) {
|
2019-03-02 23:38:22 +00:00
|
|
|
state1, _ := state.GenesisBeaconState(nil, 0, &pb.Eth1Data{})
|
|
|
|
state2, _ := state.GenesisBeaconState(nil, 0, &pb.Eth1Data{})
|
2018-12-23 22:51:04 +00:00
|
|
|
|
2019-02-26 03:42:31 +00:00
|
|
|
root1, err1 := hashutil.HashProto(state1)
|
|
|
|
root2, err2 := hashutil.HashProto(state2)
|
2018-12-23 22:51:04 +00:00
|
|
|
|
|
|
|
if err1 != nil || err2 != nil {
|
|
|
|
t.Fatalf("Failed to marshal state to bytes: %v %v", err1, err2)
|
|
|
|
}
|
|
|
|
|
2019-02-14 20:04:47 +00:00
|
|
|
if root1 != root2 {
|
|
|
|
t.Fatalf("Tree hash of two genesis states should be equal, received %#x == %#x", root1, root2)
|
2018-12-23 22:51:04 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestGenesisState_InitializesLatestBlockHashes(t *testing.T) {
|
2019-02-20 05:07:28 +00:00
|
|
|
s, _ := state.GenesisBeaconState(nil, 0, nil)
|
2019-01-05 03:58:19 +00:00
|
|
|
want, got := len(s.LatestBlockRootHash32S), int(params.BeaconConfig().LatestBlockRootsLength)
|
2018-12-23 22:51:04 +00:00
|
|
|
if want != got {
|
|
|
|
t.Errorf("Wrong number of recent block hashes. Got: %d Want: %d", got, want)
|
|
|
|
}
|
|
|
|
|
2019-01-05 03:58:19 +00:00
|
|
|
want = cap(s.LatestBlockRootHash32S)
|
2018-12-23 22:51:04 +00:00
|
|
|
if want != got {
|
|
|
|
t.Errorf("The slice underlying array capacity is wrong. Got: %d Want: %d", got, want)
|
|
|
|
}
|
|
|
|
|
2019-01-05 03:58:19 +00:00
|
|
|
for _, h := range s.LatestBlockRootHash32S {
|
2019-01-01 17:17:44 +00:00
|
|
|
if !bytes.Equal(h, params.BeaconConfig().ZeroHash[:]) {
|
2018-12-23 22:51:04 +00:00
|
|
|
t.Errorf("Unexpected non-zero hash data: %v", h)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|