package evaluators import ( "context" "fmt" "github.com/pkg/errors" "github.com/prysmaticlabs/go-bitfield" "github.com/prysmaticlabs/prysm/v3/beacon-chain/core/signing" fieldparams "github.com/prysmaticlabs/prysm/v3/config/fieldparams" "github.com/prysmaticlabs/prysm/v3/config/params" "github.com/prysmaticlabs/prysm/v3/consensus-types/blocks" types "github.com/prysmaticlabs/prysm/v3/consensus-types/primitives" "github.com/prysmaticlabs/prysm/v3/container/slice" "github.com/prysmaticlabs/prysm/v3/encoding/bytesutil" eth "github.com/prysmaticlabs/prysm/v3/proto/prysm/v1alpha1" e2e "github.com/prysmaticlabs/prysm/v3/testing/endtoend/params" "github.com/prysmaticlabs/prysm/v3/testing/endtoend/policies" e2eTypes "github.com/prysmaticlabs/prysm/v3/testing/endtoend/types" "github.com/prysmaticlabs/prysm/v3/testing/util" validatorClientFactory "github.com/prysmaticlabs/prysm/v3/validator/client/validator-client-factory" "google.golang.org/grpc" "google.golang.org/protobuf/types/known/emptypb" ) // InjectDoubleVoteOnEpoch broadcasts a double vote into the beacon node pool for the slasher to detect. var InjectDoubleVoteOnEpoch = func(n types.Epoch) e2eTypes.Evaluator { return e2eTypes.Evaluator{ Name: "inject_double_vote_%d", Policy: policies.OnEpoch(n), Evaluation: insertDoubleAttestationIntoPool, } } // InjectDoubleBlockOnEpoch proposes a double block to the beacon node for the slasher to detect. var InjectDoubleBlockOnEpoch = func(n types.Epoch) e2eTypes.Evaluator { return e2eTypes.Evaluator{ Name: "inject_double_block_%d", Policy: policies.OnEpoch(n), Evaluation: proposeDoubleBlock, } } // ValidatorsSlashedAfterEpoch ensures the expected amount of validators are slashed. var ValidatorsSlashedAfterEpoch = func(n types.Epoch) e2eTypes.Evaluator { return e2eTypes.Evaluator{ Name: "validators_slashed_epoch_%d", Policy: policies.AfterNthEpoch(n), Evaluation: validatorsSlashed, } } // SlashedValidatorsLoseBalanceAfterEpoch checks if the validators slashed lose the right balance. var SlashedValidatorsLoseBalanceAfterEpoch = func(n types.Epoch) e2eTypes.Evaluator { return e2eTypes.Evaluator{ Name: "slashed_validators_lose_valance_epoch_%d", Policy: policies.AfterNthEpoch(n), Evaluation: validatorsLoseBalance, } } var slashedIndices []uint64 func validatorsSlashed(conns ...*grpc.ClientConn) error { conn := conns[0] ctx := context.Background() client := eth.NewBeaconChainClient(conn) req := ð.GetValidatorActiveSetChangesRequest{} changes, err := client.GetValidatorActiveSetChanges(ctx, req) if err != nil { return err } if len(changes.SlashedIndices) != len(slashedIndices) { return fmt.Errorf("expected %d indices to be slashed, received %d", len(slashedIndices), len(changes.SlashedIndices)) } return nil } func validatorsLoseBalance(conns ...*grpc.ClientConn) error { conn := conns[0] ctx := context.Background() client := eth.NewBeaconChainClient(conn) for i, slashedIndex := range slashedIndices { req := ð.GetValidatorRequest{ QueryFilter: ð.GetValidatorRequest_Index{ Index: types.ValidatorIndex(slashedIndex), }, } valResp, err := client.GetValidator(ctx, req) if err != nil { return err } slashedPenalty := params.BeaconConfig().MaxEffectiveBalance / params.BeaconConfig().MinSlashingPenaltyQuotient slashedBal := params.BeaconConfig().MaxEffectiveBalance - slashedPenalty + params.BeaconConfig().EffectiveBalanceIncrement/10 if valResp.EffectiveBalance >= slashedBal { return fmt.Errorf( "expected slashed validator %d to balance less than %d, received %d", i, slashedBal, valResp.EffectiveBalance, ) } } return nil } func insertDoubleAttestationIntoPool(conns ...*grpc.ClientConn) error { conn := conns[0] valClient := validatorClientFactory.NewValidatorClient(conn) beaconClient := eth.NewBeaconChainClient(conn) ctx := context.Background() chainHead, err := beaconClient.GetChainHead(ctx, &emptypb.Empty{}) if err != nil { return errors.Wrap(err, "could not get chain head") } _, privKeys, err := util.DeterministicDepositsAndKeys(params.BeaconConfig().MinGenesisActiveValidatorCount) if err != nil { return err } pubKeys := make([][]byte, len(privKeys)) for i, priv := range privKeys { pubKeys[i] = priv.PublicKey().Marshal() } duties, err := valClient.GetDuties(ctx, ð.DutiesRequest{ Epoch: chainHead.HeadEpoch, PublicKeys: pubKeys, }) if err != nil { return errors.Wrap(err, "could not get duties") } var committeeIndex types.CommitteeIndex var committee []types.ValidatorIndex for _, duty := range duties.Duties { if duty.AttesterSlot == chainHead.HeadSlot-1 { committeeIndex = duty.CommitteeIndex committee = duty.Committee break } } attDataReq := ð.AttestationDataRequest{ CommitteeIndex: committeeIndex, Slot: chainHead.HeadSlot - 1, } attData, err := valClient.GetAttestationData(ctx, attDataReq) if err != nil { return err } blockRoot := bytesutil.ToBytes32([]byte("muahahahaha I'm an evil validator")) attData.BeaconBlockRoot = blockRoot[:] req := ð.DomainRequest{ Epoch: chainHead.HeadEpoch, Domain: params.BeaconConfig().DomainBeaconAttester[:], } resp, err := valClient.DomainData(ctx, req) if err != nil { return errors.Wrap(err, "could not get domain data") } signingRoot, err := signing.ComputeSigningRoot(attData, resp.SignatureDomain) if err != nil { return errors.Wrap(err, "could not compute signing root") } valsToSlash := uint64(2) for i := uint64(0); i < valsToSlash && i < uint64(len(committee)); i++ { if len(slice.IntersectionUint64(slashedIndices, []uint64{uint64(committee[i])})) > 0 { valsToSlash++ continue } // Set the bits of half the committee to be slashed. attBitfield := bitfield.NewBitlist(uint64(len(committee))) attBitfield.SetBitAt(i, true) att := ð.Attestation{ AggregationBits: attBitfield, Data: attData, Signature: privKeys[committee[i]].Sign(signingRoot[:]).Marshal(), } // We only broadcast to conns[0] here since we can trust that at least 1 node will be online. // Only broadcasting the attestation to one node also helps test slashing propagation. client := validatorClientFactory.NewValidatorClient(conns[0]) if _, err = client.ProposeAttestation(ctx, att); err != nil { return errors.Wrap(err, "could not propose attestation") } slashedIndices = append(slashedIndices, uint64(committee[i])) } return nil } func proposeDoubleBlock(conns ...*grpc.ClientConn) error { conn := conns[0] valClient := validatorClientFactory.NewValidatorClient(conn) beaconClient := eth.NewBeaconChainClient(conn) ctx := context.Background() chainHead, err := beaconClient.GetChainHead(ctx, &emptypb.Empty{}) if err != nil { return errors.Wrap(err, "could not get chain head") } _, privKeys, err := util.DeterministicDepositsAndKeys(params.BeaconConfig().MinGenesisActiveValidatorCount) if err != nil { return err } pubKeys := make([][]byte, len(privKeys)) for i, priv := range privKeys { pubKeys[i] = priv.PublicKey().Marshal() } duties, err := valClient.GetDuties(ctx, ð.DutiesRequest{ Epoch: chainHead.HeadEpoch, PublicKeys: pubKeys, }) if err != nil { return errors.Wrap(err, "could not get duties") } var proposerIndex types.ValidatorIndex for i, duty := range duties.CurrentEpochDuties { if slice.IsInSlots(chainHead.HeadSlot-1, duty.ProposerSlots) { proposerIndex = types.ValidatorIndex(i) break } } validatorNum := int(params.BeaconConfig().MinGenesisActiveValidatorCount) beaconNodeNum := e2e.TestParams.BeaconNodeCount if validatorNum%beaconNodeNum != 0 { return errors.New("validator count is not easily divisible by beacon node count") } validatorsPerNode := validatorNum / beaconNodeNum // If the proposer index is in the second validator client, we connect to // the corresponding beacon node instead. if proposerIndex >= types.ValidatorIndex(uint64(validatorsPerNode)) { valClient = validatorClientFactory.NewValidatorClient(conns[1]) } hashLen := 32 blk := ð.BeaconBlock{ Slot: chainHead.HeadSlot + 1, ParentRoot: chainHead.HeadBlockRoot, StateRoot: bytesutil.PadTo([]byte("bad state root"), hashLen), ProposerIndex: proposerIndex, Body: ð.BeaconBlockBody{ Eth1Data: ð.Eth1Data{ BlockHash: bytesutil.PadTo([]byte("bad block hash"), hashLen), DepositRoot: bytesutil.PadTo([]byte("bad deposit root"), hashLen), DepositCount: 1, }, RandaoReveal: bytesutil.PadTo([]byte("bad randao"), fieldparams.BLSSignatureLength), Graffiti: bytesutil.PadTo([]byte("teehee"), hashLen), ProposerSlashings: []*eth.ProposerSlashing{}, AttesterSlashings: []*eth.AttesterSlashing{}, Attestations: []*eth.Attestation{}, Deposits: []*eth.Deposit{}, VoluntaryExits: []*eth.SignedVoluntaryExit{}, }, } req := ð.DomainRequest{ Epoch: chainHead.HeadEpoch, Domain: params.BeaconConfig().DomainBeaconProposer[:], } resp, err := valClient.DomainData(ctx, req) if err != nil { return errors.Wrap(err, "could not get domain data") } signingRoot, err := signing.ComputeSigningRoot(blk, resp.SignatureDomain) if err != nil { return errors.Wrap(err, "could not compute signing root") } sig := privKeys[proposerIndex].Sign(signingRoot[:]).Marshal() signedBlk := ð.SignedBeaconBlock{ Block: blk, Signature: sig, } // We only broadcast to conns[0] here since we can trust that at least 1 node will be online. // Only broadcasting the attestation to one node also helps test slashing propagation. wb, err := blocks.NewSignedBeaconBlock(signedBlk) if err != nil { return err } b, err := wb.PbGenericBlock() if err != nil { return err } if _, err = valClient.ProposeBeaconBlock(ctx, b); err == nil { return errors.New("expected block to fail processing") } slashedIndices = append(slashedIndices, uint64(proposerIndex)) return nil }