package beacon import ( "context" "sort" "strconv" ptypes "github.com/gogo/protobuf/types" ethpb "github.com/prysmaticlabs/ethereumapis/eth/v1alpha1" "github.com/prysmaticlabs/prysm/beacon-chain/core/epoch/precompute" "github.com/prysmaticlabs/prysm/beacon-chain/core/helpers" "github.com/prysmaticlabs/prysm/beacon-chain/core/state" "github.com/prysmaticlabs/prysm/beacon-chain/core/validators" "github.com/prysmaticlabs/prysm/beacon-chain/flags" "github.com/prysmaticlabs/prysm/shared/bytesutil" "github.com/prysmaticlabs/prysm/shared/pagination" "github.com/prysmaticlabs/prysm/shared/params" "google.golang.org/grpc/codes" "google.golang.org/grpc/status" ) // ListValidatorBalances retrieves the validator balances for a given set of public keys. // An optional Epoch parameter is provided to request historical validator balances from // archived, persistent data. func (bs *Server) ListValidatorBalances( ctx context.Context, req *ethpb.ListValidatorBalancesRequest) (*ethpb.ValidatorBalances, error) { if int(req.PageSize) > flags.Get().MaxPageSize { return nil, status.Errorf(codes.InvalidArgument, "Requested page size %d can not be greater than max size %d", req.PageSize, flags.Get().MaxPageSize) } res := make([]*ethpb.ValidatorBalances_Balance, 0) filtered := map[uint64]bool{} // Track filtered validators to prevent duplication in the response. headState, err := bs.HeadFetcher.HeadState(ctx) if err != nil { return nil, status.Error(codes.Internal, "Could not get head state") } var requestingGenesis bool var epoch uint64 switch q := req.QueryFilter.(type) { case *ethpb.ListValidatorBalancesRequest_Epoch: epoch = q.Epoch case *ethpb.ListValidatorBalancesRequest_Genesis: requestingGenesis = q.Genesis default: epoch = helpers.CurrentEpoch(headState) } var balances []uint64 validators := headState.Validators() if requestingGenesis || epoch < helpers.CurrentEpoch(headState) { balances, err = bs.BeaconDB.ArchivedBalances(ctx, epoch) if err != nil { return nil, status.Errorf(codes.Internal, "Could not retrieve balances for epoch %d", epoch) } if balances == nil { return nil, status.Errorf( codes.NotFound, "Could not retrieve data for epoch %d, perhaps --archive in the running beacon node is disabled", 0, ) } } else if epoch == helpers.CurrentEpoch(headState) { balances = headState.Balances() } else { // Otherwise, we are requesting data from the future and we return an error. return nil, status.Errorf( codes.InvalidArgument, "Cannot retrieve information about an epoch in the future, current epoch %d, requesting %d", helpers.CurrentEpoch(headState), epoch, ) } balancesCount := len(balances) for _, pubKey := range req.PublicKeys { // Skip empty public key. if len(pubKey) == 0 { continue } index, ok, err := bs.BeaconDB.ValidatorIndex(ctx, pubKey) if err != nil { return nil, status.Errorf(codes.Internal, "Could not retrieve validator index: %v", err) } if !ok { return nil, status.Errorf(codes.NotFound, "Could not find validator index for public key %#x", pubKey) } filtered[index] = true if int(index) >= len(balances) { return nil, status.Errorf(codes.OutOfRange, "Validator index %d >= balance list %d", index, len(balances)) } res = append(res, ðpb.ValidatorBalances_Balance{ PublicKey: pubKey, Index: index, Balance: balances[index], }) balancesCount = len(res) } for _, index := range req.Indices { if int(index) >= len(balances) { if epoch <= helpers.CurrentEpoch(headState) { return nil, status.Errorf(codes.OutOfRange, "Validator index %d does not exist in historical balances", index) } return nil, status.Errorf(codes.OutOfRange, "Validator index %d >= balance list %d", index, len(balances)) } if !filtered[index] { res = append(res, ðpb.ValidatorBalances_Balance{ PublicKey: validators[index].PublicKey, Index: index, Balance: balances[index], }) } balancesCount = len(res) } // If there are no balances, we simply return a response specifying this. // Otherwise, attempting to paginate 0 balances below would result in an error. if balancesCount == 0 { return ðpb.ValidatorBalances{ Epoch: epoch, Balances: make([]*ethpb.ValidatorBalances_Balance, 0), TotalSize: int32(0), NextPageToken: strconv.Itoa(0), }, nil } start, end, nextPageToken, err := pagination.StartAndEndPage(req.PageToken, int(req.PageSize), balancesCount) if err != nil { return nil, status.Errorf( codes.Internal, "Could not paginate results: %v", err, ) } if len(req.Indices) == 0 && len(req.PublicKeys) == 0 { // Return everything. for i := start; i < end; i++ { pubkey := headState.PubkeyAtIndex(uint64(i)) res = append(res, ðpb.ValidatorBalances_Balance{ PublicKey: pubkey[:], Index: uint64(i), Balance: balances[i], }) } return ðpb.ValidatorBalances{ Epoch: epoch, Balances: res, TotalSize: int32(balancesCount), NextPageToken: nextPageToken, }, nil } return ðpb.ValidatorBalances{ Epoch: epoch, Balances: res[start:end], TotalSize: int32(balancesCount), NextPageToken: nextPageToken, }, nil } // ListValidators retrieves the current list of active validators with an optional historical epoch flag to // to retrieve validator set in time. func (bs *Server) ListValidators( ctx context.Context, req *ethpb.ListValidatorsRequest, ) (*ethpb.Validators, error) { if int(req.PageSize) > flags.Get().MaxPageSize { return nil, status.Errorf(codes.InvalidArgument, "Requested page size %d can not be greater than max size %d", req.PageSize, flags.Get().MaxPageSize) } headState, err := bs.HeadFetcher.HeadState(ctx) if err != nil { return nil, status.Error(codes.Internal, "Could not get head state") } currentEpoch := helpers.CurrentEpoch(headState) requestedEpoch := currentEpoch switch q := req.QueryFilter.(type) { case *ethpb.ListValidatorsRequest_Genesis: if q.Genesis { requestedEpoch = 0 } case *ethpb.ListValidatorsRequest_Epoch: requestedEpoch = q.Epoch } validatorList := make([]*ethpb.Validators_ValidatorContainer, 0) for i := 0; i < headState.NumValidators(); i++ { val, err := headState.ValidatorAtIndex(uint64(i)) if err != nil { return nil, status.Error(codes.Internal, "Could not get validator") } validatorList = append(validatorList, ðpb.Validators_ValidatorContainer{ Index: uint64(i), Validator: val, }) } if requestedEpoch < currentEpoch { stopIdx := len(validatorList) for idx, item := range validatorList { // The first time we see a validator with an activation epoch > the requested epoch, // we know this validator is from the future relative to what the request wants. if item.Validator.ActivationEpoch > requestedEpoch { stopIdx = idx break } } validatorList = validatorList[:stopIdx] } else if requestedEpoch > currentEpoch { // Otherwise, we are requesting data from the future and we return an error. return nil, status.Errorf( codes.InvalidArgument, "Cannot retrieve information about an epoch in the future, current epoch %d, requesting %d", currentEpoch, requestedEpoch, ) } // Filter active validators if the request specifies it. res := validatorList if req.Active { filteredValidators := make([]*ethpb.Validators_ValidatorContainer, 0) for _, item := range validatorList { if helpers.IsActiveValidator(item.Validator, requestedEpoch) { filteredValidators = append(filteredValidators, item) } } res = filteredValidators } validatorCount := len(res) // If there are no items, we simply return a response specifying this. // Otherwise, attempting to paginate 0 validators below would result in an error. if validatorCount == 0 { return ðpb.Validators{ ValidatorList: make([]*ethpb.Validators_ValidatorContainer, 0), TotalSize: int32(0), NextPageToken: strconv.Itoa(0), }, nil } start, end, nextPageToken, err := pagination.StartAndEndPage(req.PageToken, int(req.PageSize), validatorCount) if err != nil { return nil, status.Errorf( codes.Internal, "Could not paginate results: %v", err, ) } return ðpb.Validators{ ValidatorList: res[start:end], TotalSize: int32(validatorCount), NextPageToken: nextPageToken, }, nil } // GetValidator information from any validator in the registry by index or public key. func (bs *Server) GetValidator( ctx context.Context, req *ethpb.GetValidatorRequest, ) (*ethpb.Validator, error) { var requestingIndex bool var index uint64 var pubKey []byte switch q := req.QueryFilter.(type) { case *ethpb.GetValidatorRequest_Index: index = q.Index requestingIndex = true case *ethpb.GetValidatorRequest_PublicKey: pubKey = q.PublicKey default: return nil, status.Error( codes.InvalidArgument, "Need to specify either validator index or public key in request", ) } headState, err := bs.HeadFetcher.HeadState(ctx) if err != nil { return nil, status.Error(codes.Internal, "Could not get head state") } if requestingIndex { if index >= uint64(headState.NumValidators()) { return nil, status.Errorf( codes.OutOfRange, "Requesting index %d, but there are only %d validators", index, headState.NumValidators(), ) } return headState.ValidatorAtIndex(index) } pk48 := bytesutil.ToBytes48(pubKey) for i := 0; i < headState.NumValidators(); i++ { keyFromState := headState.PubkeyAtIndex(uint64(i)) if keyFromState == pk48 { return headState.ValidatorAtIndex(uint64(i)) } } return nil, status.Error(codes.NotFound, "No validator matched filter criteria") } // GetValidatorActiveSetChanges retrieves the active set changes for a given epoch. // // This data includes any activations, voluntary exits, and involuntary // ejections. func (bs *Server) GetValidatorActiveSetChanges( ctx context.Context, req *ethpb.GetValidatorActiveSetChangesRequest, ) (*ethpb.ActiveSetChanges, error) { headState, err := bs.HeadFetcher.HeadState(ctx) if err != nil { return nil, status.Error(codes.Internal, "Could not get head state") } currentEpoch := helpers.CurrentEpoch(headState) requestedEpoch := currentEpoch requestingGenesis := false switch q := req.QueryFilter.(type) { case *ethpb.GetValidatorActiveSetChangesRequest_Genesis: requestingGenesis = q.Genesis requestedEpoch = 0 case *ethpb.GetValidatorActiveSetChangesRequest_Epoch: requestedEpoch = q.Epoch } activatedIndices := make([]uint64, 0) slashedIndices := make([]uint64, 0) exitedIndices := make([]uint64, 0) if requestingGenesis || requestedEpoch < currentEpoch { archivedChanges, err := bs.BeaconDB.ArchivedActiveValidatorChanges(ctx, requestedEpoch) if err != nil { return nil, status.Errorf(codes.Internal, "Could not fetch archived active validator changes: %v", err) } if archivedChanges == nil { return nil, status.Errorf( codes.NotFound, "Did not find any data for epoch %d - perhaps no active set changed occurred during the epoch", requestedEpoch, ) } activatedIndices = archivedChanges.Activated slashedIndices = archivedChanges.Slashed exitedIndices = archivedChanges.Exited } else if requestedEpoch == currentEpoch { activeValidatorCount, err := helpers.ActiveValidatorCount(headState, helpers.PrevEpoch(headState)) if err != nil { return nil, status.Errorf(codes.Internal, "Could not get active validator count: %v", err) } vals := headState.Validators() activatedIndices = validators.ActivatedValidatorIndices(helpers.PrevEpoch(headState), vals) slashedIndices = validators.SlashedValidatorIndices(helpers.PrevEpoch(headState), vals) exitedIndices, err = validators.ExitedValidatorIndices(helpers.PrevEpoch(headState), vals, activeValidatorCount) if err != nil { return nil, status.Errorf(codes.Internal, "Could not determine exited validator indices: %v", err) } } else { // We are requesting data from the future and we return an error. return nil, status.Errorf( codes.InvalidArgument, "Cannot retrieve information about an epoch in the future, current epoch %d, requesting %d", currentEpoch, requestedEpoch, ) } // We retrieve the public keys for the indices. activatedKeys := make([][]byte, len(activatedIndices)) slashedKeys := make([][]byte, len(slashedIndices)) exitedKeys := make([][]byte, len(exitedIndices)) for i, idx := range activatedIndices { pubkey := headState.PubkeyAtIndex(idx) activatedKeys[i] = pubkey[:] } for i, idx := range slashedIndices { pubkey := headState.PubkeyAtIndex(idx) slashedKeys[i] = pubkey[:] } for i, idx := range exitedIndices { pubkey := headState.PubkeyAtIndex(idx) exitedKeys[i] = pubkey[:] } return ðpb.ActiveSetChanges{ Epoch: requestedEpoch, ActivatedPublicKeys: activatedKeys, ExitedPublicKeys: exitedKeys, SlashedPublicKeys: slashedKeys, }, nil } // GetValidatorParticipation retrieves the validator participation information for a given epoch, // it returns the information about validator's participation rate in voting on the proof of stake // rules based on their balance compared to the total active validator balance. func (bs *Server) GetValidatorParticipation( ctx context.Context, req *ethpb.GetValidatorParticipationRequest, ) (*ethpb.ValidatorParticipationResponse, error) { headState, err := bs.HeadFetcher.HeadState(ctx) if err != nil { return nil, status.Error(codes.Internal, "Could not get head state") } currentEpoch := helpers.CurrentEpoch(headState) prevEpoch := helpers.PrevEpoch(headState) var requestedEpoch uint64 var requestingGenesis bool switch q := req.QueryFilter.(type) { case *ethpb.GetValidatorParticipationRequest_Genesis: requestingGenesis = q.Genesis requestedEpoch = 0 case *ethpb.GetValidatorParticipationRequest_Epoch: requestedEpoch = q.Epoch default: requestedEpoch = prevEpoch } // If the request is from genesis or another past epoch, we look into our archived // data to find it and return it if it exists. if requestingGenesis || requestedEpoch < prevEpoch { participation, err := bs.BeaconDB.ArchivedValidatorParticipation(ctx, requestedEpoch) if err != nil { return nil, status.Errorf(codes.Internal, "Could not fetch archived participation: %v", err) } if participation == nil { return nil, status.Errorf( codes.NotFound, "Could not retrieve data for epoch %d, perhaps --archive in the running beacon node is disabled", 0, ) } return ðpb.ValidatorParticipationResponse{ Epoch: requestedEpoch, Finalized: requestedEpoch <= headState.FinalizedCheckpointEpoch(), Participation: participation, }, nil } else if requestedEpoch == currentEpoch { // We cannot retrieve participation for an epoch currently in progress. return nil, status.Errorf( codes.InvalidArgument, "Cannot retrieve information about an epoch currently in progress, current epoch %d, requesting %d", currentEpoch, requestedEpoch, ) } else if requestedEpoch > currentEpoch { // We are requesting data from the future and we return an error. return nil, status.Errorf( codes.InvalidArgument, "Cannot retrieve information about an epoch in the future, current epoch %d, requesting %d", currentEpoch, requestedEpoch, ) } p := bs.ParticipationFetcher.Participation(requestedEpoch) if p == nil { p = &precompute.Balance{} } participation := ðpb.ValidatorParticipation{ EligibleEther: p.PrevEpoch, VotedEther: p.PrevEpochTargetAttesters, } participation.GlobalParticipationRate = float32(0) // only divide if prevEpoch is non zero if p.PrevEpoch != 0 { participation.GlobalParticipationRate = float32(p.PrevEpochTargetAttesters) / float32(p.PrevEpoch) } return ðpb.ValidatorParticipationResponse{ Epoch: requestedEpoch, Finalized: requestedEpoch <= headState.FinalizedCheckpointEpoch(), Participation: participation, }, nil } // GetValidatorQueue retrieves the current validator queue information. func (bs *Server) GetValidatorQueue( ctx context.Context, _ *ptypes.Empty, ) (*ethpb.ValidatorQueue, error) { headState, err := bs.HeadFetcher.HeadState(ctx) if err != nil { return nil, status.Error(codes.Internal, "Could not get head state") } // Queue the validators whose eligible to activate and sort them by activation eligibility epoch number. // Additionally, determine those validators queued to exit awaitingExit := make([]uint64, 0) exitEpochs := make([]uint64, 0) activationQ := make([]uint64, 0) vals := headState.Validators() for idx, validator := range vals { eligibleActivated := validator.ActivationEligibilityEpoch != params.BeaconConfig().FarFutureEpoch canBeActive := validator.ActivationEpoch >= helpers.DelayedActivationExitEpoch(headState.FinalizedCheckpointEpoch()) if eligibleActivated && canBeActive { activationQ = append(activationQ, uint64(idx)) } if validator.ExitEpoch != params.BeaconConfig().FarFutureEpoch { exitEpochs = append(exitEpochs, validator.ExitEpoch) awaitingExit = append(awaitingExit, uint64(idx)) } } sort.Slice(activationQ, func(i, j int) bool { return vals[i].ActivationEligibilityEpoch < vals[j].ActivationEligibilityEpoch }) sort.Slice(awaitingExit, func(i, j int) bool { return vals[i].WithdrawableEpoch < vals[j].WithdrawableEpoch }) // Only activate just enough validators according to the activation churn limit. activationQueueChurn := len(activationQ) activeValidatorCount, err := helpers.ActiveValidatorCount(headState, helpers.CurrentEpoch(headState)) if err != nil { return nil, status.Errorf(codes.Internal, "Could not get active validator count: %v", err) } churnLimit, err := helpers.ValidatorChurnLimit(activeValidatorCount) if err != nil { return nil, status.Errorf(codes.Internal, "Could not compute churn limit: %v", err) } exitQueueEpoch := uint64(0) for _, i := range exitEpochs { if exitQueueEpoch < i { exitQueueEpoch = i } } exitQueueChurn := 0 for _, val := range vals { if val.ExitEpoch == exitQueueEpoch { exitQueueChurn++ } } // Prevent churn limit from causing index out of bound issues. if int(churnLimit) < activationQueueChurn { activationQueueChurn = int(churnLimit) } if int(churnLimit) < exitQueueChurn { // If we are above the churn limit, we simply increase the churn by one. exitQueueEpoch++ exitQueueChurn = int(churnLimit) } // We use the exit queue churn to determine if we have passed a churn limit. minEpoch := exitQueueEpoch + params.BeaconConfig().MinValidatorWithdrawabilityDelay exitQueueIndices := make([]uint64, 0) for _, valIdx := range awaitingExit { val := vals[valIdx] // Ensure the validator has not yet exited before adding its index to the exit queue. if val.WithdrawableEpoch < minEpoch && !validatorHasExited(val, helpers.CurrentEpoch(headState)) { exitQueueIndices = append(exitQueueIndices, valIdx) } } // Get the public keys for the validators in the queues up to the allowed churn limits. activationQueueKeys := make([][]byte, len(activationQ)) exitQueueKeys := make([][]byte, len(exitQueueIndices)) for i, idx := range activationQ { activationQueueKeys[i] = vals[idx].PublicKey } for i, idx := range exitQueueIndices { exitQueueKeys[i] = vals[idx].PublicKey } return ðpb.ValidatorQueue{ ChurnLimit: churnLimit, ActivationPublicKeys: activationQueueKeys, ExitPublicKeys: exitQueueKeys, }, nil } // GetValidatorPerformance reports the validator's latest balance along with other important metrics on // rewards and penalties throughout its lifecycle in the beacon chain. func (bs *Server) GetValidatorPerformance( ctx context.Context, req *ethpb.ValidatorPerformanceRequest, ) (*ethpb.ValidatorPerformanceResponse, error) { validatorSummary := state.ValidatorSummary reqPubKeysCount := len(req.PublicKeys) beforeTransitionBalances := make([]uint64, 0, reqPubKeysCount) afterTransitionBalances := make([]uint64, 0, reqPubKeysCount) effectiveBalances := make([]uint64, 0, reqPubKeysCount) inclusionSlots := make([]uint64, 0, reqPubKeysCount) inclusionDistances := make([]uint64, 0, reqPubKeysCount) correctlyVotedSource := make([]bool, 0, reqPubKeysCount) correctlyVotedTarget := make([]bool, 0, reqPubKeysCount) correctlyVotedHead := make([]bool, 0, reqPubKeysCount) missingValidators := make([][]byte, 0, reqPubKeysCount) // Convert the list of validator public keys to list of validator indices. // Also track missing validators using public keys. for _, key := range req.PublicKeys { idx, ok, err := bs.BeaconDB.ValidatorIndex(ctx, key) if err != nil { return nil, status.Errorf(codes.Internal, "Could not fetch validator idx for public key %#x: %v", key, err) } if !ok { missingValidators = append(missingValidators, key) continue } effectiveBalances = append(effectiveBalances, validatorSummary[idx].CurrentEpochEffectiveBalance) beforeTransitionBalances = append(beforeTransitionBalances, validatorSummary[idx].BeforeEpochTransitionBalance) afterTransitionBalances = append(afterTransitionBalances, validatorSummary[idx].AfterEpochTransitionBalance) inclusionSlots = append(inclusionSlots, validatorSummary[idx].InclusionSlot) inclusionDistances = append(inclusionDistances, validatorSummary[idx].InclusionDistance) correctlyVotedSource = append(correctlyVotedSource, validatorSummary[idx].IsPrevEpochAttester) correctlyVotedTarget = append(correctlyVotedTarget, validatorSummary[idx].IsPrevEpochTargetAttester) correctlyVotedHead = append(correctlyVotedHead, validatorSummary[idx].IsPrevEpochHeadAttester) } return ðpb.ValidatorPerformanceResponse{ InclusionSlots: inclusionSlots, InclusionDistances: inclusionDistances, CorrectlyVotedSource: correctlyVotedSource, CorrectlyVotedTarget: correctlyVotedTarget, CorrectlyVotedHead: correctlyVotedHead, CurrentEffectiveBalances: effectiveBalances, BalancesBeforeEpochTransition: beforeTransitionBalances, BalancesAfterEpochTransition: afterTransitionBalances, MissingValidators: missingValidators, }, nil } // Determines whether a validator has already exited. func validatorHasExited(validator *ethpb.Validator, currentEpoch uint64) bool { farFutureEpoch := params.BeaconConfig().FarFutureEpoch if currentEpoch < validator.ActivationEligibilityEpoch { return false } if currentEpoch < validator.ActivationEpoch { return false } if validator.ExitEpoch == farFutureEpoch { return false } if currentEpoch < validator.ExitEpoch { if validator.Slashed { return false } return false } return true }