mirror of https://github.com/certusone/wasmd.git
497 lines
15 KiB
Go
497 lines
15 KiB
Go
package keeper
|
|
|
|
import (
|
|
"context"
|
|
"encoding/binary"
|
|
"encoding/hex"
|
|
"encoding/json"
|
|
"fmt"
|
|
"runtime/debug"
|
|
|
|
"google.golang.org/grpc/codes"
|
|
"google.golang.org/grpc/status"
|
|
|
|
corestoretypes "cosmossdk.io/core/store"
|
|
errorsmod "cosmossdk.io/errors"
|
|
"cosmossdk.io/store/prefix"
|
|
storetypes "cosmossdk.io/store/types"
|
|
|
|
"github.com/cosmos/cosmos-sdk/codec"
|
|
"github.com/cosmos/cosmos-sdk/runtime"
|
|
sdk "github.com/cosmos/cosmos-sdk/types"
|
|
sdkerrors "github.com/cosmos/cosmos-sdk/types/errors"
|
|
"github.com/cosmos/cosmos-sdk/types/query"
|
|
|
|
"github.com/CosmWasm/wasmd/x/wasm/types"
|
|
)
|
|
|
|
// DefaultGasCostBuildAddress is the SDK gas cost to build a contract address
|
|
const DefaultGasCostBuildAddress = 10
|
|
|
|
var _ types.QueryServer = &GrpcQuerier{}
|
|
|
|
type GrpcQuerier struct {
|
|
cdc codec.Codec
|
|
storeService corestoretypes.KVStoreService
|
|
keeper types.ViewKeeper
|
|
queryGasLimit storetypes.Gas
|
|
}
|
|
|
|
// NewGrpcQuerier constructor
|
|
func NewGrpcQuerier(cdc codec.Codec, storeService corestoretypes.KVStoreService, keeper types.ViewKeeper, queryGasLimit storetypes.Gas) *GrpcQuerier {
|
|
return &GrpcQuerier{cdc: cdc, storeService: storeService, keeper: keeper, queryGasLimit: queryGasLimit}
|
|
}
|
|
|
|
func (q GrpcQuerier) ContractInfo(c context.Context, req *types.QueryContractInfoRequest) (*types.QueryContractInfoResponse, error) {
|
|
if req == nil {
|
|
return nil, status.Error(codes.InvalidArgument, "empty request")
|
|
}
|
|
contractAddr, err := sdk.AccAddressFromBech32(req.Address)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
rsp, err := queryContractInfo(sdk.UnwrapSDKContext(c), contractAddr, q.keeper)
|
|
switch {
|
|
case err != nil:
|
|
return nil, err
|
|
case rsp == nil:
|
|
return nil, types.ErrNoSuchContractFn(contractAddr.String()).
|
|
Wrapf("address %s", contractAddr.String())
|
|
}
|
|
return rsp, nil
|
|
}
|
|
|
|
func (q GrpcQuerier) ContractHistory(c context.Context, req *types.QueryContractHistoryRequest) (*types.QueryContractHistoryResponse, error) {
|
|
if req == nil {
|
|
return nil, status.Error(codes.InvalidArgument, "empty request")
|
|
}
|
|
contractAddr, err := sdk.AccAddressFromBech32(req.Address)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
paginationParams, err := ensurePaginationParams(req.Pagination)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
ctx := sdk.UnwrapSDKContext(c)
|
|
r := make([]types.ContractCodeHistoryEntry, 0)
|
|
|
|
prefixStore := prefix.NewStore(runtime.KVStoreAdapter(q.storeService.OpenKVStore(ctx)), types.GetContractCodeHistoryElementPrefix(contractAddr))
|
|
pageRes, err := query.FilteredPaginate(prefixStore, paginationParams, func(key, value []byte, accumulate bool) (bool, error) {
|
|
if accumulate {
|
|
var e types.ContractCodeHistoryEntry
|
|
if err := q.cdc.Unmarshal(value, &e); err != nil {
|
|
return false, err
|
|
}
|
|
r = append(r, e)
|
|
}
|
|
return true, nil
|
|
})
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
return &types.QueryContractHistoryResponse{
|
|
Entries: r,
|
|
Pagination: pageRes,
|
|
}, nil
|
|
}
|
|
|
|
// ContractsByCode lists all smart contracts for a code id
|
|
func (q GrpcQuerier) ContractsByCode(c context.Context, req *types.QueryContractsByCodeRequest) (*types.QueryContractsByCodeResponse, error) {
|
|
if req == nil {
|
|
return nil, status.Error(codes.InvalidArgument, "empty request")
|
|
}
|
|
if req.CodeId == 0 {
|
|
return nil, errorsmod.Wrap(types.ErrInvalid, "code id")
|
|
}
|
|
paginationParams, err := ensurePaginationParams(req.Pagination)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
ctx := sdk.UnwrapSDKContext(c)
|
|
r := make([]string, 0)
|
|
|
|
prefixStore := prefix.NewStore(runtime.KVStoreAdapter(q.storeService.OpenKVStore(ctx)), types.GetContractByCodeIDSecondaryIndexPrefix(req.CodeId))
|
|
pageRes, err := query.FilteredPaginate(prefixStore, paginationParams, func(key, value []byte, accumulate bool) (bool, error) {
|
|
if accumulate {
|
|
var contractAddr sdk.AccAddress = key[types.AbsoluteTxPositionLen:]
|
|
r = append(r, contractAddr.String())
|
|
}
|
|
return true, nil
|
|
})
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
return &types.QueryContractsByCodeResponse{
|
|
Contracts: r,
|
|
Pagination: pageRes,
|
|
}, nil
|
|
}
|
|
|
|
func (q GrpcQuerier) AllContractState(c context.Context, req *types.QueryAllContractStateRequest) (*types.QueryAllContractStateResponse, error) {
|
|
if req == nil {
|
|
return nil, status.Error(codes.InvalidArgument, "empty request")
|
|
}
|
|
|
|
contractAddr, err := sdk.AccAddressFromBech32(req.Address)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
paginationParams, err := ensurePaginationParams(req.Pagination)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
ctx := sdk.UnwrapSDKContext(c)
|
|
if !q.keeper.HasContractInfo(ctx, contractAddr) {
|
|
return nil, types.ErrNoSuchContractFn(contractAddr.String()).
|
|
Wrapf("address %s", contractAddr.String())
|
|
}
|
|
|
|
r := make([]types.Model, 0)
|
|
prefixStore := prefix.NewStore(runtime.KVStoreAdapter(q.storeService.OpenKVStore(ctx)), types.GetContractStorePrefix(contractAddr))
|
|
pageRes, err := query.FilteredPaginate(prefixStore, paginationParams, func(key, value []byte, accumulate bool) (bool, error) {
|
|
if accumulate {
|
|
r = append(r, types.Model{
|
|
Key: key,
|
|
Value: value,
|
|
})
|
|
}
|
|
return true, nil
|
|
})
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
return &types.QueryAllContractStateResponse{
|
|
Models: r,
|
|
Pagination: pageRes,
|
|
}, nil
|
|
}
|
|
|
|
func (q GrpcQuerier) RawContractState(c context.Context, req *types.QueryRawContractStateRequest) (*types.QueryRawContractStateResponse, error) {
|
|
if req == nil {
|
|
return nil, status.Error(codes.InvalidArgument, "empty request")
|
|
}
|
|
ctx := sdk.UnwrapSDKContext(c)
|
|
|
|
contractAddr, err := sdk.AccAddressFromBech32(req.Address)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
if !q.keeper.HasContractInfo(ctx, contractAddr) {
|
|
return nil, types.ErrNoSuchContractFn(contractAddr.String()).
|
|
Wrapf("address %s", contractAddr.String())
|
|
}
|
|
rsp := q.keeper.QueryRaw(ctx, contractAddr, req.QueryData)
|
|
return &types.QueryRawContractStateResponse{Data: rsp}, nil
|
|
}
|
|
|
|
func (q GrpcQuerier) SmartContractState(c context.Context, req *types.QuerySmartContractStateRequest) (rsp *types.QuerySmartContractStateResponse, err error) {
|
|
if req == nil {
|
|
return nil, status.Error(codes.InvalidArgument, "empty request")
|
|
}
|
|
if err := req.QueryData.ValidateBasic(); err != nil {
|
|
return nil, status.Error(codes.InvalidArgument, "invalid query data")
|
|
}
|
|
contractAddr, err := sdk.AccAddressFromBech32(req.Address)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
// limit the gas to the queryGasLimit or the remaining gas, whichever is smaller
|
|
ctx := sdk.UnwrapSDKContext(c)
|
|
gasLimit := min(ctx.GasMeter().GasRemaining(), q.queryGasLimit)
|
|
ctx = ctx.WithGasMeter(storetypes.NewGasMeter(gasLimit))
|
|
// recover from out-of-gas panic
|
|
defer func() {
|
|
if r := recover(); r != nil {
|
|
switch rType := r.(type) {
|
|
case storetypes.ErrorOutOfGas:
|
|
err = errorsmod.Wrapf(sdkerrors.ErrOutOfGas,
|
|
"out of gas in location: %v; gasWanted: %d, gasUsed: %d",
|
|
rType.Descriptor, ctx.GasMeter().Limit(), ctx.GasMeter().GasConsumed(),
|
|
)
|
|
default:
|
|
err = sdkerrors.ErrPanic
|
|
}
|
|
rsp = nil
|
|
moduleLogger(ctx).
|
|
Debug("smart query contract",
|
|
"error", "recovering panic",
|
|
"contract-address", req.Address,
|
|
"stacktrace", string(debug.Stack()))
|
|
}
|
|
}()
|
|
|
|
bz, err := q.keeper.QuerySmart(ctx, contractAddr, req.QueryData)
|
|
switch {
|
|
case err != nil:
|
|
return nil, err
|
|
case bz == nil:
|
|
return nil, types.ErrNoSuchContractFn(contractAddr.String()).
|
|
Wrapf("address %s", contractAddr.String())
|
|
}
|
|
return &types.QuerySmartContractStateResponse{Data: bz}, nil
|
|
}
|
|
|
|
func (q GrpcQuerier) Code(c context.Context, req *types.QueryCodeRequest) (*types.QueryCodeResponse, error) {
|
|
if req == nil {
|
|
return nil, status.Error(codes.InvalidArgument, "empty request")
|
|
}
|
|
if req.CodeId == 0 {
|
|
return nil, errorsmod.Wrap(types.ErrInvalid, "code id")
|
|
}
|
|
rsp, err := queryCode(sdk.UnwrapSDKContext(c), req.CodeId, q.keeper)
|
|
switch {
|
|
case err != nil:
|
|
return nil, err
|
|
case rsp == nil:
|
|
return nil, types.ErrNoSuchCodeFn(req.CodeId).Wrapf("code id %d", req.CodeId)
|
|
}
|
|
return &types.QueryCodeResponse{
|
|
CodeInfoResponse: rsp.CodeInfoResponse,
|
|
Data: rsp.Data,
|
|
}, nil
|
|
}
|
|
|
|
func (q GrpcQuerier) Codes(c context.Context, req *types.QueryCodesRequest) (*types.QueryCodesResponse, error) {
|
|
if req == nil {
|
|
return nil, status.Error(codes.InvalidArgument, "empty request")
|
|
}
|
|
paginationParams, err := ensurePaginationParams(req.Pagination)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
ctx := sdk.UnwrapSDKContext(c)
|
|
r := make([]types.CodeInfoResponse, 0)
|
|
prefixStore := prefix.NewStore(runtime.KVStoreAdapter(q.storeService.OpenKVStore(ctx)), types.CodeKeyPrefix)
|
|
pageRes, err := query.FilteredPaginate(prefixStore, paginationParams, func(key, value []byte, accumulate bool) (bool, error) {
|
|
if accumulate {
|
|
var c types.CodeInfo
|
|
if err := q.cdc.Unmarshal(value, &c); err != nil {
|
|
return false, err
|
|
}
|
|
r = append(r, types.CodeInfoResponse{
|
|
CodeID: binary.BigEndian.Uint64(key),
|
|
Creator: c.Creator,
|
|
DataHash: c.CodeHash,
|
|
InstantiatePermission: c.InstantiateConfig,
|
|
})
|
|
}
|
|
return true, nil
|
|
})
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
return &types.QueryCodesResponse{CodeInfos: r, Pagination: pageRes}, nil
|
|
}
|
|
|
|
func (q GrpcQuerier) CodeInfo(c context.Context, req *types.QueryCodeInfoRequest) (*types.QueryCodeInfoResponse, error) {
|
|
if req == nil {
|
|
return nil, status.Error(codes.InvalidArgument, "empty request")
|
|
}
|
|
if req.CodeId == 0 {
|
|
return nil, errorsmod.Wrap(types.ErrInvalid, "code id")
|
|
}
|
|
info := queryCodeInfo(sdk.UnwrapSDKContext(c), req.CodeId, q.keeper)
|
|
if info == nil {
|
|
return nil, types.ErrNoSuchCodeFn(req.CodeId).Wrapf("code id %d", req.CodeId)
|
|
}
|
|
return &types.QueryCodeInfoResponse{
|
|
CodeID: info.CodeID,
|
|
Creator: info.Creator,
|
|
Checksum: info.DataHash,
|
|
InstantiatePermission: info.InstantiatePermission,
|
|
}, nil
|
|
}
|
|
|
|
func queryContractInfo(ctx sdk.Context, addr sdk.AccAddress, keeper types.ViewKeeper) (*types.QueryContractInfoResponse, error) {
|
|
info := keeper.GetContractInfo(ctx, addr)
|
|
if info == nil {
|
|
return nil, types.ErrNoSuchContractFn(addr.String()).
|
|
Wrapf("address %s", addr.String())
|
|
}
|
|
return &types.QueryContractInfoResponse{
|
|
Address: addr.String(),
|
|
ContractInfo: *info,
|
|
}, nil
|
|
}
|
|
|
|
func queryCode(ctx sdk.Context, codeID uint64, keeper types.ViewKeeper) (*types.QueryCodeResponse, error) {
|
|
info := queryCodeInfo(ctx, codeID, keeper)
|
|
if info == nil {
|
|
// nil, nil leads to 404 in rest handler
|
|
return nil, nil
|
|
}
|
|
|
|
code, err := keeper.GetByteCode(ctx, codeID)
|
|
if err != nil {
|
|
return nil, errorsmod.Wrap(err, "loading wasm code")
|
|
}
|
|
|
|
return &types.QueryCodeResponse{CodeInfoResponse: info, Data: code}, nil
|
|
}
|
|
|
|
func queryCodeInfo(ctx sdk.Context, codeID uint64, keeper types.ViewKeeper) *types.CodeInfoResponse {
|
|
if codeID == 0 {
|
|
return nil
|
|
}
|
|
res := keeper.GetCodeInfo(ctx, codeID)
|
|
if res == nil {
|
|
return nil
|
|
}
|
|
info := types.CodeInfoResponse{
|
|
CodeID: codeID,
|
|
Creator: res.Creator,
|
|
DataHash: res.CodeHash,
|
|
InstantiatePermission: res.InstantiateConfig,
|
|
}
|
|
return &info
|
|
}
|
|
|
|
func (q GrpcQuerier) PinnedCodes(c context.Context, req *types.QueryPinnedCodesRequest) (*types.QueryPinnedCodesResponse, error) {
|
|
if req == nil {
|
|
return nil, status.Error(codes.InvalidArgument, "empty request")
|
|
}
|
|
paginationParams, err := ensurePaginationParams(req.Pagination)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
ctx := sdk.UnwrapSDKContext(c)
|
|
r := make([]uint64, 0)
|
|
|
|
prefixStore := prefix.NewStore(runtime.KVStoreAdapter(q.storeService.OpenKVStore(ctx)), types.PinnedCodeIndexPrefix)
|
|
pageRes, err := query.FilteredPaginate(prefixStore, paginationParams, func(key, _ []byte, accumulate bool) (bool, error) {
|
|
if accumulate {
|
|
r = append(r, sdk.BigEndianToUint64(key))
|
|
}
|
|
return true, nil
|
|
})
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
return &types.QueryPinnedCodesResponse{
|
|
CodeIDs: r,
|
|
Pagination: pageRes,
|
|
}, nil
|
|
}
|
|
|
|
// Params returns params of the module.
|
|
func (q GrpcQuerier) Params(c context.Context, _ *types.QueryParamsRequest) (*types.QueryParamsResponse, error) {
|
|
ctx := sdk.UnwrapSDKContext(c)
|
|
params := q.keeper.GetParams(ctx)
|
|
return &types.QueryParamsResponse{Params: params}, nil
|
|
}
|
|
|
|
func (q GrpcQuerier) ContractsByCreator(c context.Context, req *types.QueryContractsByCreatorRequest) (*types.QueryContractsByCreatorResponse, error) {
|
|
if req == nil {
|
|
return nil, status.Error(codes.InvalidArgument, "empty request")
|
|
}
|
|
paginationParams, err := ensurePaginationParams(req.Pagination)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
ctx := sdk.UnwrapSDKContext(c)
|
|
contracts := make([]string, 0)
|
|
|
|
creatorAddress, err := sdk.AccAddressFromBech32(req.CreatorAddress)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
prefixStore := prefix.NewStore(runtime.KVStoreAdapter(q.storeService.OpenKVStore(ctx)), types.GetContractsByCreatorPrefix(creatorAddress))
|
|
pageRes, err := query.FilteredPaginate(prefixStore, paginationParams, func(key, _ []byte, accumulate bool) (bool, error) {
|
|
if accumulate {
|
|
accAddress := sdk.AccAddress(key[types.AbsoluteTxPositionLen:])
|
|
contracts = append(contracts, accAddress.String())
|
|
}
|
|
return true, nil
|
|
})
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
return &types.QueryContractsByCreatorResponse{
|
|
ContractAddresses: contracts,
|
|
Pagination: pageRes,
|
|
}, nil
|
|
}
|
|
|
|
// max limit to pagination queries
|
|
const maxResultEntries = 100
|
|
|
|
var errLegacyPaginationUnsupported = status.Error(codes.InvalidArgument, "offset and count queries not supported")
|
|
|
|
// ensure that pagination is done via key iterator with reasonable limit
|
|
func ensurePaginationParams(req *query.PageRequest) (*query.PageRequest, error) {
|
|
if req == nil {
|
|
return &query.PageRequest{
|
|
Key: nil,
|
|
Limit: query.DefaultLimit,
|
|
}, nil
|
|
}
|
|
if req.Offset != 0 || req.CountTotal {
|
|
return nil, errLegacyPaginationUnsupported
|
|
}
|
|
if req.Limit > maxResultEntries || req.Limit <= 0 {
|
|
req.Limit = maxResultEntries
|
|
}
|
|
return req, nil
|
|
}
|
|
|
|
func (q GrpcQuerier) WasmLimitsConfig(c context.Context, req *types.QueryWasmLimitsConfigRequest) (*types.QueryWasmLimitsConfigResponse, error) {
|
|
json, err := json.Marshal(q.keeper.GetWasmLimits())
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
return &types.QueryWasmLimitsConfigResponse{
|
|
Config: string(json),
|
|
}, nil
|
|
}
|
|
|
|
func (q GrpcQuerier) BuildAddress(c context.Context, req *types.QueryBuildAddressRequest) (*types.QueryBuildAddressResponse, error) {
|
|
ctx := sdk.UnwrapSDKContext(c)
|
|
defer ctx.GasMeter().ConsumeGas(DefaultGasCostBuildAddress, "build address")
|
|
return BuildAddressPredictable(req)
|
|
}
|
|
|
|
func BuildAddressPredictable(req *types.QueryBuildAddressRequest) (*types.QueryBuildAddressResponse, error) {
|
|
if req == nil {
|
|
return nil, status.Error(codes.InvalidArgument, "empty request")
|
|
}
|
|
codeHash, err := hex.DecodeString(req.CodeHash)
|
|
if err != nil {
|
|
return nil, fmt.Errorf("invalid code hash: %w", err)
|
|
}
|
|
creator, err := sdk.AccAddressFromBech32(req.CreatorAddress)
|
|
if err != nil {
|
|
return nil, fmt.Errorf("invalid creator address: %w", err)
|
|
}
|
|
salt, err := hex.DecodeString(req.Salt)
|
|
if err != nil {
|
|
return nil, fmt.Errorf("invalid salt: %w", err)
|
|
}
|
|
if len(salt) == 0 {
|
|
return nil, status.Error(codes.InvalidArgument, "empty salt")
|
|
}
|
|
|
|
if req.InitArgs == nil {
|
|
return &types.QueryBuildAddressResponse{
|
|
Address: BuildContractAddressPredictable(codeHash, creator, salt, []byte{}).String(),
|
|
}, nil
|
|
}
|
|
initMsg := types.RawContractMessage(req.InitArgs)
|
|
if err := initMsg.ValidateBasic(); err != nil {
|
|
return nil, err
|
|
}
|
|
return &types.QueryBuildAddressResponse{
|
|
Address: BuildContractAddressPredictable(codeHash, creator, salt, initMsg).String(),
|
|
}, nil
|
|
}
|