package beacon import ( "context" "encoding/binary" "fmt" "strconv" "strings" "testing" "github.com/gogo/protobuf/proto" ethpb "github.com/prysmaticlabs/ethereumapis/eth/v1alpha1" mock "github.com/prysmaticlabs/prysm/beacon-chain/blockchain/testing" "github.com/prysmaticlabs/prysm/beacon-chain/core/helpers" dbTest "github.com/prysmaticlabs/prysm/beacon-chain/db/testing" "github.com/prysmaticlabs/prysm/beacon-chain/state/stategen" "github.com/prysmaticlabs/prysm/shared/cmd" "github.com/prysmaticlabs/prysm/shared/featureconfig" "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 TestServer_ListAssignments_CannotRequestFutureEpoch(t *testing.T) { resetCfg := featureconfig.InitWithReset(&featureconfig.Flags{NewStateMgmt: true}) defer resetCfg() db, _ := dbTest.SetupDB(t) ctx := context.Background() bs := &Server{ BeaconDB: db, GenesisTimeFetcher: &mock.ChainService{}, } wanted := "Cannot retrieve information about an epoch in the future" _, err := bs.ListValidatorAssignments( ctx, ðpb.ListValidatorAssignmentsRequest{ QueryFilter: ðpb.ListValidatorAssignmentsRequest_Epoch{ Epoch: helpers.SlotToEpoch(bs.GenesisTimeFetcher.CurrentSlot()) + 1, }, }, ) assert.ErrorContains(t, wanted, err) } func TestServer_ListAssignments_NoResults(t *testing.T) { resetCfg := featureconfig.InitWithReset(&featureconfig.Flags{NewStateMgmt: true}) defer resetCfg() db, sc := dbTest.SetupDB(t) ctx := context.Background() st := testutil.NewBeaconState() b := testutil.NewBeaconBlock() require.NoError(t, db.SaveBlock(ctx, b)) gRoot, err := b.Block.HashTreeRoot() require.NoError(t, err) require.NoError(t, db.SaveGenesisBlockRoot(ctx, gRoot)) require.NoError(t, db.SaveState(ctx, st, gRoot)) bs := &Server{ BeaconDB: db, GenesisTimeFetcher: &mock.ChainService{}, StateGen: stategen.New(db, sc), } wanted := ðpb.ValidatorAssignments{ Assignments: make([]*ethpb.ValidatorAssignments_CommitteeAssignment, 0), TotalSize: int32(0), NextPageToken: strconv.Itoa(0), } res, err := bs.ListValidatorAssignments( ctx, ðpb.ListValidatorAssignmentsRequest{ QueryFilter: ðpb.ListValidatorAssignmentsRequest_Genesis{ Genesis: true, }, }, ) require.NoError(t, err) if !proto.Equal(wanted, res) { t.Errorf("Wanted %v, received %v", wanted, res) } } func TestServer_ListAssignments_Pagination_InputOutOfRange(t *testing.T) { resetCfg := featureconfig.InitWithReset(&featureconfig.Flags{NewStateMgmt: true}) defer resetCfg() db, sc := dbTest.SetupDB(t) ctx := context.Background() setupValidators(t, db, 1) headState, err := db.HeadState(ctx) require.NoError(t, err) b := testutil.NewBeaconBlock() require.NoError(t, db.SaveBlock(ctx, b)) gRoot, err := b.Block.HashTreeRoot() require.NoError(t, err) require.NoError(t, db.SaveGenesisBlockRoot(ctx, gRoot)) require.NoError(t, db.SaveState(ctx, headState, gRoot)) bs := &Server{ BeaconDB: db, GenesisTimeFetcher: &mock.ChainService{}, StateGen: stategen.New(db, sc), } wanted := fmt.Sprintf("page start %d >= list %d", 0, 0) if _, err := bs.ListValidatorAssignments( context.Background(), ðpb.ListValidatorAssignmentsRequest{ QueryFilter: ðpb.ListValidatorAssignmentsRequest_Genesis{Genesis: true}, }, ); err != nil && !strings.Contains(err.Error(), wanted) { t.Errorf("Expected error %v, received %v", wanted, err) } } func TestServer_ListAssignments_Pagination_ExceedsMaxPageSize(t *testing.T) { bs := &Server{} exceedsMax := int32(cmd.Get().MaxRPCPageSize + 1) wanted := fmt.Sprintf("Requested page size %d can not be greater than max size %d", exceedsMax, cmd.Get().MaxRPCPageSize) req := ðpb.ListValidatorAssignmentsRequest{ PageToken: strconv.Itoa(0), PageSize: exceedsMax, } _, err := bs.ListValidatorAssignments(context.Background(), req) assert.ErrorContains(t, wanted, err) } func TestServer_ListAssignments_Pagination_DefaultPageSize_NoArchive(t *testing.T) { helpers.ClearCache() db, sc := dbTest.SetupDB(t) ctx := context.Background() count := 500 validators := make([]*ethpb.Validator, 0, count) for i := 0; i < count; i++ { pubKey := make([]byte, params.BeaconConfig().BLSPubkeyLength) withdrawalCred := make([]byte, 32) binary.LittleEndian.PutUint64(pubKey, uint64(i)) // Mark the validators with index divisible by 3 inactive. if i%3 == 0 { validators = append(validators, ðpb.Validator{ PublicKey: pubKey, WithdrawalCredentials: withdrawalCred, ExitEpoch: 0, ActivationEpoch: 0, EffectiveBalance: params.BeaconConfig().MaxEffectiveBalance, }) } else { validators = append(validators, ðpb.Validator{ PublicKey: pubKey, WithdrawalCredentials: withdrawalCred, ExitEpoch: params.BeaconConfig().FarFutureEpoch, EffectiveBalance: params.BeaconConfig().MaxEffectiveBalance, ActivationEpoch: 0, }) } } blk := testutil.NewBeaconBlock().Block blockRoot, err := blk.HashTreeRoot() require.NoError(t, err) s := testutil.NewBeaconState() require.NoError(t, s.SetValidators(validators)) require.NoError(t, db.SaveState(ctx, s, blockRoot)) require.NoError(t, db.SaveGenesisBlockRoot(ctx, blockRoot)) bs := &Server{ BeaconDB: db, HeadFetcher: &mock.ChainService{ State: s, }, FinalizationFetcher: &mock.ChainService{ FinalizedCheckPoint: ðpb.Checkpoint{ Epoch: 0, }, }, GenesisTimeFetcher: &mock.ChainService{}, StateGen: stategen.New(db, sc), } res, err := bs.ListValidatorAssignments(context.Background(), ðpb.ListValidatorAssignmentsRequest{ QueryFilter: ðpb.ListValidatorAssignmentsRequest_Genesis{Genesis: true}, }) require.NoError(t, err) // Construct the wanted assignments. var wanted []*ethpb.ValidatorAssignments_CommitteeAssignment activeIndices, err := helpers.ActiveValidatorIndices(s, 0) require.NoError(t, err) committeeAssignments, proposerIndexToSlots, err := helpers.CommitteeAssignments(s, 0) require.NoError(t, err) for _, index := range activeIndices[0:params.BeaconConfig().DefaultPageSize] { val, err := s.ValidatorAtIndex(index) require.NoError(t, err) wanted = append(wanted, ðpb.ValidatorAssignments_CommitteeAssignment{ BeaconCommittees: committeeAssignments[index].Committee, CommitteeIndex: committeeAssignments[index].CommitteeIndex, AttesterSlot: committeeAssignments[index].AttesterSlot, ProposerSlots: proposerIndexToSlots[index], PublicKey: val.PublicKey, ValidatorIndex: index, }) } assert.DeepEqual(t, wanted, res.Assignments, "Did not receive wanted assignments") } func TestServer_ListAssignments_FilterPubkeysIndices_NoPagination(t *testing.T) { helpers.ClearCache() db, sc := dbTest.SetupDB(t) resetCfg := featureconfig.InitWithReset(&featureconfig.Flags{NewStateMgmt: true}) defer resetCfg() ctx := context.Background() count := 100 validators := make([]*ethpb.Validator, 0, count) withdrawCreds := make([]byte, 32) for i := 0; i < count; i++ { pubKey := make([]byte, params.BeaconConfig().BLSPubkeyLength) binary.LittleEndian.PutUint64(pubKey, uint64(i)) val := ðpb.Validator{ PublicKey: pubKey, WithdrawalCredentials: withdrawCreds, ExitEpoch: params.BeaconConfig().FarFutureEpoch, } validators = append(validators, val) } blk := testutil.NewBeaconBlock().Block blockRoot, err := blk.HashTreeRoot() require.NoError(t, err) s := testutil.NewBeaconState() require.NoError(t, s.SetValidators(validators)) require.NoError(t, db.SaveState(ctx, s, blockRoot)) require.NoError(t, db.SaveGenesisBlockRoot(ctx, blockRoot)) bs := &Server{ BeaconDB: db, FinalizationFetcher: &mock.ChainService{ FinalizedCheckPoint: ðpb.Checkpoint{ Epoch: 0, }, }, GenesisTimeFetcher: &mock.ChainService{}, StateGen: stategen.New(db, sc), } pubKey1 := make([]byte, params.BeaconConfig().BLSPubkeyLength) binary.LittleEndian.PutUint64(pubKey1, 1) pubKey2 := make([]byte, params.BeaconConfig().BLSPubkeyLength) binary.LittleEndian.PutUint64(pubKey2, 2) req := ðpb.ListValidatorAssignmentsRequest{PublicKeys: [][]byte{pubKey1, pubKey2}, Indices: []uint64{2, 3}} res, err := bs.ListValidatorAssignments(context.Background(), req) require.NoError(t, err) // Construct the wanted assignments. var wanted []*ethpb.ValidatorAssignments_CommitteeAssignment activeIndices, err := helpers.ActiveValidatorIndices(s, 0) require.NoError(t, err) committeeAssignments, proposerIndexToSlots, err := helpers.CommitteeAssignments(s, 0) require.NoError(t, err) for _, index := range activeIndices[1:4] { val, err := s.ValidatorAtIndex(index) require.NoError(t, err) wanted = append(wanted, ðpb.ValidatorAssignments_CommitteeAssignment{ BeaconCommittees: committeeAssignments[index].Committee, CommitteeIndex: committeeAssignments[index].CommitteeIndex, AttesterSlot: committeeAssignments[index].AttesterSlot, ProposerSlots: proposerIndexToSlots[index], PublicKey: val.PublicKey, ValidatorIndex: index, }) } assert.DeepEqual(t, wanted, res.Assignments, "Did not receive wanted assignments") } func TestServer_ListAssignments_CanFilterPubkeysIndices_WithPagination(t *testing.T) { helpers.ClearCache() db, sc := dbTest.SetupDB(t) ctx := context.Background() count := 100 validators := make([]*ethpb.Validator, 0, count) withdrawCred := make([]byte, 32) for i := 0; i < count; i++ { pubKey := make([]byte, params.BeaconConfig().BLSPubkeyLength) binary.LittleEndian.PutUint64(pubKey, uint64(i)) val := ðpb.Validator{ PublicKey: pubKey, WithdrawalCredentials: withdrawCred, ExitEpoch: params.BeaconConfig().FarFutureEpoch, } validators = append(validators, val) } blk := testutil.NewBeaconBlock().Block blockRoot, err := blk.HashTreeRoot() require.NoError(t, err) s := testutil.NewBeaconState() require.NoError(t, s.SetValidators(validators)) require.NoError(t, db.SaveState(ctx, s, blockRoot)) require.NoError(t, db.SaveGenesisBlockRoot(ctx, blockRoot)) bs := &Server{ BeaconDB: db, FinalizationFetcher: &mock.ChainService{ FinalizedCheckPoint: ðpb.Checkpoint{ Epoch: 0, }, }, GenesisTimeFetcher: &mock.ChainService{}, StateGen: stategen.New(db, sc), } req := ðpb.ListValidatorAssignmentsRequest{Indices: []uint64{1, 2, 3, 4, 5, 6}, PageSize: 2, PageToken: "1"} res, err := bs.ListValidatorAssignments(context.Background(), req) require.NoError(t, err) // Construct the wanted assignments. var assignments []*ethpb.ValidatorAssignments_CommitteeAssignment activeIndices, err := helpers.ActiveValidatorIndices(s, 0) require.NoError(t, err) committeeAssignments, proposerIndexToSlots, err := helpers.CommitteeAssignments(s, 0) require.NoError(t, err) for _, index := range activeIndices[3:5] { val, err := s.ValidatorAtIndex(index) require.NoError(t, err) assignments = append(assignments, ðpb.ValidatorAssignments_CommitteeAssignment{ BeaconCommittees: committeeAssignments[index].Committee, CommitteeIndex: committeeAssignments[index].CommitteeIndex, AttesterSlot: committeeAssignments[index].AttesterSlot, ProposerSlots: proposerIndexToSlots[index], PublicKey: val.PublicKey, ValidatorIndex: index, }) } wantedRes := ðpb.ValidatorAssignments{ Assignments: assignments, TotalSize: int32(len(req.Indices)), NextPageToken: "2", } assert.DeepEqual(t, wantedRes, res, "Did not get wanted assignments") // Test the wrap around scenario. assignments = nil req = ðpb.ListValidatorAssignmentsRequest{Indices: []uint64{1, 2, 3, 4, 5, 6}, PageSize: 5, PageToken: "1"} res, err = bs.ListValidatorAssignments(context.Background(), req) require.NoError(t, err) cAssignments, proposerIndexToSlots, err := helpers.CommitteeAssignments(s, 0) require.NoError(t, err) for _, index := range activeIndices[6:7] { val, err := s.ValidatorAtIndex(index) require.NoError(t, err) assignments = append(assignments, ðpb.ValidatorAssignments_CommitteeAssignment{ BeaconCommittees: cAssignments[index].Committee, CommitteeIndex: cAssignments[index].CommitteeIndex, AttesterSlot: cAssignments[index].AttesterSlot, ProposerSlots: proposerIndexToSlots[index], PublicKey: val.PublicKey, ValidatorIndex: index, }) } wantedRes = ðpb.ValidatorAssignments{ Assignments: assignments, TotalSize: int32(len(req.Indices)), NextPageToken: "", } assert.DeepEqual(t, wantedRes, res, "Did not receive wanted assignments") }