package beacon import ( "context" "encoding/binary" "fmt" "reflect" "sort" "strconv" "strings" "testing" "time" "github.com/gogo/protobuf/proto" ptypes "github.com/gogo/protobuf/types" ethpb "github.com/prysmaticlabs/ethereumapis/eth/v1alpha1" "github.com/prysmaticlabs/go-bitfield" "github.com/prysmaticlabs/go-ssz" mock "github.com/prysmaticlabs/prysm/beacon-chain/blockchain/testing" "github.com/prysmaticlabs/prysm/beacon-chain/cache" "github.com/prysmaticlabs/prysm/beacon-chain/core/epoch/precompute" "github.com/prysmaticlabs/prysm/beacon-chain/core/helpers" "github.com/prysmaticlabs/prysm/beacon-chain/db" dbTest "github.com/prysmaticlabs/prysm/beacon-chain/db/testing" "github.com/prysmaticlabs/prysm/beacon-chain/flags" stateTrie "github.com/prysmaticlabs/prysm/beacon-chain/state" "github.com/prysmaticlabs/prysm/beacon-chain/state/stategen" "github.com/prysmaticlabs/prysm/beacon-chain/state/stateutil" mockSync "github.com/prysmaticlabs/prysm/beacon-chain/sync/initial-sync/testing" pb "github.com/prysmaticlabs/prysm/proto/beacon/p2p/v1" pbp2p "github.com/prysmaticlabs/prysm/proto/beacon/p2p/v1" "github.com/prysmaticlabs/prysm/shared/bytesutil" "github.com/prysmaticlabs/prysm/shared/featureconfig" "github.com/prysmaticlabs/prysm/shared/params" "github.com/prysmaticlabs/prysm/shared/testutil" ) func TestServer_GetValidatorActiveSetChanges_CannotRequestFutureEpoch(t *testing.T) { db := dbTest.SetupDB(t) ctx := context.Background() st := testutil.NewBeaconState() if err := st.SetSlot(0); err != nil { t.Fatal(err) } bs := &Server{ GenesisTimeFetcher: &mock.ChainService{}, HeadFetcher: &mock.ChainService{ State: st, }, BeaconDB: db, } wanted := "Cannot retrieve information about an epoch in the future" if _, err := bs.GetValidatorActiveSetChanges( ctx, ðpb.GetValidatorActiveSetChangesRequest{ QueryFilter: ðpb.GetValidatorActiveSetChangesRequest_Epoch{ Epoch: helpers.SlotToEpoch(bs.GenesisTimeFetcher.CurrentSlot()) + 1, }, }, ); err != nil && !strings.Contains(err.Error(), wanted) { t.Errorf("Expected error %v, received %v", wanted, err) } } func TestServer_ListValidatorBalances_CannotRequestFutureEpoch(t *testing.T) { db := dbTest.SetupDB(t) ctx := context.Background() st := testutil.NewBeaconState() if err := st.SetSlot(0); err != nil { t.Fatal(err) } bs := &Server{ BeaconDB: db, HeadFetcher: &mock.ChainService{ State: st, }, GenesisTimeFetcher: &mock.ChainService{}, } wanted := "Cannot retrieve information about an epoch in the future" if _, err := bs.ListValidatorBalances( ctx, ðpb.ListValidatorBalancesRequest{ QueryFilter: ðpb.ListValidatorBalancesRequest_Epoch{ Epoch: helpers.SlotToEpoch(bs.GenesisTimeFetcher.CurrentSlot()) + 1, }, }, ); err != nil && !strings.Contains(err.Error(), wanted) { t.Errorf("Expected error %v, received %v", wanted, err) } } func TestServer_ListValidatorBalances_NoResults(t *testing.T) { db := dbTest.SetupDB(t) resetCfg := featureconfig.InitWithReset(&featureconfig.Flags{NewStateMgmt: true}) defer resetCfg() ctx := context.Background() st := testutil.NewBeaconState() if err := st.SetSlot(0); err != nil { t.Fatal(err) } bs := &Server{ GenesisTimeFetcher: &mock.ChainService{}, StateGen: stategen.New(db, cache.NewStateSummaryCache()), } headState := testutil.NewBeaconState() b := ðpb.SignedBeaconBlock{Block: ðpb.BeaconBlock{}} if err := db.SaveBlock(ctx, b); err != nil { t.Fatal(err) } gRoot, err := stateutil.BlockRoot(b.Block) if err != nil { t.Fatal(err) } if err := db.SaveGenesisBlockRoot(ctx, gRoot); err != nil { t.Fatal(err) } if err := db.SaveState(ctx, headState, gRoot); err != nil { t.Fatal(err) } wanted := ðpb.ValidatorBalances{ Balances: make([]*ethpb.ValidatorBalances_Balance, 0), TotalSize: int32(0), NextPageToken: strconv.Itoa(0), } res, err := bs.ListValidatorBalances( ctx, ðpb.ListValidatorBalancesRequest{ QueryFilter: ðpb.ListValidatorBalancesRequest_Epoch{ Epoch: 0, }, }, ) if err != nil { t.Fatal(err) } if !proto.Equal(wanted, res) { t.Errorf("Wanted %v, received %v", wanted, res) } } func TestServer_ListValidatorBalances_DefaultResponse_NoArchive(t *testing.T) { db := dbTest.SetupDB(t) ctx := context.Background() numItems := 100 validators := make([]*ethpb.Validator, numItems) balances := make([]uint64, numItems) balancesResponse := make([]*ethpb.ValidatorBalances_Balance, numItems) for i := 0; i < numItems; i++ { validators[i] = ðpb.Validator{ PublicKey: pubKey(uint64(i)), WithdrawalCredentials: make([]byte, 32), } balances[i] = params.BeaconConfig().MaxEffectiveBalance balancesResponse[i] = ðpb.ValidatorBalances_Balance{ PublicKey: pubKey(uint64(i)), Index: uint64(i), Balance: params.BeaconConfig().MaxEffectiveBalance, } } st := testutil.NewBeaconState() if err := st.SetSlot(0); err != nil { t.Fatal(err) } if err := st.SetValidators(validators); err != nil { t.Fatal(err) } if err := st.SetBalances(balances); err != nil { t.Fatal(err) } b := ðpb.SignedBeaconBlock{Block: ðpb.BeaconBlock{}} if err := db.SaveBlock(ctx, b); err != nil { t.Fatal(err) } gRoot, err := stateutil.BlockRoot(b.Block) if err != nil { t.Fatal(err) } if err := db.SaveGenesisBlockRoot(ctx, gRoot); err != nil { t.Fatal(err) } if err := db.SaveState(ctx, st, gRoot); err != nil { t.Fatal(err) } bs := &Server{ GenesisTimeFetcher: &mock.ChainService{}, StateGen: stategen.New(db, cache.NewStateSummaryCache()), HeadFetcher: &mock.ChainService{ State: st, }, } res, err := bs.ListValidatorBalances( ctx, ðpb.ListValidatorBalancesRequest{ QueryFilter: ðpb.ListValidatorBalancesRequest_Epoch{Epoch: 0}, }, ) if err != nil { t.Fatal(err) } if !reflect.DeepEqual(balancesResponse, res.Balances) { t.Errorf("Wanted %v, received %v", balancesResponse, res.Balances) } } func TestServer_ListValidatorBalances_PaginationOutOfRange(t *testing.T) { db := dbTest.SetupDB(t) ctx := context.Background() setupValidators(t, db, 3) st := testutil.NewBeaconState() b := ðpb.SignedBeaconBlock{Block: ðpb.BeaconBlock{}} if err := db.SaveBlock(ctx, b); err != nil { t.Fatal(err) } gRoot, err := stateutil.BlockRoot(b.Block) if err != nil { t.Fatal(err) } if err := db.SaveGenesisBlockRoot(ctx, gRoot); err != nil { t.Fatal(err) } if err := db.SaveState(ctx, st, gRoot); err != nil { t.Fatal(err) } bs := &Server{ GenesisTimeFetcher: &mock.ChainService{}, StateGen: stategen.New(db, cache.NewStateSummaryCache()), HeadFetcher: &mock.ChainService{ State: st, }, } req := ðpb.ListValidatorBalancesRequest{PageToken: strconv.Itoa(1), PageSize: 100, QueryFilter: ðpb.ListValidatorBalancesRequest_Epoch{Epoch: 0}} wanted := fmt.Sprintf("page start %d >= list %d", req.PageSize, len(st.Balances())) if _, err := bs.ListValidatorBalances(context.Background(), req); err != nil && !strings.Contains(err.Error(), wanted) { t.Errorf("Expected error %v, received %v", wanted, err) } } func TestServer_ListValidatorBalances_ExceedsMaxPageSize(t *testing.T) { bs := &Server{} exceedsMax := int32(flags.Get().MaxPageSize + 1) wanted := fmt.Sprintf( "Requested page size %d can not be greater than max size %d", exceedsMax, flags.Get().MaxPageSize, ) req := ðpb.ListValidatorBalancesRequest{PageSize: exceedsMax} if _, err := bs.ListValidatorBalances(context.Background(), req); err != nil && !strings.Contains(err.Error(), wanted) { t.Errorf("Expected error %v, received %v", wanted, err) } } func pubKey(i uint64) []byte { pubKey := make([]byte, params.BeaconConfig().BLSPubkeyLength) binary.LittleEndian.PutUint64(pubKey, i) return pubKey } func TestServer_ListValidatorBalances_Pagination_Default(t *testing.T) { db := dbTest.SetupDB(t) ctx := context.Background() setupValidators(t, db, 100) headState, err := db.HeadState(context.Background()) if err != nil { t.Fatal(err) } b := ðpb.SignedBeaconBlock{Block: ðpb.BeaconBlock{}} gRoot, err := stateutil.BlockRoot(b.Block) if err != nil { t.Fatal(err) } if err := db.SaveGenesisBlockRoot(ctx, gRoot); err != nil { t.Fatal(err) } if err := db.SaveState(ctx, headState, gRoot); err != nil { t.Fatal(err) } bs := &Server{ GenesisTimeFetcher: &mock.ChainService{}, StateGen: stategen.New(db, cache.NewStateSummaryCache()), HeadFetcher: &mock.ChainService{ State: headState, }, } tests := []struct { req *ethpb.ListValidatorBalancesRequest res *ethpb.ValidatorBalances }{ {req: ðpb.ListValidatorBalancesRequest{PublicKeys: [][]byte{pubKey(99)}, QueryFilter: ðpb.ListValidatorBalancesRequest_Epoch{Epoch: 0}}, res: ðpb.ValidatorBalances{ Balances: []*ethpb.ValidatorBalances_Balance{ {Index: 99, PublicKey: pubKey(99), Balance: 99}, }, NextPageToken: "", TotalSize: 1, }, }, {req: ðpb.ListValidatorBalancesRequest{Indices: []uint64{1, 2, 3}, QueryFilter: ðpb.ListValidatorBalancesRequest_Epoch{Epoch: 0}}, res: ðpb.ValidatorBalances{ Balances: []*ethpb.ValidatorBalances_Balance{ {Index: 1, PublicKey: pubKey(1), Balance: 1}, {Index: 2, PublicKey: pubKey(2), Balance: 2}, {Index: 3, PublicKey: pubKey(3), Balance: 3}, }, NextPageToken: "", TotalSize: 3, }, }, {req: ðpb.ListValidatorBalancesRequest{PublicKeys: [][]byte{pubKey(10), pubKey(11), pubKey(12)}, QueryFilter: ðpb.ListValidatorBalancesRequest_Epoch{Epoch: 0}}, res: ðpb.ValidatorBalances{ Balances: []*ethpb.ValidatorBalances_Balance{ {Index: 10, PublicKey: pubKey(10), Balance: 10}, {Index: 11, PublicKey: pubKey(11), Balance: 11}, {Index: 12, PublicKey: pubKey(12), Balance: 12}, }, NextPageToken: "", TotalSize: 3, }}, {req: ðpb.ListValidatorBalancesRequest{PublicKeys: [][]byte{pubKey(2), pubKey(3)}, Indices: []uint64{3, 4}, QueryFilter: ðpb.ListValidatorBalancesRequest_Epoch{Epoch: 0}}, // Duplication res: ðpb.ValidatorBalances{ Balances: []*ethpb.ValidatorBalances_Balance{ {Index: 2, PublicKey: pubKey(2), Balance: 2}, {Index: 3, PublicKey: pubKey(3), Balance: 3}, {Index: 4, PublicKey: pubKey(4), Balance: 4}, }, NextPageToken: "", TotalSize: 3, }}, {req: ðpb.ListValidatorBalancesRequest{PublicKeys: [][]byte{{}}, Indices: []uint64{3, 4}, QueryFilter: ðpb.ListValidatorBalancesRequest_Epoch{Epoch: 0}}, // Public key has a blank value res: ðpb.ValidatorBalances{ Balances: []*ethpb.ValidatorBalances_Balance{ {Index: 3, PublicKey: pubKey(3), Balance: 3}, {Index: 4, PublicKey: pubKey(4), Balance: 4}, }, NextPageToken: "", TotalSize: 2, }}, } 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.Errorf("Expected %v, received %v", test.res, res) } } } func TestServer_ListValidatorBalances_Pagination_CustomPageSizes(t *testing.T) { db := dbTest.SetupDB(t) ctx := context.Background() count := 1000 setupValidators(t, db, count) headState, err := db.HeadState(context.Background()) if err != nil { t.Fatal(err) } b := ðpb.SignedBeaconBlock{Block: ðpb.BeaconBlock{}} gRoot, err := stateutil.BlockRoot(b.Block) if err != nil { t.Fatal(err) } if err := db.SaveGenesisBlockRoot(ctx, gRoot); err != nil { t.Fatal(err) } if err := db.SaveState(ctx, headState, gRoot); err != nil { t.Fatal(err) } bs := &Server{ GenesisTimeFetcher: &mock.ChainService{}, StateGen: stategen.New(db, cache.NewStateSummaryCache()), HeadFetcher: &mock.ChainService{ State: headState, }, } tests := []struct { req *ethpb.ListValidatorBalancesRequest res *ethpb.ValidatorBalances }{ {req: ðpb.ListValidatorBalancesRequest{PageToken: strconv.Itoa(1), PageSize: 3, QueryFilter: ðpb.ListValidatorBalancesRequest_Epoch{Epoch: 0}}, res: ðpb.ValidatorBalances{ Balances: []*ethpb.ValidatorBalances_Balance{ {PublicKey: pubKey(3), Index: 3, Balance: uint64(3)}, {PublicKey: pubKey(4), Index: 4, Balance: uint64(4)}, {PublicKey: pubKey(5), Index: 5, Balance: uint64(5)}}, NextPageToken: strconv.Itoa(2), TotalSize: int32(count)}}, {req: ðpb.ListValidatorBalancesRequest{PageToken: strconv.Itoa(10), PageSize: 5, QueryFilter: ðpb.ListValidatorBalancesRequest_Epoch{Epoch: 0}}, res: ðpb.ValidatorBalances{ Balances: []*ethpb.ValidatorBalances_Balance{ {PublicKey: pubKey(50), Index: 50, Balance: uint64(50)}, {PublicKey: pubKey(51), Index: 51, Balance: uint64(51)}, {PublicKey: pubKey(52), Index: 52, Balance: uint64(52)}, {PublicKey: pubKey(53), Index: 53, Balance: uint64(53)}, {PublicKey: pubKey(54), Index: 54, Balance: uint64(54)}}, NextPageToken: strconv.Itoa(11), TotalSize: int32(count)}}, {req: ðpb.ListValidatorBalancesRequest{PageToken: strconv.Itoa(33), PageSize: 3, QueryFilter: ðpb.ListValidatorBalancesRequest_Epoch{Epoch: 0}}, res: ðpb.ValidatorBalances{ Balances: []*ethpb.ValidatorBalances_Balance{ {PublicKey: pubKey(99), Index: 99, Balance: uint64(99)}, {PublicKey: pubKey(100), Index: 100, Balance: uint64(100)}, {PublicKey: pubKey(101), Index: 101, Balance: uint64(101)}, }, NextPageToken: "34", TotalSize: int32(count)}}, {req: ðpb.ListValidatorBalancesRequest{PageSize: 2, QueryFilter: ðpb.ListValidatorBalancesRequest_Epoch{Epoch: 0}}, res: ðpb.ValidatorBalances{ Balances: []*ethpb.ValidatorBalances_Balance{ {PublicKey: pubKey(0), Index: 0, Balance: uint64(0)}, {PublicKey: pubKey(1), Index: 1, Balance: uint64(1)}}, NextPageToken: strconv.Itoa(1), TotalSize: int32(count)}}, } 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.Errorf("Expected %v, received %v", test.res, res) } } } func TestServer_ListValidatorBalances_OutOfRange(t *testing.T) { db := dbTest.SetupDB(t) resetCfg := featureconfig.InitWithReset(&featureconfig.Flags{NewStateMgmt: true}) defer resetCfg() ctx := context.Background() setupValidators(t, db, 1) headState, err := db.HeadState(context.Background()) if err != nil { t.Fatal(err) } b := ðpb.SignedBeaconBlock{Block: ðpb.BeaconBlock{}} gRoot, err := stateutil.BlockRoot(b.Block) if err != nil { t.Fatal(err) } if err := db.SaveGenesisBlockRoot(ctx, gRoot); err != nil { t.Fatal(err) } if err := db.SaveState(ctx, headState, gRoot); err != nil { t.Fatal(err) } bs := &Server{ GenesisTimeFetcher: &mock.ChainService{}, StateGen: stategen.New(db, cache.NewStateSummaryCache()), HeadFetcher: &mock.ChainService{ State: headState, }, } req := ðpb.ListValidatorBalancesRequest{Indices: []uint64{uint64(1)}, QueryFilter: ðpb.ListValidatorBalancesRequest_Epoch{Epoch: 0}} wanted := "Validator index 1 >= balance list 1" if _, err := bs.ListValidatorBalances(context.Background(), req); err == nil || !strings.Contains(err.Error(), wanted) { t.Errorf("Expected error %v, received %v", wanted, err) } } func TestServer_ListValidators_CannotRequestFutureEpoch(t *testing.T) { db := dbTest.SetupDB(t) ctx := context.Background() st := testutil.NewBeaconState() if err := st.SetSlot(0); err != nil { t.Fatal(err) } bs := &Server{ BeaconDB: db, GenesisTimeFetcher: &mock.ChainService{ // We are in epoch 0. Genesis: time.Now(), }, HeadFetcher: &mock.ChainService{ State: st, }, } wanted := "Cannot retrieve information about an epoch in the future" if _, err := bs.ListValidators( ctx, ðpb.ListValidatorsRequest{ QueryFilter: ðpb.ListValidatorsRequest_Epoch{ Epoch: 1, }, }, ); err != nil && !strings.Contains(err.Error(), wanted) { t.Errorf("Expected error %v, received %v", wanted, err) } } func TestServer_ListValidatorBalances_FromArchive(t *testing.T) { db := dbTest.SetupDB(t) ctx := context.Background() resetCfg := featureconfig.InitWithReset(&featureconfig.Flags{NewStateMgmt: false}) defer resetCfg() epoch := uint64(0) validators, balances := setupValidators(t, db, 100) if err := db.SaveArchivedBalances(ctx, epoch, balances); err != nil { t.Fatal(err) } newerBalances := make([]uint64, len(balances)) for i := 0; i < len(newerBalances); i++ { newerBalances[i] = balances[i] * 2 } st := testutil.NewBeaconState() if err := st.SetSlot(params.BeaconConfig().SlotsPerEpoch * 3); err != nil { t.Fatal(err) } if err := st.SetValidators(validators); err != nil { t.Fatal(err) } if err := st.SetBalances(newerBalances); err != nil { t.Fatal(err) } bs := &Server{ BeaconDB: db, HeadFetcher: &mock.ChainService{ State: st, }, GenesisTimeFetcher: &mock.ChainService{}, } req := ðpb.ListValidatorBalancesRequest{ QueryFilter: ðpb.ListValidatorBalancesRequest_Epoch{Epoch: 0}, Indices: []uint64{uint64(1)}, } res, err := bs.ListValidatorBalances(context.Background(), req) if err != nil { t.Fatal(err) } // We should expect a response containing the old balance from epoch 0, // not the new balance from the current state. want := []*ethpb.ValidatorBalances_Balance{ { PublicKey: validators[1].PublicKey, Index: 1, Balance: balances[1], }, } if !reflect.DeepEqual(want, res.Balances) { t.Errorf("Wanted %v, received %v", want, res.Balances) } } func TestServer_ListValidatorBalances_FromArchive_NewValidatorNotFound(t *testing.T) { db := dbTest.SetupDB(t) ctx := context.Background() resetCfg := featureconfig.InitWithReset(&featureconfig.Flags{NewStateMgmt: false}) defer resetCfg() epoch := uint64(0) _, balances := setupValidators(t, db, 100) if err := db.SaveArchivedBalances(ctx, epoch, balances); err != nil { t.Fatal(err) } newValidators, newBalances := setupValidators(t, db, 200) st := testutil.NewBeaconState() if err := st.SetSlot(params.BeaconConfig().SlotsPerEpoch * 3); err != nil { t.Fatal(err) } if err := st.SetValidators(newValidators); err != nil { t.Fatal(err) } if err := st.SetBalances(newBalances); err != nil { t.Fatal(err) } bs := &Server{ BeaconDB: db, HeadFetcher: &mock.ChainService{ State: st, }, } req := ðpb.ListValidatorBalancesRequest{ QueryFilter: ðpb.ListValidatorBalancesRequest_Epoch{Epoch: 0}, Indices: []uint64{1, 150, 161}, } if _, err := bs.ListValidatorBalances(context.Background(), req); err == nil || !strings.Contains(err.Error(), "does not exist") { t.Errorf("Wanted out of range error for including newer validators in the arguments, received %v", err) } } func TestServer_ListValidators_NoResults(t *testing.T) { db := dbTest.SetupDB(t) resetCfg := featureconfig.InitWithReset(&featureconfig.Flags{NewStateMgmt: true}) defer resetCfg() ctx := context.Background() st := testutil.NewBeaconState() if err := st.SetSlot(0); err != nil { t.Fatal(err) } bs := &Server{ BeaconDB: db, GenesisTimeFetcher: &mock.ChainService{ // We are in epoch 0. Genesis: time.Now(), }, HeadFetcher: &mock.ChainService{ State: st, }, } wanted := ðpb.Validators{ ValidatorList: make([]*ethpb.Validators_ValidatorContainer, 0), TotalSize: int32(0), NextPageToken: strconv.Itoa(0), } res, err := bs.ListValidators( ctx, ðpb.ListValidatorsRequest{ QueryFilter: ðpb.ListValidatorsRequest_Epoch{ Epoch: 0, }, }, ) if err != nil { t.Fatal(err) } if !proto.Equal(wanted, res) { t.Errorf("Wanted %v, received %v", wanted, res) } } func TestServer_ListValidators_OnlyActiveValidators(t *testing.T) { ctx := context.Background() count := 100 balances := make([]uint64, count) validators := make([]*ethpb.Validator, count) activeValidators := make([]*ethpb.Validators_ValidatorContainer, 0) for i := 0; i < count; i++ { pubKey := pubKey(uint64(i)) balances[i] = params.BeaconConfig().MaxEffectiveBalance // We mark even validators as active, and odd validators as inactive. if i%2 == 0 { val := ðpb.Validator{ PublicKey: pubKey, WithdrawalCredentials: make([]byte, 32), ActivationEpoch: 0, ExitEpoch: params.BeaconConfig().FarFutureEpoch, } validators[i] = val activeValidators = append(activeValidators, ðpb.Validators_ValidatorContainer{ Index: uint64(i), Validator: val, }) } else { validators[i] = ðpb.Validator{ PublicKey: pubKey, WithdrawalCredentials: make([]byte, 32), ActivationEpoch: 0, ExitEpoch: 0, } } } st := testutil.NewBeaconState() if err := st.SetValidators(validators); err != nil { t.Fatal(err) } if err := st.SetBalances(balances); err != nil { t.Fatal(err) } bs := &Server{ HeadFetcher: &mock.ChainService{ State: st, }, GenesisTimeFetcher: &mock.ChainService{ // We are in epoch 0. Genesis: time.Now(), }, } received, err := bs.ListValidators(ctx, ðpb.ListValidatorsRequest{ Active: true, }) if err != nil { t.Fatal(err) } if !reflect.DeepEqual(activeValidators, received.ValidatorList) { t.Errorf("Wanted %v, received %v", activeValidators, received.ValidatorList) } } func TestServer_ListValidators_NoPagination(t *testing.T) { db := dbTest.SetupDB(t) validators, _ := setupValidators(t, db, 100) want := make([]*ethpb.Validators_ValidatorContainer, len(validators)) for i := 0; i < len(validators); i++ { want[i] = ðpb.Validators_ValidatorContainer{ Index: uint64(i), Validator: validators[i], } } headState, err := db.HeadState(context.Background()) if err != nil { t.Fatal(err) } bs := &Server{ HeadFetcher: &mock.ChainService{ State: headState, }, GenesisTimeFetcher: &mock.ChainService{ // We are in epoch 0. Genesis: time.Now(), }, FinalizationFetcher: &mock.ChainService{ FinalizedCheckPoint: ðpb.Checkpoint{ Epoch: 0, }, }, } received, err := bs.ListValidators(context.Background(), ðpb.ListValidatorsRequest{}) if err != nil { t.Fatal(err) } if !reflect.DeepEqual(want, received.ValidatorList) { t.Fatal("Incorrect respond of validators") } } func TestServer_ListValidators_IndicesPubKeys(t *testing.T) { db := dbTest.SetupDB(t) validators, _ := setupValidators(t, db, 100) indicesWanted := []uint64{2, 7, 11, 17} pubkeyIndicesWanted := []uint64{3, 5, 9, 15} allIndicesWanted := append(indicesWanted, pubkeyIndicesWanted...) want := make([]*ethpb.Validators_ValidatorContainer, len(allIndicesWanted)) for i, idx := range allIndicesWanted { want[i] = ðpb.Validators_ValidatorContainer{ Index: idx, Validator: validators[idx], } } sort.Slice(want, func(i int, j int) bool { return want[i].Index < want[j].Index }) headState, err := db.HeadState(context.Background()) if err != nil { t.Fatal(err) } bs := &Server{ HeadFetcher: &mock.ChainService{ State: headState, }, FinalizationFetcher: &mock.ChainService{ FinalizedCheckPoint: ðpb.Checkpoint{ Epoch: 0, }, }, GenesisTimeFetcher: &mock.ChainService{ // We are in epoch 0. Genesis: time.Now(), }, } pubKeysWanted := make([][]byte, len(pubkeyIndicesWanted)) for i, indice := range pubkeyIndicesWanted { pubKeysWanted[i] = pubKey(indice) } req := ðpb.ListValidatorsRequest{ Indices: indicesWanted, PublicKeys: pubKeysWanted, } received, err := bs.ListValidators(context.Background(), req) if err != nil { t.Fatal(err) } if !reflect.DeepEqual(want, received.ValidatorList) { t.Fatal("Incorrect respond of validators") } } func TestServer_ListValidators_Pagination(t *testing.T) { db := dbTest.SetupDB(t) count := 100 setupValidators(t, db, count) headState, err := db.HeadState(context.Background()) if err != nil { t.Fatal(err) } bs := &Server{ HeadFetcher: &mock.ChainService{ State: headState, }, FinalizationFetcher: &mock.ChainService{ FinalizedCheckPoint: ðpb.Checkpoint{ Epoch: 0, }, }, GenesisTimeFetcher: &mock.ChainService{ // We are in epoch 0. Genesis: time.Now(), }, } tests := []struct { req *ethpb.ListValidatorsRequest res *ethpb.Validators }{ {req: ðpb.ListValidatorsRequest{PageToken: strconv.Itoa(1), PageSize: 3}, res: ðpb.Validators{ ValidatorList: []*ethpb.Validators_ValidatorContainer{ { Validator: ðpb.Validator{ PublicKey: pubKey(3), WithdrawalCredentials: make([]byte, 32), }, Index: 3, }, { Validator: ðpb.Validator{ PublicKey: pubKey(4), WithdrawalCredentials: make([]byte, 32), }, Index: 4, }, { Validator: ðpb.Validator{ PublicKey: pubKey(5), WithdrawalCredentials: make([]byte, 32), }, Index: 5, }, }, NextPageToken: strconv.Itoa(2), TotalSize: int32(count)}}, {req: ðpb.ListValidatorsRequest{PageToken: strconv.Itoa(10), PageSize: 5}, res: ðpb.Validators{ ValidatorList: []*ethpb.Validators_ValidatorContainer{ { Validator: ðpb.Validator{ PublicKey: pubKey(50), WithdrawalCredentials: make([]byte, 32), }, Index: 50, }, { Validator: ðpb.Validator{ PublicKey: pubKey(51), WithdrawalCredentials: make([]byte, 32), }, Index: 51, }, { Validator: ðpb.Validator{ PublicKey: pubKey(52), WithdrawalCredentials: make([]byte, 32), }, Index: 52, }, { Validator: ðpb.Validator{ PublicKey: pubKey(53), WithdrawalCredentials: make([]byte, 32), }, Index: 53, }, { Validator: ðpb.Validator{ PublicKey: pubKey(54), WithdrawalCredentials: make([]byte, 32), }, Index: 54, }, }, NextPageToken: strconv.Itoa(11), TotalSize: int32(count)}}, {req: ðpb.ListValidatorsRequest{PageToken: strconv.Itoa(33), PageSize: 3}, res: ðpb.Validators{ ValidatorList: []*ethpb.Validators_ValidatorContainer{ { Validator: ðpb.Validator{ PublicKey: pubKey(99), WithdrawalCredentials: make([]byte, 32), }, Index: 99, }, }, NextPageToken: "", TotalSize: int32(count)}}, {req: ðpb.ListValidatorsRequest{PageSize: 2}, res: ðpb.Validators{ ValidatorList: []*ethpb.Validators_ValidatorContainer{ { Validator: ðpb.Validator{ PublicKey: pubKey(0), WithdrawalCredentials: make([]byte, 32), }, Index: 0, }, { Validator: ðpb.Validator{ PublicKey: pubKey(1), WithdrawalCredentials: make([]byte, 32), }, Index: 1, }, }, NextPageToken: strconv.Itoa(1), TotalSize: int32(count)}}, } for _, test := range tests { res, err := bs.ListValidators(context.Background(), test.req) if err != nil { t.Fatal(err) } if !proto.Equal(res, test.res) { t.Errorf("Incorrect validator response, wanted %v, received %v", test.res, res) } } } func TestServer_ListValidators_PaginationOutOfRange(t *testing.T) { db := dbTest.SetupDB(t) count := 1 validators, _ := setupValidators(t, db, count) headState, err := db.HeadState(context.Background()) if err != nil { t.Fatal(err) } bs := &Server{ HeadFetcher: &mock.ChainService{ State: headState, }, FinalizationFetcher: &mock.ChainService{ FinalizedCheckPoint: ðpb.Checkpoint{ Epoch: 0, }, }, GenesisTimeFetcher: &mock.ChainService{ // We are in epoch 0. Genesis: time.Now(), }, } req := ðpb.ListValidatorsRequest{PageToken: strconv.Itoa(1), PageSize: 100} wanted := fmt.Sprintf("page start %d >= list %d", req.PageSize, len(validators)) if _, err := bs.ListValidators(context.Background(), req); err == nil || !strings.Contains(err.Error(), wanted) { t.Errorf("Expected error %v, received %v", wanted, err) } } func TestServer_ListValidators_ExceedsMaxPageSize(t *testing.T) { bs := &Server{} exceedsMax := int32(flags.Get().MaxPageSize + 1) wanted := fmt.Sprintf("Requested page size %d can not be greater than max size %d", exceedsMax, flags.Get().MaxPageSize) req := ðpb.ListValidatorsRequest{PageToken: strconv.Itoa(0), PageSize: exceedsMax} if _, err := bs.ListValidators(context.Background(), req); err == nil || !strings.Contains(err.Error(), wanted) { t.Errorf("Expected error %v, received %v", wanted, err) } } func TestServer_ListValidators_DefaultPageSize(t *testing.T) { db := dbTest.SetupDB(t) validators, _ := setupValidators(t, db, 1000) want := make([]*ethpb.Validators_ValidatorContainer, len(validators)) for i := 0; i < len(validators); i++ { want[i] = ðpb.Validators_ValidatorContainer{ Index: uint64(i), Validator: validators[i], } } headState, err := db.HeadState(context.Background()) if err != nil { t.Fatal(err) } bs := &Server{ HeadFetcher: &mock.ChainService{ State: headState, }, FinalizationFetcher: &mock.ChainService{ FinalizedCheckPoint: ðpb.Checkpoint{ Epoch: 0, }, }, GenesisTimeFetcher: &mock.ChainService{ // We are in epoch 0. Genesis: time.Now(), }, } req := ðpb.ListValidatorsRequest{} res, err := bs.ListValidators(context.Background(), req) if err != nil { t.Fatal(err) } i := 0 j := params.BeaconConfig().DefaultPageSize if !reflect.DeepEqual(res.ValidatorList, want[i:j]) { t.Error("Incorrect respond of validators") } } func TestServer_ListValidators_FromOldEpoch(t *testing.T) { numEpochs := 30 validators := make([]*ethpb.Validator, numEpochs) for i := 0; i < numEpochs; i++ { validators[i] = ðpb.Validator{ ActivationEpoch: uint64(i), PublicKey: make([]byte, 48), WithdrawalCredentials: make([]byte, 32), } } want := make([]*ethpb.Validators_ValidatorContainer, len(validators)) for i := 0; i < len(validators); i++ { want[i] = ðpb.Validators_ValidatorContainer{ Index: uint64(i), Validator: validators[i], } } st := testutil.NewBeaconState() if err := st.SetSlot(helpers.StartSlot(30)); err != nil { t.Fatal(err) } if err := st.SetValidators(validators); err != nil { t.Fatal(err) } secondsPerEpoch := params.BeaconConfig().SecondsPerSlot * params.BeaconConfig().SlotsPerEpoch bs := &Server{ HeadFetcher: &mock.ChainService{ State: st, }, GenesisTimeFetcher: &mock.ChainService{ // We are in epoch 30 Genesis: time.Now().Add(time.Duration(-1*int64(30*secondsPerEpoch)) * time.Second), }, } req := ðpb.ListValidatorsRequest{ QueryFilter: ðpb.ListValidatorsRequest_Genesis{ Genesis: true, }, } res, err := bs.ListValidators(context.Background(), req) if err != nil { t.Fatal(err) } if len(res.ValidatorList) != 1 { t.Errorf("Wanted 1 validator at genesis, received %d", len(res.ValidatorList)) } req = ðpb.ListValidatorsRequest{ QueryFilter: ðpb.ListValidatorsRequest_Epoch{ Epoch: 20, }, } res, err = bs.ListValidators(context.Background(), req) if err != nil { t.Fatal(err) } if !reflect.DeepEqual(res.ValidatorList, want[:21]) { t.Errorf("Incorrect number of validators, wanted %d received %d", len(want[:21]), len(res.ValidatorList)) } } func TestServer_ListValidators_ProcessHeadStateSlots(t *testing.T) { params.SetupTestConfigCleanup(t) params.OverrideBeaconConfig(params.MinimalSpecConfig()) headSlot := uint64(0) numValidators := uint64(64) validators := make([]*ethpb.Validator, numValidators) balances := make([]uint64, numValidators) for i := uint64(0); i < numValidators; i++ { validators[i] = ðpb.Validator{ ActivationEpoch: 0, PublicKey: make([]byte, 48), WithdrawalCredentials: make([]byte, 32), EffectiveBalance: params.BeaconConfig().MaxEffectiveBalance, } balances[i] = params.BeaconConfig().MaxEffectiveBalance } want := make([]*ethpb.Validators_ValidatorContainer, len(validators)) for i := 0; i < len(validators); i++ { want[i] = ðpb.Validators_ValidatorContainer{ Index: uint64(i), Validator: validators[i], } } st := testutil.NewBeaconState() if err := st.SetSlot(headSlot); err != nil { t.Fatal(err) } if err := st.SetValidators(validators); err != nil { t.Fatal(err) } if err := st.SetBalances(balances); err != nil { t.Fatal(err) } blockRoots := make([][]byte, params.BeaconConfig().SlotsPerHistoricalRoot) stateRoots := make([][]byte, params.BeaconConfig().SlotsPerHistoricalRoot) for i := 0; i < len(blockRoots); i++ { blockRoots[i] = make([]byte, 32) stateRoots[i] = make([]byte, 32) } randaoMixes := make([][]byte, params.BeaconConfig().EpochsPerHistoricalVector) for i := 0; i < len(randaoMixes); i++ { randaoMixes[i] = make([]byte, 32) } slashings := make([]uint64, params.BeaconConfig().EpochsPerSlashingsVector) if err := st.SetBlockRoots(blockRoots); err != nil { t.Fatal(err) } if err := st.SetStateRoots(stateRoots); err != nil { t.Fatal(err) } if err := st.SetRandaoMixes(randaoMixes); err != nil { t.Fatal(err) } if err := st.SetSlashings(slashings); err != nil { t.Fatal(err) } secondsPerEpoch := params.BeaconConfig().SecondsPerSlot * params.BeaconConfig().SlotsPerEpoch bs := &Server{ HeadFetcher: &mock.ChainService{ State: st, }, GenesisTimeFetcher: &mock.ChainService{ Genesis: time.Now().Add(time.Duration(-1*int64(secondsPerEpoch)) * time.Second), }, } req := ðpb.ListValidatorsRequest{ QueryFilter: ðpb.ListValidatorsRequest_Epoch{ Epoch: 1, }, } res, err := bs.ListValidators(context.Background(), req) if err != nil { t.Fatal(err) } if len(res.ValidatorList) != len(want) { t.Errorf("Incorrect number of validators, wanted %d received %d", len(want), len(res.ValidatorList)) } for i := 0; i < len(res.ValidatorList); i++ { if !reflect.DeepEqual(res.ValidatorList[i], want[i]) { t.Errorf("Wanted validator %d: %v, got %v", i, want[i], res.ValidatorList[i]) } } } func TestServer_GetValidator(t *testing.T) { count := 30 validators := make([]*ethpb.Validator, count) for i := 0; i < count; i++ { validators[i] = ðpb.Validator{ ActivationEpoch: uint64(i), PublicKey: pubKey(uint64(i)), WithdrawalCredentials: make([]byte, 32), } } st := testutil.NewBeaconState() if err := st.SetValidators(validators); err != nil { t.Fatal(err) } bs := &Server{ HeadFetcher: &mock.ChainService{ State: st, }, } tests := []struct { req *ethpb.GetValidatorRequest res *ethpb.Validator wantErr bool err string }{ { req: ðpb.GetValidatorRequest{ QueryFilter: ðpb.GetValidatorRequest_Index{ Index: 0, }, }, res: validators[0], wantErr: false, }, { req: ðpb.GetValidatorRequest{ QueryFilter: ðpb.GetValidatorRequest_Index{ Index: uint64(count - 1), }, }, res: validators[count-1], wantErr: false, }, { req: ðpb.GetValidatorRequest{ QueryFilter: ðpb.GetValidatorRequest_PublicKey{ PublicKey: pubKey(5), }, }, res: validators[5], wantErr: false, }, { req: ðpb.GetValidatorRequest{ QueryFilter: ðpb.GetValidatorRequest_PublicKey{ PublicKey: []byte("bad-keyxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxx"), }, }, res: nil, wantErr: true, err: "No validator matched filter criteria", }, { req: ðpb.GetValidatorRequest{ QueryFilter: ðpb.GetValidatorRequest_Index{ Index: uint64(len(validators)), }, }, res: nil, wantErr: true, err: fmt.Sprintf("there are only %d validators", len(validators)), }, } for _, test := range tests { res, err := bs.GetValidator(context.Background(), test.req) if test.wantErr && err != nil { if !strings.Contains(err.Error(), test.err) { t.Fatalf("Wanted %v, received %v", test.err, err) } } else if err != nil { t.Fatal(err) } if !reflect.DeepEqual(test.res, res) { t.Errorf("Wanted %v, got %v", test.res, res) } } } func TestServer_GetValidatorActiveSetChanges(t *testing.T) { db := dbTest.SetupDB(t) resetCfg := featureconfig.InitWithReset(&featureconfig.Flags{NewStateMgmt: true}) defer resetCfg() ctx := context.Background() validators := make([]*ethpb.Validator, 8) headState := testutil.NewBeaconState() if err := headState.SetSlot(0); err != nil { t.Fatal(err) } if err := headState.SetValidators(validators); err != nil { t.Fatal(err) } for i := 0; i < len(validators); i++ { activationEpoch := params.BeaconConfig().FarFutureEpoch withdrawableEpoch := params.BeaconConfig().FarFutureEpoch exitEpoch := params.BeaconConfig().FarFutureEpoch slashed := false balance := params.BeaconConfig().MaxEffectiveBalance // Mark indices divisible by two as activated. if i%2 == 0 { activationEpoch = 0 } else if i%3 == 0 { // Mark indices divisible by 3 as slashed. withdrawableEpoch = params.BeaconConfig().EpochsPerSlashingsVector slashed = true } else if i%5 == 0 { // Mark indices divisible by 5 as exited. exitEpoch = 0 withdrawableEpoch = params.BeaconConfig().MinValidatorWithdrawabilityDelay } else if i%7 == 0 { // Mark indices divisible by 7 as ejected. exitEpoch = 0 withdrawableEpoch = params.BeaconConfig().MinValidatorWithdrawabilityDelay balance = params.BeaconConfig().EjectionBalance } if err := headState.UpdateValidatorAtIndex(uint64(i), ðpb.Validator{ ActivationEpoch: activationEpoch, PublicKey: pubKey(uint64(i)), EffectiveBalance: balance, WithdrawalCredentials: make([]byte, 32), WithdrawableEpoch: withdrawableEpoch, Slashed: slashed, ExitEpoch: exitEpoch, }); err != nil { t.Fatal(err) } } b := ðpb.SignedBeaconBlock{Block: ðpb.BeaconBlock{}} if err := db.SaveBlock(ctx, b); err != nil { t.Fatal(err) } gRoot, err := stateutil.BlockRoot(b.Block) if err != nil { t.Fatal(err) } if err := db.SaveGenesisBlockRoot(ctx, gRoot); err != nil { t.Fatal(err) } if err := db.SaveState(ctx, headState, gRoot); err != nil { t.Fatal(err) } bs := &Server{ FinalizationFetcher: &mock.ChainService{ FinalizedCheckPoint: ðpb.Checkpoint{Epoch: 0}, }, GenesisTimeFetcher: &mock.ChainService{}, StateGen: stategen.New(db, cache.NewStateSummaryCache()), } res, err := bs.GetValidatorActiveSetChanges(ctx, ðpb.GetValidatorActiveSetChangesRequest{ QueryFilter: ðpb.GetValidatorActiveSetChangesRequest_Genesis{Genesis: true}, }) if err != nil { t.Fatal(err) } wantedActive := [][]byte{ pubKey(0), pubKey(2), pubKey(4), pubKey(6), } wantedActiveIndices := []uint64{0, 2, 4, 6} wantedExited := [][]byte{ pubKey(5), } wantedExitedIndices := []uint64{5} wantedSlashed := [][]byte{ pubKey(3), } wantedSlashedIndices := []uint64{3} wantedEjected := [][]byte{ pubKey(7), } wantedEjectedIndices := []uint64{7} wanted := ðpb.ActiveSetChanges{ Epoch: 0, ActivatedPublicKeys: wantedActive, ActivatedIndices: wantedActiveIndices, ExitedPublicKeys: wantedExited, ExitedIndices: wantedExitedIndices, SlashedPublicKeys: wantedSlashed, SlashedIndices: wantedSlashedIndices, EjectedPublicKeys: wantedEjected, EjectedIndices: wantedEjectedIndices, } if !proto.Equal(wanted, res) { t.Errorf("Wanted \n%v, received \n%v", wanted, res) } } func TestServer_GetValidatorActiveSetChanges_FromArchive(t *testing.T) { resetCfg := featureconfig.InitWithReset(&featureconfig.Flags{NewStateMgmt: false}) defer resetCfg() db := dbTest.SetupDB(t) ctx := context.Background() validators := make([]*ethpb.Validator, 8) headState := testutil.NewBeaconState() if err := headState.SetSlot(helpers.StartSlot(100)); err != nil { t.Fatal(err) } if err := headState.SetValidators(validators); err != nil { t.Fatal(err) } activatedIndices := make([]uint64, 0) exitedIndices := make([]uint64, 0) slashedIndices := make([]uint64, 0) ejectedIndices := make([]uint64, 0) for i := 0; i < len(validators); i++ { // Mark indices divisible by two as activated. if i%2 == 0 { activatedIndices = append(activatedIndices, uint64(i)) } else if i%3 == 0 { // Mark indices divisible by 3 as slashed. slashedIndices = append(slashedIndices, uint64(i)) } else if i%5 == 0 { // Mark indices divisible by 5 as exited. exitedIndices = append(exitedIndices, uint64(i)) } else if i%7 == 0 { // Mark indices divisible by 7 as ejected. ejectedIndices = append(ejectedIndices, uint64(i)) } key := make([]byte, 48) copy(key, strconv.Itoa(i)) if err := headState.UpdateValidatorAtIndex(uint64(i), ðpb.Validator{ PublicKey: key, }); err != nil { t.Fatal(err) } } archivedChanges := &pbp2p.ArchivedActiveSetChanges{ Activated: activatedIndices, Exited: exitedIndices, Slashed: slashedIndices, Ejected: ejectedIndices, } // We store the changes during the genesis epoch. if err := db.SaveArchivedActiveValidatorChanges(ctx, 0, archivedChanges); err != nil { t.Fatal(err) } // We store the same changes during epoch 5 for further testing. if err := db.SaveArchivedActiveValidatorChanges(ctx, 5, archivedChanges); err != nil { t.Fatal(err) } bs := &Server{ BeaconDB: db, HeadFetcher: &mock.ChainService{ State: headState, }, } res, err := bs.GetValidatorActiveSetChanges(ctx, ðpb.GetValidatorActiveSetChangesRequest{ QueryFilter: ðpb.GetValidatorActiveSetChangesRequest_Genesis{Genesis: true}, }) if err != nil { t.Fatal(err) } wantedKeys := make([][]byte, 8) for i := 0; i < len(wantedKeys); i++ { k := make([]byte, 48) copy(k, strconv.Itoa(i)) wantedKeys[i] = k } wantedActive := [][]byte{ wantedKeys[0], wantedKeys[2], wantedKeys[4], wantedKeys[6], } wantedActiveIndices := []uint64{0, 2, 4, 6} wantedExited := [][]byte{ wantedKeys[5], } wantedExitedIndices := []uint64{5} wantedSlashed := [][]byte{ wantedKeys[3], } wantedSlashedIndices := []uint64{3} wantedEjected := [][]byte{ wantedKeys[7], } wantedEjectedIndices := []uint64{7} wanted := ðpb.ActiveSetChanges{ Epoch: 0, ActivatedPublicKeys: wantedActive, ActivatedIndices: wantedActiveIndices, ExitedPublicKeys: wantedExited, ExitedIndices: wantedExitedIndices, SlashedPublicKeys: wantedSlashed, SlashedIndices: wantedSlashedIndices, EjectedPublicKeys: wantedEjected, EjectedIndices: wantedEjectedIndices, } if !proto.Equal(wanted, res) { t.Errorf("Wanted \n%v, received \n%v", wanted, res) } res, err = bs.GetValidatorActiveSetChanges(ctx, ðpb.GetValidatorActiveSetChangesRequest{ QueryFilter: ðpb.GetValidatorActiveSetChangesRequest_Epoch{Epoch: 5}, }) if err != nil { t.Fatal(err) } wanted.Epoch = 5 if !proto.Equal(wanted, res) { t.Errorf("Wanted \n%v, received \n%v", wanted, res) } } func TestServer_GetValidatorQueue_PendingActivation(t *testing.T) { headState, err := stateTrie.InitializeFromProto(&pbp2p.BeaconState{ Validators: []*ethpb.Validator{ { ActivationEpoch: helpers.ActivationExitEpoch(0), ActivationEligibilityEpoch: 3, PublicKey: pubKey(3), WithdrawalCredentials: make([]byte, 32), }, { ActivationEpoch: helpers.ActivationExitEpoch(0), ActivationEligibilityEpoch: 2, PublicKey: pubKey(2), WithdrawalCredentials: make([]byte, 32), }, { ActivationEpoch: helpers.ActivationExitEpoch(0), ActivationEligibilityEpoch: 1, PublicKey: pubKey(1), WithdrawalCredentials: make([]byte, 32), }, }, FinalizedCheckpoint: ðpb.Checkpoint{ Epoch: 0, }, }) if err != nil { t.Fatal(err) } bs := &Server{ HeadFetcher: &mock.ChainService{ State: headState, }, } res, err := bs.GetValidatorQueue(context.Background(), &ptypes.Empty{}) if err != nil { t.Fatal(err) } // We verify the keys are properly sorted by the validators' activation eligibility epoch. wanted := [][]byte{ pubKey(1), pubKey(2), pubKey(3), } activeValidatorCount, err := helpers.ActiveValidatorCount(headState, helpers.CurrentEpoch(headState)) if err != nil { t.Fatal(err) } wantChurn, err := helpers.ValidatorChurnLimit(activeValidatorCount) if err != nil { t.Fatal(err) } if res.ChurnLimit != wantChurn { t.Errorf("Wanted churn %d, received %d", wantChurn, res.ChurnLimit) } if !reflect.DeepEqual(res.ActivationPublicKeys, wanted) { t.Errorf("Wanted %v, received %v", wanted, res.ActivationPublicKeys) } wantedActiveIndices := []uint64{2, 1, 0} if !reflect.DeepEqual(res.ActivationValidatorIndices, wantedActiveIndices) { t.Errorf("wanted %v, received %v", wantedActiveIndices, res.ActivationValidatorIndices) } } func TestServer_GetValidatorQueue_ExitedValidatorLeavesQueue(t *testing.T) { validators := []*ethpb.Validator{ { ActivationEpoch: 0, ExitEpoch: params.BeaconConfig().FarFutureEpoch, WithdrawableEpoch: params.BeaconConfig().FarFutureEpoch, PublicKey: []byte("1"), }, { ActivationEpoch: 0, ExitEpoch: 4, WithdrawableEpoch: 6, PublicKey: []byte("2"), }, } headState := testutil.NewBeaconState() if err := headState.SetValidators(validators); err != nil { t.Fatal(err) } if err := headState.SetFinalizedCheckpoint(ðpb.Checkpoint{Epoch: 0}); err != nil { t.Fatal(err) } bs := &Server{ HeadFetcher: &mock.ChainService{ State: headState, }, } // First we check if validator with index 1 is in the exit queue. res, err := bs.GetValidatorQueue(context.Background(), &ptypes.Empty{}) if err != nil { t.Fatal(err) } wanted := [][]byte{ []byte("2"), } activeValidatorCount, err := helpers.ActiveValidatorCount(headState, helpers.CurrentEpoch(headState)) if err != nil { t.Fatal(err) } wantChurn, err := helpers.ValidatorChurnLimit(activeValidatorCount) if err != nil { t.Fatal(err) } if res.ChurnLimit != wantChurn { t.Errorf("Wanted churn %d, received %d", wantChurn, res.ChurnLimit) } if !reflect.DeepEqual(res.ExitPublicKeys, wanted) { t.Errorf("Wanted %v, received %v", wanted, res.ExitPublicKeys) } wantedExitIndices := []uint64{1} if !reflect.DeepEqual(res.ExitValidatorIndices, wantedExitIndices) { t.Errorf("wanted %v, received %v", wantedExitIndices, res.ExitValidatorIndices) } // Now, we move the state.slot past the exit epoch of the validator, and now // the validator should no longer exist in the queue. if err := headState.SetSlot(helpers.StartSlot(validators[1].ExitEpoch + 1)); err != nil { t.Fatal(err) } res, err = bs.GetValidatorQueue(context.Background(), &ptypes.Empty{}) if err != nil { t.Fatal(err) } if len(res.ExitPublicKeys) != 0 { t.Errorf("Wanted empty exit queue, received %v", res.ExitPublicKeys) } } func TestServer_GetValidatorQueue_PendingExit(t *testing.T) { headState, err := stateTrie.InitializeFromProto(&pbp2p.BeaconState{ Validators: []*ethpb.Validator{ { ActivationEpoch: 0, ExitEpoch: 4, WithdrawableEpoch: 3, PublicKey: pubKey(3), WithdrawalCredentials: make([]byte, 32), }, { ActivationEpoch: 0, ExitEpoch: 4, WithdrawableEpoch: 2, PublicKey: pubKey(2), WithdrawalCredentials: make([]byte, 32), }, { ActivationEpoch: 0, ExitEpoch: 4, WithdrawableEpoch: 1, PublicKey: pubKey(1), WithdrawalCredentials: make([]byte, 32), }, }, FinalizedCheckpoint: ðpb.Checkpoint{ Epoch: 0, }, }) if err != nil { t.Fatal(err) } bs := &Server{ HeadFetcher: &mock.ChainService{ State: headState, }, } res, err := bs.GetValidatorQueue(context.Background(), &ptypes.Empty{}) if err != nil { t.Fatal(err) } // We verify the keys are properly sorted by the validators' withdrawable epoch. wanted := [][]byte{ pubKey(1), pubKey(2), pubKey(3), } activeValidatorCount, err := helpers.ActiveValidatorCount(headState, helpers.CurrentEpoch(headState)) if err != nil { t.Fatal(err) } wantChurn, err := helpers.ValidatorChurnLimit(activeValidatorCount) if err != nil { t.Fatal(err) } if res.ChurnLimit != wantChurn { t.Errorf("Wanted churn %d, received %d", wantChurn, res.ChurnLimit) } if !reflect.DeepEqual(res.ExitPublicKeys, wanted) { t.Errorf("Wanted %v, received %v", wanted, res.ExitPublicKeys) } } func TestServer_GetValidatorParticipation_CannotRequestCurrentEpoch(t *testing.T) { db := dbTest.SetupDB(t) ctx := context.Background() headState := testutil.NewBeaconState() if err := headState.SetSlot(helpers.StartSlot(2)); err != nil { t.Fatal(err) } bs := &Server{ BeaconDB: db, HeadFetcher: &mock.ChainService{ State: headState, }, } wanted := "Cannot retrieve information about an epoch currently in progress" if _, err := bs.GetValidatorParticipation( ctx, ðpb.GetValidatorParticipationRequest{ QueryFilter: ðpb.GetValidatorParticipationRequest_Epoch{ Epoch: 2, }, }, ); err != nil && !strings.Contains(err.Error(), wanted) { t.Errorf("Expected error %v, received %v", wanted, err) } } func TestServer_GetValidatorParticipation_CannotRequestFutureEpoch(t *testing.T) { db := dbTest.SetupDB(t) resetCfg := featureconfig.InitWithReset(&featureconfig.Flags{NewStateMgmt: false}) defer resetCfg() ctx := context.Background() headState := testutil.NewBeaconState() if err := headState.SetSlot(0); err != nil { t.Fatal(err) } bs := &Server{ BeaconDB: db, HeadFetcher: &mock.ChainService{ State: headState, }, GenesisTimeFetcher: &mock.ChainService{}, } wanted := "Cannot retrieve information about an epoch in the future" if _, err := bs.GetValidatorParticipation( ctx, ðpb.GetValidatorParticipationRequest{ QueryFilter: ðpb.GetValidatorParticipationRequest_Epoch{ Epoch: helpers.SlotToEpoch(bs.GenesisTimeFetcher.CurrentSlot()) + 1, }, }, ); err != nil && !strings.Contains(err.Error(), wanted) { t.Errorf("Expected error %v, received %v", wanted, err) } } func TestServer_GetValidatorParticipation_FromArchive(t *testing.T) { db := dbTest.SetupDB(t) ctx := context.Background() resetCfg := featureconfig.InitWithReset(&featureconfig.Flags{NewStateMgmt: false}) defer resetCfg() epoch := uint64(4) part := ðpb.ValidatorParticipation{ GlobalParticipationRate: 1.0, VotedEther: 20, EligibleEther: 20, } if err := db.SaveArchivedValidatorParticipation(ctx, epoch-2, part); err != nil { t.Fatal(err) } headState := testutil.NewBeaconState() if err := headState.SetSlot(helpers.StartSlot(epoch + 1)); err != nil { t.Fatal(err) } if err := headState.SetFinalizedCheckpoint(ðpb.Checkpoint{Epoch: epoch + 1}); err != nil { t.Fatal(err) } bs := &Server{ BeaconDB: db, HeadFetcher: &mock.ChainService{ State: headState, }, } if _, err := bs.GetValidatorParticipation(ctx, ðpb.GetValidatorParticipationRequest{ QueryFilter: ðpb.GetValidatorParticipationRequest_Epoch{ Epoch: epoch + 2, }, }); err == nil { t.Error("Expected error when requesting future epoch, received nil") } // We request data from epoch 0, which we didn't archive, so we should expect an error. if _, err := bs.GetValidatorParticipation(ctx, ðpb.GetValidatorParticipationRequest{ QueryFilter: ðpb.GetValidatorParticipationRequest_Genesis{ Genesis: true, }, }); err == nil { t.Error("Expected error when data from archive is not found, received nil") } want := ðpb.ValidatorParticipationResponse{ Epoch: epoch - 2, Finalized: true, Participation: part, } res, err := bs.GetValidatorParticipation(ctx, ðpb.GetValidatorParticipationRequest{ QueryFilter: ðpb.GetValidatorParticipationRequest_Epoch{ Epoch: epoch - 2, }, }) if err != nil { t.Fatal(err) } if !proto.Equal(want, res) { t.Errorf("Wanted %v, received %v", want, res) } } func TestServer_GetValidatorParticipation_PrevEpoch(t *testing.T) { db := dbTest.SetupDB(t) resetCfg := featureconfig.InitWithReset(&featureconfig.Flags{NewStateMgmt: true}) defer resetCfg() ctx := context.Background() 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{Target: ðpb.Checkpoint{}}}} headState := testutil.NewBeaconState() if err := headState.SetSlot(params.BeaconConfig().SlotsPerEpoch); err != nil { t.Fatal(err) } if err := headState.SetValidators(validators); err != nil { t.Fatal(err) } if err := headState.SetBalances(balances); err != nil { t.Fatal(err) } if err := headState.SetPreviousEpochAttestations(atts); err != nil { t.Fatal(err) } b := ðpb.SignedBeaconBlock{Block: ðpb.BeaconBlock{Slot: params.BeaconConfig().SlotsPerEpoch}} if err := db.SaveBlock(ctx, b); err != nil { t.Fatal(err) } bRoot, err := stateutil.BlockRoot(b.Block) if err != nil { t.Fatal(err) } if err := db.SaveState(ctx, headState, bRoot); err != nil { t.Fatal(err) } m := &mock.ChainService{State: headState} bs := &Server{ BeaconDB: db, HeadFetcher: m, ParticipationFetcher: m, GenesisTimeFetcher: &mock.ChainService{}, StateGen: stategen.New(db, cache.NewStateSummaryCache()), } res, err := bs.GetValidatorParticipation(ctx, ðpb.GetValidatorParticipationRequest{QueryFilter: ðpb.GetValidatorParticipationRequest_Epoch{Epoch: 0}}) if err != nil { t.Fatal(err) } wanted := ðpb.ValidatorParticipation{EligibleEther: validatorCount * params.BeaconConfig().MaxEffectiveBalance, VotedEther: params.BeaconConfig().EffectiveBalanceIncrement, GlobalParticipationRate: float32(params.BeaconConfig().EffectiveBalanceIncrement) / float32(validatorCount*params.BeaconConfig().MaxEffectiveBalance)} if !reflect.DeepEqual(res.Participation, wanted) { t.Error("Incorrect validator participation respond") } } func TestServer_GetValidatorParticipation_DoesntExist(t *testing.T) { db := dbTest.SetupDB(t) ctx := context.Background() headState := testutil.NewBeaconState() if err := headState.SetSlot(params.BeaconConfig().SlotsPerEpoch); err != nil { t.Fatal(err) } b := ðpb.SignedBeaconBlock{Block: ðpb.BeaconBlock{Slot: params.BeaconConfig().SlotsPerEpoch}} if err := db.SaveBlock(ctx, b); err != nil { t.Fatal(err) } bRoot, err := stateutil.BlockRoot(b.Block) if err != nil { t.Fatal(err) } if err := db.SaveState(ctx, headState, bRoot); err != nil { t.Fatal(err) } m := &mock.ChainService{State: headState} bs := &Server{ BeaconDB: db, HeadFetcher: m, ParticipationFetcher: m, GenesisTimeFetcher: &mock.ChainService{}, StateGen: stategen.New(db, cache.NewStateSummaryCache()), } wanted := "Participation information for epoch 0 is not yet available" if _, err := bs.GetValidatorParticipation(ctx, ðpb.GetValidatorParticipationRequest{QueryFilter: ðpb.GetValidatorParticipationRequest_Epoch{Epoch: 0}}); err != nil && !strings.Contains(err.Error(), wanted) { t.Errorf("Expected error %v, received %v", wanted, err) } } func TestServer_GetValidatorParticipation_FromArchive_FinalizedEpoch(t *testing.T) { db := dbTest.SetupDB(t) ctx := context.Background() resetCfg := featureconfig.InitWithReset(&featureconfig.Flags{NewStateMgmt: false}) defer resetCfg() part := ðpb.ValidatorParticipation{ GlobalParticipationRate: 1.0, VotedEther: 20, EligibleEther: 20, } epoch := uint64(1) // We archive data for epoch 1. if err := db.SaveArchivedValidatorParticipation(ctx, epoch, part); err != nil { t.Fatal(err) } headState := testutil.NewBeaconState() if err := headState.SetSlot(helpers.StartSlot(epoch + 10)); err != nil { t.Fatal(err) } if err := headState.SetFinalizedCheckpoint(ðpb.Checkpoint{Epoch: epoch + 5}); err != nil { t.Fatal(err) } bs := &Server{ BeaconDB: db, HeadFetcher: &mock.ChainService{ // 10 epochs into the future. State: headState, }, } want := ðpb.ValidatorParticipationResponse{ Epoch: epoch, Finalized: true, Participation: part, } // We request epoch 1. res, err := bs.GetValidatorParticipation(ctx, ðpb.GetValidatorParticipationRequest{ QueryFilter: ðpb.GetValidatorParticipationRequest_Epoch{ Epoch: epoch, }, }) if err != nil { t.Fatal(err) } if !proto.Equal(want, res) { t.Errorf("Wanted %v, received %v", want, res) } } func TestGetValidatorPerformance_Syncing(t *testing.T) { ctx := context.Background() bs := &Server{ SyncChecker: &mockSync.Sync{IsSyncing: true}, } wanted := "Syncing to latest head, not ready to respond" if _, err := bs.GetValidatorPerformance(ctx, nil); err == nil || !strings.Contains(err.Error(), wanted) { t.Error("Did not receive wanted error") } } func TestGetValidatorPerformance_OK(t *testing.T) { ctx := context.Background() epoch := uint64(1) headState := testutil.NewBeaconState() if err := headState.SetSlot(helpers.StartSlot(epoch + 1)); err != nil { t.Fatal(err) } atts := make([]*pb.PendingAttestation, 3) for i := 0; i < len(atts); i++ { atts[i] = &pb.PendingAttestation{ Data: ðpb.AttestationData{ Target: ðpb.Checkpoint{}, Source: ðpb.Checkpoint{}, }, AggregationBits: bitfield.Bitlist{0xC0, 0xC0, 0xC0, 0xC0, 0x01}, InclusionDelay: 1, } } if err := headState.SetPreviousEpochAttestations(atts); err != nil { t.Fatal(err) } defaultBal := params.BeaconConfig().MaxEffectiveBalance extraBal := params.BeaconConfig().MaxEffectiveBalance + params.BeaconConfig().GweiPerEth balances := []uint64{defaultBal, extraBal, extraBal + params.BeaconConfig().GweiPerEth} if err := headState.SetBalances(balances); err != nil { t.Fatal(err) } publicKey1 := bytesutil.ToBytes48([]byte{1}) publicKey2 := bytesutil.ToBytes48([]byte{2}) publicKey3 := bytesutil.ToBytes48([]byte{3}) validators := []*ethpb.Validator{ { PublicKey: publicKey1[:], ActivationEpoch: 5, ExitEpoch: params.BeaconConfig().FarFutureEpoch, }, { PublicKey: publicKey2[:], EffectiveBalance: defaultBal, ActivationEpoch: 0, ExitEpoch: params.BeaconConfig().FarFutureEpoch, }, { PublicKey: publicKey3[:], EffectiveBalance: defaultBal, ActivationEpoch: 0, ExitEpoch: params.BeaconConfig().FarFutureEpoch, }, } if err := headState.SetValidators(validators); err != nil { t.Fatal(err) } if err := headState.SetBalances([]uint64{100, 101, 102}); err != nil { t.Fatal(err) } bs := &Server{ HeadFetcher: &mock.ChainService{ State: headState, }, GenesisTimeFetcher: &mock.ChainService{Genesis: time.Now().Add(time.Duration(-1*int64(headState.Slot()*params.BeaconConfig().SecondsPerSlot)) * time.Second)}, SyncChecker: &mockSync.Sync{IsSyncing: false}, } farFuture := params.BeaconConfig().FarFutureEpoch want := ðpb.ValidatorPerformanceResponse{ PublicKeys: [][]byte{publicKey2[:], publicKey3[:]}, CurrentEffectiveBalances: []uint64{params.BeaconConfig().MaxEffectiveBalance, params.BeaconConfig().MaxEffectiveBalance}, InclusionSlots: []uint64{farFuture, farFuture}, InclusionDistances: []uint64{farFuture, farFuture}, CorrectlyVotedSource: []bool{false, false}, CorrectlyVotedTarget: []bool{false, false}, CorrectlyVotedHead: []bool{false, false}, BalancesBeforeEpochTransition: []uint64{101, 102}, BalancesAfterEpochTransition: []uint64{0, 0}, MissingValidators: [][]byte{publicKey1[:]}, } res, err := bs.GetValidatorPerformance(ctx, ðpb.ValidatorPerformanceRequest{ PublicKeys: [][]byte{publicKey1[:], publicKey3[:], publicKey2[:]}, }) if err != nil { t.Fatal(err) } if !proto.Equal(want, res) { t.Errorf("Wanted %v\nReceived %v", want, res) } } func TestGetValidatorPerformance_Indices(t *testing.T) { ctx := context.Background() epoch := uint64(1) defaultBal := params.BeaconConfig().MaxEffectiveBalance extraBal := params.BeaconConfig().MaxEffectiveBalance + params.BeaconConfig().GweiPerEth headState := testutil.NewBeaconState() if err := headState.SetSlot(helpers.StartSlot(epoch + 1)); err != nil { t.Fatal(err) } balances := []uint64{defaultBal, extraBal, extraBal + params.BeaconConfig().GweiPerEth} if err := headState.SetBalances(balances); err != nil { t.Fatal(err) } publicKey1 := bytesutil.ToBytes48([]byte{1}) publicKey2 := bytesutil.ToBytes48([]byte{2}) publicKey3 := bytesutil.ToBytes48([]byte{3}) validators := []*ethpb.Validator{ { PublicKey: publicKey1[:], ActivationEpoch: 5, ExitEpoch: params.BeaconConfig().FarFutureEpoch, }, { PublicKey: publicKey2[:], EffectiveBalance: defaultBal, ActivationEpoch: 0, ExitEpoch: params.BeaconConfig().FarFutureEpoch, }, { PublicKey: publicKey3[:], EffectiveBalance: defaultBal, ActivationEpoch: 0, ExitEpoch: params.BeaconConfig().FarFutureEpoch, }, } if err := headState.SetValidators(validators); err != nil { t.Fatal(err) } bs := &Server{ HeadFetcher: &mock.ChainService{ // 10 epochs into the future. State: headState, }, SyncChecker: &mockSync.Sync{IsSyncing: false}, GenesisTimeFetcher: &mock.ChainService{Genesis: time.Now().Add(time.Duration(-1*int64(headState.Slot()*params.BeaconConfig().SecondsPerSlot)) * time.Second)}, } c := headState.Copy() vp, bp, err := precompute.New(ctx, c) if err != nil { t.Fatal(err) } vp, bp, err = precompute.ProcessAttestations(ctx, c, vp, bp) if err != nil { t.Fatal(err) } c, err = precompute.ProcessRewardsAndPenaltiesPrecompute(c, bp, vp) if err != nil { t.Fatal(err) } farFuture := params.BeaconConfig().FarFutureEpoch want := ðpb.ValidatorPerformanceResponse{ PublicKeys: [][]byte{publicKey2[:], publicKey3[:]}, CurrentEffectiveBalances: []uint64{params.BeaconConfig().MaxEffectiveBalance, params.BeaconConfig().MaxEffectiveBalance}, InclusionSlots: []uint64{farFuture, farFuture}, InclusionDistances: []uint64{farFuture, farFuture}, CorrectlyVotedSource: []bool{false, false}, CorrectlyVotedTarget: []bool{false, false}, CorrectlyVotedHead: []bool{false, false}, BalancesBeforeEpochTransition: []uint64{extraBal, extraBal + params.BeaconConfig().GweiPerEth}, BalancesAfterEpochTransition: []uint64{vp[1].AfterEpochTransitionBalance, vp[2].AfterEpochTransitionBalance}, MissingValidators: [][]byte{publicKey1[:]}, } res, err := bs.GetValidatorPerformance(ctx, ðpb.ValidatorPerformanceRequest{ Indices: []uint64{2, 1, 0}, }) if err != nil { t.Fatal(err) } if !proto.Equal(want, res) { t.Errorf("Wanted %v\nReceived %v", want, res) } } func TestGetValidatorPerformance_IndicesPubkeys(t *testing.T) { ctx := context.Background() epoch := uint64(1) defaultBal := params.BeaconConfig().MaxEffectiveBalance extraBal := params.BeaconConfig().MaxEffectiveBalance + params.BeaconConfig().GweiPerEth headState := testutil.NewBeaconState() if err := headState.SetSlot(helpers.StartSlot(epoch + 1)); err != nil { t.Fatal(err) } balances := []uint64{defaultBal, extraBal, extraBal + params.BeaconConfig().GweiPerEth} if err := headState.SetBalances(balances); err != nil { t.Fatal(err) } publicKey1 := bytesutil.ToBytes48([]byte{1}) publicKey2 := bytesutil.ToBytes48([]byte{2}) publicKey3 := bytesutil.ToBytes48([]byte{3}) validators := []*ethpb.Validator{ { PublicKey: publicKey1[:], ActivationEpoch: 5, ExitEpoch: params.BeaconConfig().FarFutureEpoch, }, { PublicKey: publicKey2[:], EffectiveBalance: defaultBal, ActivationEpoch: 0, ExitEpoch: params.BeaconConfig().FarFutureEpoch, }, { PublicKey: publicKey3[:], EffectiveBalance: defaultBal, ActivationEpoch: 0, ExitEpoch: params.BeaconConfig().FarFutureEpoch, }, } if err := headState.SetValidators(validators); err != nil { t.Fatal(err) } bs := &Server{ HeadFetcher: &mock.ChainService{ // 10 epochs into the future. State: headState, }, SyncChecker: &mockSync.Sync{IsSyncing: false}, GenesisTimeFetcher: &mock.ChainService{Genesis: time.Now().Add(time.Duration(-1*int64(headState.Slot()*params.BeaconConfig().SecondsPerSlot)) * time.Second)}, } c := headState.Copy() vp, bp, err := precompute.New(ctx, c) if err != nil { t.Fatal(err) } vp, bp, err = precompute.ProcessAttestations(ctx, c, vp, bp) if err != nil { t.Fatal(err) } c, err = precompute.ProcessRewardsAndPenaltiesPrecompute(c, bp, vp) if err != nil { t.Fatal(err) } farFuture := params.BeaconConfig().FarFutureEpoch want := ðpb.ValidatorPerformanceResponse{ PublicKeys: [][]byte{publicKey2[:], publicKey3[:]}, CurrentEffectiveBalances: []uint64{params.BeaconConfig().MaxEffectiveBalance, params.BeaconConfig().MaxEffectiveBalance}, InclusionSlots: []uint64{farFuture, farFuture}, InclusionDistances: []uint64{farFuture, farFuture}, CorrectlyVotedSource: []bool{false, false}, CorrectlyVotedTarget: []bool{false, false}, CorrectlyVotedHead: []bool{false, false}, BalancesBeforeEpochTransition: []uint64{extraBal, extraBal + params.BeaconConfig().GweiPerEth}, BalancesAfterEpochTransition: []uint64{vp[1].AfterEpochTransitionBalance, vp[2].AfterEpochTransitionBalance}, MissingValidators: [][]byte{publicKey1[:]}, } // Index 2 and publicKey3 points to the same validator. // Should not return duplicates. res, err := bs.GetValidatorPerformance(ctx, ðpb.ValidatorPerformanceRequest{ PublicKeys: [][]byte{publicKey1[:], publicKey3[:]}, Indices: []uint64{1, 2}, }) if err != nil { t.Fatal(err) } if !proto.Equal(want, res) { t.Errorf("Wanted %v\nReceived %v", want, res) } } func BenchmarkListValidatorBalances(b *testing.B) { b.StopTimer() db := dbTest.SetupDB(b) ctx := context.Background() count := 1000 setupValidators(b, db, count) headState, err := db.HeadState(ctx) if err != nil { b.Fatal(err) } bs := &Server{ HeadFetcher: &mock.ChainService{ State: headState, }, } req := ðpb.ListValidatorBalancesRequest{PageSize: 100} b.StartTimer() for i := 0; i < b.N; i++ { if _, err := bs.ListValidatorBalances(ctx, req); err != nil { b.Fatal(err) } } } func BenchmarkListValidatorBalances_FromArchive(b *testing.B) { b.StopTimer() db := dbTest.SetupDB(b) ctx := context.Background() currentNumValidators := 1000 numOldBalances := 50 validators := make([]*ethpb.Validator, currentNumValidators) oldBalances := make([]uint64, numOldBalances) for i := 0; i < currentNumValidators; i++ { validators[i] = ðpb.Validator{ PublicKey: []byte(strconv.Itoa(i)), } } for i := 0; i < numOldBalances; i++ { oldBalances[i] = params.BeaconConfig().MaxEffectiveBalance } // We archive old balances for epoch 50. if err := db.SaveArchivedBalances(ctx, 50, oldBalances); err != nil { b.Fatal(err) } headState := testutil.NewBeaconState() if err := headState.SetSlot(helpers.StartSlot(100 /* epoch 100 */)); err != nil { b.Fatal(err) } if err := headState.SetValidators(validators); err != nil { b.Fatal(err) } bs := &Server{ BeaconDB: db, HeadFetcher: &mock.ChainService{ State: headState, }, } b.StartTimer() for i := 0; i < b.N; i++ { if _, err := bs.ListValidatorBalances( ctx, ðpb.ListValidatorBalancesRequest{ QueryFilter: ðpb.ListValidatorBalancesRequest_Epoch{ Epoch: 50, }, PageSize: 100, }, ); err != nil { b.Fatal(err) } } } func setupValidators(t testing.TB, 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++ { pubKey := pubKey(uint64(i)) balances[i] = uint64(i) validators = append(validators, ðpb.Validator{ PublicKey: pubKey, WithdrawalCredentials: make([]byte, 32), }) } blk := ðpb.BeaconBlock{ Slot: 0, } blockRoot, err := ssz.HashTreeRoot(blk) if err != nil { t.Fatal(err) } s := testutil.NewBeaconState() if err := s.SetValidators(validators); err != nil { t.Fatal(err) } if err := s.SetBalances(balances); err != nil { t.Fatal(err) } if err := db.SaveState( context.Background(), s, blockRoot, ); err != nil { t.Fatal(err) } if err := db.SaveHeadBlockRoot(ctx, blockRoot); err != nil { t.Fatal(err) } return validators, balances } func TestServer_GetIndividualVotes_RequestFutureSlot(t *testing.T) { resetCfg := featureconfig.InitWithReset(&featureconfig.Flags{NewStateMgmt: true}) defer resetCfg() ds := &Server{GenesisTimeFetcher: &mock.ChainService{}} req := ðpb.IndividualVotesRequest{ Epoch: helpers.SlotToEpoch(ds.GenesisTimeFetcher.CurrentSlot()) + 1, } wanted := "Cannot retrieve information about an epoch in the future" if _, err := ds.GetIndividualVotes(context.Background(), req); err != nil && !strings.Contains(err.Error(), wanted) { t.Errorf("Expected error %v, received %v", wanted, err) } } func TestServer_GetIndividualVotes_ValidatorsDontExist(t *testing.T) { resetCfg := featureconfig.InitWithReset(&featureconfig.Flags{NewStateMgmt: true}) defer resetCfg() params.UseMinimalConfig() defer params.UseMainnetConfig() db := dbTest.SetupDB(t) ctx := context.Background() validators := uint64(64) stateWithValidators, _ := testutil.DeterministicGenesisState(t, validators) beaconState := testutil.NewBeaconState() if err := beaconState.SetValidators(stateWithValidators.Validators()); err != nil { t.Fatal(err) } if err := beaconState.SetSlot(params.BeaconConfig().SlotsPerEpoch); err != nil { t.Fatal(err) } b := testutil.NewBeaconBlock() b.Block.Slot = params.BeaconConfig().SlotsPerEpoch if err := db.SaveBlock(ctx, b); err != nil { t.Fatal(err) } gRoot, err := stateutil.BlockRoot(b.Block) if err != nil { t.Fatal(err) } gen := stategen.New(db, cache.NewStateSummaryCache()) if err := gen.SaveState(ctx, gRoot, beaconState); err != nil { t.Fatal(err) } if err := db.SaveState(ctx, beaconState, gRoot); err != nil { t.Fatal(err) } if err := db.SaveGenesisBlockRoot(ctx, gRoot); err != nil { t.Fatal(err) } bs := &Server{ StateGen: gen, GenesisTimeFetcher: &mock.ChainService{}, } // Test non exist public key. res, err := bs.GetIndividualVotes(ctx, ðpb.IndividualVotesRequest{ PublicKeys: [][]byte{{'a'}}, Epoch: 0, }) if err != nil { t.Fatal(err) } wanted := ðpb.IndividualVotesRespond{ IndividualVotes: []*ethpb.IndividualVotesRespond_IndividualVote{ {PublicKey: []byte{'a'}, ValidatorIndex: ^uint64(0)}, }, } if !reflect.DeepEqual(res, wanted) { t.Error("Did not get wanted respond") } // Test non-existent validator index. res, err = bs.GetIndividualVotes(ctx, ðpb.IndividualVotesRequest{ Indices: []uint64{100}, Epoch: 0, }) if err != nil { t.Fatal(err) } wanted = ðpb.IndividualVotesRespond{ IndividualVotes: []*ethpb.IndividualVotesRespond_IndividualVote{ {ValidatorIndex: 100}, }, } if !reflect.DeepEqual(res, wanted) { t.Log(res, wanted) t.Error("Did not get wanted respond") } // Test both. res, err = bs.GetIndividualVotes(ctx, ðpb.IndividualVotesRequest{ PublicKeys: [][]byte{{'a'}, {'b'}}, Indices: []uint64{100, 101}, Epoch: 0, }) if err != nil { t.Fatal(err) } wanted = ðpb.IndividualVotesRespond{ IndividualVotes: []*ethpb.IndividualVotesRespond_IndividualVote{ {PublicKey: []byte{'a'}, ValidatorIndex: ^uint64(0)}, {PublicKey: []byte{'b'}, ValidatorIndex: ^uint64(0)}, {ValidatorIndex: 100}, {ValidatorIndex: 101}, }, } if !reflect.DeepEqual(res, wanted) { t.Log(res, wanted) t.Error("Did not get wanted respond") } } func TestServer_GetIndividualVotes_Working(t *testing.T) { resetCfg := featureconfig.InitWithReset(&featureconfig.Flags{NewStateMgmt: true}) defer resetCfg() params.UseMinimalConfig() defer params.UseMainnetConfig() db := dbTest.SetupDB(t) ctx := context.Background() validators := uint64(64) stateWithValidators, _ := testutil.DeterministicGenesisState(t, validators) beaconState := testutil.NewBeaconState() if err := beaconState.SetValidators(stateWithValidators.Validators()); err != nil { t.Fatal(err) } if err := beaconState.SetSlot(params.BeaconConfig().SlotsPerEpoch); err != nil { t.Fatal(err) } bf := []byte{0xff} att1 := ðpb.Attestation{Data: ðpb.AttestationData{ Target: ðpb.Checkpoint{Epoch: 0}}, AggregationBits: bf} att2 := ðpb.Attestation{Data: ðpb.AttestationData{ Target: ðpb.Checkpoint{Epoch: 0}}, AggregationBits: bf} rt := [32]byte{'A'} att1.Data.Target.Root = rt[:] att1.Data.BeaconBlockRoot = rt[:] br := beaconState.BlockRoots() newRt := [32]byte{'B'} br[0] = newRt[:] if err := beaconState.SetBlockRoots(br); err != nil { t.Fatal(err) } att2.Data.Target.Root = rt[:] att2.Data.BeaconBlockRoot = newRt[:] err := beaconState.SetPreviousEpochAttestations([]*pb.PendingAttestation{{Data: att1.Data, AggregationBits: bf}}) if err != nil { t.Fatal(err) } err = beaconState.SetCurrentEpochAttestations([]*pb.PendingAttestation{{Data: att2.Data, AggregationBits: bf}}) if err != nil { t.Fatal(err) } b := testutil.NewBeaconBlock() b.Block.Slot = params.BeaconConfig().SlotsPerEpoch if err := db.SaveBlock(ctx, b); err != nil { t.Fatal(err) } gRoot, err := stateutil.BlockRoot(b.Block) if err != nil { t.Fatal(err) } gen := stategen.New(db, cache.NewStateSummaryCache()) if err := gen.SaveState(ctx, gRoot, beaconState); err != nil { t.Fatal(err) } if err := db.SaveState(ctx, beaconState, gRoot); err != nil { t.Fatal(err) } if err := db.SaveGenesisBlockRoot(ctx, gRoot); err != nil { t.Fatal(err) } bs := &Server{ StateGen: gen, GenesisTimeFetcher: &mock.ChainService{}, } res, err := bs.GetIndividualVotes(ctx, ðpb.IndividualVotesRequest{ Indices: []uint64{0, 1}, Epoch: 0, }) if err != nil { t.Fatal(err) } wanted := ðpb.IndividualVotesRespond{ IndividualVotes: []*ethpb.IndividualVotesRespond_IndividualVote{ {ValidatorIndex: 0, PublicKey: beaconState.Validators()[0].PublicKey, IsActiveInCurrentEpoch: true, IsActiveInPreviousEpoch: true, CurrentEpochEffectiveBalanceGwei: params.BeaconConfig().MaxEffectiveBalance, InclusionSlot: params.BeaconConfig().FarFutureEpoch, InclusionDistance: params.BeaconConfig().FarFutureEpoch}, {ValidatorIndex: 1, PublicKey: beaconState.Validators()[1].PublicKey, IsActiveInCurrentEpoch: true, IsActiveInPreviousEpoch: true, CurrentEpochEffectiveBalanceGwei: params.BeaconConfig().MaxEffectiveBalance, InclusionSlot: params.BeaconConfig().FarFutureEpoch, InclusionDistance: params.BeaconConfig().FarFutureEpoch}, }, } if !reflect.DeepEqual(res, wanted) { t.Log(res, wanted) t.Error("Did not get wanted respond") } }