Some checks failed
Build SimApp / build (amd64) (push) Waiting to run
Build SimApp / build (arm64) (push) Waiting to run
CodeQL / Analyze (push) Waiting to run
Build & Push / build (push) Waiting to run
Run Gosec / Gosec (push) Waiting to run
Lint / golangci-lint (push) Waiting to run
Checks dependencies and mocks generation / Check go mod tidy (push) Waiting to run
Checks dependencies and mocks generation / Check up to date mocks (push) Waiting to run
System Tests / setup (push) Waiting to run
System Tests / test-system (push) Blocked by required conditions
System Tests / test-system-legacy (push) Blocked by required conditions
Tests / Code Coverage / split-test-files (push) Waiting to run
Tests / Code Coverage / tests (00) (push) Blocked by required conditions
Tests / Code Coverage / tests (01) (push) Blocked by required conditions
Tests / Code Coverage / tests (02) (push) Blocked by required conditions
Tests / Code Coverage / tests (03) (push) Blocked by required conditions
Tests / Code Coverage / test-integration (push) Waiting to run
Tests / Code Coverage / test-e2e (push) Waiting to run
Tests / Code Coverage / repo-analysis (push) Blocked by required conditions
Tests / Code Coverage / test-sim-nondeterminism (push) Waiting to run
Tests / Code Coverage / test-clientv2 (push) Waiting to run
Tests / Code Coverage / test-core (push) Waiting to run
Tests / Code Coverage / test-depinject (push) Waiting to run
Tests / Code Coverage / test-errors (push) Waiting to run
Tests / Code Coverage / test-math (push) Waiting to run
Tests / Code Coverage / test-schema (push) Waiting to run
Tests / Code Coverage / test-collections (push) Waiting to run
Tests / Code Coverage / test-cosmovisor (push) Waiting to run
Tests / Code Coverage / test-confix (push) Waiting to run
Tests / Code Coverage / test-store (push) Waiting to run
Tests / Code Coverage / test-log (push) Waiting to run
Tests / Code Coverage / test-x-tx (push) Waiting to run
Tests / Code Coverage / test-x-nft (push) Waiting to run
Tests / Code Coverage / test-x-circuit (push) Waiting to run
Tests / Code Coverage / test-x-feegrant (push) Waiting to run
Tests / Code Coverage / test-x-evidence (push) Waiting to run
Tests / Code Coverage / test-x-upgrade (push) Waiting to run
Tests / Code Coverage / test-tools-benchmark (push) Waiting to run
Build & Push SDK Proto Builder / build (push) Has been cancelled
318 lines
10 KiB
Go
318 lines
10 KiB
Go
package keeper
|
|
|
|
import (
|
|
"context"
|
|
|
|
"google.golang.org/grpc/codes"
|
|
"google.golang.org/grpc/status"
|
|
|
|
"cosmossdk.io/collections"
|
|
"cosmossdk.io/math"
|
|
"cosmossdk.io/store/prefix"
|
|
|
|
"github.com/cosmos/cosmos-sdk/runtime"
|
|
sdk "github.com/cosmos/cosmos-sdk/types"
|
|
"github.com/cosmos/cosmos-sdk/types/query"
|
|
"github.com/cosmos/cosmos-sdk/x/bank/types"
|
|
)
|
|
|
|
type Querier struct {
|
|
BaseKeeper
|
|
}
|
|
|
|
var _ types.QueryServer = BaseKeeper{}
|
|
|
|
func NewQuerier(keeper *BaseKeeper) Querier {
|
|
return Querier{BaseKeeper: *keeper}
|
|
}
|
|
|
|
// Balance implements the Query/Balance gRPC method
|
|
func (k BaseKeeper) Balance(ctx context.Context, req *types.QueryBalanceRequest) (*types.QueryBalanceResponse, error) {
|
|
if req == nil {
|
|
return nil, status.Error(codes.InvalidArgument, "empty request")
|
|
}
|
|
|
|
if err := sdk.ValidateDenom(req.Denom); err != nil {
|
|
return nil, status.Error(codes.InvalidArgument, err.Error())
|
|
}
|
|
|
|
sdkCtx := sdk.UnwrapSDKContext(ctx)
|
|
address, err := k.ak.AddressCodec().StringToBytes(req.Address)
|
|
if err != nil {
|
|
return nil, status.Errorf(codes.InvalidArgument, "invalid address: %s", err.Error())
|
|
}
|
|
|
|
balance := k.GetBalance(sdkCtx, address, req.Denom)
|
|
|
|
return &types.QueryBalanceResponse{Balance: &balance}, nil
|
|
}
|
|
|
|
// AllBalances implements the Query/AllBalances gRPC method
|
|
func (k BaseKeeper) AllBalances(ctx context.Context, req *types.QueryAllBalancesRequest) (*types.QueryAllBalancesResponse, error) {
|
|
if req == nil {
|
|
return nil, status.Error(codes.InvalidArgument, "empty request")
|
|
}
|
|
|
|
addr, err := k.ak.AddressCodec().StringToBytes(req.Address)
|
|
if err != nil {
|
|
return nil, status.Errorf(codes.InvalidArgument, "invalid address: %s", err.Error())
|
|
}
|
|
|
|
sdkCtx := sdk.UnwrapSDKContext(ctx)
|
|
balances, pageRes, err := query.CollectionPaginate(
|
|
ctx,
|
|
k.Balances,
|
|
req.Pagination,
|
|
func(key collections.Pair[sdk.AccAddress, string], value math.Int) (sdk.Coin, error) {
|
|
if req.ResolveDenom {
|
|
if metadata, ok := k.GetDenomMetaData(sdkCtx, key.K2()); ok {
|
|
return sdk.NewCoin(metadata.Display, value), nil
|
|
}
|
|
}
|
|
return sdk.NewCoin(key.K2(), value), nil
|
|
},
|
|
query.WithCollectionPaginationPairPrefix[sdk.AccAddress, string](addr),
|
|
)
|
|
if err != nil {
|
|
return nil, status.Errorf(codes.InvalidArgument, "paginate: %v", err)
|
|
}
|
|
|
|
return &types.QueryAllBalancesResponse{Balances: balances, Pagination: pageRes}, nil
|
|
}
|
|
|
|
// SpendableBalances implements a gRPC query handler for retrieving an account's
|
|
// spendable balances.
|
|
func (k BaseKeeper) SpendableBalances(ctx context.Context, req *types.QuerySpendableBalancesRequest) (*types.QuerySpendableBalancesResponse, error) {
|
|
if req == nil {
|
|
return nil, status.Error(codes.InvalidArgument, "empty request")
|
|
}
|
|
|
|
addr, err := k.ak.AddressCodec().StringToBytes(req.Address)
|
|
if err != nil {
|
|
return nil, status.Errorf(codes.InvalidArgument, "invalid address: %s", err.Error())
|
|
}
|
|
|
|
sdkCtx := sdk.UnwrapSDKContext(ctx)
|
|
|
|
zeroAmt := math.ZeroInt()
|
|
|
|
balances, pageRes, err := query.CollectionPaginate(ctx, k.Balances, req.Pagination, func(key collections.Pair[sdk.AccAddress, string], _ math.Int) (coin sdk.Coin, err error) {
|
|
return sdk.NewCoin(key.K2(), zeroAmt), nil
|
|
}, query.WithCollectionPaginationPairPrefix[sdk.AccAddress, string](addr))
|
|
if err != nil {
|
|
return nil, status.Errorf(codes.InvalidArgument, "paginate: %v", err)
|
|
}
|
|
|
|
result := sdk.NewCoins()
|
|
spendable := k.SpendableCoins(sdkCtx, addr)
|
|
|
|
for _, c := range balances {
|
|
result = append(result, sdk.NewCoin(c.Denom, spendable.AmountOf(c.Denom)))
|
|
}
|
|
|
|
return &types.QuerySpendableBalancesResponse{Balances: result, Pagination: pageRes}, nil
|
|
}
|
|
|
|
// SpendableBalanceByDenom implements a gRPC query handler for retrieving an account's
|
|
// spendable balance for a specific denom.
|
|
func (k BaseKeeper) SpendableBalanceByDenom(ctx context.Context, req *types.QuerySpendableBalanceByDenomRequest) (*types.QuerySpendableBalanceByDenomResponse, error) {
|
|
if req == nil {
|
|
return nil, status.Error(codes.InvalidArgument, "empty request")
|
|
}
|
|
|
|
addr, err := k.ak.AddressCodec().StringToBytes(req.Address)
|
|
if err != nil {
|
|
return nil, status.Errorf(codes.InvalidArgument, "invalid address: %s", err.Error())
|
|
}
|
|
|
|
if err := sdk.ValidateDenom(req.Denom); err != nil {
|
|
return nil, status.Error(codes.InvalidArgument, err.Error())
|
|
}
|
|
|
|
sdkCtx := sdk.UnwrapSDKContext(ctx)
|
|
|
|
spendable := k.SpendableCoin(sdkCtx, addr, req.Denom)
|
|
|
|
return &types.QuerySpendableBalanceByDenomResponse{Balance: &spendable}, nil
|
|
}
|
|
|
|
// TotalSupply implements the Query/TotalSupply gRPC method
|
|
func (k BaseKeeper) TotalSupply(ctx context.Context, req *types.QueryTotalSupplyRequest) (*types.QueryTotalSupplyResponse, error) {
|
|
sdkCtx := sdk.UnwrapSDKContext(ctx)
|
|
totalSupply, pageRes, err := k.GetPaginatedTotalSupply(sdkCtx, req.Pagination)
|
|
if err != nil {
|
|
return nil, status.Error(codes.Internal, err.Error())
|
|
}
|
|
|
|
return &types.QueryTotalSupplyResponse{Supply: totalSupply, Pagination: pageRes}, nil
|
|
}
|
|
|
|
// SupplyOf implements the Query/SupplyOf gRPC method
|
|
func (k BaseKeeper) SupplyOf(c context.Context, req *types.QuerySupplyOfRequest) (*types.QuerySupplyOfResponse, error) {
|
|
if req == nil {
|
|
return nil, status.Error(codes.InvalidArgument, "empty request")
|
|
}
|
|
|
|
if err := sdk.ValidateDenom(req.Denom); err != nil {
|
|
return nil, status.Error(codes.InvalidArgument, err.Error())
|
|
}
|
|
|
|
ctx := sdk.UnwrapSDKContext(c)
|
|
supply := k.GetSupply(ctx, req.Denom)
|
|
|
|
return &types.QuerySupplyOfResponse{Amount: sdk.NewCoin(req.Denom, supply.Amount)}, nil
|
|
}
|
|
|
|
// Params implements the gRPC service handler for querying x/bank parameters.
|
|
func (k BaseKeeper) Params(ctx context.Context, req *types.QueryParamsRequest) (*types.QueryParamsResponse, error) {
|
|
if req == nil {
|
|
return nil, status.Errorf(codes.InvalidArgument, "empty request")
|
|
}
|
|
|
|
sdkCtx := sdk.UnwrapSDKContext(ctx)
|
|
params := k.GetParams(sdkCtx)
|
|
|
|
return &types.QueryParamsResponse{Params: params}, nil
|
|
}
|
|
|
|
// DenomsMetadata implements Query/DenomsMetadata gRPC method.
|
|
func (k BaseKeeper) DenomsMetadata(c context.Context, req *types.QueryDenomsMetadataRequest) (*types.QueryDenomsMetadataResponse, error) {
|
|
if req == nil {
|
|
return nil, status.Errorf(codes.InvalidArgument, "empty request")
|
|
}
|
|
kvStore := runtime.KVStoreAdapter(k.storeService.OpenKVStore(c))
|
|
store := prefix.NewStore(kvStore, types.DenomMetadataPrefix)
|
|
|
|
metadatas := []types.Metadata{}
|
|
pageRes, err := query.Paginate(store, req.Pagination, func(_, value []byte) error {
|
|
var metadata types.Metadata
|
|
k.cdc.MustUnmarshal(value, &metadata)
|
|
|
|
metadatas = append(metadatas, metadata)
|
|
return nil
|
|
})
|
|
if err != nil {
|
|
return nil, status.Error(codes.Internal, err.Error())
|
|
}
|
|
|
|
return &types.QueryDenomsMetadataResponse{
|
|
Metadatas: metadatas,
|
|
Pagination: pageRes,
|
|
}, nil
|
|
}
|
|
|
|
// DenomMetadata implements Query/DenomMetadata gRPC method.
|
|
func (k BaseKeeper) DenomMetadata(c context.Context, req *types.QueryDenomMetadataRequest) (*types.QueryDenomMetadataResponse, error) {
|
|
if req == nil {
|
|
return nil, status.Errorf(codes.InvalidArgument, "empty request")
|
|
}
|
|
|
|
if err := sdk.ValidateDenom(req.Denom); err != nil {
|
|
return nil, status.Error(codes.InvalidArgument, err.Error())
|
|
}
|
|
|
|
ctx := sdk.UnwrapSDKContext(c)
|
|
|
|
metadata, found := k.GetDenomMetaData(ctx, req.Denom)
|
|
if !found {
|
|
return nil, status.Errorf(codes.NotFound, "client metadata for denom %s", req.Denom)
|
|
}
|
|
|
|
return &types.QueryDenomMetadataResponse{
|
|
Metadata: metadata,
|
|
}, nil
|
|
}
|
|
|
|
// DenomMetadataByQueryString is identical to DenomMetadata query, but receives request via query string.
|
|
func (k BaseKeeper) DenomMetadataByQueryString(c context.Context, req *types.QueryDenomMetadataByQueryStringRequest) (*types.QueryDenomMetadataByQueryStringResponse, error) {
|
|
if req == nil {
|
|
return nil, status.Errorf(codes.InvalidArgument, "empty request")
|
|
}
|
|
|
|
res, err := k.DenomMetadata(c, &types.QueryDenomMetadataRequest{
|
|
Denom: req.Denom,
|
|
})
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
return &types.QueryDenomMetadataByQueryStringResponse{Metadata: res.Metadata}, nil
|
|
}
|
|
|
|
func (k BaseKeeper) DenomOwners(
|
|
ctx context.Context,
|
|
req *types.QueryDenomOwnersRequest,
|
|
) (*types.QueryDenomOwnersResponse, error) {
|
|
if req == nil {
|
|
return nil, status.Errorf(codes.InvalidArgument, "empty request")
|
|
}
|
|
|
|
if err := sdk.ValidateDenom(req.Denom); err != nil {
|
|
return nil, status.Error(codes.InvalidArgument, err.Error())
|
|
}
|
|
|
|
denomOwners, pageRes, err := query.CollectionPaginate(
|
|
ctx,
|
|
k.Balances.Indexes.Denom,
|
|
req.Pagination,
|
|
func(key collections.Pair[string, sdk.AccAddress], value collections.NoValue) (*types.DenomOwner, error) {
|
|
amt, err := k.Balances.Get(ctx, collections.Join(key.K2(), req.Denom))
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
return &types.DenomOwner{Address: key.K2().String(), Balance: sdk.NewCoin(req.Denom, amt)}, nil
|
|
},
|
|
query.WithCollectionPaginationPairPrefix[string, sdk.AccAddress](req.Denom),
|
|
)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
return &types.QueryDenomOwnersResponse{DenomOwners: denomOwners, Pagination: pageRes}, nil
|
|
}
|
|
|
|
func (k BaseKeeper) SendEnabled(goCtx context.Context, req *types.QuerySendEnabledRequest) (*types.QuerySendEnabledResponse, error) {
|
|
if req == nil {
|
|
return nil, status.Errorf(codes.InvalidArgument, "empty request")
|
|
}
|
|
ctx := sdk.UnwrapSDKContext(goCtx)
|
|
resp := &types.QuerySendEnabledResponse{}
|
|
if len(req.Denoms) > 0 {
|
|
for _, denom := range req.Denoms {
|
|
if se, ok := k.getSendEnabled(ctx, denom); ok {
|
|
resp.SendEnabled = append(resp.SendEnabled, types.NewSendEnabled(denom, se))
|
|
}
|
|
}
|
|
} else {
|
|
results, pageResp, err := query.CollectionPaginate(
|
|
ctx,
|
|
k.BaseViewKeeper.SendEnabled,
|
|
req.Pagination, func(key string, value bool) (*types.SendEnabled, error) {
|
|
return types.NewSendEnabled(key, value), nil
|
|
},
|
|
)
|
|
if err != nil {
|
|
return nil, status.Error(codes.Internal, err.Error())
|
|
}
|
|
resp.SendEnabled = results
|
|
resp.Pagination = pageResp
|
|
}
|
|
|
|
return resp, nil
|
|
}
|
|
|
|
// DenomOwnersByQuery is identical to DenomOwner query, but receives denom values via query string.
|
|
func (k BaseKeeper) DenomOwnersByQuery(ctx context.Context, req *types.QueryDenomOwnersByQueryRequest) (*types.QueryDenomOwnersByQueryResponse, error) {
|
|
if req == nil {
|
|
return nil, status.Errorf(codes.InvalidArgument, "empty request")
|
|
}
|
|
resp, err := k.DenomOwners(ctx, &types.QueryDenomOwnersRequest{
|
|
Denom: req.Denom,
|
|
Pagination: req.Pagination,
|
|
})
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
return &types.QueryDenomOwnersByQueryResponse{DenomOwners: resp.DenomOwners, Pagination: resp.Pagination}, nil
|
|
}
|