2022-03-22 21:13:27 +00:00
|
|
|
package keeper
|
|
|
|
|
|
|
|
import (
|
|
|
|
"context"
|
2022-11-08 17:43:26 +00:00
|
|
|
"strings"
|
2022-03-22 21:13:27 +00:00
|
|
|
|
|
|
|
"google.golang.org/grpc/codes"
|
|
|
|
"google.golang.org/grpc/status"
|
|
|
|
|
2023-04-05 23:21:59 +00:00
|
|
|
errorsmod "cosmossdk.io/errors"
|
2022-03-24 16:43:03 +00:00
|
|
|
"github.com/cosmos/cosmos-sdk/client"
|
2022-03-22 21:13:27 +00:00
|
|
|
sdk "github.com/cosmos/cosmos-sdk/types"
|
2022-03-24 16:43:03 +00:00
|
|
|
sdkerrors "github.com/cosmos/cosmos-sdk/types/errors"
|
|
|
|
"github.com/cosmos/cosmos-sdk/types/query"
|
2022-03-22 21:13:27 +00:00
|
|
|
|
2024-05-01 03:17:24 +00:00
|
|
|
"github.com/0glabs/0g-chain/x/savings/types"
|
2022-03-22 21:13:27 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
type queryServer struct {
|
|
|
|
keeper Keeper
|
|
|
|
}
|
|
|
|
|
|
|
|
// NewQueryServerImpl creates a new server for handling gRPC queries.
|
|
|
|
func NewQueryServerImpl(k Keeper) types.QueryServer {
|
|
|
|
return &queryServer{keeper: k}
|
|
|
|
}
|
|
|
|
|
|
|
|
var _ types.QueryServer = queryServer{}
|
|
|
|
|
|
|
|
// Params implements the gRPC service handler for querying x/savings parameters.
|
|
|
|
func (s queryServer) Params(c context.Context, req *types.QueryParamsRequest) (*types.QueryParamsResponse, error) {
|
|
|
|
if req == nil {
|
|
|
|
return nil, status.Errorf(codes.InvalidArgument, "empty request")
|
|
|
|
}
|
|
|
|
|
|
|
|
sdkCtx := sdk.UnwrapSDKContext(c)
|
|
|
|
params := s.keeper.GetParams(sdkCtx)
|
|
|
|
|
|
|
|
return &types.QueryParamsResponse{Params: params}, nil
|
|
|
|
}
|
2022-03-24 16:43:03 +00:00
|
|
|
|
|
|
|
func (s queryServer) Deposits(ctx context.Context, req *types.QueryDepositsRequest) (*types.QueryDepositsResponse, error) {
|
|
|
|
if req == nil {
|
|
|
|
return nil, status.Errorf(codes.InvalidArgument, "empty request")
|
|
|
|
}
|
|
|
|
|
|
|
|
sdkCtx := sdk.UnwrapSDKContext(ctx)
|
|
|
|
|
|
|
|
hasDenom := len(req.Denom) > 0
|
|
|
|
hasOwner := len(req.Owner) > 0
|
|
|
|
|
|
|
|
var owner sdk.AccAddress
|
|
|
|
var err error
|
|
|
|
if hasOwner {
|
|
|
|
owner, err = sdk.AccAddressFromBech32(req.Owner)
|
|
|
|
if err != nil {
|
2023-04-05 23:21:59 +00:00
|
|
|
return nil, errorsmod.Wrap(sdkerrors.ErrInvalidAddress, err.Error())
|
2022-03-24 16:43:03 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
var deposits types.Deposits
|
|
|
|
switch {
|
|
|
|
case hasOwner && hasDenom:
|
|
|
|
deposit, found := s.keeper.GetDeposit(sdkCtx, owner)
|
|
|
|
if found {
|
|
|
|
for _, coin := range deposit.Amount {
|
|
|
|
if coin.Denom == req.Denom {
|
|
|
|
deposits = append(deposits, deposit)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
case hasOwner:
|
|
|
|
deposit, found := s.keeper.GetDeposit(sdkCtx, owner)
|
|
|
|
if found {
|
|
|
|
deposits = append(deposits, deposit)
|
|
|
|
}
|
|
|
|
case hasDenom:
|
|
|
|
s.keeper.IterateDeposits(sdkCtx, func(deposit types.Deposit) (stop bool) {
|
|
|
|
if deposit.Amount.AmountOf(req.Denom).IsPositive() {
|
|
|
|
deposits = append(deposits, deposit)
|
|
|
|
}
|
|
|
|
return false
|
|
|
|
})
|
|
|
|
default:
|
|
|
|
s.keeper.IterateDeposits(sdkCtx, func(deposit types.Deposit) (stop bool) {
|
|
|
|
deposits = append(deposits, deposit)
|
|
|
|
return false
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
page, limit, err := query.ParsePagination(req.Pagination)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
start, end := client.Paginate(len(deposits), page, limit, 100)
|
|
|
|
if start < 0 || end < 0 {
|
|
|
|
deposits = types.Deposits{}
|
|
|
|
} else {
|
|
|
|
deposits = deposits[start:end]
|
|
|
|
}
|
|
|
|
|
|
|
|
return &types.QueryDepositsResponse{
|
|
|
|
Deposits: deposits,
|
|
|
|
Pagination: nil,
|
|
|
|
}, nil
|
|
|
|
}
|
2022-11-08 17:43:26 +00:00
|
|
|
|
|
|
|
func (s queryServer) TotalSupply(ctx context.Context, req *types.QueryTotalSupplyRequest) (*types.QueryTotalSupplyResponse, error) {
|
|
|
|
sdkCtx := sdk.UnwrapSDKContext(ctx)
|
|
|
|
totalSupply := sdk.NewCoins()
|
|
|
|
liquidStakedDerivatives := sdk.NewCoins()
|
|
|
|
|
|
|
|
s.keeper.IterateDeposits(sdkCtx, func(deposit types.Deposit) (stop bool) {
|
|
|
|
for _, c := range deposit.Amount {
|
|
|
|
// separate out bkava denoms
|
|
|
|
if strings.HasPrefix(c.Denom, bkavaPrefix) {
|
|
|
|
liquidStakedDerivatives = liquidStakedDerivatives.Add(c)
|
|
|
|
} else {
|
|
|
|
totalSupply = totalSupply.Add(c)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return false
|
|
|
|
})
|
|
|
|
|
|
|
|
// determine underlying value of bkava denoms
|
|
|
|
if len(liquidStakedDerivatives) > 0 {
|
|
|
|
underlyingValue, err := s.keeper.liquidKeeper.GetStakedTokensForDerivatives(
|
|
|
|
sdkCtx,
|
|
|
|
liquidStakedDerivatives,
|
|
|
|
)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
totalSupply = totalSupply.Add(sdk.NewCoin(bkavaDenom, underlyingValue.Amount))
|
|
|
|
}
|
|
|
|
|
|
|
|
return &types.QueryTotalSupplyResponse{
|
|
|
|
Height: sdkCtx.BlockHeight(),
|
|
|
|
Result: totalSupply,
|
|
|
|
}, nil
|
|
|
|
}
|