mirror of
https://gitlab.com/pulsechaincom/prysm-pulse.git
synced 2025-01-19 08:14:15 +00:00
528 lines
14 KiB
Go
528 lines
14 KiB
Go
package state
|
|
|
|
import (
|
|
"fmt"
|
|
"strings"
|
|
"testing"
|
|
|
|
pb "github.com/prysmaticlabs/prysm/proto/beacon/p2p/v1"
|
|
"github.com/prysmaticlabs/prysm/shared/params"
|
|
)
|
|
|
|
func TestProcessBlock_IncorrectSlot(t *testing.T) {
|
|
beaconState := &pb.BeaconState{
|
|
Slot: 5,
|
|
}
|
|
block := &pb.BeaconBlock{
|
|
Slot: 4,
|
|
}
|
|
want := fmt.Sprintf(
|
|
"block.slot != state.slot, block.slot = %d, state.slot = %d",
|
|
4,
|
|
5,
|
|
)
|
|
if _, err := ProcessBlock(beaconState, block); !strings.Contains(err.Error(), want) {
|
|
t.Errorf("Expected %s, received %v", want, err)
|
|
}
|
|
}
|
|
|
|
func TestProcessBlock_IncorrectBlockRandao(t *testing.T) {
|
|
registry := []*pb.ValidatorRecord{
|
|
{
|
|
Status: pb.ValidatorRecord_ACTIVE,
|
|
LatestStatusChangeSlot: 0,
|
|
RandaoCommitmentHash32: []byte{0},
|
|
RandaoLayers: 0,
|
|
},
|
|
{
|
|
Status: pb.ValidatorRecord_ACTIVE,
|
|
LatestStatusChangeSlot: 0,
|
|
RandaoCommitmentHash32: []byte{0},
|
|
RandaoLayers: 0,
|
|
},
|
|
}
|
|
beaconState := &pb.BeaconState{
|
|
Slot: 0,
|
|
ValidatorRegistry: registry,
|
|
ShardAndCommitteesAtSlots: []*pb.ShardAndCommitteeArray{
|
|
{
|
|
ArrayShardAndCommittee: []*pb.ShardAndCommittee{
|
|
{
|
|
Shard: 0,
|
|
Committee: []uint32{0, 1},
|
|
},
|
|
},
|
|
},
|
|
},
|
|
}
|
|
block := &pb.BeaconBlock{
|
|
Slot: 0,
|
|
RandaoRevealHash32: []byte{1},
|
|
Body: &pb.BeaconBlockBody{},
|
|
}
|
|
want := "could not verify and process block randao"
|
|
if _, err := ProcessBlock(beaconState, block); !strings.Contains(err.Error(), want) {
|
|
t.Errorf("Expected %s, received %v", want, err)
|
|
}
|
|
}
|
|
|
|
func TestProcessBlock_IncorrectProposerSlashing(t *testing.T) {
|
|
registry := []*pb.ValidatorRecord{
|
|
{
|
|
Status: pb.ValidatorRecord_ACTIVE,
|
|
LatestStatusChangeSlot: 0,
|
|
RandaoCommitmentHash32: []byte{1},
|
|
RandaoLayers: 0,
|
|
},
|
|
{
|
|
Status: pb.ValidatorRecord_ACTIVE,
|
|
LatestStatusChangeSlot: 0,
|
|
RandaoCommitmentHash32: []byte{1},
|
|
RandaoLayers: 0,
|
|
},
|
|
}
|
|
slashings := make([]*pb.ProposerSlashing, params.BeaconConfig().MaxProposerSlashings+1)
|
|
shardAndCommittees := make([]*pb.ShardAndCommitteeArray, 64)
|
|
shardAndCommittees[5] = &pb.ShardAndCommitteeArray{
|
|
ArrayShardAndCommittee: []*pb.ShardAndCommittee{
|
|
{
|
|
Shard: 0,
|
|
Committee: []uint32{0, 1},
|
|
},
|
|
},
|
|
}
|
|
latestMixes := make([][]byte, params.BeaconConfig().LatestRandaoMixesLength)
|
|
beaconState := &pb.BeaconState{
|
|
LatestRandaoMixesHash32S: latestMixes,
|
|
ValidatorRegistry: registry,
|
|
ShardAndCommitteesAtSlots: shardAndCommittees,
|
|
Slot: 5,
|
|
}
|
|
block := &pb.BeaconBlock{
|
|
Slot: 5,
|
|
RandaoRevealHash32: []byte{1},
|
|
Body: &pb.BeaconBlockBody{
|
|
ProposerSlashings: slashings,
|
|
},
|
|
}
|
|
want := "could not verify block proposer slashing"
|
|
if _, err := ProcessBlock(beaconState, block); !strings.Contains(err.Error(), want) {
|
|
t.Errorf("Expected %s, received %v", want, err)
|
|
}
|
|
}
|
|
|
|
func TestProcessBlock_IncorrectCasperSlashing(t *testing.T) {
|
|
registry := []*pb.ValidatorRecord{
|
|
{
|
|
Status: pb.ValidatorRecord_ACTIVE,
|
|
LatestStatusChangeSlot: 0,
|
|
RandaoCommitmentHash32: []byte{1},
|
|
RandaoLayers: 0,
|
|
},
|
|
{
|
|
Status: pb.ValidatorRecord_ACTIVE,
|
|
LatestStatusChangeSlot: 0,
|
|
RandaoCommitmentHash32: []byte{1},
|
|
RandaoLayers: 0,
|
|
},
|
|
}
|
|
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},
|
|
},
|
|
},
|
|
}
|
|
casperSlashings := make([]*pb.CasperSlashing, params.BeaconConfig().MaxCasperSlashings+1)
|
|
shardAndCommittees := make([]*pb.ShardAndCommitteeArray, 64)
|
|
shardAndCommittees[5] = &pb.ShardAndCommitteeArray{
|
|
ArrayShardAndCommittee: []*pb.ShardAndCommittee{
|
|
{
|
|
Shard: 0,
|
|
Committee: []uint32{0, 1},
|
|
},
|
|
},
|
|
}
|
|
latestMixes := make([][]byte, params.BeaconConfig().LatestRandaoMixesLength)
|
|
beaconState := &pb.BeaconState{
|
|
LatestRandaoMixesHash32S: latestMixes,
|
|
Slot: 5,
|
|
ValidatorRegistry: registry,
|
|
ShardAndCommitteesAtSlots: shardAndCommittees,
|
|
}
|
|
block := &pb.BeaconBlock{
|
|
Slot: 5,
|
|
RandaoRevealHash32: []byte{1},
|
|
Body: &pb.BeaconBlockBody{
|
|
ProposerSlashings: slashings,
|
|
CasperSlashings: casperSlashings,
|
|
},
|
|
}
|
|
want := "could not verify block casper slashing"
|
|
if _, err := ProcessBlock(beaconState, block); !strings.Contains(err.Error(), want) {
|
|
t.Errorf("Expected %s, received %v", want, err)
|
|
}
|
|
}
|
|
|
|
func TestProcessBlock_IncorrectProcessBlockAttestations(t *testing.T) {
|
|
registry := []*pb.ValidatorRecord{
|
|
{
|
|
Status: pb.ValidatorRecord_ACTIVE,
|
|
LatestStatusChangeSlot: 0,
|
|
RandaoCommitmentHash32: []byte{1},
|
|
RandaoLayers: 0,
|
|
},
|
|
{
|
|
Status: pb.ValidatorRecord_ACTIVE,
|
|
LatestStatusChangeSlot: 0,
|
|
RandaoCommitmentHash32: []byte{1},
|
|
RandaoLayers: 0,
|
|
},
|
|
}
|
|
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{
|
|
Slot: 5,
|
|
JustifiedSlot: 5,
|
|
}
|
|
att2 := &pb.AttestationData{
|
|
Slot: 5,
|
|
JustifiedSlot: 4,
|
|
}
|
|
casperSlashings := []*pb.CasperSlashing{
|
|
{
|
|
Votes_1: &pb.SlashableVoteData{
|
|
Data: att1,
|
|
AggregateSignaturePoc_0Indices: []uint32{0, 1},
|
|
AggregateSignaturePoc_1Indices: []uint32{2, 3},
|
|
},
|
|
Votes_2: &pb.SlashableVoteData{
|
|
Data: att2,
|
|
AggregateSignaturePoc_0Indices: []uint32{4, 5},
|
|
AggregateSignaturePoc_1Indices: []uint32{6, 1},
|
|
},
|
|
},
|
|
}
|
|
blockAttestations := make([]*pb.Attestation, params.BeaconConfig().MaxAttestations+1)
|
|
shardAndCommittees := make([]*pb.ShardAndCommitteeArray, 64)
|
|
shardAndCommittees[5] = &pb.ShardAndCommitteeArray{
|
|
ArrayShardAndCommittee: []*pb.ShardAndCommittee{
|
|
{
|
|
Shard: 0,
|
|
Committee: []uint32{0, 1},
|
|
},
|
|
},
|
|
}
|
|
latestMixes := make([][]byte, params.BeaconConfig().LatestRandaoMixesLength)
|
|
beaconState := &pb.BeaconState{
|
|
LatestRandaoMixesHash32S: latestMixes,
|
|
Slot: 5,
|
|
ValidatorRegistry: registry,
|
|
ShardAndCommitteesAtSlots: shardAndCommittees,
|
|
}
|
|
block := &pb.BeaconBlock{
|
|
Slot: 5,
|
|
RandaoRevealHash32: []byte{1},
|
|
Body: &pb.BeaconBlockBody{
|
|
ProposerSlashings: proposerSlashings,
|
|
CasperSlashings: casperSlashings,
|
|
Attestations: blockAttestations,
|
|
},
|
|
}
|
|
want := "could not process block attestations"
|
|
if _, err := ProcessBlock(beaconState, block); !strings.Contains(err.Error(), want) {
|
|
t.Errorf("Expected %s, received %v", want, err)
|
|
}
|
|
}
|
|
|
|
func TestProcessBlock_IncorrectProcessExits(t *testing.T) {
|
|
registry := []*pb.ValidatorRecord{
|
|
{
|
|
Status: pb.ValidatorRecord_ACTIVE,
|
|
LatestStatusChangeSlot: 0,
|
|
RandaoCommitmentHash32: []byte{1},
|
|
RandaoLayers: 0,
|
|
},
|
|
{
|
|
Status: pb.ValidatorRecord_ACTIVE,
|
|
LatestStatusChangeSlot: 0,
|
|
RandaoCommitmentHash32: []byte{1},
|
|
RandaoLayers: 0,
|
|
},
|
|
}
|
|
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{
|
|
Slot: 5,
|
|
JustifiedSlot: 5,
|
|
}
|
|
att2 := &pb.AttestationData{
|
|
Slot: 5,
|
|
JustifiedSlot: 4,
|
|
}
|
|
casperSlashings := []*pb.CasperSlashing{
|
|
{
|
|
Votes_1: &pb.SlashableVoteData{
|
|
Data: att1,
|
|
AggregateSignaturePoc_0Indices: []uint32{0, 1},
|
|
AggregateSignaturePoc_1Indices: []uint32{2, 3},
|
|
},
|
|
Votes_2: &pb.SlashableVoteData{
|
|
Data: att2,
|
|
AggregateSignaturePoc_0Indices: []uint32{4, 5},
|
|
AggregateSignaturePoc_1Indices: []uint32{6, 1},
|
|
},
|
|
},
|
|
}
|
|
var blockRoots [][]byte
|
|
for i := uint64(0); i < 2*params.BeaconConfig().EpochLength; i++ {
|
|
blockRoots = append(blockRoots, []byte{byte(i)})
|
|
}
|
|
stateLatestCrosslinks := []*pb.CrosslinkRecord{
|
|
{
|
|
ShardBlockRootHash32: []byte{1},
|
|
},
|
|
}
|
|
blockAtt := &pb.Attestation{
|
|
Data: &pb.AttestationData{
|
|
Shard: 0,
|
|
Slot: 20,
|
|
JustifiedSlot: 10,
|
|
JustifiedBlockRootHash32: blockRoots[10],
|
|
LatestCrosslinkRootHash32: []byte{1},
|
|
ShardBlockRootHash32: []byte{},
|
|
},
|
|
ParticipationBitfield: []byte{1},
|
|
CustodyBitfield: []byte{1},
|
|
}
|
|
attestations := []*pb.Attestation{blockAtt}
|
|
shardAndCommittees := make([]*pb.ShardAndCommitteeArray, 128)
|
|
shardAndCommittees[64] = &pb.ShardAndCommitteeArray{
|
|
ArrayShardAndCommittee: []*pb.ShardAndCommittee{
|
|
{
|
|
Shard: 0,
|
|
Committee: []uint32{0, 1},
|
|
},
|
|
},
|
|
}
|
|
latestMixes := make([][]byte, params.BeaconConfig().LatestRandaoMixesLength)
|
|
beaconState := &pb.BeaconState{
|
|
LatestRandaoMixesHash32S: latestMixes,
|
|
ValidatorRegistry: registry,
|
|
Slot: 64,
|
|
PreviousJustifiedSlot: 10,
|
|
LatestBlockRootHash32S: blockRoots,
|
|
LatestCrosslinks: stateLatestCrosslinks,
|
|
ShardAndCommitteesAtSlots: shardAndCommittees,
|
|
}
|
|
exits := make([]*pb.Exit, params.BeaconConfig().MaxExits+1)
|
|
block := &pb.BeaconBlock{
|
|
Slot: 64,
|
|
RandaoRevealHash32: []byte{1},
|
|
Body: &pb.BeaconBlockBody{
|
|
ProposerSlashings: proposerSlashings,
|
|
CasperSlashings: casperSlashings,
|
|
Attestations: attestations,
|
|
Exits: exits,
|
|
},
|
|
}
|
|
want := "could not process validator exits"
|
|
if _, err := ProcessBlock(beaconState, block); !strings.Contains(err.Error(), want) {
|
|
t.Errorf("Expected %s, received %v", want, err)
|
|
}
|
|
}
|
|
|
|
func TestProcessBlock_PassesProcessingConditions(t *testing.T) {
|
|
registry := []*pb.ValidatorRecord{
|
|
{
|
|
Status: pb.ValidatorRecord_ACTIVE,
|
|
LatestStatusChangeSlot: 0,
|
|
RandaoCommitmentHash32: []byte{1},
|
|
RandaoLayers: 0,
|
|
},
|
|
{
|
|
Status: pb.ValidatorRecord_ACTIVE,
|
|
LatestStatusChangeSlot: 0,
|
|
RandaoCommitmentHash32: []byte{1},
|
|
RandaoLayers: 0,
|
|
},
|
|
}
|
|
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{
|
|
Slot: 5,
|
|
JustifiedSlot: 5,
|
|
}
|
|
att2 := &pb.AttestationData{
|
|
Slot: 5,
|
|
JustifiedSlot: 4,
|
|
}
|
|
casperSlashings := []*pb.CasperSlashing{
|
|
{
|
|
Votes_1: &pb.SlashableVoteData{
|
|
Data: att1,
|
|
AggregateSignaturePoc_0Indices: []uint32{0, 1},
|
|
AggregateSignaturePoc_1Indices: []uint32{2, 3},
|
|
},
|
|
Votes_2: &pb.SlashableVoteData{
|
|
Data: att2,
|
|
AggregateSignaturePoc_0Indices: []uint32{4, 5},
|
|
AggregateSignaturePoc_1Indices: []uint32{6, 1},
|
|
},
|
|
},
|
|
}
|
|
var blockRoots [][]byte
|
|
for i := uint64(0); i < 2*params.BeaconConfig().EpochLength; i++ {
|
|
blockRoots = append(blockRoots, []byte{byte(i)})
|
|
}
|
|
stateLatestCrosslinks := []*pb.CrosslinkRecord{
|
|
{
|
|
ShardBlockRootHash32: []byte{1},
|
|
},
|
|
}
|
|
blockAtt := &pb.Attestation{
|
|
Data: &pb.AttestationData{
|
|
Shard: 0,
|
|
Slot: 20,
|
|
JustifiedSlot: 10,
|
|
JustifiedBlockRootHash32: blockRoots[10],
|
|
LatestCrosslinkRootHash32: []byte{1},
|
|
ShardBlockRootHash32: []byte{},
|
|
},
|
|
ParticipationBitfield: []byte{1},
|
|
CustodyBitfield: []byte{1},
|
|
}
|
|
attestations := []*pb.Attestation{blockAtt}
|
|
shardAndCommittees := make([]*pb.ShardAndCommitteeArray, 128)
|
|
shardAndCommittees[64] = &pb.ShardAndCommitteeArray{
|
|
ArrayShardAndCommittee: []*pb.ShardAndCommittee{
|
|
{
|
|
Shard: 0,
|
|
Committee: []uint32{0, 1},
|
|
},
|
|
},
|
|
}
|
|
latestMixes := make([][]byte, params.BeaconConfig().LatestRandaoMixesLength)
|
|
beaconState := &pb.BeaconState{
|
|
LatestRandaoMixesHash32S: latestMixes,
|
|
ValidatorRegistry: registry,
|
|
Slot: 64,
|
|
ShardAndCommitteesAtSlots: shardAndCommittees,
|
|
PreviousJustifiedSlot: 10,
|
|
LatestBlockRootHash32S: blockRoots,
|
|
LatestCrosslinks: stateLatestCrosslinks,
|
|
}
|
|
exits := []*pb.Exit{
|
|
{
|
|
ValidatorIndex: 0,
|
|
Slot: 0,
|
|
},
|
|
}
|
|
block := &pb.BeaconBlock{
|
|
Slot: 64,
|
|
RandaoRevealHash32: []byte{1},
|
|
Body: &pb.BeaconBlockBody{
|
|
ProposerSlashings: proposerSlashings,
|
|
CasperSlashings: casperSlashings,
|
|
Attestations: attestations,
|
|
Exits: exits,
|
|
},
|
|
}
|
|
if _, err := ProcessBlock(beaconState, block); err != nil {
|
|
t.Errorf("Expected block to pass processing conditions: %v", err)
|
|
}
|
|
}
|
|
|
|
func TestIsNewValidatorSetTransition(t *testing.T) {
|
|
beaconState, err := NewGenesisBeaconState(nil)
|
|
if err != nil {
|
|
t.Fatalf("Failed to initialize state: %v", err)
|
|
}
|
|
beaconState.ValidatorRegistryLastChangeSlot = 1
|
|
if IsValidatorSetChange(beaconState, 0) {
|
|
t.Errorf("Is new validator set change should be false, last changed slot greater than finalized slot")
|
|
}
|
|
beaconState.FinalizedSlot = 2
|
|
if IsValidatorSetChange(beaconState, 2) {
|
|
t.Errorf("Is new validator set change should be false, MinValidatorSetChangeInterval has not reached")
|
|
}
|
|
shardCommitteeForSlots := []*pb.ShardAndCommitteeArray{{
|
|
ArrayShardAndCommittee: []*pb.ShardAndCommittee{
|
|
{Shard: 0},
|
|
{Shard: 1},
|
|
{Shard: 2},
|
|
},
|
|
},
|
|
}
|
|
beaconState.ShardAndCommitteesAtSlots = shardCommitteeForSlots
|
|
|
|
crosslinks := []*pb.CrosslinkRecord{
|
|
{Slot: 1},
|
|
{Slot: 1},
|
|
{Slot: 1},
|
|
}
|
|
beaconState.LatestCrosslinks = crosslinks
|
|
|
|
if IsValidatorSetChange(beaconState, params.BeaconConfig().MinValidatorSetChangeInterval+1) {
|
|
t.Errorf("Is new validator set change should be false, crosslink slot record is higher than current slot")
|
|
}
|
|
|
|
crosslinks = []*pb.CrosslinkRecord{
|
|
{Slot: 2},
|
|
{Slot: 2},
|
|
{Slot: 2},
|
|
}
|
|
beaconState.LatestCrosslinks = crosslinks
|
|
|
|
if !IsValidatorSetChange(beaconState, params.BeaconConfig().MinValidatorSetChangeInterval+1) {
|
|
t.Errorf("New validator set change failed should have been true")
|
|
}
|
|
}
|