package beaconv1 import ( "context" "testing" "github.com/gogo/protobuf/types" eth2types "github.com/prysmaticlabs/eth2-types" ethpb "github.com/prysmaticlabs/ethereumapis/eth/v1" eth "github.com/prysmaticlabs/ethereumapis/eth/v1alpha1" chainMock "github.com/prysmaticlabs/prysm/beacon-chain/blockchain/testing" "github.com/prysmaticlabs/prysm/beacon-chain/core/helpers" "github.com/prysmaticlabs/prysm/beacon-chain/operations/slashings" "github.com/prysmaticlabs/prysm/beacon-chain/operations/voluntaryexits" p2pMock "github.com/prysmaticlabs/prysm/beacon-chain/p2p/testing" pb "github.com/prysmaticlabs/prysm/proto/beacon/p2p/v1" "github.com/prysmaticlabs/prysm/proto/migration" "github.com/prysmaticlabs/prysm/shared/bls" "github.com/prysmaticlabs/prysm/shared/bytesutil" "github.com/prysmaticlabs/prysm/shared/params" "github.com/prysmaticlabs/prysm/shared/testutil" "github.com/prysmaticlabs/prysm/shared/testutil/assert" "github.com/prysmaticlabs/prysm/shared/testutil/require" ) func TestListPoolAttesterSlashings(t *testing.T) { state, err := testutil.NewBeaconState() require.NoError(t, err) slashing1 := ð.AttesterSlashing{ Attestation_1: ð.IndexedAttestation{ AttestingIndices: []uint64{1, 10}, Data: ð.AttestationData{ Slot: 1, CommitteeIndex: 1, BeaconBlockRoot: bytesutil.PadTo([]byte("blockroot1"), 32), Source: ð.Checkpoint{ Epoch: 1, Root: bytesutil.PadTo([]byte("sourceroot1"), 32), }, Target: ð.Checkpoint{ Epoch: 10, Root: bytesutil.PadTo([]byte("targetroot1"), 32), }, }, Signature: bytesutil.PadTo([]byte("signature1"), 96), }, Attestation_2: ð.IndexedAttestation{ AttestingIndices: []uint64{2, 20}, Data: ð.AttestationData{ Slot: 2, CommitteeIndex: 2, BeaconBlockRoot: bytesutil.PadTo([]byte("blockroot2"), 32), Source: ð.Checkpoint{ Epoch: 2, Root: bytesutil.PadTo([]byte("sourceroot2"), 32), }, Target: ð.Checkpoint{ Epoch: 20, Root: bytesutil.PadTo([]byte("targetroot2"), 32), }, }, Signature: bytesutil.PadTo([]byte("signature2"), 96), }, } slashing2 := ð.AttesterSlashing{ Attestation_1: ð.IndexedAttestation{ AttestingIndices: []uint64{3, 30}, Data: ð.AttestationData{ Slot: 3, CommitteeIndex: 3, BeaconBlockRoot: bytesutil.PadTo([]byte("blockroot3"), 32), Source: ð.Checkpoint{ Epoch: 3, Root: bytesutil.PadTo([]byte("sourceroot3"), 32), }, Target: ð.Checkpoint{ Epoch: 30, Root: bytesutil.PadTo([]byte("targetroot3"), 32), }, }, Signature: bytesutil.PadTo([]byte("signature3"), 96), }, Attestation_2: ð.IndexedAttestation{ AttestingIndices: []uint64{4, 40}, Data: ð.AttestationData{ Slot: 4, CommitteeIndex: 4, BeaconBlockRoot: bytesutil.PadTo([]byte("blockroot4"), 32), Source: ð.Checkpoint{ Epoch: 4, Root: bytesutil.PadTo([]byte("sourceroot4"), 32), }, Target: ð.Checkpoint{ Epoch: 40, Root: bytesutil.PadTo([]byte("targetroot4"), 32), }, }, Signature: bytesutil.PadTo([]byte("signature4"), 96), }, } s := &Server{ ChainInfoFetcher: &chainMock.ChainService{State: state}, SlashingsPool: &slashings.PoolMock{PendingAttSlashings: []*eth.AttesterSlashing{slashing1, slashing2}}, } resp, err := s.ListPoolAttesterSlashings(context.Background(), &types.Empty{}) require.NoError(t, err) require.Equal(t, 2, len(resp.Data)) assert.DeepEqual(t, migration.V1Alpha1AttSlashingToV1(slashing1), resp.Data[0]) assert.DeepEqual(t, migration.V1Alpha1AttSlashingToV1(slashing2), resp.Data[1]) } func TestListPoolProposerSlashings(t *testing.T) { state, err := testutil.NewBeaconState() require.NoError(t, err) slashing1 := ð.ProposerSlashing{ Header_1: ð.SignedBeaconBlockHeader{ Header: ð.BeaconBlockHeader{ Slot: 1, ProposerIndex: 1, ParentRoot: bytesutil.PadTo([]byte("parentroot1"), 32), StateRoot: bytesutil.PadTo([]byte("stateroot1"), 32), BodyRoot: bytesutil.PadTo([]byte("bodyroot1"), 32), }, Signature: bytesutil.PadTo([]byte("signature1"), 96), }, Header_2: ð.SignedBeaconBlockHeader{ Header: ð.BeaconBlockHeader{ Slot: 2, ProposerIndex: 2, ParentRoot: bytesutil.PadTo([]byte("parentroot2"), 32), StateRoot: bytesutil.PadTo([]byte("stateroot2"), 32), BodyRoot: bytesutil.PadTo([]byte("bodyroot2"), 32), }, Signature: bytesutil.PadTo([]byte("signature2"), 96), }, } slashing2 := ð.ProposerSlashing{ Header_1: ð.SignedBeaconBlockHeader{ Header: ð.BeaconBlockHeader{ Slot: 3, ProposerIndex: 3, ParentRoot: bytesutil.PadTo([]byte("parentroot3"), 32), StateRoot: bytesutil.PadTo([]byte("stateroot3"), 32), BodyRoot: bytesutil.PadTo([]byte("bodyroot3"), 32), }, Signature: bytesutil.PadTo([]byte("signature3"), 96), }, Header_2: ð.SignedBeaconBlockHeader{ Header: ð.BeaconBlockHeader{ Slot: 4, ProposerIndex: 4, ParentRoot: bytesutil.PadTo([]byte("parentroot4"), 32), StateRoot: bytesutil.PadTo([]byte("stateroot4"), 32), BodyRoot: bytesutil.PadTo([]byte("bodyroot4"), 32), }, Signature: bytesutil.PadTo([]byte("signature4"), 96), }, } s := &Server{ ChainInfoFetcher: &chainMock.ChainService{State: state}, SlashingsPool: &slashings.PoolMock{PendingPropSlashings: []*eth.ProposerSlashing{slashing1, slashing2}}, } resp, err := s.ListPoolProposerSlashings(context.Background(), &types.Empty{}) require.NoError(t, err) require.Equal(t, 2, len(resp.Data)) assert.DeepEqual(t, migration.V1Alpha1ProposerSlashingToV1(slashing1), resp.Data[0]) assert.DeepEqual(t, migration.V1Alpha1ProposerSlashingToV1(slashing2), resp.Data[1]) } func TestListPoolVoluntaryExits(t *testing.T) { state, err := testutil.NewBeaconState() require.NoError(t, err) exit1 := ð.SignedVoluntaryExit{ Exit: ð.VoluntaryExit{ Epoch: 1, ValidatorIndex: 1, }, Signature: bytesutil.PadTo([]byte("signature1"), 96), } exit2 := ð.SignedVoluntaryExit{ Exit: ð.VoluntaryExit{ Epoch: 2, ValidatorIndex: 2, }, Signature: bytesutil.PadTo([]byte("signature2"), 96), } s := &Server{ ChainInfoFetcher: &chainMock.ChainService{State: state}, VoluntaryExitsPool: &voluntaryexits.PoolMock{Exits: []*eth.SignedVoluntaryExit{exit1, exit2}}, } resp, err := s.ListPoolVoluntaryExits(context.Background(), &types.Empty{}) require.NoError(t, err) require.Equal(t, 2, len(resp.Data)) assert.DeepEqual(t, migration.V1Alpha1ExitToV1(exit1), resp.Data[0]) assert.DeepEqual(t, migration.V1Alpha1ExitToV1(exit2), resp.Data[1]) } func TestSubmitAttesterSlashing_Ok(t *testing.T) { ctx := context.Background() _, keys, err := testutil.DeterministicDepositsAndKeys(1) require.NoError(t, err) validator := ð.Validator{ PublicKey: keys[0].PublicKey().Marshal(), } state, err := testutil.NewBeaconState(func(state *pb.BeaconState) { state.Validators = []*eth.Validator{validator} }) require.NoError(t, err) slashing := ðpb.AttesterSlashing{ Attestation_1: ðpb.IndexedAttestation{ AttestingIndices: []uint64{0}, Data: ðpb.AttestationData{ Slot: 1, CommitteeIndex: 1, BeaconBlockRoot: bytesutil.PadTo([]byte("blockroot1"), 32), Source: ðpb.Checkpoint{ Epoch: 1, Root: bytesutil.PadTo([]byte("sourceroot1"), 32), }, Target: ðpb.Checkpoint{ Epoch: 10, Root: bytesutil.PadTo([]byte("targetroot1"), 32), }, }, Signature: make([]byte, 96), }, Attestation_2: ðpb.IndexedAttestation{ AttestingIndices: []uint64{0}, Data: ðpb.AttestationData{ Slot: 1, CommitteeIndex: 1, BeaconBlockRoot: bytesutil.PadTo([]byte("blockroot2"), 32), Source: ðpb.Checkpoint{ Epoch: 1, Root: bytesutil.PadTo([]byte("sourceroot2"), 32), }, Target: ðpb.Checkpoint{ Epoch: 10, Root: bytesutil.PadTo([]byte("targetroot2"), 32), }, }, Signature: make([]byte, 96), }, } for _, att := range []*ethpb.IndexedAttestation{slashing.Attestation_1, slashing.Attestation_2} { sb, err := helpers.ComputeDomainAndSign(state, att.Data.Target.Epoch, att.Data, params.BeaconConfig().DomainBeaconAttester, keys[0]) require.NoError(t, err) sig, err := bls.SignatureFromBytes(sb) require.NoError(t, err) att.Signature = sig.Marshal() } broadcaster := &p2pMock.MockBroadcaster{} s := &Server{ ChainInfoFetcher: &chainMock.ChainService{State: state}, SlashingsPool: &slashings.PoolMock{}, Broadcaster: broadcaster, } _, err = s.SubmitAttesterSlashing(ctx, slashing) require.NoError(t, err) pendingSlashings := s.SlashingsPool.PendingAttesterSlashings(ctx, state, true) require.Equal(t, 1, len(pendingSlashings)) assert.DeepEqual(t, migration.V1AttSlashingToV1Alpha1(slashing), pendingSlashings[0]) assert.Equal(t, true, broadcaster.BroadcastCalled) } func TestSubmitAttesterSlashing_InvalidSlashing(t *testing.T) { ctx := context.Background() state, err := testutil.NewBeaconState() require.NoError(t, err) attestation := ðpb.IndexedAttestation{ AttestingIndices: []uint64{0}, Data: ðpb.AttestationData{ Slot: 1, CommitteeIndex: 1, BeaconBlockRoot: bytesutil.PadTo([]byte("blockroot1"), 32), Source: ðpb.Checkpoint{ Epoch: 1, Root: bytesutil.PadTo([]byte("sourceroot1"), 32), }, Target: ðpb.Checkpoint{ Epoch: 10, Root: bytesutil.PadTo([]byte("targetroot1"), 32), }, }, Signature: make([]byte, 96), } slashing := ðpb.AttesterSlashing{ Attestation_1: attestation, Attestation_2: attestation, } broadcaster := &p2pMock.MockBroadcaster{} s := &Server{ ChainInfoFetcher: &chainMock.ChainService{State: state}, SlashingsPool: &slashings.PoolMock{}, Broadcaster: broadcaster, } _, err = s.SubmitAttesterSlashing(ctx, slashing) require.ErrorContains(t, "Invalid attester slashing", err) assert.Equal(t, false, broadcaster.BroadcastCalled) } func TestSubmitProposerSlashing_Ok(t *testing.T) { ctx := context.Background() _, keys, err := testutil.DeterministicDepositsAndKeys(1) require.NoError(t, err) validator := ð.Validator{ PublicKey: keys[0].PublicKey().Marshal(), WithdrawableEpoch: eth2types.Epoch(1), } state, err := testutil.NewBeaconState(func(state *pb.BeaconState) { state.Validators = []*eth.Validator{validator} }) require.NoError(t, err) slashing := ðpb.ProposerSlashing{ Header_1: ðpb.SignedBeaconBlockHeader{ Header: ðpb.BeaconBlockHeader{ Slot: 1, ProposerIndex: 0, ParentRoot: bytesutil.PadTo([]byte("parentroot1"), 32), StateRoot: bytesutil.PadTo([]byte("stateroot1"), 32), BodyRoot: bytesutil.PadTo([]byte("bodyroot1"), 32), }, Signature: make([]byte, 96), }, Header_2: ðpb.SignedBeaconBlockHeader{ Header: ðpb.BeaconBlockHeader{ Slot: 1, ProposerIndex: 0, ParentRoot: bytesutil.PadTo([]byte("parentroot2"), 32), StateRoot: bytesutil.PadTo([]byte("stateroot2"), 32), BodyRoot: bytesutil.PadTo([]byte("bodyroot2"), 32), }, Signature: make([]byte, 96), }, } for _, h := range []*ethpb.SignedBeaconBlockHeader{slashing.Header_1, slashing.Header_2} { sb, err := helpers.ComputeDomainAndSign( state, helpers.SlotToEpoch(h.Header.Slot), h.Header, params.BeaconConfig().DomainBeaconProposer, keys[0], ) require.NoError(t, err) sig, err := bls.SignatureFromBytes(sb) require.NoError(t, err) h.Signature = sig.Marshal() } broadcaster := &p2pMock.MockBroadcaster{} s := &Server{ ChainInfoFetcher: &chainMock.ChainService{State: state}, SlashingsPool: &slashings.PoolMock{}, Broadcaster: broadcaster, } _, err = s.SubmitProposerSlashing(ctx, slashing) require.NoError(t, err) pendingSlashings := s.SlashingsPool.PendingProposerSlashings(ctx, state, true) require.Equal(t, 1, len(pendingSlashings)) assert.DeepEqual(t, migration.V1ProposerSlashingToV1Alpha1(slashing), pendingSlashings[0]) assert.Equal(t, true, broadcaster.BroadcastCalled) } func TestSubmitProposerSlashing_InvalidSlashing(t *testing.T) { ctx := context.Background() state, err := testutil.NewBeaconState() require.NoError(t, err) header := ðpb.SignedBeaconBlockHeader{ Header: ðpb.BeaconBlockHeader{ Slot: 1, ProposerIndex: 0, ParentRoot: bytesutil.PadTo([]byte("parentroot1"), 32), StateRoot: bytesutil.PadTo([]byte("stateroot1"), 32), BodyRoot: bytesutil.PadTo([]byte("bodyroot1"), 32), }, Signature: make([]byte, 96), } slashing := ðpb.ProposerSlashing{ Header_1: header, Header_2: header, } broadcaster := &p2pMock.MockBroadcaster{} s := &Server{ ChainInfoFetcher: &chainMock.ChainService{State: state}, SlashingsPool: &slashings.PoolMock{}, Broadcaster: broadcaster, } _, err = s.SubmitProposerSlashing(ctx, slashing) require.ErrorContains(t, "Invalid proposer slashing", err) assert.Equal(t, false, broadcaster.BroadcastCalled) } func TestSubmitVoluntaryExit_Ok(t *testing.T) { ctx := context.Background() _, keys, err := testutil.DeterministicDepositsAndKeys(1) require.NoError(t, err) validator := ð.Validator{ ExitEpoch: params.BeaconConfig().FarFutureEpoch, PublicKey: keys[0].PublicKey().Marshal(), } state, err := testutil.NewBeaconState(func(state *pb.BeaconState) { state.Validators = []*eth.Validator{validator} // Satisfy activity time required before exiting. state.Slot = params.BeaconConfig().SlotsPerEpoch.Mul(uint64(params.BeaconConfig().ShardCommitteePeriod)) }) require.NoError(t, err) exit := ðpb.SignedVoluntaryExit{ Exit: ðpb.VoluntaryExit{ Epoch: 0, ValidatorIndex: 0, }, Signature: make([]byte, 96), } sb, err := helpers.ComputeDomainAndSign(state, exit.Exit.Epoch, exit.Exit, params.BeaconConfig().DomainVoluntaryExit, keys[0]) require.NoError(t, err) sig, err := bls.SignatureFromBytes(sb) require.NoError(t, err) exit.Signature = sig.Marshal() broadcaster := &p2pMock.MockBroadcaster{} s := &Server{ ChainInfoFetcher: &chainMock.ChainService{State: state}, VoluntaryExitsPool: &voluntaryexits.PoolMock{}, Broadcaster: broadcaster, } _, err = s.SubmitVoluntaryExit(ctx, exit) require.NoError(t, err) pendingExits := s.VoluntaryExitsPool.PendingExits(state, state.Slot(), true) require.Equal(t, 1, len(pendingExits)) assert.DeepEqual(t, migration.V1ExitToV1Alpha1(exit), pendingExits[0]) assert.Equal(t, true, broadcaster.BroadcastCalled) } func TestSubmitVoluntaryExit_InvalidValidatorIndex(t *testing.T) { ctx := context.Background() _, keys, err := testutil.DeterministicDepositsAndKeys(1) require.NoError(t, err) validator := ð.Validator{ ExitEpoch: params.BeaconConfig().FarFutureEpoch, PublicKey: keys[0].PublicKey().Marshal(), } state, err := testutil.NewBeaconState(func(state *pb.BeaconState) { state.Validators = []*eth.Validator{validator} }) require.NoError(t, err) exit := ðpb.SignedVoluntaryExit{ Exit: ðpb.VoluntaryExit{ Epoch: 0, ValidatorIndex: 99, }, Signature: make([]byte, 96), } broadcaster := &p2pMock.MockBroadcaster{} s := &Server{ ChainInfoFetcher: &chainMock.ChainService{State: state}, VoluntaryExitsPool: &voluntaryexits.PoolMock{}, Broadcaster: broadcaster, } _, err = s.SubmitVoluntaryExit(ctx, exit) require.ErrorContains(t, "Could not get exiting validator", err) assert.Equal(t, false, broadcaster.BroadcastCalled) } func TestSubmitVoluntaryExit_InvalidExit(t *testing.T) { ctx := context.Background() _, keys, err := testutil.DeterministicDepositsAndKeys(1) require.NoError(t, err) validator := ð.Validator{ ExitEpoch: params.BeaconConfig().FarFutureEpoch, PublicKey: keys[0].PublicKey().Marshal(), } state, err := testutil.NewBeaconState(func(state *pb.BeaconState) { state.Validators = []*eth.Validator{validator} }) require.NoError(t, err) exit := ðpb.SignedVoluntaryExit{ Exit: ðpb.VoluntaryExit{ Epoch: 0, ValidatorIndex: 0, }, Signature: make([]byte, 96), } broadcaster := &p2pMock.MockBroadcaster{} s := &Server{ ChainInfoFetcher: &chainMock.ChainService{State: state}, VoluntaryExitsPool: &voluntaryexits.PoolMock{}, Broadcaster: broadcaster, } _, err = s.SubmitVoluntaryExit(ctx, exit) require.ErrorContains(t, "Invalid voluntary exit", err) assert.Equal(t, false, broadcaster.BroadcastCalled) }