215 lines
5.6 KiB
Go
215 lines
5.6 KiB
Go
package keeper
|
|
|
|
import (
|
|
"context"
|
|
|
|
"github.com/hashicorp/go-metrics"
|
|
|
|
errorsmod "cosmossdk.io/errors"
|
|
"cosmossdk.io/x/bank/types"
|
|
|
|
"github.com/cosmos/cosmos-sdk/telemetry"
|
|
sdk "github.com/cosmos/cosmos-sdk/types"
|
|
sdkerrors "github.com/cosmos/cosmos-sdk/types/errors"
|
|
)
|
|
|
|
type msgServer struct {
|
|
Keeper
|
|
}
|
|
|
|
var _ types.MsgServer = msgServer{}
|
|
|
|
// NewMsgServerImpl returns an implementation of the bank MsgServer interface
|
|
// for the provided Keeper.
|
|
func NewMsgServerImpl(keeper Keeper) types.MsgServer {
|
|
return &msgServer{Keeper: keeper}
|
|
}
|
|
|
|
func (k msgServer) Send(ctx context.Context, msg *types.MsgSend) (*types.MsgSendResponse, error) {
|
|
var (
|
|
from, to []byte
|
|
err error
|
|
)
|
|
|
|
if base, ok := k.Keeper.(BaseKeeper); ok {
|
|
from, err = base.ak.AddressCodec().StringToBytes(msg.FromAddress)
|
|
if err != nil {
|
|
return nil, sdkerrors.ErrInvalidAddress.Wrapf("invalid from address: %s", err)
|
|
}
|
|
to, err = base.ak.AddressCodec().StringToBytes(msg.ToAddress)
|
|
if err != nil {
|
|
return nil, sdkerrors.ErrInvalidAddress.Wrapf("invalid to address: %s", err)
|
|
}
|
|
} else {
|
|
return nil, sdkerrors.ErrInvalidRequest.Wrapf("invalid keeper type: %T", k.Keeper)
|
|
}
|
|
|
|
if !msg.Amount.IsValid() {
|
|
return nil, errorsmod.Wrap(sdkerrors.ErrInvalidCoins, msg.Amount.String())
|
|
}
|
|
|
|
if !msg.Amount.IsAllPositive() {
|
|
return nil, errorsmod.Wrap(sdkerrors.ErrInvalidCoins, msg.Amount.String())
|
|
}
|
|
|
|
if err := k.IsSendEnabledCoins(ctx, msg.Amount...); err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
if k.BlockedAddr(to) {
|
|
return nil, errorsmod.Wrapf(sdkerrors.ErrUnauthorized, "%s is not allowed to receive funds", msg.ToAddress)
|
|
}
|
|
|
|
err = k.SendCoins(ctx, from, to, msg.Amount)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
defer func() {
|
|
for _, a := range msg.Amount {
|
|
if a.Amount.IsInt64() {
|
|
telemetry.SetGaugeWithLabels(
|
|
[]string{"tx", "msg", "send"},
|
|
float32(a.Amount.Int64()),
|
|
[]metrics.Label{telemetry.NewLabel("denom", a.Denom)},
|
|
)
|
|
}
|
|
}
|
|
}()
|
|
|
|
return &types.MsgSendResponse{}, nil
|
|
}
|
|
|
|
func (k msgServer) MultiSend(ctx context.Context, msg *types.MsgMultiSend) (*types.MsgMultiSendResponse, error) {
|
|
if len(msg.Inputs) == 0 {
|
|
return nil, types.ErrNoInputs
|
|
}
|
|
|
|
if len(msg.Inputs) != 1 {
|
|
return nil, types.ErrMultipleSenders
|
|
}
|
|
|
|
if len(msg.Outputs) == 0 {
|
|
return nil, types.ErrNoOutputs
|
|
}
|
|
|
|
if err := types.ValidateInputOutputs(msg.Inputs[0], msg.Outputs); err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
// NOTE: totalIn == totalOut should already have been checked
|
|
for _, in := range msg.Inputs {
|
|
if err := k.IsSendEnabledCoins(ctx, in.Coins...); err != nil {
|
|
return nil, err
|
|
}
|
|
}
|
|
|
|
for _, out := range msg.Outputs {
|
|
if base, ok := k.Keeper.(BaseKeeper); ok {
|
|
accAddr, err := base.ak.AddressCodec().StringToBytes(out.Address)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
if k.BlockedAddr(accAddr) {
|
|
return nil, errorsmod.Wrapf(sdkerrors.ErrUnauthorized, "%s is not allowed to receive funds", out.Address)
|
|
}
|
|
} else {
|
|
return nil, sdkerrors.ErrInvalidRequest.Wrapf("invalid keeper type: %T", k.Keeper)
|
|
}
|
|
}
|
|
|
|
err := k.InputOutputCoins(ctx, msg.Inputs[0], msg.Outputs)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
return &types.MsgMultiSendResponse{}, nil
|
|
}
|
|
|
|
func (k msgServer) UpdateParams(ctx context.Context, req *types.MsgUpdateParams) (*types.MsgUpdateParamsResponse, error) {
|
|
if k.GetAuthority() != req.Authority {
|
|
return nil, errorsmod.Wrapf(types.ErrInvalidSigner, "invalid authority; expected %s, got %s", k.GetAuthority(), req.Authority)
|
|
}
|
|
|
|
if err := req.Params.Validate(); err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
if err := k.SetParams(ctx, req.Params); err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
return &types.MsgUpdateParamsResponse{}, nil
|
|
}
|
|
|
|
func (k msgServer) SetSendEnabled(ctx context.Context, msg *types.MsgSetSendEnabled) (*types.MsgSetSendEnabledResponse, error) {
|
|
if k.GetAuthority() != msg.Authority {
|
|
return nil, errorsmod.Wrapf(types.ErrInvalidSigner, "invalid authority; expected %s, got %s", k.GetAuthority(), msg.Authority)
|
|
}
|
|
|
|
seen := map[string]bool{}
|
|
for _, se := range msg.SendEnabled {
|
|
if _, alreadySeen := seen[se.Denom]; alreadySeen {
|
|
return nil, sdkerrors.ErrInvalidRequest.Wrapf("duplicate denom entries found for %q", se.Denom)
|
|
}
|
|
|
|
seen[se.Denom] = true
|
|
|
|
if err := se.Validate(); err != nil {
|
|
return nil, sdkerrors.ErrInvalidRequest.Wrapf("invalid SendEnabled denom %q: %s", se.Denom, err)
|
|
}
|
|
}
|
|
|
|
for _, denom := range msg.UseDefaultFor {
|
|
if err := sdk.ValidateDenom(denom); err != nil {
|
|
return nil, sdkerrors.ErrInvalidRequest.Wrapf("invalid UseDefaultFor denom %q: %s", denom, err)
|
|
}
|
|
}
|
|
|
|
if len(msg.SendEnabled) > 0 {
|
|
k.SetAllSendEnabled(ctx, msg.SendEnabled)
|
|
}
|
|
if len(msg.UseDefaultFor) > 0 {
|
|
k.DeleteSendEnabled(ctx, msg.UseDefaultFor...)
|
|
}
|
|
|
|
return &types.MsgSetSendEnabledResponse{}, nil
|
|
}
|
|
|
|
func (k msgServer) Burn(ctx context.Context, msg *types.MsgBurn) (*types.MsgBurnResponse, error) {
|
|
var (
|
|
from []byte
|
|
err error
|
|
)
|
|
|
|
var coins sdk.Coins
|
|
for _, coin := range msg.Amount {
|
|
coins = coins.Add(sdk.NewCoin(coin.Denom, coin.Amount))
|
|
}
|
|
|
|
if base, ok := k.Keeper.(BaseKeeper); ok {
|
|
from, err = base.ak.AddressCodec().StringToBytes(msg.FromAddress)
|
|
if err != nil {
|
|
return nil, sdkerrors.ErrInvalidAddress.Wrapf("invalid from address: %s", err)
|
|
}
|
|
} else {
|
|
return nil, sdkerrors.ErrInvalidRequest.Wrapf("invalid keeper type: %T", k.Keeper)
|
|
}
|
|
|
|
if !coins.IsValid() {
|
|
return nil, errorsmod.Wrap(sdkerrors.ErrInvalidCoins, coins.String())
|
|
}
|
|
|
|
if !coins.IsAllPositive() {
|
|
return nil, errorsmod.Wrap(sdkerrors.ErrInvalidCoins, coins.String())
|
|
}
|
|
|
|
err = k.BurnCoins(ctx, from, coins)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
return &types.MsgBurnResponse{}, nil
|
|
}
|