package rpc import ( "context" "fmt" "reflect" "strconv" "strings" "testing" "github.com/gogo/protobuf/proto" ptypes "github.com/gogo/protobuf/types" "github.com/prysmaticlabs/go-bitfield" "github.com/prysmaticlabs/go-ssz" "github.com/prysmaticlabs/prysm/beacon-chain/core/helpers" "github.com/prysmaticlabs/prysm/beacon-chain/db" testutil "github.com/prysmaticlabs/prysm/beacon-chain/db/testing" pbp2p "github.com/prysmaticlabs/prysm/proto/beacon/p2p/v1" ethpb "github.com/prysmaticlabs/prysm/proto/eth/v1alpha1" "github.com/prysmaticlabs/prysm/shared/params" ) type mockPool struct{} func (m *mockPool) AttestationPool(ctx context.Context, expectedSlot uint64) ([]*ethpb.Attestation, error) { return []*ethpb.Attestation{ { Data: ðpb.AttestationData{ BeaconBlockRoot: []byte("1"), }, }, { Data: ðpb.AttestationData{ BeaconBlockRoot: []byte("2"), }, }, }, nil } func TestBeaconChainServer_ListAttestationsNoPagination(t *testing.T) { db := testutil.SetupDB(t) defer testutil.TeardownDB(t, db) ctx := context.Background() count := uint64(10) atts := make([]*ethpb.Attestation, 0, count) for i := uint64(0); i < count; i++ { attExample := ðpb.Attestation{ Data: ðpb.AttestationData{ Crosslink: ðpb.Crosslink{ Shard: i, }, }, } if err := db.SaveAttestation(ctx, attExample); err != nil { t.Fatal(err) } atts = append(atts, attExample) } bs := &BeaconChainServer{ beaconDB: db, } received, err := bs.ListAttestations(ctx, ðpb.ListAttestationsRequest{}) if err != nil { t.Fatal(err) } if !reflect.DeepEqual(atts, received.Attestations) { t.Fatalf("incorrect attestations response: wanted %v, received %v", atts, received.Attestations) } } func TestBeaconChainServer_ListAttestationsPagination(t *testing.T) { db := testutil.SetupDB(t) defer testutil.TeardownDB(t, db) ctx := context.Background() count := uint64(100) atts := make([]*ethpb.Attestation, 0, count) for i := uint64(0); i < count; i++ { attExample := ðpb.Attestation{ Data: ðpb.AttestationData{ Crosslink: ðpb.Crosslink{ Shard: i, }, }, } if err := db.SaveAttestation(ctx, attExample); err != nil { t.Fatal(err) } atts = append(atts, attExample) } bs := &BeaconChainServer{ beaconDB: db, } tests := []struct { req *ethpb.ListAttestationsRequest res *ethpb.ListAttestationsResponse }{ {req: ðpb.ListAttestationsRequest{PageToken: strconv.Itoa(1), PageSize: 3}, res: ðpb.ListAttestationsResponse{ Attestations: []*ethpb.Attestation{ {Data: ðpb.AttestationData{ Crosslink: ðpb.Crosslink{Shard: 3}, }}, {Data: ðpb.AttestationData{ Crosslink: ðpb.Crosslink{Shard: 4}, }}, {Data: ðpb.AttestationData{ Crosslink: ðpb.Crosslink{Shard: 5}, }}, }, NextPageToken: strconv.Itoa(2), TotalSize: int32(count)}}, {req: ðpb.ListAttestationsRequest{PageToken: strconv.Itoa(10), PageSize: 5}, res: ðpb.ListAttestationsResponse{ Attestations: []*ethpb.Attestation{ {Data: ðpb.AttestationData{ Crosslink: ðpb.Crosslink{Shard: 50}, }}, {Data: ðpb.AttestationData{ Crosslink: ðpb.Crosslink{Shard: 51}, }}, {Data: ðpb.AttestationData{ Crosslink: ðpb.Crosslink{Shard: 52}, }}, {Data: ðpb.AttestationData{ Crosslink: ðpb.Crosslink{Shard: 53}, }}, {Data: ðpb.AttestationData{ Crosslink: ðpb.Crosslink{Shard: 54}, }}, }, NextPageToken: strconv.Itoa(11), TotalSize: int32(count)}}, {req: ðpb.ListAttestationsRequest{PageToken: strconv.Itoa(33), PageSize: 3}, res: ðpb.ListAttestationsResponse{ Attestations: []*ethpb.Attestation{ {Data: ðpb.AttestationData{ Crosslink: ðpb.Crosslink{Shard: 99}, }}, }, NextPageToken: strconv.Itoa(34), TotalSize: int32(count)}}, {req: ðpb.ListAttestationsRequest{PageSize: 2}, res: ðpb.ListAttestationsResponse{ Attestations: []*ethpb.Attestation{ {Data: ðpb.AttestationData{ Crosslink: ðpb.Crosslink{Shard: 0}, }}, {Data: ðpb.AttestationData{ Crosslink: ðpb.Crosslink{Shard: 1}, }}, }, NextPageToken: strconv.Itoa(1), TotalSize: int32(count)}}, } for _, test := range tests { res, err := bs.ListAttestations(ctx, test.req) if err != nil { t.Fatal(err) } if !proto.Equal(res, test.res) { t.Error("Incorrect attestations response") } } } func TestBeaconChainServer_ListAttestationsPaginationOutOfRange(t *testing.T) { db := testutil.SetupDB(t) defer testutil.TeardownDB(t, db) ctx := context.Background() count := uint64(1) atts := make([]*ethpb.Attestation, 0, count) for i := uint64(0); i < count; i++ { attExample := ðpb.Attestation{ Data: ðpb.AttestationData{ Crosslink: ðpb.Crosslink{ Shard: i, }, }, } if err := db.SaveAttestation(ctx, attExample); err != nil { t.Fatal(err) } atts = append(atts, attExample) } bs := &BeaconChainServer{ beaconDB: db, } req := ðpb.ListAttestationsRequest{PageToken: strconv.Itoa(1), PageSize: 100} wanted := fmt.Sprintf("page start %d >= list %d", req.PageSize, len(atts)) if _, err := bs.ListAttestations(ctx, req); !strings.Contains(err.Error(), wanted) { t.Errorf("Expected error %v, received %v", wanted, err) } } func TestBeaconChainServer_ListAttestationsExceedsMaxPageSize(t *testing.T) { ctx := context.Background() bs := &BeaconChainServer{} exceedsMax := int32(params.BeaconConfig().MaxPageSize + 1) wanted := fmt.Sprintf("requested page size %d can not be greater than max size %d", exceedsMax, params.BeaconConfig().MaxPageSize) req := ðpb.ListAttestationsRequest{PageToken: strconv.Itoa(0), PageSize: exceedsMax} if _, err := bs.ListAttestations(ctx, req); !strings.Contains(err.Error(), wanted) { t.Errorf("Expected error %v, received %v", wanted, err) } } func TestBeaconChainServer_ListAttestationsDefaultPageSize(t *testing.T) { db := testutil.SetupDB(t) defer testutil.TeardownDB(t, db) ctx := context.Background() count := uint64(params.BeaconConfig().DefaultPageSize) atts := make([]*ethpb.Attestation, 0, count) for i := uint64(0); i < count; i++ { attExample := ðpb.Attestation{ Data: ðpb.AttestationData{ Crosslink: ðpb.Crosslink{ Shard: i, }, }, } if err := db.SaveAttestation(ctx, attExample); err != nil { t.Fatal(err) } atts = append(atts, attExample) } bs := &BeaconChainServer{ beaconDB: db, } req := ðpb.ListAttestationsRequest{} res, err := bs.ListAttestations(ctx, req) if err != nil { t.Fatal(err) } i := 0 j := params.BeaconConfig().DefaultPageSize if !reflect.DeepEqual(res.Attestations, atts[i:j]) { t.Error("Incorrect attestations response") } } func TestBeaconChainServer_AttestationPool(t *testing.T) { ctx := context.Background() db := testutil.SetupDB(t) defer testutil.TeardownDB(t, db) bs := &BeaconChainServer{ pool: &mockPool{}, beaconDB: db, } block := ðpb.BeaconBlock{ Slot: 10, } blockRoot, err := ssz.SigningRoot(block) if err != nil { t.Fatal(err) } if err := bs.beaconDB.SaveBlock(ctx, block); err != nil { t.Fatal(err) } if err := bs.beaconDB.SaveHeadBlockRoot(ctx, blockRoot); err != nil { t.Fatal(err) } if err := bs.beaconDB.SaveState(ctx, &pbp2p.BeaconState{Slot: 10}, blockRoot); err != nil { t.Fatal(err) } res, err := bs.AttestationPool(ctx, &ptypes.Empty{}) if err != nil { t.Fatal(err) } want, _ := bs.pool.AttestationPool(ctx, 10) if !reflect.DeepEqual(res.Attestations, want) { t.Errorf("Wanted AttestationPool() = %v, received %v", want, res.Attestations) } } func TestBeaconChainServer_ListValidatorBalances(t *testing.T) { db := testutil.SetupDB(t) defer testutil.TeardownDB(t, db) setupValidators(t, db, 100) bs := &BeaconChainServer{ beaconDB: db, } tests := []struct { req *ethpb.GetValidatorBalancesRequest res *ethpb.ValidatorBalances }{ {req: ðpb.GetValidatorBalancesRequest{PublicKeys: [][]byte{{99}}}, res: ðpb.ValidatorBalances{Balances: []*ethpb.ValidatorBalances_Balance{{ Index: 99, PublicKey: []byte{99}, Balance: 99}}, }}, {req: ðpb.GetValidatorBalancesRequest{Indices: []uint64{1, 2, 3}}, res: ðpb.ValidatorBalances{Balances: []*ethpb.ValidatorBalances_Balance{ {Index: 1, PublicKey: []byte{1}, Balance: 1}, {Index: 2, PublicKey: []byte{2}, Balance: 2}, {Index: 3, PublicKey: []byte{3}, Balance: 3}}, }}, {req: ðpb.GetValidatorBalancesRequest{PublicKeys: [][]byte{{10}, {11}, {12}}}, res: ðpb.ValidatorBalances{Balances: []*ethpb.ValidatorBalances_Balance{ {Index: 10, PublicKey: []byte{10}, Balance: 10}, {Index: 11, PublicKey: []byte{11}, Balance: 11}, {Index: 12, PublicKey: []byte{12}, Balance: 12}}, }}, {req: ðpb.GetValidatorBalancesRequest{PublicKeys: [][]byte{{2}, {3}}, Indices: []uint64{3, 4}}, // Duplication res: ðpb.ValidatorBalances{Balances: []*ethpb.ValidatorBalances_Balance{ {Index: 2, PublicKey: []byte{2}, Balance: 2}, {Index: 3, PublicKey: []byte{3}, Balance: 3}, {Index: 4, PublicKey: []byte{4}, Balance: 4}}, }}, {req: ðpb.GetValidatorBalancesRequest{PublicKeys: [][]byte{{}}, Indices: []uint64{3, 4}}, // Public key has a blank value res: ðpb.ValidatorBalances{Balances: []*ethpb.ValidatorBalances_Balance{ {Index: 3, PublicKey: []byte{3}, Balance: 3}, {Index: 4, PublicKey: []byte{4}, Balance: 4}}, }}, } for _, test := range tests { res, err := bs.ListValidatorBalances(context.Background(), test.req) if err != nil { t.Fatal(err) } if !proto.Equal(res, test.res) { t.Error("Incorrect respond of validator balances") } } } func TestBeaconChainServer_ListValidatorBalancesOutOfRange(t *testing.T) { db := testutil.SetupDB(t) defer testutil.TeardownDB(t, db) _, balances := setupValidators(t, db, 1) bs := &BeaconChainServer{ beaconDB: db, } req := ðpb.GetValidatorBalancesRequest{Indices: []uint64{uint64(1)}} wanted := fmt.Sprintf("validator index %d >= balance list %d", 1, len(balances)) if _, err := bs.ListValidatorBalances(context.Background(), req); !strings.Contains(err.Error(), wanted) { t.Errorf("Expected error %v, received %v", wanted, err) } } func TestBeaconChainServer_GetValidatorsNoPagination(t *testing.T) { db := testutil.SetupDB(t) defer testutil.TeardownDB(t, db) validators, _ := setupValidators(t, db, 100) bs := &BeaconChainServer{ beaconDB: db, } received, err := bs.GetValidators(context.Background(), ðpb.GetValidatorsRequest{}) if err != nil { t.Fatal(err) } if !reflect.DeepEqual(validators, received.Validators) { t.Fatal("Incorrect respond of validators") } } func TestBeaconChainServer_GetValidatorsPagination(t *testing.T) { db := testutil.SetupDB(t) defer testutil.TeardownDB(t, db) count := 100 setupValidators(t, db, count) bs := &BeaconChainServer{ beaconDB: db, } tests := []struct { req *ethpb.GetValidatorsRequest res *ethpb.Validators }{ {req: ðpb.GetValidatorsRequest{PageToken: strconv.Itoa(1), PageSize: 3}, res: ðpb.Validators{ Validators: []*ethpb.Validator{ {PublicKey: []byte{3}}, {PublicKey: []byte{4}}, {PublicKey: []byte{5}}}, NextPageToken: strconv.Itoa(2), TotalSize: int32(count)}}, {req: ðpb.GetValidatorsRequest{PageToken: strconv.Itoa(10), PageSize: 5}, res: ðpb.Validators{ Validators: []*ethpb.Validator{ {PublicKey: []byte{50}}, {PublicKey: []byte{51}}, {PublicKey: []byte{52}}, {PublicKey: []byte{53}}, {PublicKey: []byte{54}}}, NextPageToken: strconv.Itoa(11), TotalSize: int32(count)}}, {req: ðpb.GetValidatorsRequest{PageToken: strconv.Itoa(33), PageSize: 3}, res: ðpb.Validators{ Validators: []*ethpb.Validator{ {PublicKey: []byte{99}}}, NextPageToken: strconv.Itoa(34), TotalSize: int32(count)}}, {req: ðpb.GetValidatorsRequest{PageSize: 2}, res: ðpb.Validators{ Validators: []*ethpb.Validator{ {PublicKey: []byte{0}}, {PublicKey: []byte{1}}}, NextPageToken: strconv.Itoa(1), TotalSize: int32(count)}}, } for _, test := range tests { res, err := bs.GetValidators(context.Background(), test.req) if err != nil { t.Fatal(err) } if !proto.Equal(res, test.res) { t.Error("Incorrect respond of validators") } } } func TestBeaconChainServer_GetValidatorsPaginationOutOfRange(t *testing.T) { db := testutil.SetupDB(t) defer testutil.TeardownDB(t, db) count := 1 validators, _ := setupValidators(t, db, count) bs := &BeaconChainServer{ beaconDB: db, } req := ðpb.GetValidatorsRequest{PageToken: strconv.Itoa(1), PageSize: 100} wanted := fmt.Sprintf("page start %d >= list %d", req.PageSize, len(validators)) if _, err := bs.GetValidators(context.Background(), req); !strings.Contains(err.Error(), wanted) { t.Errorf("Expected error %v, received %v", wanted, err) } } func TestBeaconChainServer_GetValidatorsExceedsMaxPageSize(t *testing.T) { bs := &BeaconChainServer{} exceedsMax := int32(params.BeaconConfig().MaxPageSize + 1) wanted := fmt.Sprintf("requested page size %d can not be greater than max size %d", exceedsMax, params.BeaconConfig().MaxPageSize) req := ðpb.GetValidatorsRequest{PageToken: strconv.Itoa(0), PageSize: exceedsMax} if _, err := bs.GetValidators(context.Background(), req); !strings.Contains(err.Error(), wanted) { t.Errorf("Expected error %v, received %v", wanted, err) } } func TestBeaconChainServer_GetValidatorsDefaultPageSize(t *testing.T) { db := testutil.SetupDB(t) defer testutil.TeardownDB(t, db) validators, _ := setupValidators(t, db, 1000) bs := &BeaconChainServer{ beaconDB: db, } req := ðpb.GetValidatorsRequest{} res, err := bs.GetValidators(context.Background(), req) if err != nil { t.Fatal(err) } i := 0 j := params.BeaconConfig().DefaultPageSize if !reflect.DeepEqual(res.Validators, validators[i:j]) { t.Error("Incorrect respond of validators") } } func TestBeaconChainServer_ListAssignmentsInputOutOfRange(t *testing.T) { db := testutil.SetupDB(t) defer testutil.TeardownDB(t, db) setupValidators(t, db, 1) bs := &BeaconChainServer{beaconDB: db} wanted := fmt.Sprintf("page start %d >= list %d", 0, 0) if _, err := bs.ListValidatorAssignments(context.Background(), ðpb.ListValidatorAssignmentsRequest{Epoch: 0}); !strings.Contains(err.Error(), wanted) { t.Errorf("Expected error %v, received %v", wanted, err) } } func TestBeaconChainServer_ListAssignmentsExceedsMaxPageSize(t *testing.T) { bs := &BeaconChainServer{} exceedsMax := int32(params.BeaconConfig().MaxPageSize + 1) wanted := fmt.Sprintf("requested page size %d can not be greater than max size %d", exceedsMax, params.BeaconConfig().MaxPageSize) req := ðpb.ListValidatorAssignmentsRequest{PageToken: strconv.Itoa(0), PageSize: exceedsMax} if _, err := bs.ListValidatorAssignments(context.Background(), req); !strings.Contains(err.Error(), wanted) { t.Errorf("Expected error %v, received %v", wanted, err) } } func TestBeaconChainServer_ListAssignmentsDefaultPageSize(t *testing.T) { db := testutil.SetupDB(t) defer testutil.TeardownDB(t, db) ctx := context.Background() count := 1000 validators := make([]*ethpb.Validator, 0, count) for i := 0; i < count; i++ { if err := db.SaveValidatorIndex(ctx, [48]byte{byte(i)}, uint64(i)); err != nil { t.Fatal(err) } // Mark the validators with index divisible by 3 inactive. if i%3 == 0 { validators = append(validators, ðpb.Validator{PublicKey: []byte{byte(i)}, ExitEpoch: 0}) } else { validators = append(validators, ðpb.Validator{PublicKey: []byte{byte(i)}, ExitEpoch: params.BeaconConfig().FarFutureEpoch}) } } blk := ðpb.BeaconBlock{ Slot: 0, } blockRoot, err := ssz.SigningRoot(blk) if err != nil { t.Fatal(err) } if err := db.SaveHeadBlockRoot(ctx, blockRoot); err != nil { t.Fatal(err) } s := &pbp2p.BeaconState{ Validators: validators, RandaoMixes: make([][]byte, params.BeaconConfig().EpochsPerHistoricalVector), ActiveIndexRoots: make([][]byte, params.BeaconConfig().EpochsPerHistoricalVector)} if err := db.SaveState(ctx, s, blockRoot); err != nil { t.Fatal(err) } bs := &BeaconChainServer{ beaconDB: db, } res, err := bs.ListValidatorAssignments(context.Background(), ðpb.ListValidatorAssignmentsRequest{Epoch: 0}) if err != nil { t.Fatal(err) } // Construct the wanted assignments var wanted []*ethpb.ValidatorAssignments_CommitteeAssignment activeIndices, err := helpers.ActiveValidatorIndices(s, 0) if err != nil { t.Fatal(err) } for _, index := range activeIndices[0:params.BeaconConfig().DefaultPageSize] { committee, shard, slot, isProposer, err := helpers.CommitteeAssignment(s, 0, index) if err != nil { t.Fatal(err) } wanted = append(wanted, ðpb.ValidatorAssignments_CommitteeAssignment{ CrosslinkCommittees: committee, Shard: shard, Slot: slot, Proposer: isProposer, PublicKey: s.Validators[index].PublicKey, }) } if !reflect.DeepEqual(res.Assignments, wanted) { t.Error("Did not receive wanted assignments") } } func TestBeaconChainServer_ListAssignmentsFilterPubkeysIndicesNoPage(t *testing.T) { helpers.ClearAllCaches() db := testutil.SetupDB(t) defer testutil.TeardownDB(t, db) ctx := context.Background() count := 100 validators := make([]*ethpb.Validator, 0, count) for i := 0; i < count; i++ { if err := db.SaveValidatorIndex(ctx, [48]byte{byte(i)}, uint64(i)); err != nil { t.Fatal(err) } validators = append(validators, ðpb.Validator{PublicKey: []byte{byte(i)}, ExitEpoch: params.BeaconConfig().FarFutureEpoch}) } blk := ðpb.BeaconBlock{ Slot: 0, } blockRoot, err := ssz.SigningRoot(blk) if err != nil { t.Fatal(err) } if err := db.SaveHeadBlockRoot(ctx, blockRoot); err != nil { t.Fatal(err) } s := &pbp2p.BeaconState{ Validators: validators, RandaoMixes: make([][]byte, params.BeaconConfig().EpochsPerHistoricalVector), ActiveIndexRoots: make([][]byte, params.BeaconConfig().EpochsPerHistoricalVector)} if err := db.SaveState(ctx, s, blockRoot); err != nil { t.Fatal(err) } bs := &BeaconChainServer{ beaconDB: db, } req := ðpb.ListValidatorAssignmentsRequest{Epoch: 0, PublicKeys: [][]byte{{1}, {2}}, Indices: []uint64{2, 3}} res, err := bs.ListValidatorAssignments(context.Background(), req) if err != nil { t.Fatal(err) } // Construct the wanted assignments var wanted []*ethpb.ValidatorAssignments_CommitteeAssignment activeIndices, err := helpers.ActiveValidatorIndices(s, 0) if err != nil { t.Fatal(err) } for _, index := range activeIndices[1:4] { committee, shard, slot, isProposer, err := helpers.CommitteeAssignment(s, 0, index) if err != nil { t.Fatal(err) } wanted = append(wanted, ðpb.ValidatorAssignments_CommitteeAssignment{ CrosslinkCommittees: committee, Shard: shard, Slot: slot, Proposer: isProposer, PublicKey: s.Validators[index].PublicKey, }) } if !reflect.DeepEqual(res.Assignments, wanted) { t.Error("Did not receive wanted assignments") } } func TestBeaconChainServer_ListAssignmentsCanFilterPubkeysIndicesWithPages(t *testing.T) { helpers.ClearAllCaches() db := testutil.SetupDB(t) defer testutil.TeardownDB(t, db) ctx := context.Background() count := 100 validators := make([]*ethpb.Validator, 0, count) for i := 0; i < count; i++ { if err := db.SaveValidatorIndex(ctx, [48]byte{byte(i)}, uint64(i)); err != nil { t.Fatal(err) } validators = append(validators, ðpb.Validator{PublicKey: []byte{byte(i)}, ExitEpoch: params.BeaconConfig().FarFutureEpoch}) } blk := ðpb.BeaconBlock{ Slot: 0, } blockRoot, err := ssz.SigningRoot(blk) if err != nil { t.Fatal(err) } if err := db.SaveHeadBlockRoot(ctx, blockRoot); err != nil { t.Fatal(err) } s := &pbp2p.BeaconState{ Validators: validators, RandaoMixes: make([][]byte, params.BeaconConfig().EpochsPerHistoricalVector), ActiveIndexRoots: make([][]byte, params.BeaconConfig().EpochsPerHistoricalVector)} if err := db.SaveState(ctx, s, blockRoot); err != nil { t.Fatal(err) } bs := &BeaconChainServer{ beaconDB: db, } req := ðpb.ListValidatorAssignmentsRequest{Epoch: 0, Indices: []uint64{1, 2, 3, 4, 5, 6}, PageSize: 2, PageToken: "1"} res, err := bs.ListValidatorAssignments(context.Background(), req) if err != nil { t.Fatal(err) } // Construct the wanted assignments var assignments []*ethpb.ValidatorAssignments_CommitteeAssignment activeIndices, err := helpers.ActiveValidatorIndices(s, 0) if err != nil { t.Fatal(err) } for _, index := range activeIndices[3:5] { committee, shard, slot, isProposer, err := helpers.CommitteeAssignment(s, 0, index) if err != nil { t.Fatal(err) } assignments = append(assignments, ðpb.ValidatorAssignments_CommitteeAssignment{ CrosslinkCommittees: committee, Shard: shard, Slot: slot, Proposer: isProposer, PublicKey: s.Validators[index].PublicKey, }) } wantedRes := ðpb.ValidatorAssignments{ Assignments: assignments, TotalSize: int32(len(req.Indices)), NextPageToken: "2", } if !reflect.DeepEqual(res, wantedRes) { t.Error("Did not receive wanted assignments") } // Test the wrap around scenario assignments = nil req = ðpb.ListValidatorAssignmentsRequest{Epoch: 0, Indices: []uint64{1, 2, 3, 4, 5, 6}, PageSize: 5, PageToken: "1"} res, err = bs.ListValidatorAssignments(context.Background(), req) if err != nil { t.Fatal(err) } for _, index := range activeIndices[6:7] { committee, shard, slot, isProposer, err := helpers.CommitteeAssignment(s, 0, index) if err != nil { t.Fatal(err) } assignments = append(assignments, ðpb.ValidatorAssignments_CommitteeAssignment{ CrosslinkCommittees: committee, Shard: shard, Slot: slot, Proposer: isProposer, PublicKey: s.Validators[index].PublicKey, }) } wantedRes = ðpb.ValidatorAssignments{ Assignments: assignments, TotalSize: int32(len(req.Indices)), NextPageToken: "2", } if !reflect.DeepEqual(res, wantedRes) { t.Error("Did not receive wanted assignments") } } func TestBeaconChainServer_GetValidatorsParticipation(t *testing.T) { helpers.ClearAllCaches() db := testutil.SetupDB(t) defer testutil.TeardownDB(t, db) ctx := context.Background() epoch := uint64(1) attestedBalance := uint64(1) validatorCount := uint64(100) validators := make([]*ethpb.Validator, validatorCount) balances := make([]uint64, validatorCount) for i := 0; i < len(validators); i++ { validators[i] = ðpb.Validator{ ExitEpoch: params.BeaconConfig().FarFutureEpoch, EffectiveBalance: params.BeaconConfig().MaxEffectiveBalance, } balances[i] = params.BeaconConfig().MaxEffectiveBalance } atts := []*pbp2p.PendingAttestation{{Data: ðpb.AttestationData{Crosslink: ðpb.Crosslink{Shard: 0}, Target: ðpb.Checkpoint{}}}} var crosslinks []*ethpb.Crosslink for i := uint64(0); i < params.BeaconConfig().ShardCount; i++ { crosslinks = append(crosslinks, ðpb.Crosslink{ StartEpoch: 0, DataRoot: []byte{'A'}, }) } s := &pbp2p.BeaconState{ Slot: epoch*params.BeaconConfig().SlotsPerEpoch + 1, Validators: validators, Balances: balances, BlockRoots: make([][]byte, 128), Slashings: []uint64{0, 1e9, 1e9}, RandaoMixes: make([][]byte, params.BeaconConfig().EpochsPerHistoricalVector), ActiveIndexRoots: make([][]byte, params.BeaconConfig().EpochsPerHistoricalVector), CompactCommitteesRoots: make([][]byte, params.BeaconConfig().EpochsPerHistoricalVector), CurrentCrosslinks: crosslinks, CurrentEpochAttestations: atts, FinalizedCheckpoint: ðpb.Checkpoint{}, JustificationBits: bitfield.Bitvector4{0x00}, CurrentJustifiedCheckpoint: ðpb.Checkpoint{}, } bs := &BeaconChainServer{ beaconDB: db, } block := ðpb.BeaconBlock{ Slot: 1, } blockRoot, err := ssz.SigningRoot(block) if err != nil { t.Fatal(err) } if err := bs.beaconDB.SaveHeadBlockRoot(ctx, blockRoot); err != nil { t.Fatal(err) } if err := bs.beaconDB.SaveState(ctx, s, blockRoot); err != nil { t.Fatal(err) } res, err := bs.GetValidatorParticipation(ctx, ðpb.GetValidatorParticipationRequest{Epoch: epoch}) if err != nil { t.Fatal(err) } wanted := ðpb.ValidatorParticipation{ Epoch: epoch, VotedEther: attestedBalance, EligibleEther: validatorCount * params.BeaconConfig().MaxEffectiveBalance, GlobalParticipationRate: float32(attestedBalance) / float32(validatorCount*params.BeaconConfig().MaxEffectiveBalance), } if !reflect.DeepEqual(res, wanted) { t.Error("Incorrect validator participation respond") } } func TestBeaconChainServer_ListBlocksPagination(t *testing.T) { db := testutil.SetupDB(t) defer testutil.TeardownDB(t, db) ctx := context.Background() count := uint64(100) blks := make([]*ethpb.BeaconBlock, count) for i := uint64(0); i < count; i++ { b := ðpb.BeaconBlock{ Slot: i, } blks[i] = b } if err := db.SaveBlocks(ctx, blks); err != nil { t.Fatal(err) } root6, err := ssz.SigningRoot(ðpb.BeaconBlock{Slot: 6}) if err != nil { t.Fatal(err) } bs := &BeaconChainServer{ beaconDB: db, } tests := []struct { req *ethpb.ListBlocksRequest res *ethpb.ListBlocksResponse }{ {req: ðpb.ListBlocksRequest{ PageToken: strconv.Itoa(0), QueryFilter: ðpb.ListBlocksRequest_Slot{Slot: 5}, PageSize: 3}, res: ðpb.ListBlocksResponse{ Blocks: []*ethpb.BeaconBlock{{Slot: 5}}, NextPageToken: strconv.Itoa(1), TotalSize: 1}}, {req: ðpb.ListBlocksRequest{ PageToken: strconv.Itoa(0), QueryFilter: ðpb.ListBlocksRequest_Root{Root: root6[:]}, PageSize: 3}, res: ðpb.ListBlocksResponse{ Blocks: []*ethpb.BeaconBlock{{Slot: 6}}, TotalSize: 1}}, {req: ðpb.ListBlocksRequest{QueryFilter: ðpb.ListBlocksRequest_Root{Root: root6[:]}}, res: ðpb.ListBlocksResponse{ Blocks: []*ethpb.BeaconBlock{{Slot: 6}}, TotalSize: 1}}, {req: ðpb.ListBlocksRequest{ PageToken: strconv.Itoa(0), QueryFilter: ðpb.ListBlocksRequest_Epoch{Epoch: 0}, PageSize: 100}, res: ðpb.ListBlocksResponse{ Blocks: blks[0:params.BeaconConfig().SlotsPerEpoch], NextPageToken: strconv.Itoa(1), TotalSize: int32(params.BeaconConfig().SlotsPerEpoch)}}, {req: ðpb.ListBlocksRequest{ PageToken: strconv.Itoa(1), QueryFilter: ðpb.ListBlocksRequest_Epoch{Epoch: 5}, PageSize: 3}, res: ðpb.ListBlocksResponse{ Blocks: blks[43:46], NextPageToken: strconv.Itoa(2), TotalSize: int32(params.BeaconConfig().SlotsPerEpoch)}}, {req: ðpb.ListBlocksRequest{ PageToken: strconv.Itoa(1), QueryFilter: ðpb.ListBlocksRequest_Epoch{Epoch: 11}, PageSize: 7}, res: ðpb.ListBlocksResponse{ Blocks: blks[95:96], NextPageToken: strconv.Itoa(2), TotalSize: int32(params.BeaconConfig().SlotsPerEpoch)}}, {req: ðpb.ListBlocksRequest{ PageToken: strconv.Itoa(0), QueryFilter: ðpb.ListBlocksRequest_Epoch{Epoch: 12}, PageSize: 4}, res: ðpb.ListBlocksResponse{ Blocks: blks[96:100], NextPageToken: strconv.Itoa(1), TotalSize: int32(params.BeaconConfig().SlotsPerEpoch / 2)}}, } for _, test := range tests { res, err := bs.ListBlocks(ctx, test.req) if err != nil { t.Fatal(err) } if !proto.Equal(res, test.res) { t.Errorf("Incorrect blocks response, wanted %d, received %d", len(test.res.Blocks), len(res.Blocks)) } } } func TestBeaconChainServer_ListBlocksErrors(t *testing.T) { db := testutil.SetupDB(t) defer testutil.TeardownDB(t, db) ctx := context.Background() bs := &BeaconChainServer{beaconDB: db} exceedsMax := int32(params.BeaconConfig().MaxPageSize + 1) wanted := fmt.Sprintf("requested page size %d can not be greater than max size %d", exceedsMax, params.BeaconConfig().MaxPageSize) req := ðpb.ListBlocksRequest{PageToken: strconv.Itoa(0), PageSize: exceedsMax} if _, err := bs.ListBlocks(ctx, req); !strings.Contains(err.Error(), wanted) { t.Errorf("Expected error %v, received %v", wanted, err) } wanted = "must satisfy one of the filter requirement" req = ðpb.ListBlocksRequest{} if _, err := bs.ListBlocks(ctx, req); !strings.Contains(err.Error(), wanted) { t.Errorf("Expected error %v, received %v", wanted, err) } req = ðpb.ListBlocksRequest{QueryFilter: ðpb.ListBlocksRequest_Epoch{}} res, err := bs.ListBlocks(ctx, req) if err != nil { t.Fatal(err) } if len(res.Blocks) != 0 { t.Errorf("wanted empty list, got a list of %d", len(res.Blocks)) } if res.TotalSize != 0 { t.Errorf("wanted total size 0, got size %d", res.TotalSize) } req = ðpb.ListBlocksRequest{QueryFilter: ðpb.ListBlocksRequest_Slot{}} res, err = bs.ListBlocks(ctx, req) if err != nil { t.Fatal(err) } if len(res.Blocks) != 0 { t.Errorf("wanted empty list, got a list of %d", len(res.Blocks)) } if res.TotalSize != 0 { t.Errorf("wanted total size 0, got size %d", res.TotalSize) } req = ðpb.ListBlocksRequest{QueryFilter: ðpb.ListBlocksRequest_Root{Root: []byte{'A'}}} res, err = bs.ListBlocks(ctx, req) if err != nil { t.Fatal(err) } if len(res.Blocks) != 0 { t.Errorf("wanted empty list, got a list of %d", len(res.Blocks)) } if res.TotalSize != 0 { t.Errorf("wanted total size 0, got size %d", res.TotalSize) } req = ðpb.ListBlocksRequest{QueryFilter: ðpb.ListBlocksRequest_Root{Root: []byte{'A'}}} res, err = bs.ListBlocks(ctx, req) if err != nil { t.Fatal(err) } if len(res.Blocks) != 0 { t.Errorf("wanted empty list, got a list of %d", len(res.Blocks)) } if res.TotalSize != 0 { t.Errorf("wanted total size 0, got size %d", res.TotalSize) } } func setupValidators(t *testing.T, db db.Database, count int) ([]*ethpb.Validator, []uint64) { ctx := context.Background() balances := make([]uint64, count) validators := make([]*ethpb.Validator, 0, count) for i := 0; i < count; i++ { if err := db.SaveValidatorIndex(ctx, [48]byte{byte(i)}, uint64(i)); err != nil { t.Fatal(err) } balances[i] = uint64(i) validators = append(validators, ðpb.Validator{PublicKey: []byte{byte(i)}}) } blk := ðpb.BeaconBlock{ Slot: 0, } blockRoot, err := ssz.SigningRoot(blk) if err != nil { t.Fatal(err) } if err := db.SaveHeadBlockRoot(ctx, blockRoot); err != nil { t.Fatal(err) } if err := db.SaveState( context.Background(), &pbp2p.BeaconState{Validators: validators, Balances: balances}, blockRoot, ); err != nil { t.Fatal(err) } return validators, balances }