package keeper import ( "context" "google.golang.org/grpc/codes" "google.golang.org/grpc/status" "github.com/cosmos/cosmos-sdk/store/prefix" sdk "github.com/cosmos/cosmos-sdk/types" sdkerrors "github.com/cosmos/cosmos-sdk/types/errors" "github.com/cosmos/cosmos-sdk/types/query" "github.com/cosmos/cosmos-sdk/x/distribution/types" stakingtypes "github.com/cosmos/cosmos-sdk/x/staking/types" ) var _ types.QueryServer = Keeper{} // Params queries params of distribution module func (k Keeper) Params(c context.Context, req *types.QueryParamsRequest) (*types.QueryParamsResponse, error) { ctx := sdk.UnwrapSDKContext(c) var params types.Params k.paramSpace.GetParamSet(ctx, ¶ms) return &types.QueryParamsResponse{Params: params}, nil } // ValidatorOutstandingRewards queries rewards of a validator address func (k Keeper) ValidatorOutstandingRewards(c context.Context, req *types.QueryValidatorOutstandingRewardsRequest) (*types.QueryValidatorOutstandingRewardsResponse, error) { if req == nil { return nil, status.Error(codes.InvalidArgument, "invalid request") } if req.ValidatorAddress == "" { return nil, status.Error(codes.InvalidArgument, "empty validator address") } ctx := sdk.UnwrapSDKContext(c) valAdr, err := sdk.ValAddressFromBech32(req.ValidatorAddress) if err != nil { return nil, err } rewards := k.GetValidatorOutstandingRewards(ctx, valAdr) return &types.QueryValidatorOutstandingRewardsResponse{Rewards: rewards}, nil } // ValidatorCommission queries accumulated commission for a validator func (k Keeper) ValidatorCommission(c context.Context, req *types.QueryValidatorCommissionRequest) (*types.QueryValidatorCommissionResponse, error) { if req == nil { return nil, status.Error(codes.InvalidArgument, "invalid request") } if req.ValidatorAddress == "" { return nil, status.Error(codes.InvalidArgument, "empty validator address") } ctx := sdk.UnwrapSDKContext(c) valAdr, err := sdk.ValAddressFromBech32(req.ValidatorAddress) if err != nil { return nil, err } commission := k.GetValidatorAccumulatedCommission(ctx, valAdr) return &types.QueryValidatorCommissionResponse{Commission: commission}, nil } // ValidatorSlashes queries slash events of a validator func (k Keeper) ValidatorSlashes(c context.Context, req *types.QueryValidatorSlashesRequest) (*types.QueryValidatorSlashesResponse, error) { if req == nil { return nil, status.Error(codes.InvalidArgument, "invalid request") } if req.ValidatorAddress == "" { return nil, status.Error(codes.InvalidArgument, "empty validator address") } if req.EndingHeight < req.StartingHeight { return nil, status.Errorf(codes.InvalidArgument, "starting height greater than ending height (%d > %d)", req.StartingHeight, req.EndingHeight) } ctx := sdk.UnwrapSDKContext(c) events := make([]types.ValidatorSlashEvent, 0) store := ctx.KVStore(k.storeKey) valAddr, err := sdk.ValAddressFromBech32(req.ValidatorAddress) if err != nil { return nil, status.Errorf(codes.InvalidArgument, "invalid validator address") } slashesStore := prefix.NewStore(store, types.GetValidatorSlashEventPrefix(valAddr)) pageRes, err := query.FilteredPaginate(slashesStore, req.Pagination, func(key []byte, value []byte, accumulate bool) (bool, error) { var result types.ValidatorSlashEvent err := k.cdc.UnmarshalBinaryBare(value, &result) if err != nil { return false, err } if result.ValidatorPeriod < req.StartingHeight || result.ValidatorPeriod > req.EndingHeight { return false, nil } if accumulate { events = append(events, result) } return true, nil }) if err != nil { return nil, err } return &types.QueryValidatorSlashesResponse{Slashes: events, Pagination: pageRes}, nil } // DelegationRewards the total rewards accrued by a delegation func (k Keeper) DelegationRewards(c context.Context, req *types.QueryDelegationRewardsRequest) (*types.QueryDelegationRewardsResponse, error) { if req == nil { return nil, status.Error(codes.InvalidArgument, "invalid request") } if req.DelegatorAddress == "" { return nil, status.Error(codes.InvalidArgument, "empty delegator address") } if req.ValidatorAddress == "" { return nil, status.Error(codes.InvalidArgument, "empty validator address") } ctx := sdk.UnwrapSDKContext(c) valAdr, err := sdk.ValAddressFromBech32(req.ValidatorAddress) if err != nil { return nil, err } val := k.stakingKeeper.Validator(ctx, valAdr) if val == nil { return nil, sdkerrors.Wrap(types.ErrNoValidatorExists, req.ValidatorAddress) } delAdr, err := sdk.AccAddressFromBech32(req.DelegatorAddress) if err != nil { return nil, err } del := k.stakingKeeper.Delegation(ctx, delAdr, valAdr) if del == nil { return nil, types.ErrNoDelegationExists } endingPeriod := k.IncrementValidatorPeriod(ctx, val) rewards := k.CalculateDelegationRewards(ctx, val, del, endingPeriod) return &types.QueryDelegationRewardsResponse{Rewards: rewards}, nil } // DelegationTotalRewards the total rewards accrued by a each validator func (k Keeper) DelegationTotalRewards(c context.Context, req *types.QueryDelegationTotalRewardsRequest) (*types.QueryDelegationTotalRewardsResponse, error) { if req == nil { return nil, status.Error(codes.InvalidArgument, "invalid request") } if req.DelegatorAddress == "" { return nil, status.Error(codes.InvalidArgument, "empty delegator address") } ctx := sdk.UnwrapSDKContext(c) total := sdk.DecCoins{} var delRewards []types.DelegationDelegatorReward delAdr, err := sdk.AccAddressFromBech32(req.DelegatorAddress) if err != nil { return nil, err } k.stakingKeeper.IterateDelegations( ctx, delAdr, func(_ int64, del stakingtypes.DelegationI) (stop bool) { valAddr := del.GetValidatorAddr() val := k.stakingKeeper.Validator(ctx, valAddr) endingPeriod := k.IncrementValidatorPeriod(ctx, val) delReward := k.CalculateDelegationRewards(ctx, val, del, endingPeriod) delRewards = append(delRewards, types.NewDelegationDelegatorReward(valAddr, delReward)) total = total.Add(delReward...) return false }, ) return &types.QueryDelegationTotalRewardsResponse{Rewards: delRewards, Total: total}, nil } // DelegatorValidators queries the validators list of a delegator func (k Keeper) DelegatorValidators(c context.Context, req *types.QueryDelegatorValidatorsRequest) (*types.QueryDelegatorValidatorsResponse, error) { if req == nil { return nil, status.Error(codes.InvalidArgument, "invalid request") } if req.DelegatorAddress == "" { return nil, status.Error(codes.InvalidArgument, "empty delegator address") } ctx := sdk.UnwrapSDKContext(c) delAdr, err := sdk.AccAddressFromBech32(req.DelegatorAddress) if err != nil { return nil, err } var validators []string k.stakingKeeper.IterateDelegations( ctx, delAdr, func(_ int64, del stakingtypes.DelegationI) (stop bool) { validators = append(validators, del.GetValidatorAddr().String()) return false }, ) return &types.QueryDelegatorValidatorsResponse{Validators: validators}, nil } // DelegatorWithdrawAddress queries Query/delegatorWithdrawAddress func (k Keeper) DelegatorWithdrawAddress(c context.Context, req *types.QueryDelegatorWithdrawAddressRequest) (*types.QueryDelegatorWithdrawAddressResponse, error) { if req == nil { return nil, status.Error(codes.InvalidArgument, "invalid request") } if req.DelegatorAddress == "" { return nil, status.Error(codes.InvalidArgument, "empty delegator address") } delAdr, err := sdk.AccAddressFromBech32(req.DelegatorAddress) if err != nil { return nil, err } ctx := sdk.UnwrapSDKContext(c) withdrawAddr := k.GetDelegatorWithdrawAddr(ctx, delAdr) return &types.QueryDelegatorWithdrawAddressResponse{WithdrawAddress: withdrawAddr.String()}, nil } // CommunityPool queries the community pool coins func (k Keeper) CommunityPool(c context.Context, req *types.QueryCommunityPoolRequest) (*types.QueryCommunityPoolResponse, error) { ctx := sdk.UnwrapSDKContext(c) pool := k.GetFeePoolCommunityCoins(ctx) return &types.QueryCommunityPoolResponse{Pool: pool}, nil }