2019-02-20 05:07:28 +00:00
|
|
|
package state_test
|
2018-11-24 18:57:07 +00:00
|
|
|
|
|
|
|
import (
|
2019-03-05 21:09:50 +00:00
|
|
|
"context"
|
2019-02-20 05:07:28 +00:00
|
|
|
"crypto/rand"
|
|
|
|
"encoding/binary"
|
2018-12-27 03:15:12 +00:00
|
|
|
"fmt"
|
|
|
|
"strings"
|
2018-11-24 18:57:07 +00:00
|
|
|
"testing"
|
2019-02-20 05:07:28 +00:00
|
|
|
"time"
|
|
|
|
|
|
|
|
"github.com/prysmaticlabs/prysm/beacon-chain/core/helpers"
|
|
|
|
"github.com/prysmaticlabs/prysm/beacon-chain/core/state"
|
2018-11-24 18:57:07 +00:00
|
|
|
pb "github.com/prysmaticlabs/prysm/proto/beacon/p2p/v1"
|
2019-02-20 05:07:28 +00:00
|
|
|
"github.com/prysmaticlabs/prysm/shared/bls"
|
2019-03-18 06:19:44 +00:00
|
|
|
"github.com/prysmaticlabs/prysm/shared/forkutil"
|
2019-02-07 06:07:25 +00:00
|
|
|
"github.com/prysmaticlabs/prysm/shared/params"
|
2018-11-24 18:57:07 +00:00
|
|
|
)
|
|
|
|
|
2019-02-20 05:07:28 +00:00
|
|
|
func setupInitialDeposits(t *testing.T, numDeposits uint64) ([]*pb.Deposit, []*bls.SecretKey) {
|
|
|
|
privKeys := make([]*bls.SecretKey, numDeposits)
|
|
|
|
deposits := make([]*pb.Deposit, numDeposits)
|
|
|
|
for i := 0; i < len(deposits); i++ {
|
|
|
|
priv, err := bls.RandKey(rand.Reader)
|
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
depositInput := &pb.DepositInput{
|
|
|
|
Pubkey: priv.PublicKey().Marshal(),
|
|
|
|
}
|
|
|
|
balance := params.BeaconConfig().MaxDepositAmount
|
|
|
|
depositData, err := helpers.EncodeDepositData(depositInput, balance, time.Now().Unix())
|
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("Cannot encode data: %v", err)
|
|
|
|
}
|
|
|
|
deposits[i] = &pb.Deposit{DepositData: depositData}
|
|
|
|
privKeys[i] = priv
|
|
|
|
}
|
|
|
|
return deposits, privKeys
|
|
|
|
}
|
|
|
|
|
|
|
|
func createRandaoReveal(t *testing.T, beaconState *pb.BeaconState, privKeys []*bls.SecretKey) []byte {
|
|
|
|
// We fetch the proposer's index as that is whom the RANDAO will be verified against.
|
|
|
|
proposerIdx, err := helpers.BeaconProposerIndex(beaconState, beaconState.Slot)
|
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
epoch := helpers.SlotToEpoch(params.BeaconConfig().GenesisSlot)
|
|
|
|
buf := make([]byte, 32)
|
|
|
|
binary.LittleEndian.PutUint64(buf, epoch)
|
2019-03-12 23:39:13 +00:00
|
|
|
domain := forkutil.DomainVersion(beaconState.Fork, epoch, params.BeaconConfig().DomainRandao)
|
2019-02-20 05:07:28 +00:00
|
|
|
// We make the previous validator's index sign the message instead of the proposer.
|
|
|
|
epochSignature := privKeys[proposerIdx].Sign(buf, domain)
|
|
|
|
return epochSignature.Marshal()
|
|
|
|
}
|
|
|
|
|
2018-12-27 03:15:12 +00:00
|
|
|
func TestProcessBlock_IncorrectSlot(t *testing.T) {
|
|
|
|
beaconState := &pb.BeaconState{
|
2019-02-25 02:09:45 +00:00
|
|
|
Slot: params.BeaconConfig().GenesisSlot + 5,
|
2018-12-27 03:15:12 +00:00
|
|
|
}
|
|
|
|
block := &pb.BeaconBlock{
|
2019-02-25 02:09:45 +00:00
|
|
|
Slot: params.BeaconConfig().GenesisSlot + 4,
|
2018-12-27 03:15:12 +00:00
|
|
|
}
|
|
|
|
want := fmt.Sprintf(
|
|
|
|
"block.slot != state.slot, block.slot = %d, state.slot = %d",
|
|
|
|
4,
|
|
|
|
5,
|
|
|
|
)
|
2019-03-18 06:19:44 +00:00
|
|
|
if _, err := state.ProcessBlock(context.Background(), beaconState, block, state.DefaultConfig()); !strings.Contains(err.Error(), want) {
|
2018-12-27 03:15:12 +00:00
|
|
|
t.Errorf("Expected %s, received %v", want, err)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestProcessBlock_IncorrectProposerSlashing(t *testing.T) {
|
2019-02-20 05:07:28 +00:00
|
|
|
deposits, privKeys := setupInitialDeposits(t, params.BeaconConfig().SlotsPerEpoch)
|
2019-03-02 23:38:22 +00:00
|
|
|
beaconState, err := state.GenesisBeaconState(deposits, uint64(0), &pb.Eth1Data{})
|
2019-02-20 05:07:28 +00:00
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
2018-12-27 03:15:12 +00:00
|
|
|
}
|
2019-02-25 02:09:45 +00:00
|
|
|
var slashings []*pb.ProposerSlashing
|
|
|
|
for i := uint64(0); i < params.BeaconConfig().MaxProposerSlashings+1; i++ {
|
|
|
|
slashings = append(slashings, &pb.ProposerSlashing{})
|
|
|
|
}
|
2019-02-20 05:07:28 +00:00
|
|
|
randaoReveal := createRandaoReveal(t, beaconState, privKeys)
|
2018-12-27 03:15:12 +00:00
|
|
|
block := &pb.BeaconBlock{
|
2019-02-20 05:07:28 +00:00
|
|
|
Slot: params.BeaconConfig().GenesisSlot,
|
|
|
|
RandaoReveal: randaoReveal,
|
2019-02-02 21:24:42 +00:00
|
|
|
Eth1Data: &pb.Eth1Data{
|
|
|
|
DepositRootHash32: []byte{2},
|
|
|
|
BlockHash32: []byte{3},
|
|
|
|
},
|
2018-12-27 03:15:12 +00:00
|
|
|
Body: &pb.BeaconBlockBody{
|
|
|
|
ProposerSlashings: slashings,
|
|
|
|
},
|
|
|
|
}
|
|
|
|
want := "could not verify block proposer slashing"
|
2019-03-18 06:19:44 +00:00
|
|
|
if _, err := state.ProcessBlock(context.Background(), beaconState, block, state.DefaultConfig()); !strings.Contains(err.Error(), want) {
|
2018-12-27 03:15:12 +00:00
|
|
|
t.Errorf("Expected %s, received %v", want, err)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-01-30 10:11:13 +00:00
|
|
|
func TestProcessBlock_IncorrectAttesterSlashing(t *testing.T) {
|
2019-02-20 05:07:28 +00:00
|
|
|
deposits, privKeys := setupInitialDeposits(t, params.BeaconConfig().SlotsPerEpoch)
|
2019-03-02 23:38:22 +00:00
|
|
|
beaconState, err := state.GenesisBeaconState(deposits, uint64(0), &pb.Eth1Data{})
|
2019-02-20 05:07:28 +00:00
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
2018-12-27 03:15:12 +00:00
|
|
|
slashings := []*pb.ProposerSlashing{
|
|
|
|
{
|
|
|
|
ProposerIndex: 1,
|
|
|
|
ProposalData_1: &pb.ProposalSignedData{
|
|
|
|
Slot: 1,
|
|
|
|
Shard: 1,
|
|
|
|
BlockRootHash32: []byte{0, 1, 0},
|
|
|
|
},
|
|
|
|
ProposalData_2: &pb.ProposalSignedData{
|
|
|
|
Slot: 1,
|
|
|
|
Shard: 1,
|
|
|
|
BlockRootHash32: []byte{0, 1, 0},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
}
|
2019-02-25 02:09:45 +00:00
|
|
|
var attesterSlashings []*pb.AttesterSlashing
|
|
|
|
for i := uint64(0); i < params.BeaconConfig().MaxAttesterSlashings+1; i++ {
|
|
|
|
attesterSlashings = append(attesterSlashings, &pb.AttesterSlashing{})
|
|
|
|
}
|
2019-02-20 05:07:28 +00:00
|
|
|
randaoReveal := createRandaoReveal(t, beaconState, privKeys)
|
2018-12-27 03:15:12 +00:00
|
|
|
block := &pb.BeaconBlock{
|
2019-02-20 05:07:28 +00:00
|
|
|
Slot: params.BeaconConfig().GenesisSlot,
|
|
|
|
RandaoReveal: randaoReveal,
|
2019-02-02 21:24:42 +00:00
|
|
|
Eth1Data: &pb.Eth1Data{
|
|
|
|
DepositRootHash32: []byte{2},
|
|
|
|
BlockHash32: []byte{3},
|
|
|
|
},
|
2018-12-27 03:15:12 +00:00
|
|
|
Body: &pb.BeaconBlockBody{
|
|
|
|
ProposerSlashings: slashings,
|
2019-01-30 10:11:13 +00:00
|
|
|
AttesterSlashings: attesterSlashings,
|
2018-12-27 03:15:12 +00:00
|
|
|
},
|
|
|
|
}
|
2019-01-30 10:11:13 +00:00
|
|
|
want := "could not verify block attester slashing"
|
2019-03-18 06:19:44 +00:00
|
|
|
if _, err := state.ProcessBlock(context.Background(), beaconState, block, state.DefaultConfig()); !strings.Contains(err.Error(), want) {
|
2018-12-27 03:15:12 +00:00
|
|
|
t.Errorf("Expected %s, received %v", want, err)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestProcessBlock_IncorrectProcessBlockAttestations(t *testing.T) {
|
2019-02-20 05:07:28 +00:00
|
|
|
deposits, privKeys := setupInitialDeposits(t, params.BeaconConfig().SlotsPerEpoch)
|
2019-03-02 23:38:22 +00:00
|
|
|
beaconState, err := state.GenesisBeaconState(deposits, uint64(0), &pb.Eth1Data{})
|
2019-02-20 05:07:28 +00:00
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
2019-02-08 21:53:36 +00:00
|
|
|
}
|
2018-12-27 03:15:12 +00:00
|
|
|
proposerSlashings := []*pb.ProposerSlashing{
|
|
|
|
{
|
|
|
|
ProposerIndex: 1,
|
|
|
|
ProposalData_1: &pb.ProposalSignedData{
|
|
|
|
Slot: 1,
|
|
|
|
Shard: 1,
|
|
|
|
BlockRootHash32: []byte{0, 1, 0},
|
|
|
|
},
|
|
|
|
ProposalData_2: &pb.ProposalSignedData{
|
|
|
|
Slot: 1,
|
|
|
|
Shard: 1,
|
|
|
|
BlockRootHash32: []byte{0, 1, 0},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
}
|
|
|
|
att1 := &pb.AttestationData{
|
2019-02-15 23:19:36 +00:00
|
|
|
Slot: params.BeaconConfig().GenesisSlot + 5,
|
|
|
|
JustifiedEpoch: params.BeaconConfig().GenesisEpoch + 5,
|
2018-12-27 03:15:12 +00:00
|
|
|
}
|
|
|
|
att2 := &pb.AttestationData{
|
2019-02-15 23:19:36 +00:00
|
|
|
Slot: params.BeaconConfig().GenesisSlot + 5,
|
|
|
|
JustifiedEpoch: params.BeaconConfig().GenesisEpoch + 4,
|
2018-12-27 03:15:12 +00:00
|
|
|
}
|
2019-01-30 10:11:13 +00:00
|
|
|
attesterSlashings := []*pb.AttesterSlashing{
|
2018-12-27 03:15:12 +00:00
|
|
|
{
|
2019-02-08 21:53:36 +00:00
|
|
|
SlashableAttestation_1: &pb.SlashableAttestation{
|
2019-01-30 10:11:13 +00:00
|
|
|
Data: att1,
|
2019-02-14 01:17:54 +00:00
|
|
|
ValidatorIndices: []uint64{1, 2, 3, 4, 5, 6, 7, 8},
|
2019-01-30 10:11:13 +00:00
|
|
|
CustodyBitfield: []byte{0xFF},
|
2018-12-27 03:15:12 +00:00
|
|
|
},
|
2019-02-08 21:53:36 +00:00
|
|
|
SlashableAttestation_2: &pb.SlashableAttestation{
|
2019-01-30 10:11:13 +00:00
|
|
|
Data: att2,
|
2019-02-14 01:17:54 +00:00
|
|
|
ValidatorIndices: []uint64{1, 2, 3, 4, 5, 6, 7, 8},
|
2019-01-30 10:11:13 +00:00
|
|
|
CustodyBitfield: []byte{0xFF},
|
2018-12-27 03:15:12 +00:00
|
|
|
},
|
|
|
|
},
|
|
|
|
}
|
2019-01-09 17:26:40 +00:00
|
|
|
|
2019-02-25 02:09:45 +00:00
|
|
|
var blockAttestations []*pb.Attestation
|
|
|
|
for i := uint64(0); i < params.BeaconConfig().MaxAttestations+1; i++ {
|
|
|
|
blockAttestations = append(blockAttestations, &pb.Attestation{})
|
|
|
|
}
|
2019-02-20 05:07:28 +00:00
|
|
|
randaoReveal := createRandaoReveal(t, beaconState, privKeys)
|
2018-12-27 03:15:12 +00:00
|
|
|
block := &pb.BeaconBlock{
|
2019-02-20 05:07:28 +00:00
|
|
|
Slot: params.BeaconConfig().GenesisSlot,
|
|
|
|
RandaoReveal: randaoReveal,
|
2019-02-02 21:24:42 +00:00
|
|
|
Eth1Data: &pb.Eth1Data{
|
|
|
|
DepositRootHash32: []byte{2},
|
|
|
|
BlockHash32: []byte{3},
|
|
|
|
},
|
2018-12-27 03:15:12 +00:00
|
|
|
Body: &pb.BeaconBlockBody{
|
|
|
|
ProposerSlashings: proposerSlashings,
|
2019-01-30 10:11:13 +00:00
|
|
|
AttesterSlashings: attesterSlashings,
|
2018-12-27 03:15:12 +00:00
|
|
|
Attestations: blockAttestations,
|
|
|
|
},
|
|
|
|
}
|
|
|
|
want := "could not process block attestations"
|
2019-03-18 06:19:44 +00:00
|
|
|
if _, err := state.ProcessBlock(context.Background(), beaconState, block, state.DefaultConfig()); !strings.Contains(err.Error(), want) {
|
2018-12-27 03:15:12 +00:00
|
|
|
t.Errorf("Expected %s, received %v", want, err)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestProcessBlock_IncorrectProcessExits(t *testing.T) {
|
2019-02-20 05:07:28 +00:00
|
|
|
deposits, privKeys := setupInitialDeposits(t, params.BeaconConfig().SlotsPerEpoch)
|
2019-03-02 23:38:22 +00:00
|
|
|
beaconState, err := state.GenesisBeaconState(deposits, uint64(0), &pb.Eth1Data{})
|
2019-02-20 05:07:28 +00:00
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
2019-02-08 21:53:36 +00:00
|
|
|
}
|
2018-12-27 03:15:12 +00:00
|
|
|
proposerSlashings := []*pb.ProposerSlashing{
|
|
|
|
{
|
|
|
|
ProposerIndex: 1,
|
|
|
|
ProposalData_1: &pb.ProposalSignedData{
|
2019-02-15 23:19:36 +00:00
|
|
|
Slot: params.BeaconConfig().GenesisSlot + 1,
|
2018-12-27 03:15:12 +00:00
|
|
|
Shard: 1,
|
|
|
|
BlockRootHash32: []byte{0, 1, 0},
|
|
|
|
},
|
|
|
|
ProposalData_2: &pb.ProposalSignedData{
|
2019-02-15 23:19:36 +00:00
|
|
|
Slot: params.BeaconConfig().GenesisSlot + 1,
|
2018-12-27 03:15:12 +00:00
|
|
|
Shard: 1,
|
|
|
|
BlockRootHash32: []byte{0, 1, 0},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
}
|
|
|
|
att1 := &pb.AttestationData{
|
2019-02-15 23:19:36 +00:00
|
|
|
Slot: params.BeaconConfig().GenesisSlot + 5,
|
|
|
|
JustifiedEpoch: params.BeaconConfig().GenesisEpoch + 5,
|
2018-12-27 03:15:12 +00:00
|
|
|
}
|
|
|
|
att2 := &pb.AttestationData{
|
2019-02-15 23:19:36 +00:00
|
|
|
Slot: params.BeaconConfig().GenesisSlot + 5,
|
|
|
|
JustifiedEpoch: params.BeaconConfig().GenesisEpoch + 4,
|
2018-12-27 03:15:12 +00:00
|
|
|
}
|
2019-01-30 10:11:13 +00:00
|
|
|
attesterSlashings := []*pb.AttesterSlashing{
|
2018-12-27 03:15:12 +00:00
|
|
|
{
|
2019-02-08 21:53:36 +00:00
|
|
|
SlashableAttestation_1: &pb.SlashableAttestation{
|
2019-01-30 10:11:13 +00:00
|
|
|
Data: att1,
|
2019-02-14 01:17:54 +00:00
|
|
|
ValidatorIndices: []uint64{1, 2, 3, 4, 5, 6, 7, 8},
|
2019-01-30 10:11:13 +00:00
|
|
|
CustodyBitfield: []byte{0xFF},
|
2018-12-27 03:15:12 +00:00
|
|
|
},
|
2019-02-08 21:53:36 +00:00
|
|
|
SlashableAttestation_2: &pb.SlashableAttestation{
|
2019-01-30 10:11:13 +00:00
|
|
|
Data: att2,
|
2019-02-14 01:17:54 +00:00
|
|
|
ValidatorIndices: []uint64{1, 2, 3, 4, 5, 6, 7, 8},
|
2019-01-30 10:11:13 +00:00
|
|
|
CustodyBitfield: []byte{0xFF},
|
2018-12-27 03:15:12 +00:00
|
|
|
},
|
|
|
|
},
|
|
|
|
}
|
|
|
|
var blockRoots [][]byte
|
2019-02-15 23:19:36 +00:00
|
|
|
for i := uint64(0); i < params.BeaconConfig().LatestBlockRootsLength; i++ {
|
2018-12-27 03:15:12 +00:00
|
|
|
blockRoots = append(blockRoots, []byte{byte(i)})
|
|
|
|
}
|
2019-02-20 05:07:28 +00:00
|
|
|
beaconState.LatestBlockRootHash32S = blockRoots
|
|
|
|
beaconState.LatestCrosslinks = []*pb.Crosslink{
|
2018-12-27 03:15:12 +00:00
|
|
|
{
|
2019-02-27 23:07:39 +00:00
|
|
|
CrosslinkDataRootHash32: []byte{1},
|
2018-12-27 03:15:12 +00:00
|
|
|
},
|
|
|
|
}
|
2019-02-20 05:07:28 +00:00
|
|
|
beaconState.Slot = params.BeaconConfig().GenesisSlot + 10
|
2018-12-27 03:15:12 +00:00
|
|
|
blockAtt := &pb.Attestation{
|
|
|
|
Data: &pb.AttestationData{
|
2019-02-10 17:59:17 +00:00
|
|
|
Shard: 0,
|
2019-02-20 05:07:28 +00:00
|
|
|
Slot: params.BeaconConfig().GenesisSlot,
|
2019-02-15 23:19:36 +00:00
|
|
|
JustifiedEpoch: params.BeaconConfig().GenesisEpoch,
|
2019-02-10 17:59:17 +00:00
|
|
|
JustifiedBlockRootHash32: blockRoots[0],
|
2019-02-27 23:07:39 +00:00
|
|
|
LatestCrosslink: &pb.Crosslink{CrosslinkDataRootHash32: []byte{1}},
|
|
|
|
CrosslinkDataRootHash32: params.BeaconConfig().ZeroHash[:],
|
2018-12-27 03:15:12 +00:00
|
|
|
},
|
2019-02-06 16:20:38 +00:00
|
|
|
AggregationBitfield: []byte{1},
|
|
|
|
CustodyBitfield: []byte{1},
|
2018-12-27 03:15:12 +00:00
|
|
|
}
|
|
|
|
attestations := []*pb.Attestation{blockAtt}
|
2019-02-25 02:09:45 +00:00
|
|
|
var exits []*pb.VoluntaryExit
|
|
|
|
for i := uint64(0); i < params.BeaconConfig().MaxVoluntaryExits+1; i++ {
|
|
|
|
exits = append(exits, &pb.VoluntaryExit{})
|
|
|
|
}
|
2019-02-20 05:07:28 +00:00
|
|
|
randaoReveal := createRandaoReveal(t, beaconState, privKeys)
|
2018-12-27 03:15:12 +00:00
|
|
|
block := &pb.BeaconBlock{
|
2019-02-20 05:07:28 +00:00
|
|
|
Slot: params.BeaconConfig().GenesisSlot + 10,
|
|
|
|
RandaoReveal: randaoReveal,
|
2019-02-02 21:24:42 +00:00
|
|
|
Eth1Data: &pb.Eth1Data{
|
|
|
|
DepositRootHash32: []byte{2},
|
|
|
|
BlockHash32: []byte{3},
|
|
|
|
},
|
2018-12-27 03:15:12 +00:00
|
|
|
Body: &pb.BeaconBlockBody{
|
|
|
|
ProposerSlashings: proposerSlashings,
|
2019-01-30 10:11:13 +00:00
|
|
|
AttesterSlashings: attesterSlashings,
|
2018-12-27 03:15:12 +00:00
|
|
|
Attestations: attestations,
|
2019-02-18 16:52:16 +00:00
|
|
|
VoluntaryExits: exits,
|
2018-12-27 03:15:12 +00:00
|
|
|
},
|
|
|
|
}
|
|
|
|
want := "could not process validator exits"
|
2019-03-18 06:19:44 +00:00
|
|
|
if _, err := state.ProcessBlock(context.Background(), beaconState, block, state.DefaultConfig()); !strings.Contains(err.Error(), want) {
|
2018-12-27 03:15:12 +00:00
|
|
|
t.Errorf("Expected %s, received %v", want, err)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestProcessBlock_PassesProcessingConditions(t *testing.T) {
|
2019-02-20 05:07:28 +00:00
|
|
|
deposits, privKeys := setupInitialDeposits(t, params.BeaconConfig().SlotsPerEpoch)
|
2019-03-02 23:38:22 +00:00
|
|
|
beaconState, err := state.GenesisBeaconState(deposits, uint64(0), &pb.Eth1Data{})
|
2019-02-20 05:07:28 +00:00
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
2019-02-08 21:53:36 +00:00
|
|
|
}
|
2018-12-27 03:15:12 +00:00
|
|
|
proposerSlashings := []*pb.ProposerSlashing{
|
|
|
|
{
|
|
|
|
ProposerIndex: 1,
|
|
|
|
ProposalData_1: &pb.ProposalSignedData{
|
|
|
|
Slot: 1,
|
|
|
|
Shard: 1,
|
|
|
|
BlockRootHash32: []byte{0, 1, 0},
|
|
|
|
},
|
|
|
|
ProposalData_2: &pb.ProposalSignedData{
|
|
|
|
Slot: 1,
|
|
|
|
Shard: 1,
|
|
|
|
BlockRootHash32: []byte{0, 1, 0},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
}
|
|
|
|
att1 := &pb.AttestationData{
|
2019-02-13 16:42:33 +00:00
|
|
|
Slot: 5,
|
2019-02-15 23:19:36 +00:00
|
|
|
JustifiedEpoch: params.BeaconConfig().GenesisEpoch + 5,
|
2018-12-27 03:15:12 +00:00
|
|
|
}
|
|
|
|
att2 := &pb.AttestationData{
|
2019-02-13 16:42:33 +00:00
|
|
|
Slot: 5,
|
2019-02-15 23:19:36 +00:00
|
|
|
JustifiedEpoch: params.BeaconConfig().GenesisEpoch + 4,
|
2018-12-27 03:15:12 +00:00
|
|
|
}
|
2019-01-30 10:11:13 +00:00
|
|
|
attesterSlashings := []*pb.AttesterSlashing{
|
2018-12-27 03:15:12 +00:00
|
|
|
{
|
2019-02-08 21:53:36 +00:00
|
|
|
SlashableAttestation_1: &pb.SlashableAttestation{
|
2019-01-30 10:11:13 +00:00
|
|
|
Data: att1,
|
2019-02-14 01:17:54 +00:00
|
|
|
ValidatorIndices: []uint64{1, 2, 3, 4, 5, 6, 7, 8},
|
2019-01-30 10:11:13 +00:00
|
|
|
CustodyBitfield: []byte{0xFF},
|
2018-12-27 03:15:12 +00:00
|
|
|
},
|
2019-02-08 21:53:36 +00:00
|
|
|
SlashableAttestation_2: &pb.SlashableAttestation{
|
2019-01-30 10:11:13 +00:00
|
|
|
Data: att2,
|
2019-02-14 01:17:54 +00:00
|
|
|
ValidatorIndices: []uint64{1, 2, 3, 4, 5, 6, 7, 8},
|
2019-01-30 10:11:13 +00:00
|
|
|
CustodyBitfield: []byte{0xFF},
|
2018-12-27 03:15:12 +00:00
|
|
|
},
|
|
|
|
},
|
|
|
|
}
|
|
|
|
var blockRoots [][]byte
|
2019-02-15 23:19:36 +00:00
|
|
|
for i := uint64(0); i < params.BeaconConfig().LatestBlockRootsLength; i++ {
|
2018-12-27 03:15:12 +00:00
|
|
|
blockRoots = append(blockRoots, []byte{byte(i)})
|
|
|
|
}
|
2019-02-20 05:07:28 +00:00
|
|
|
beaconState.LatestBlockRootHash32S = blockRoots
|
|
|
|
beaconState.LatestCrosslinks = []*pb.Crosslink{
|
2018-12-27 03:15:12 +00:00
|
|
|
{
|
2019-02-27 23:07:39 +00:00
|
|
|
CrosslinkDataRootHash32: []byte{1},
|
2018-12-27 03:15:12 +00:00
|
|
|
},
|
|
|
|
}
|
2019-02-20 05:07:28 +00:00
|
|
|
beaconState.Slot = params.BeaconConfig().GenesisSlot + 10
|
2018-12-27 03:15:12 +00:00
|
|
|
blockAtt := &pb.Attestation{
|
|
|
|
Data: &pb.AttestationData{
|
2019-02-10 17:59:17 +00:00
|
|
|
Shard: 0,
|
2019-02-20 05:07:28 +00:00
|
|
|
Slot: params.BeaconConfig().GenesisSlot,
|
2019-02-15 23:19:36 +00:00
|
|
|
JustifiedEpoch: params.BeaconConfig().GenesisEpoch,
|
2019-02-10 17:59:17 +00:00
|
|
|
JustifiedBlockRootHash32: blockRoots[0],
|
2019-02-27 23:07:39 +00:00
|
|
|
LatestCrosslink: &pb.Crosslink{CrosslinkDataRootHash32: []byte{1}},
|
|
|
|
CrosslinkDataRootHash32: params.BeaconConfig().ZeroHash[:],
|
2018-12-27 03:15:12 +00:00
|
|
|
},
|
2019-02-06 16:20:38 +00:00
|
|
|
AggregationBitfield: []byte{1},
|
|
|
|
CustodyBitfield: []byte{1},
|
2018-12-27 03:15:12 +00:00
|
|
|
}
|
|
|
|
attestations := []*pb.Attestation{blockAtt}
|
2019-02-18 16:52:16 +00:00
|
|
|
exits := []*pb.VoluntaryExit{
|
2018-12-27 03:15:12 +00:00
|
|
|
{
|
2019-02-08 21:53:36 +00:00
|
|
|
ValidatorIndex: 10,
|
2019-02-15 23:19:36 +00:00
|
|
|
Epoch: params.BeaconConfig().GenesisEpoch,
|
2018-12-27 03:15:12 +00:00
|
|
|
},
|
|
|
|
}
|
2019-02-20 05:07:28 +00:00
|
|
|
randaoReveal := createRandaoReveal(t, beaconState, privKeys)
|
2018-12-27 03:15:12 +00:00
|
|
|
block := &pb.BeaconBlock{
|
2019-02-20 05:07:28 +00:00
|
|
|
Slot: params.BeaconConfig().GenesisSlot + 10,
|
|
|
|
RandaoReveal: randaoReveal,
|
2019-02-02 21:24:42 +00:00
|
|
|
Eth1Data: &pb.Eth1Data{
|
|
|
|
DepositRootHash32: []byte{2},
|
|
|
|
BlockHash32: []byte{3},
|
|
|
|
},
|
2018-12-27 03:15:12 +00:00
|
|
|
Body: &pb.BeaconBlockBody{
|
|
|
|
ProposerSlashings: proposerSlashings,
|
2019-01-30 10:11:13 +00:00
|
|
|
AttesterSlashings: attesterSlashings,
|
2018-12-27 03:15:12 +00:00
|
|
|
Attestations: attestations,
|
2019-02-18 16:52:16 +00:00
|
|
|
VoluntaryExits: exits,
|
2018-12-27 03:15:12 +00:00
|
|
|
},
|
|
|
|
}
|
2019-03-18 06:19:44 +00:00
|
|
|
if _, err := state.ProcessBlock(context.Background(), beaconState, block, state.DefaultConfig()); err != nil {
|
2018-12-27 03:15:12 +00:00
|
|
|
t.Errorf("Expected block to pass processing conditions: %v", err)
|
|
|
|
}
|
|
|
|
}
|
2019-01-09 17:26:40 +00:00
|
|
|
|
|
|
|
func TestProcessEpoch_PassesProcessingConditions(t *testing.T) {
|
2019-02-06 21:46:05 +00:00
|
|
|
var validatorRegistry []*pb.Validator
|
2019-02-01 16:52:35 +00:00
|
|
|
for i := uint64(0); i < 10; i++ {
|
|
|
|
validatorRegistry = append(validatorRegistry,
|
2019-02-06 21:46:05 +00:00
|
|
|
&pb.Validator{
|
2019-02-07 06:07:25 +00:00
|
|
|
ExitEpoch: params.BeaconConfig().FarFutureEpoch,
|
2019-02-01 16:52:35 +00:00
|
|
|
})
|
|
|
|
}
|
2019-01-25 16:20:02 +00:00
|
|
|
validatorBalances := make([]uint64, len(validatorRegistry))
|
|
|
|
for i := 0; i < len(validatorBalances); i++ {
|
2019-02-09 13:09:09 +00:00
|
|
|
validatorBalances[i] = params.BeaconConfig().MaxDepositAmount
|
2019-01-09 17:26:40 +00:00
|
|
|
}
|
|
|
|
|
2019-02-12 18:04:30 +00:00
|
|
|
var attestations []*pb.PendingAttestation
|
2019-02-18 16:52:16 +00:00
|
|
|
for i := uint64(0); i < params.BeaconConfig().SlotsPerEpoch*2; i++ {
|
2019-02-12 18:04:30 +00:00
|
|
|
attestations = append(attestations, &pb.PendingAttestation{
|
2019-01-09 17:26:40 +00:00
|
|
|
Data: &pb.AttestationData{
|
2019-02-18 16:52:16 +00:00
|
|
|
Slot: i + params.BeaconConfig().SlotsPerEpoch + params.BeaconConfig().GenesisSlot,
|
2019-01-09 17:26:40 +00:00
|
|
|
Shard: 1,
|
2019-02-15 23:19:36 +00:00
|
|
|
JustifiedEpoch: params.BeaconConfig().GenesisEpoch + 1,
|
2019-01-09 17:26:40 +00:00
|
|
|
JustifiedBlockRootHash32: []byte{0},
|
|
|
|
},
|
2019-02-18 16:52:16 +00:00
|
|
|
InclusionSlot: i + params.BeaconConfig().SlotsPerEpoch + 1 + params.BeaconConfig().GenesisSlot,
|
2019-01-09 17:26:40 +00:00
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
var blockRoots [][]byte
|
2019-02-15 23:19:36 +00:00
|
|
|
for i := uint64(0); i < params.BeaconConfig().LatestBlockRootsLength; i++ {
|
2019-01-09 17:26:40 +00:00
|
|
|
blockRoots = append(blockRoots, []byte{byte(i)})
|
|
|
|
}
|
|
|
|
|
|
|
|
var randaoHashes [][]byte
|
2019-02-18 16:52:16 +00:00
|
|
|
for i := uint64(0); i < params.BeaconConfig().SlotsPerEpoch; i++ {
|
2019-01-09 17:26:40 +00:00
|
|
|
randaoHashes = append(randaoHashes, []byte{byte(i)})
|
|
|
|
}
|
|
|
|
|
2019-03-01 02:47:32 +00:00
|
|
|
crosslinkRecord := make([]*pb.Crosslink, 64)
|
2019-02-20 05:07:28 +00:00
|
|
|
newState := &pb.BeaconState{
|
2019-02-19 20:24:00 +00:00
|
|
|
Slot: params.BeaconConfig().SlotsPerEpoch + params.BeaconConfig().GenesisSlot + 1,
|
|
|
|
LatestAttestations: attestations,
|
|
|
|
ValidatorBalances: validatorBalances,
|
|
|
|
ValidatorRegistry: validatorRegistry,
|
|
|
|
LatestBlockRootHash32S: blockRoots,
|
|
|
|
LatestCrosslinks: crosslinkRecord,
|
|
|
|
LatestRandaoMixes: randaoHashes,
|
2019-02-09 21:14:10 +00:00
|
|
|
LatestIndexRootHash32S: make([][]byte,
|
2019-02-18 16:52:16 +00:00
|
|
|
params.BeaconConfig().LatestActiveIndexRootsLength),
|
|
|
|
LatestSlashedBalances: make([]uint64,
|
|
|
|
params.BeaconConfig().LatestSlashedExitLength),
|
2019-01-09 17:26:40 +00:00
|
|
|
}
|
|
|
|
|
2019-03-18 06:19:44 +00:00
|
|
|
_, err := state.ProcessEpoch(context.Background(), newState, state.DefaultConfig())
|
2019-01-09 17:26:40 +00:00
|
|
|
if err != nil {
|
|
|
|
t.Errorf("Expected epoch transition to pass processing conditions: %v", err)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestProcessEpoch_InactiveConditions(t *testing.T) {
|
2019-02-09 13:09:09 +00:00
|
|
|
defaultBalance := params.BeaconConfig().MaxDepositAmount
|
2019-01-09 17:26:40 +00:00
|
|
|
|
2019-02-06 21:46:05 +00:00
|
|
|
validatorRegistry := []*pb.Validator{
|
2019-02-07 06:07:25 +00:00
|
|
|
{ExitEpoch: params.BeaconConfig().FarFutureEpoch}, {ExitEpoch: params.BeaconConfig().FarFutureEpoch},
|
|
|
|
{ExitEpoch: params.BeaconConfig().FarFutureEpoch}, {ExitEpoch: params.BeaconConfig().FarFutureEpoch},
|
|
|
|
{ExitEpoch: params.BeaconConfig().FarFutureEpoch}, {ExitEpoch: params.BeaconConfig().FarFutureEpoch},
|
|
|
|
{ExitEpoch: params.BeaconConfig().FarFutureEpoch}, {ExitEpoch: params.BeaconConfig().FarFutureEpoch}}
|
2019-01-09 17:26:40 +00:00
|
|
|
|
|
|
|
validatorBalances := []uint64{
|
|
|
|
defaultBalance, defaultBalance, defaultBalance, defaultBalance,
|
|
|
|
defaultBalance, defaultBalance, defaultBalance, defaultBalance,
|
|
|
|
}
|
|
|
|
|
2019-02-12 18:04:30 +00:00
|
|
|
var attestations []*pb.PendingAttestation
|
2019-02-18 16:52:16 +00:00
|
|
|
for i := uint64(0); i < params.BeaconConfig().SlotsPerEpoch*2; i++ {
|
2019-02-12 18:04:30 +00:00
|
|
|
attestations = append(attestations, &pb.PendingAttestation{
|
2019-01-09 17:26:40 +00:00
|
|
|
Data: &pb.AttestationData{
|
2019-02-18 16:52:16 +00:00
|
|
|
Slot: i + params.BeaconConfig().SlotsPerEpoch + params.BeaconConfig().GenesisSlot,
|
2019-01-09 17:26:40 +00:00
|
|
|
Shard: 1,
|
2019-02-15 23:19:36 +00:00
|
|
|
JustifiedEpoch: params.BeaconConfig().GenesisEpoch + 1,
|
2019-01-09 17:26:40 +00:00
|
|
|
JustifiedBlockRootHash32: []byte{0},
|
|
|
|
},
|
2019-02-06 16:20:38 +00:00
|
|
|
AggregationBitfield: []byte{},
|
2019-02-18 16:52:16 +00:00
|
|
|
InclusionSlot: i + params.BeaconConfig().SlotsPerEpoch + 1 + params.BeaconConfig().GenesisSlot,
|
2019-01-09 17:26:40 +00:00
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
var blockRoots [][]byte
|
2019-02-18 16:52:16 +00:00
|
|
|
for i := uint64(0); i < 2*params.BeaconConfig().SlotsPerEpoch; i++ {
|
2019-01-09 17:26:40 +00:00
|
|
|
blockRoots = append(blockRoots, []byte{byte(i)})
|
|
|
|
}
|
|
|
|
|
|
|
|
var randaoHashes [][]byte
|
2019-02-18 16:52:16 +00:00
|
|
|
for i := uint64(0); i < 5*params.BeaconConfig().SlotsPerEpoch; i++ {
|
2019-01-09 17:26:40 +00:00
|
|
|
randaoHashes = append(randaoHashes, []byte{byte(i)})
|
|
|
|
}
|
|
|
|
|
2019-03-01 02:47:32 +00:00
|
|
|
crosslinkRecord := make([]*pb.Crosslink, 64)
|
2019-01-09 17:26:40 +00:00
|
|
|
|
2019-02-20 05:07:28 +00:00
|
|
|
newState := &pb.BeaconState{
|
2019-02-19 20:24:00 +00:00
|
|
|
Slot: params.BeaconConfig().SlotsPerEpoch + params.BeaconConfig().GenesisSlot + 1,
|
|
|
|
LatestAttestations: attestations,
|
|
|
|
ValidatorBalances: validatorBalances,
|
|
|
|
ValidatorRegistry: validatorRegistry,
|
|
|
|
LatestBlockRootHash32S: blockRoots,
|
|
|
|
LatestCrosslinks: crosslinkRecord,
|
|
|
|
LatestRandaoMixes: randaoHashes,
|
2019-02-09 21:14:10 +00:00
|
|
|
LatestIndexRootHash32S: make([][]byte,
|
2019-02-18 16:52:16 +00:00
|
|
|
params.BeaconConfig().LatestActiveIndexRootsLength),
|
|
|
|
LatestSlashedBalances: make([]uint64,
|
|
|
|
params.BeaconConfig().LatestSlashedExitLength),
|
2019-01-09 17:26:40 +00:00
|
|
|
}
|
|
|
|
|
2019-03-18 06:19:44 +00:00
|
|
|
_, err := state.ProcessEpoch(context.Background(), newState, state.DefaultConfig())
|
2019-01-09 17:26:40 +00:00
|
|
|
if err != nil {
|
|
|
|
t.Errorf("Expected epoch transition to pass processing conditions: %v", err)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestProcessEpoch_CantGetBoundaryAttestation(t *testing.T) {
|
2019-02-20 05:07:28 +00:00
|
|
|
newState := &pb.BeaconState{
|
2019-02-18 16:52:16 +00:00
|
|
|
Slot: params.BeaconConfig().GenesisSlot + params.BeaconConfig().SlotsPerEpoch,
|
2019-02-12 18:04:30 +00:00
|
|
|
LatestAttestations: []*pb.PendingAttestation{
|
2019-02-15 23:19:36 +00:00
|
|
|
{Data: &pb.AttestationData{Slot: params.BeaconConfig().GenesisSlot + 100}},
|
2019-01-09 17:26:40 +00:00
|
|
|
}}
|
|
|
|
|
|
|
|
want := fmt.Sprintf(
|
2019-02-05 17:55:15 +00:00
|
|
|
"could not get current boundary attestations: slot %d is not within expected range of %d to %d",
|
2019-02-25 02:09:45 +00:00
|
|
|
newState.Slot-params.BeaconConfig().GenesisSlot,
|
|
|
|
0,
|
|
|
|
newState.Slot-params.BeaconConfig().GenesisSlot,
|
2019-01-09 17:26:40 +00:00
|
|
|
)
|
2019-03-18 06:19:44 +00:00
|
|
|
if _, err := state.ProcessEpoch(context.Background(), newState, state.DefaultConfig()); !strings.Contains(err.Error(), want) {
|
2019-01-09 17:26:40 +00:00
|
|
|
t.Errorf("Expected: %s, received: %v", want, err)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestProcessEpoch_CantGetCurrentValidatorIndices(t *testing.T) {
|
2019-02-07 06:07:25 +00:00
|
|
|
latestBlockRoots := make([][]byte, params.BeaconConfig().LatestBlockRootsLength)
|
2019-01-09 17:26:40 +00:00
|
|
|
for i := 0; i < len(latestBlockRoots); i++ {
|
2019-02-07 06:07:25 +00:00
|
|
|
latestBlockRoots[i] = params.BeaconConfig().ZeroHash[:]
|
2019-01-09 17:26:40 +00:00
|
|
|
}
|
|
|
|
|
2019-02-12 18:04:30 +00:00
|
|
|
var attestations []*pb.PendingAttestation
|
2019-02-18 16:52:16 +00:00
|
|
|
for i := uint64(0); i < params.BeaconConfig().SlotsPerEpoch*2; i++ {
|
2019-02-12 18:04:30 +00:00
|
|
|
attestations = append(attestations, &pb.PendingAttestation{
|
2019-01-09 17:26:40 +00:00
|
|
|
Data: &pb.AttestationData{
|
2019-02-15 23:19:36 +00:00
|
|
|
Slot: params.BeaconConfig().GenesisSlot + 1,
|
2019-01-09 17:26:40 +00:00
|
|
|
Shard: 1,
|
|
|
|
JustifiedBlockRootHash32: make([]byte, 32),
|
|
|
|
},
|
2019-02-06 16:20:38 +00:00
|
|
|
AggregationBitfield: []byte{0xff},
|
2019-01-09 17:26:40 +00:00
|
|
|
})
|
|
|
|
}
|
|
|
|
|
2019-02-20 05:07:28 +00:00
|
|
|
newState := &pb.BeaconState{
|
2019-02-18 16:52:16 +00:00
|
|
|
Slot: params.BeaconConfig().GenesisSlot + params.BeaconConfig().SlotsPerEpoch,
|
2019-01-09 17:26:40 +00:00
|
|
|
LatestAttestations: attestations,
|
|
|
|
LatestBlockRootHash32S: latestBlockRoots,
|
|
|
|
}
|
|
|
|
|
2019-01-29 13:56:32 +00:00
|
|
|
wanted := fmt.Sprintf("wanted participants bitfield length %d, got: %d", 0, 1)
|
2019-03-18 06:19:44 +00:00
|
|
|
if _, err := state.ProcessEpoch(context.Background(), newState, state.DefaultConfig()); !strings.Contains(err.Error(), wanted) {
|
2019-01-29 13:56:32 +00:00
|
|
|
t.Errorf("Expected: %s, received: %v", wanted, err)
|
2019-01-09 17:26:40 +00:00
|
|
|
}
|
|
|
|
}
|