From 9dfec55540e80d36ba3780584d259bb7b11be1d8 Mon Sep 17 00:00:00 2001 From: Dzmitry Hil Date: Thu, 21 Nov 2024 14:15:01 +0300 Subject: [PATCH 1/5] Integrate asset FT extensions with the DEX. --- build/coreum/build.go | 2 +- build/coreum/generate-proto-breaking.go | 3 - .../modules/assetft_extension_test.go | 202 +++++++++++++- x/asset/ft/keeper/before_send.go | 16 +- x/asset/ft/keeper/keeper.go | 27 +- x/asset/ft/keeper/keeper_dex.go | 193 +++++++++----- x/asset/ft/keeper/keeper_dex_test.go | 249 ++++++++++++++---- .../asset-extension/src/contract.rs | 41 ++- .../asset-extension/src/error.rs | 3 + .../test-contracts/asset-extension/src/msg.rs | 21 +- x/dex/keeper/keeper_ft_test.go | 120 +++++++-- x/dex/keeper/keeper_matching.go | 5 +- x/dex/keeper/keeper_matching_fuzz_test.go | 9 +- x/dex/types/expected_keepers.go | 2 +- x/wbank/keeper/keeper.go | 38 +-- x/wbank/types/expected_keepers.go | 2 +- 16 files changed, 719 insertions(+), 214 deletions(-) diff --git a/build/coreum/build.go b/build/coreum/build.go index 019531ae9..0812ca0e5 100644 --- a/build/coreum/build.go +++ b/build/coreum/build.go @@ -195,7 +195,7 @@ func Lint(ctx context.Context, deps types.DepsFunc) error { CompileAllSmartContracts, formatProto, lintProto, - // breakingProto, // FIXME: uncomment in next PR + breakingProto, ) return golang.Lint(ctx, deps) } diff --git a/build/coreum/generate-proto-breaking.go b/build/coreum/generate-proto-breaking.go index 7a724c59e..6df51dae3 100644 --- a/build/coreum/generate-proto-breaking.go +++ b/build/coreum/generate-proto-breaking.go @@ -1,4 +1,3 @@ -//nolint:unused package coreum import ( @@ -22,10 +21,8 @@ import ( ) //go:embed proto-breaking.tmpl.json -//nolint:unused // FIXME: uncomment in next PR var configBreakingTmpl string -//nolint:deadcode func breakingProto(ctx context.Context, deps types.DepsFunc) error { deps(golang.Tidy, tools.EnsureProtoc, tools.EnsureProtocGenBufBreaking) diff --git a/integration-tests/modules/assetft_extension_test.go b/integration-tests/modules/assetft_extension_test.go index bfb0e2c8a..769d57700 100644 --- a/integration-tests/modules/assetft_extension_test.go +++ b/integration-tests/modules/assetft_extension_test.go @@ -24,6 +24,7 @@ import ( "github.com/CoreumFoundation/coreum/v5/testutil/integration" testcontracts "github.com/CoreumFoundation/coreum/v5/x/asset/ft/keeper/test-contracts" assetfttypes "github.com/CoreumFoundation/coreum/v5/x/asset/ft/types" + dextypes "github.com/CoreumFoundation/coreum/v5/x/dex/types" ) const ( @@ -49,7 +50,7 @@ func TestAssetFTExtensionIssue(t *testing.T) { issuer := chain.GenAccount() chain.FundAccountWithOptions(ctx, t, issuer, integration.BalancesOptions{ Amount: chain.QueryAssetFTParams(ctx, t).IssueFee.Amount. - Add(sdkmath.NewInt(1_000_000)). // one million added for uploading wasm code + Add(sdkmath.NewInt(500_000)). Add(sdkmath.NewInt(3 * 500_000)), // give 500k gas for each message since extensions are nondeterministic }) @@ -205,7 +206,7 @@ func TestAssetFTExtensionWhitelist(t *testing.T) { &assetfttypes.MsgSetWhitelistedLimit{}, }, Amount: chain.QueryAssetFTParams(ctx, t).IssueFee.Amount.Mul(sdkmath.NewInt(2)). - Add(sdkmath.NewInt(1_000_000)). // added 1 million for smart contract upload + Add(sdkmath.NewInt(500_000)). // added 1 million for smart contract upload Add(sdkmath.NewInt(10 * 500_000)), // give 500k gas for each message since extensions are nondeterministic }) chain.FundAccountWithOptions(ctx, t, nonIssuer, integration.BalancesOptions{ @@ -463,7 +464,7 @@ func TestAssetFTExtensionFreeze(t *testing.T) { &assetfttypes.MsgUnfreeze{}, }, Amount: chain.QueryAssetFTParams(ctx, t).IssueFee.Amount. - Add(sdkmath.NewInt(1_000_000)). // added 1 million for smart contract upload + Add(sdkmath.NewInt(500_000)). // added 1 million for smart contract upload Add(sdkmath.NewInt(2 * 500_000)), // add 500k for each message with extenstion transfer }) chain.FundAccountWithOptions(ctx, t, recipient, integration.BalancesOptions{ @@ -616,7 +617,7 @@ func TestAssetFTExtensionBurn(t *testing.T) { chain.FundAccountWithOptions(ctx, t, issuer, integration.BalancesOptions{ Messages: []sdk.Msg{}, Amount: chain.QueryAssetFTParams(ctx, t).IssueFee.Amount.MulRaw(2). - Add(sdkmath.NewInt(1_000_000)). // added 1 million for smart contract upload + Add(sdkmath.NewInt(500_000)). // added 1 million for smart contract upload Add(sdkmath.NewInt(6 * 500_000)), // add 500k for each message with extenstion transfer }) @@ -803,7 +804,7 @@ func TestAssetFTExtensionMint(t *testing.T) { chain.FundAccountWithOptions(ctx, t, issuer, integration.BalancesOptions{ Amount: chain.QueryAssetFTParams(ctx, t).IssueFee.Amount.MulRaw(2). - Add(sdkmath.NewInt(1_000_000)). // added 1 million for smart contract upload + Add(sdkmath.NewInt(500_000)). // added 1 million for smart contract upload Add(sdkmath.NewInt(7 * 500_000)), // add 500k for each message with extenstion transfer }) @@ -812,7 +813,7 @@ func TestAssetFTExtensionMint(t *testing.T) { }) chain.FundAccountWithOptions(ctx, t, admin, integration.BalancesOptions{ - Amount: sdkmath.NewInt(1_000_000), + Amount: sdkmath.NewInt(500_000), }) codeID, err := chain.Wasm.DeployWASMContract( @@ -1010,7 +1011,7 @@ func TestAssetFTExtensionSendingToSmartContractIsDenied(t *testing.T) { chain.FundAccountWithOptions(ctx, t, issuer, integration.BalancesOptions{ Amount: chain.QueryAssetFTParams(ctx, t).IssueFee.Amount. - Add(sdkmath.NewInt(1_000_000)), + Add(sdkmath.NewInt(500_000)), }) clientCtx := chain.ClientContext @@ -1117,7 +1118,7 @@ func TestAssetFTExtensionAttachingToSmartContractCallIsDenied(t *testing.T) { requireT := require.New(t) chain.FundAccountWithOptions(ctx, t, issuer, integration.BalancesOptions{ Amount: chain.QueryAssetFTParams(ctx, t).IssueFee.Amount. - Add(sdkmath.NewInt(1_000_000)), + Add(sdkmath.NewInt(500_000)), }) codeID, err := chain.Wasm.DeployWASMContract( @@ -1198,7 +1199,7 @@ func TestAssetFTExtensionIssuingSmartContractIsAllowedToSendAndReceive(t *testin requireT := require.New(t) chain.FundAccountWithOptions(ctx, t, admin, integration.BalancesOptions{ Amount: chain.QueryAssetFTParams(ctx, t).IssueFee.Amount.MulRaw(2). - Add(sdkmath.NewInt(1_000_000)), + Add(sdkmath.NewInt(500_000)), }) chain.FundAccountWithOptions(ctx, t, recipient, integration.BalancesOptions{ Amount: sdkmath.NewInt(500_000), @@ -1312,7 +1313,7 @@ func TestAssetFTExtensionAttachingToSmartContractInstantiationIsDenied(t *testin requireT := require.New(t) chain.FundAccountWithOptions(ctx, t, issuer, integration.BalancesOptions{ Amount: chain.QueryAssetFTParams(ctx, t).IssueFee.Amount. - Add(sdkmath.NewInt(1_000_000)), + Add(sdkmath.NewInt(500_000)), }) codeID, err := chain.Wasm.DeployWASMContract( @@ -1389,7 +1390,7 @@ func TestAssetFTExtensionMintingAndSendingOnBehalfOfIssuingSmartContractIsPossib requireT := require.New(t) chain.FundAccountWithOptions(ctx, t, admin, integration.BalancesOptions{ Amount: chain.QueryAssetFTParams(ctx, t).IssueFee.Amount. - Add(sdkmath.NewInt(1_000_000)), + Add(sdkmath.NewInt(500_000)), }) codeID, err := chain.Wasm.DeployWASMContract( @@ -1530,7 +1531,7 @@ func TestAssetFTExtensionBurnRate(t *testing.T) { chain.FundAccountWithOptions(ctx, t, admin, integration.BalancesOptions{ Amount: chain.QueryAssetFTParams(ctx, t).IssueFee.Amount. - Add(sdkmath.NewInt(1_000_000)). // added 1 million for smart contract upload + Add(sdkmath.NewInt(500_000)). // added 1 million for smart contract upload Add(sdkmath.NewInt(2 * 500_000)), }) chain.FundAccountWithOptions(ctx, t, recipient1, integration.BalancesOptions{ @@ -1672,7 +1673,7 @@ func TestAssetFTExtensionSendCommissionRate(t *testing.T) { chain.FundAccountWithOptions(ctx, t, admin, integration.BalancesOptions{ Amount: chain.QueryAssetFTParams(ctx, t).IssueFee.Amount. - Add(sdkmath.NewInt(1_000_000)). // added 1 million for smart contract upload + Add(sdkmath.NewInt(500_000)). // added 1 million for smart contract upload Add(sdkmath.NewInt(2 * 500_000)), }) chain.FundAccountWithOptions(ctx, t, recipient1, integration.BalancesOptions{ @@ -1806,3 +1807,178 @@ func TestAssetFTExtensionSendCommissionRate(t *testing.T) { &extension: 30, // 25 + 5 (50% of the commission to the extension) }) } + +// TestAssetFTExtensionDEX checks extension works with the DEX. +func TestAssetFTExtensionDEX(t *testing.T) { + t.Parallel() + + ctx, chain := integrationtests.NewCoreumTestingContext(t) + + requireT := require.New(t) + + assetFTClint := assetfttypes.NewQueryClient(chain.ClientContext) + dexClient := dextypes.NewQueryClient(chain.ClientContext) + + dexParamsRes, err := dexClient.Params(ctx, &dextypes.QueryParamsRequest{}) + requireT.NoError(err) + dexReserver := dexParamsRes.Params.OrderReserve + + admin := chain.GenAccount() + acc1 := chain.GenAccount() + acc2 := chain.GenAccount() + + chain.FundAccountWithOptions(ctx, t, admin, integration.BalancesOptions{ + Amount: chain.QueryAssetFTParams(ctx, t).IssueFee.Amount. + AddRaw(500_000). // added 1 million for smart contract upload + AddRaw(2 * 500_000), + }) + chain.FundAccountWithOptions(ctx, t, acc1, integration.BalancesOptions{ + Amount: sdkmath.NewInt(500_000). // message + order reserve + Add(dexReserver.Amount), + }) + chain.FundAccountWithOptions(ctx, t, acc2, integration.BalancesOptions{ + Amount: sdkmath.NewInt(500_000). + AddRaw(100). + Add(dexReserver.Amount), // message + balance to place an order + order reserve + }) + + codeID, err := chain.Wasm.DeployWASMContract( + ctx, chain.TxFactory().WithSimulateAndExecute(true), admin, testcontracts.AssetExtensionWasm, + ) + requireT.NoError(err) + + issueMsg := &assetfttypes.MsgIssue{ + Issuer: admin.String(), + Symbol: "EXABC", + Subunit: "extabc", + Precision: 6, + InitialAmount: sdkmath.NewInt(1000), + Features: []assetfttypes.Feature{ + assetfttypes.Feature_extension, + }, + ExtensionSettings: &assetfttypes.ExtensionIssueSettings{ + CodeId: codeID, + Label: "testing-dex", + }, + } + + res, err := client.BroadcastTx( + ctx, + chain.ClientContext.WithFromAddress(admin), + chain.TxFactoryAuto(), + issueMsg, + ) + requireT.NoError(err) + tokenIssuedEvents, err := event.FindTypedEvents[*assetfttypes.EventIssued](res.Events) + requireT.NoError(err) + denomWithExtension := tokenIssuedEvents[0].Denom + + // send from admin to acc1 to place an order + sendMsg := &banktypes.MsgSend{ + FromAddress: admin.String(), + ToAddress: acc1.String(), + Amount: sdk.NewCoins(sdk.NewCoin(denomWithExtension, sdkmath.NewInt(400))), + } + _, err = client.BroadcastTx( + ctx, + chain.ClientContext.WithFromAddress(admin), + chain.TxFactoryAuto(), + sendMsg, + ) + requireT.NoError(err) + + placeSellOrderMsg := &dextypes.MsgPlaceOrder{ + Sender: acc1.String(), + Type: dextypes.ORDER_TYPE_LIMIT, + ID: "id1", + BaseDenom: denomWithExtension, + QuoteDenom: chain.ChainSettings.Denom, + Price: lo.ToPtr(dextypes.MustNewPriceFromString("1")), + Quantity: sdkmath.NewInt(104), // 104 is prohibited by extensions smart contract + Side: dextypes.SIDE_SELL, + TimeInForce: dextypes.TIME_IN_FORCE_GTC, + } + _, err = client.BroadcastTx( + ctx, + chain.ClientContext.WithFromAddress(acc1), + chain.TxFactoryAuto(), + placeSellOrderMsg, + ) + requireT.ErrorContains(err, "wasm error: DEX order placement is failed") + + // update to allowed + placeSellOrderMsg.Quantity = sdkmath.NewInt(100) + _, err = client.BroadcastTx( + ctx, + chain.ClientContext.WithFromAddress(acc1), + chain.TxFactoryAuto(), + placeSellOrderMsg, + ) + requireT.NoError(err) + + acc1BalanceRes, err := assetFTClint.Balance(ctx, &assetfttypes.QueryBalanceRequest{ + Account: acc1.String(), + Denom: denomWithExtension, + }) + requireT.NoError(err) + requireT.True(acc1BalanceRes.ExpectedToReceiveInDEX.IsZero()) + requireT.Equal(sdkmath.NewInt(100).String(), acc1BalanceRes.LockedInDEX.String()) + + // place buy order from acc2 + + acc2BalanceRes, err := assetFTClint.Balance(ctx, &assetfttypes.QueryBalanceRequest{ + Account: acc2.String(), + Denom: denomWithExtension, + }) + requireT.NoError(err) + // no coins of the denomWithExtension + requireT.True(acc2BalanceRes.Balance.IsZero()) + + placeBuyOrderMsg := &dextypes.MsgPlaceOrder{ + Sender: acc2.String(), + Type: dextypes.ORDER_TYPE_LIMIT, + ID: "id1", + BaseDenom: denomWithExtension, + QuoteDenom: chain.ChainSettings.Denom, + Price: lo.ToPtr(dextypes.MustNewPriceFromString("1")), + Quantity: sdkmath.NewInt(103), // 103 is prohibited by extensions smart contract + Side: dextypes.SIDE_BUY, + TimeInForce: dextypes.TIME_IN_FORCE_GTC, + } + _, err = client.BroadcastTx( + ctx, + chain.ClientContext.WithFromAddress(acc2), + chain.TxFactoryAuto(), + placeBuyOrderMsg, + ) + requireT.ErrorContains(err, "wasm error: DEX order placement is failed") + + // update to allowed + placeBuyOrderMsg.Quantity = sdkmath.NewInt(100) + _, err = client.BroadcastTx( + ctx, + chain.ClientContext.WithFromAddress(acc2), + chain.TxFactoryAuto(), + placeBuyOrderMsg, + ) + requireT.NoError(err) + + // both order are executed and closed + acc2BalanceRes, err = assetFTClint.Balance(ctx, &assetfttypes.QueryBalanceRequest{ + Account: acc2.String(), + Denom: denomWithExtension, + }) + requireT.NoError(err) + // bought expected quantity + requireT.Equal(sdkmath.NewInt(100).String(), acc2BalanceRes.Balance.String()) + requireT.True(acc2BalanceRes.LockedInDEX.IsZero()) + requireT.True(acc2BalanceRes.ExpectedToReceiveInDEX.IsZero()) + + acc1BalanceRes, err = assetFTClint.Balance(ctx, &assetfttypes.QueryBalanceRequest{ + Account: acc1.String(), + Denom: denomWithExtension, + }) + requireT.NoError(err) + requireT.True(acc1BalanceRes.LockedInDEX.IsZero()) + requireT.True(acc1BalanceRes.ExpectedToReceiveInDEX.IsZero()) +} diff --git a/x/asset/ft/keeper/before_send.go b/x/asset/ft/keeper/before_send.go index f2ec5bae8..67f5853f3 100644 --- a/x/asset/ft/keeper/before_send.go +++ b/x/asset/ft/keeper/before_send.go @@ -7,6 +7,7 @@ import ( sdkmath "cosmossdk.io/math" sdk "github.com/cosmos/cosmos-sdk/types" banktypes "github.com/cosmos/cosmos-sdk/x/bank/types" + "github.com/pkg/errors" "github.com/CoreumFoundation/coreum/v5/x/asset/ft/types" "github.com/CoreumFoundation/coreum/v5/x/wasm" @@ -14,12 +15,9 @@ import ( wibctransfertypes "github.com/CoreumFoundation/coreum/v5/x/wibctransfer/types" ) -// extension method calls. -const ( - // the function name of the extension smart contract, which will be invoked - // when doing the transfer. - ExtenstionTransferMethod = "extension_transfer" -) +// ExtensionTransferMethod the function name of the extension smart contract, which will be invoked +// when doing the transfer. +const ExtensionTransferMethod = "extension_transfer" // sudoExtensionTransferMsg contains the fields passed to extension method call. // @@ -201,7 +199,7 @@ func (k Keeper) invokeAssetExtension( wasm.IsSmartContract(ctx, recipient, k.wasmKeeper) contractMsg := map[string]interface{}{ - ExtenstionTransferMethod: sudoExtensionTransferMsg{ + ExtensionTransferMethod: sudoExtensionTransferMsg{ Sender: sender.String(), Recipient: recipient.String(), TransferAmount: sendAmount.Amount, @@ -216,7 +214,7 @@ func (k Keeper) invokeAssetExtension( } contractMsgBytes, err := json.Marshal(contractMsg) if err != nil { - return err + return errors.Wrapf(err, "failed to marshal contract msg") } _, err = k.wasmPermissionedKeeper.Sudo( @@ -225,7 +223,7 @@ func (k Keeper) invokeAssetExtension( contractMsgBytes, ) if err != nil { - return types.ErrExtensionCallFailed.Wrapf("was error: %s", err) + return types.ErrExtensionCallFailed.Wrapf("wasm error: %s", err) } return nil } diff --git a/x/asset/ft/keeper/keeper.go b/x/asset/ft/keeper/keeper.go index 4f37f7485..3f0d8ed8b 100644 --- a/x/asset/ft/keeper/keeper.go +++ b/x/asset/ft/keeper/keeper.go @@ -707,25 +707,36 @@ func (k Keeper) GetSpendableBalance( ctx sdk.Context, addr sdk.AccAddress, denom string, -) sdk.Coin { +) (sdk.Coin, error) { balance := k.bankKeeper.GetBalance(ctx, addr, denom) if balance.Amount.IsZero() { - return balance + return balance, nil } notLockedAmt := balance.Amount. Sub(k.GetDEXLockedBalance(ctx, addr, denom).Amount). Sub(k.bankKeeper.LockedCoins(ctx, addr).AmountOf(denom)) if notLockedAmt.IsNegative() { - return sdk.NewCoin(denom, sdkmath.ZeroInt()) + return sdk.NewCoin(denom, sdkmath.ZeroInt()), nil } - notFrozenAmt := balance.Amount.Sub(k.GetFrozenBalance(ctx, addr, denom).Amount) - if notFrozenAmt.IsNegative() { - return sdk.NewCoin(denom, sdkmath.ZeroInt()) + + def, err := k.getDefinitionOrNil(ctx, denom) + if err != nil { + return sdk.Coin{}, err + } + // the spendable balance counts the frozen balance, but if extensions are not enabled + if def != nil && + def.IsFeatureEnabled(types.Feature_freezing) && + !def.IsFeatureEnabled(types.Feature_extension) { + notFrozenAmt := balance.Amount.Sub(k.GetFrozenBalance(ctx, addr, denom).Amount) + if notFrozenAmt.IsNegative() { + return sdk.NewCoin(denom, sdkmath.ZeroInt()), nil + } + spendableAmount := sdkmath.MinInt(notLockedAmt, notFrozenAmt) + return sdk.NewCoin(denom, spendableAmount), nil } - spendableAmount := sdkmath.MinInt(notLockedAmt, notFrozenAmt) - return sdk.NewCoin(denom, spendableAmount) + return sdk.NewCoin(denom, notLockedAmt), nil } // TransferAdmin changes admin of a fungible token. diff --git a/x/asset/ft/keeper/keeper_dex.go b/x/asset/ft/keeper/keeper_dex.go index 73f6a0cf9..a181ab50c 100644 --- a/x/asset/ft/keeper/keeper_dex.go +++ b/x/asset/ft/keeper/keeper_dex.go @@ -1,6 +1,8 @@ package keeper import ( + "encoding/json" + sdkerrors "cosmossdk.io/errors" sdkmath "cosmossdk.io/math" "cosmossdk.io/store/prefix" @@ -14,6 +16,19 @@ import ( cwasmtypes "github.com/CoreumFoundation/coreum/v5/x/wasm/types" ) +// ExtensionPlaceOrderMethod is the function name of the extension smart contract, which will be invoked +// when place and DEX order. +const ExtensionPlaceOrderMethod = "extension_place_order" + +// sudoExtensionPlaceOrderMsg contains the fields passed to extension method call. +// +//nolint:tagliatelle // these will be exposed to rust and must be snake case. +type sudoExtensionPlaceOrderMsg struct { + Order types.DEXOrder `json:"order"` + ExpectedToSpend sdk.Coin `json:"expected_to_spend"` + ExpectedToReceive sdk.Coin `json:"expected_to_receive"` +} + // DEXExecuteActions executes a series of DEX actions which include checking order amounts, // adjusting locked balances, and updating expected to receive balances. It performs necessary // validations and updates the state accordingly based on the provided actions. @@ -87,26 +102,11 @@ func (k Keeper) DEXCheckOrderAmounts( order types.DEXOrder, expectedToSpend, expectedToReceive sdk.Coin, ) error { - spendDef, err := k.getDefinitionOrNil(ctx, expectedToSpend.Denom) - if err != nil { - return err - } - if err := k.dexChecksForDenom(ctx, order.Creator, spendDef, expectedToReceive.Denom); err != nil { - return err - } - if err := k.dexExpectedToSpendChecks(ctx, order.Creator, spendDef, expectedToSpend); err != nil { - return err - } - - receiveDef, err := k.getDefinitionOrNil(ctx, expectedToReceive.Denom) - if err != nil { - return err - } - if err := k.dexChecksForDenom(ctx, order.Creator, receiveDef, expectedToSpend.Denom); err != nil { + if err := k.dexCheckExpectedToSpend(ctx, order, expectedToSpend, expectedToReceive); err != nil { return err } - return k.dexExpectedToReceiveChecks(ctx, order.Creator, receiveDef, expectedToReceive) + return k.dexCheckExpectedToReceive(ctx, order, expectedToSpend, expectedToReceive) } // SetDEXSettings sets the DEX settings of a specified denom. @@ -384,20 +384,118 @@ func (k Keeper) ValidateDEXCancelOrdersByDenomIsAllowed(ctx sdk.Context, addr sd return nil } -func (k Keeper) dexExpectedToReceiveChecks( +func (k Keeper) dexCheckExpectedToSpend( ctx sdk.Context, - addr sdk.AccAddress, - def *types.Definition, - coin sdk.Coin, + order types.DEXOrder, + expectedToSpend, expectedToReceive sdk.Coin, +) error { + // validate that the order creator has enough balance, for both extension and non-extension coin + balance := k.bankKeeper.GetBalance(ctx, order.Creator, expectedToSpend.Denom) + if err := k.validateCoinIsNotLockedByDEXAndBank(ctx, order.Creator, balance, expectedToSpend); err != nil { + return sdkerrors.Wrapf(types.ErrDEXInsufficientSpendableBalance, "%s", err) + } + + spendDef, err := k.getDefinitionOrNil(ctx, expectedToSpend.Denom) + if err != nil { + return err + } + + if spendDef == nil { + return nil + } + + if spendDef.IsFeatureEnabled(types.Feature_extension) { + extensionContract, err := sdk.AccAddressFromBech32(spendDef.ExtensionCWAddress) + if err != nil { + return err + } + return k.dexCallExtensionPlaceOrder( + ctx, extensionContract, order, expectedToSpend, expectedToReceive, + ) + } + + if err := k.dexChecksForDenom(ctx, order.Creator, spendDef, expectedToReceive.Denom); err != nil { + return err + } + + if spendDef.IsFeatureEnabled(types.Feature_freezing) && !spendDef.HasAdminPrivileges(order.Creator) { + frozenAmt := k.GetFrozenBalance(ctx, order.Creator, expectedToSpend.Denom).Amount + notFrozenTotalAmt := balance.Amount.Sub(frozenAmt) + if notFrozenTotalAmt.LT(expectedToSpend.Amount) { + return sdkerrors.Wrapf( + types.ErrDEXInsufficientSpendableBalance, + "failed to DEX lock %s available %s%s", + expectedToSpend.String(), + notFrozenTotalAmt, + expectedToSpend.Denom, + ) + } + } + + return nil +} + +func (k Keeper) dexCheckExpectedToReceive( + ctx sdk.Context, + order types.DEXOrder, + expectedToSpend, expectedToReceive sdk.Coin, ) error { - if coin.IsZero() || def == nil { + receiveDef, err := k.getDefinitionOrNil(ctx, expectedToReceive.Denom) + if err != nil { + return err + } + if receiveDef == nil { return nil } - if def.IsFeatureEnabled(types.Feature_whitelisting) && !def.HasAdminPrivileges(addr) { - if err := k.validateWhitelistedBalance(ctx, addr, coin); err != nil { + if receiveDef.IsFeatureEnabled(types.Feature_extension) { + extensionContract, err := sdk.AccAddressFromBech32(receiveDef.ExtensionCWAddress) + if err != nil { return err } + return k.dexCallExtensionPlaceOrder( + ctx, extensionContract, order, expectedToSpend, expectedToReceive, + ) + } + + if err := k.dexChecksForDenom(ctx, order.Creator, receiveDef, expectedToSpend.Denom); err != nil { + return err + } + + if receiveDef.IsFeatureEnabled(types.Feature_whitelisting) && !receiveDef.HasAdminPrivileges(order.Creator) { + if err := k.validateWhitelistedBalance(ctx, order.Creator, expectedToReceive); err != nil { + return err + } + } + + return nil +} + +func (k Keeper) dexCallExtensionPlaceOrder( + ctx sdk.Context, + extensionContract sdk.AccAddress, + order types.DEXOrder, + expectedToSpend, expectedToReceive sdk.Coin, +) error { + contractMsg := map[string]interface{}{ + ExtensionPlaceOrderMethod: sudoExtensionPlaceOrderMsg{ + Order: order, + ExpectedToSpend: expectedToSpend, + ExpectedToReceive: expectedToReceive, + }, + } + contractMsgBytes, err := json.Marshal(contractMsg) + if err != nil { + return errors.Wrapf(err, "failed to marshal contract msg") + } + + _, err = k.wasmPermissionedKeeper.Sudo( + ctx, + extensionContract, + contractMsgBytes, + ) + if err != nil { + return types.ErrExtensionCallFailed.Wrapf("wasm error: %s", err) } return nil @@ -440,14 +538,6 @@ func (k Keeper) dexChecksForDenom( } func (k Keeper) dexChecksForDefinition(ctx sdk.Context, acc sdk.AccAddress, def types.Definition) error { - if def.ExtensionCWAddress != "" { - return sdkerrors.Wrapf( - types.ErrInvalidInput, - "usage of %s is not supported for DEX, the token has extensions", - def.Denom, - ) - } - if def.IsFeatureEnabled(types.Feature_dex_block) { return sdkerrors.Wrapf( cosmoserrors.ErrUnauthorized, @@ -482,43 +572,6 @@ func (k Keeper) dexChecksForDefinition(ctx sdk.Context, acc sdk.AccAddress, def return nil } -func (k Keeper) dexExpectedToSpendChecks( - ctx sdk.Context, - addr sdk.AccAddress, - def *types.Definition, - coin sdk.Coin, -) error { - if coin.IsZero() { - return nil - } - - // validate locked balance for any token - balance := k.bankKeeper.GetBalance(ctx, addr, coin.Denom) - if err := k.validateCoinIsNotLockedByDEXAndBank(ctx, addr, balance, coin); err != nil { - return sdkerrors.Wrapf(types.ErrDEXInsufficientSpendableBalance, "%s", err) - } - - if def == nil { - return nil - } - - if def.IsFeatureEnabled(types.Feature_freezing) && !def.HasAdminPrivileges(addr) { - frozenAmt := k.GetFrozenBalance(ctx, addr, coin.Denom).Amount - notFrozenTotalAmt := balance.Amount.Sub(frozenAmt) - if notFrozenTotalAmt.LT(coin.Amount) { - return sdkerrors.Wrapf( - types.ErrDEXInsufficientSpendableBalance, - "failed to DEX lock %s available %s%s", - coin.String(), - notFrozenTotalAmt, - coin.Denom, - ) - } - } - - return nil -} - func (k Keeper) updateDEXSettings( ctx sdk.Context, sender sdk.AccAddress, @@ -615,7 +668,7 @@ func (k Keeper) shouldRecordExpectedToReceiveBalance(ctx sdk.Context, denom stri return false, err } // increase for FT with the whitelisting enabled only - if def != nil && def.IsFeatureEnabled(types.Feature_whitelisting) { + if def != nil && (def.IsFeatureEnabled(types.Feature_whitelisting) || def.IsFeatureEnabled(types.Feature_extension)) { return true, nil } diff --git a/x/asset/ft/keeper/keeper_dex_test.go b/x/asset/ft/keeper/keeper_dex_test.go index 950a6353d..942497612 100644 --- a/x/asset/ft/keeper/keeper_dex_test.go +++ b/x/asset/ft/keeper/keeper_dex_test.go @@ -30,7 +30,10 @@ func TestKeeper_DEXExpectedToReceive(t *testing.T) { requireT := require.New(t) testApp := simapp.New() - ctx := testApp.BaseApp.NewContextLegacy(false, tmproto.Header{}) + ctx := testApp.BaseApp.NewContextLegacy(false, tmproto.Header{ + Time: time.Now(), + AppHash: []byte("some-hash"), + }) ftKeeper := testApp.AssetFTKeeper bankKeeper := testApp.BankKeeper @@ -39,20 +42,7 @@ func TestKeeper_DEXExpectedToReceive(t *testing.T) { sender := sdk.AccAddress(secp256k1.GenPrivKey().PubKey().Address()) recipient := sdk.AccAddress(secp256k1.GenPrivKey().PubKey().Address()) - settings := types.IssueSettings{ - Issuer: issuer, - Symbol: "DEF", - Subunit: "def", - Precision: 1, - Description: "DEF Desc", - InitialAmount: sdkmath.NewInt(666), - Features: []types.Feature{types.Feature_whitelisting}, - } - - denom, err := ftKeeper.Issue(ctx, settings) - requireT.NoError(err) - - unwhitelistableSettings := types.IssueSettings{ + noFeaturesIssuanceSettings := types.IssueSettings{ Issuer: issuer, Symbol: "ABC", Subunit: "abc", @@ -62,16 +52,16 @@ func TestKeeper_DEXExpectedToReceive(t *testing.T) { Features: []types.Feature{}, } - unwhitelistableDenom, err := ftKeeper.Issue(ctx, unwhitelistableSettings) + noFeaturesDenom, err := ftKeeper.Issue(ctx, noFeaturesIssuanceSettings) requireT.NoError(err) - _, err = ftKeeper.GetToken(ctx, unwhitelistableDenom) + _, err = ftKeeper.GetToken(ctx, noFeaturesDenom) requireT.NoError(err) // function passed but nothing is reserved requireT.NoError(ftKeeper.DEXIncreaseExpectedToReceive( - ctx, recipient, sdk.NewCoin(unwhitelistableDenom, sdkmath.NewInt(1)), + ctx, recipient, sdk.NewCoin(noFeaturesDenom, sdkmath.NewInt(1)), )) - requireT.True(ftKeeper.GetDEXExpectedToReceivedBalance(ctx, recipient, unwhitelistableDenom).IsZero()) + requireT.True(ftKeeper.GetDEXExpectedToReceivedBalance(ctx, recipient, noFeaturesDenom).IsZero()) // increase for not asset FT denom, passes but nothing is reserved notFTDenom := types.BuildDenom("nonexist", issuer) @@ -82,9 +72,22 @@ func TestKeeper_DEXExpectedToReceive(t *testing.T) { ftKeeper.GetDEXExpectedToReceivedBalance(ctx, recipient, "nonexist").IsZero(), ) + whitelistingIssuanceSettings := types.IssueSettings{ + Issuer: issuer, + Symbol: "DEF", + Subunit: "def", + Precision: 1, + Description: "DEF Desc", + InitialAmount: sdkmath.NewInt(666), + Features: []types.Feature{types.Feature_whitelisting}, + } + + whitelistingDenom, err := ftKeeper.Issue(ctx, whitelistingIssuanceSettings) + requireT.NoError(err) + // set whitelisted balance - coinToSend := sdk.NewCoin(denom, sdkmath.NewInt(100)) - // whitelist sender and fund + coinToSend := sdk.NewCoin(whitelistingDenom, sdkmath.NewInt(100)) + // whitelist sender and whitelistingDenom requireT.NoError(ftKeeper.SetWhitelistedBalance(ctx, issuer, sender, coinToSend)) requireT.NoError(bankKeeper.SendCoins(ctx, issuer, sender, sdk.NewCoins(coinToSend))) // send without the expected to received balance @@ -93,11 +96,11 @@ func TestKeeper_DEXExpectedToReceive(t *testing.T) { // return coin requireT.NoError(bankKeeper.SendCoins(ctx, recipient, sender, sdk.NewCoins(coinToSend))) // increase expected to received balance - coinToIncreaseExpectedToReceive := sdk.NewCoin(denom, sdkmath.NewInt(1)) + coinToIncreaseExpectedToReceive := sdk.NewCoin(whitelistingDenom, sdkmath.NewInt(1)) requireT.NoError(ftKeeper.DEXIncreaseExpectedToReceive(ctx, recipient, coinToIncreaseExpectedToReceive)) requireT.Equal( coinToIncreaseExpectedToReceive.String(), - ftKeeper.GetDEXExpectedToReceivedBalance(ctx, recipient, denom).String(), + ftKeeper.GetDEXExpectedToReceivedBalance(ctx, recipient, whitelistingDenom).String(), ) // try to send with the increased part requireT.ErrorIs( @@ -114,9 +117,39 @@ func TestKeeper_DEXExpectedToReceive(t *testing.T) { ) requireT.NoError(ftKeeper.DEXDecreaseExpectedToReceive(ctx, recipient, coinToIncreaseExpectedToReceive)) - requireT.True(ftKeeper.GetDEXExpectedToReceivedBalance(ctx, recipient, denom).IsZero()) + requireT.True(ftKeeper.GetDEXExpectedToReceivedBalance(ctx, recipient, whitelistingDenom).IsZero()) // send without decreased amount requireT.NoError(bankKeeper.SendCoins(ctx, sender, recipient, sdk.NewCoins(coinToSend))) + + // extension + codeID, _, err := testApp.WasmPermissionedKeeper.Create( + ctx, issuer, testcontracts.AssetExtensionWasm, &wasmtypes.AllowEverybody, + ) + requireT.NoError(err) + settingsWithExtension := types.IssueSettings{ + Issuer: issuer, + Symbol: "DEFEXT", + Subunit: "defext", + Precision: 6, + InitialAmount: sdkmath.NewIntWithDecimal(1, 10), + Features: []types.Feature{types.Feature_extension}, + ExtensionSettings: &types.ExtensionIssueSettings{ + CodeId: codeID, + }, + } + extensionDenom, err := ftKeeper.Issue(ctx, settingsWithExtension) + requireT.NoError(err) + coinToIncreaseExpectedToReceive = sdk.NewCoin(extensionDenom, sdkmath.NewInt(1)) + requireT.NoError(ftKeeper.DEXIncreaseExpectedToReceive(ctx, recipient, coinToIncreaseExpectedToReceive)) + requireT.Equal( + coinToIncreaseExpectedToReceive.String(), + ftKeeper.GetDEXExpectedToReceivedBalance(ctx, recipient, extensionDenom).String(), + ) + + requireT.NoError(ftKeeper.DEXDecreaseExpectedToReceive(ctx, recipient, coinToIncreaseExpectedToReceive)) + requireT.True( + ftKeeper.GetDEXExpectedToReceivedBalance(ctx, recipient, extensionDenom).IsZero(), + ) } func TestKeeper_DEXLocked(t *testing.T) { @@ -218,12 +251,16 @@ func TestKeeper_DEXLocked(t *testing.T) { // decrease locked part requireT.NoError(ftKeeper.DEXDecreaseLocked(ctx, acc, sdk.NewInt64Coin(denom, 400))) requireT.Equal(sdk.NewInt64Coin(denom, 600).String(), ftKeeper.GetDEXLockedBalance(ctx, acc, denom).String()) - requireT.Equal(sdk.NewInt64Coin(denom, 400).String(), ftKeeper.GetSpendableBalance(ctx, acc, denom).String()) + spendableBalance, err := ftKeeper.GetSpendableBalance(ctx, acc, denom) + requireT.NoError(err) + requireT.Equal(sdk.NewInt64Coin(denom, 400).String(), spendableBalance.String()) // freeze locked balance requireT.NoError(ftKeeper.Freeze(ctx, issuer, acc, coinToSend)) // 1050 - total, 600 locked by dex, 50 locked by bank, 1000 frozen - requireT.Equal(sdk.NewInt64Coin(denom, 50).String(), ftKeeper.GetSpendableBalance(ctx, acc, denom).String()) + spendableBalance, err = ftKeeper.GetSpendableBalance(ctx, acc, denom) + requireT.NoError(err) + requireT.Equal(sdk.NewInt64Coin(denom, 50).String(), spendableBalance.String()) // decrease locked 2d part, even when it's frozen we allow it requireT.NoError(ftKeeper.DEXDecreaseLocked(ctx, acc, sdk.NewInt64Coin(denom, 600))) @@ -271,7 +308,9 @@ func TestKeeper_DEXLocked(t *testing.T) { ), fmt.Sprintf("usage of %s for DEX is blocked because the token is globally frozen", denom), ) - requireT.True(ftKeeper.GetSpendableBalance(ctx, acc, denom).IsZero()) + spendableBalance, err = ftKeeper.GetSpendableBalance(ctx, acc, denom) + requireT.NoError(err) + requireT.True(spendableBalance.IsZero()) // globally unfreeze now and check that we can use the previously locked amount requireT.NoError(ftKeeper.GloballyUnfreeze(ctx, issuer, denom)) requireT.NoError( @@ -286,6 +325,8 @@ func TestKeeper_DEXLocked(t *testing.T) { // freeze more than balance requireT.NoError(ftKeeper.Freeze(ctx, issuer, acc, sdk.NewInt64Coin(denom, 1_000_000))) + // check freezing with extensions + // extension codeID, _, err := testApp.WasmPermissionedKeeper.Create( ctx, issuer, testcontracts.AssetExtensionWasm, &wasmtypes.AllowEverybody, @@ -297,25 +338,25 @@ func TestKeeper_DEXLocked(t *testing.T) { Subunit: "defext", Precision: 6, InitialAmount: sdkmath.NewIntWithDecimal(1, 10), - Features: []types.Feature{types.Feature_extension}, - + Features: []types.Feature{ + types.Feature_freezing, + types.Feature_extension, + }, ExtensionSettings: &types.ExtensionIssueSettings{ CodeId: codeID, }, } - denomWithExtension, err := ftKeeper.Issue(ctx, settingsWithExtension) + extensionDenom, err := ftKeeper.Issue(ctx, settingsWithExtension) requireT.NoError(err) - extensionCoin := sdk.NewInt64Coin(denomWithExtension, 50) - requireT.NoError(bankKeeper.SendCoins(ctx, issuer, acc, sdk.NewCoins(extensionCoin))) - requireT.ErrorContains( - ftKeeper.DEXCheckOrderAmounts( - ctx, - types.DEXOrder{Creator: acc}, - extensionCoin, - sdk.NewInt64Coin(denom1, 0), - ), - "the token has extensions", - ) + coinWithExtensionToSend := sdk.NewCoin(extensionDenom, sdkmath.NewInt(33)) + requireT.NoError(bankKeeper.SendCoins(ctx, issuer, acc, sdk.NewCoins(coinWithExtensionToSend))) + + requireT.NoError(ftKeeper.Freeze(ctx, issuer, acc, coinWithExtensionToSend)) + + spendableBalance, err = ftKeeper.GetSpendableBalance(ctx, acc, extensionDenom) + requireT.NoError(err) + // the balance is frozen, not we don't count it as spendable, since the extensions control it + requireT.Equal(coinWithExtensionToSend.String(), spendableBalance.String()) } func TestKeeper_DEXBlockSmartContracts(t *testing.T) { @@ -409,6 +450,7 @@ func TestKeeper_DEXSettings_BlockDEX(t *testing.T) { }) ftKeeper := testApp.AssetFTKeeper + bankKeeper := testApp.BankKeeper issuer := sdk.AccAddress(secp256k1.GenPrivKey().PubKey().Address()) acc := sdk.AccAddress(secp256k1.GenPrivKey().PubKey().Address()) @@ -435,19 +477,21 @@ func TestKeeper_DEXSettings_BlockDEX(t *testing.T) { ft1Denom, err := ftKeeper.Issue(ctx, ft1Settings) requireT.NoError(err) + dexBlockedCoin := sdk.NewInt64Coin(ft1Denom, 50) + requireT.NoError(bankKeeper.SendCoins(ctx, issuer, acc, sdk.NewCoins(dexBlockedCoin))) errStr := fmt.Sprintf("usage of %s is not supported for DEX, the token has dex_block", ft1Denom) requireT.ErrorContains(ftKeeper.DEXCheckOrderAmounts( ctx, types.DEXOrder{Creator: acc}, - sdk.NewInt64Coin(ft1Denom, 50), + dexBlockedCoin, sdk.NewInt64Coin(denom1, 0), ), errStr) requireT.ErrorContains(ftKeeper.DEXCheckOrderAmounts( ctx, types.DEXOrder{Creator: acc}, sdk.NewInt64Coin(denom1, 0), - sdk.NewInt64Coin(ft1Denom, 50), + dexBlockedCoin, ), errStr) } @@ -664,7 +708,7 @@ func TestKeeper_DEXLimitsWithGlobalFreeze(t *testing.T) { // admin still can increase the limits requireT.NoError( ftKeeper.DEXCheckOrderAmounts( - simapp.CopyContextWithMultiStore(ctx), + ctx, types.DEXOrder{Creator: issuer}, ft1CoinToSend, ft2CoinToSend, @@ -977,3 +1021,120 @@ func TestKeeper_UpdateDEXWhitelistedDenoms(t *testing.T) { WhitelistedDenoms: whitelistedDenoms, }, dexSettings) } + +func TestKeeper_DEXExtensions(t *testing.T) { + requireT := require.New(t) + + testApp := simapp.New() + ctx := testApp.BaseApp.NewContextLegacy(false, tmproto.Header{ + Time: time.Now(), + AppHash: []byte("some-hash"), + }) + + ftKeeper := testApp.AssetFTKeeper + bankKeeper := testApp.BankKeeper + + issuer := sdk.AccAddress(secp256k1.GenPrivKey().PubKey().Address()) + acc := sdk.AccAddress(secp256k1.GenPrivKey().PubKey().Address()) + + // extension + codeID, _, err := testApp.WasmPermissionedKeeper.Create( + ctx, issuer, testcontracts.AssetExtensionWasm, &wasmtypes.AllowEverybody, + ) + requireT.NoError(err) + settingsWithExtension := types.IssueSettings{ + Issuer: issuer, + Symbol: "DEFEXT", + Subunit: "defext", + Precision: 6, + InitialAmount: sdkmath.NewIntWithDecimal(1, 10), + Features: []types.Feature{ + types.Feature_freezing, + types.Feature_extension, + types.Feature_whitelisting, + }, + ExtensionSettings: &types.ExtensionIssueSettings{ + CodeId: codeID, + }, + } + extensionDenom, err := ftKeeper.Issue(ctx, settingsWithExtension) + requireT.NoError(err) + coinWithExtension := sdk.NewCoin(extensionDenom, sdkmath.NewInt(200)) + requireT.NoError(ftKeeper.SetWhitelistedBalance(ctx, issuer, acc, coinWithExtension)) + coinWithoutExtension := sdk.NewInt64Coin(denom1, 123) + testApp.MintAndSendCoin(t, ctx, acc, sdk.NewCoins(coinWithoutExtension)) + + // the extension controls all features, bot to locked, amount + requireT.ErrorIs( + ftKeeper.DEXCheckOrderAmounts( + simapp.CopyContextWithMultiStore(ctx), + types.DEXOrder{Creator: acc}, + coinWithExtension, + coinWithoutExtension, + ), + types.ErrDEXInsufficientSpendableBalance, + ) + // send coins now to lock + requireT.NoError(bankKeeper.SendCoins(ctx, issuer, acc, sdk.NewCoins(coinWithExtension))) + + // freeze locked balance, to check that is extension controls the freezing, but asset FT ignores + requireT.NoError(ftKeeper.Freeze(ctx, issuer, acc, coinWithExtension)) + requireT.NoError( + ftKeeper.DEXCheckOrderAmounts( + ctx, + types.DEXOrder{Creator: acc}, + coinWithExtension, + coinWithoutExtension, + ), + ) + requireT.NoError(ftKeeper.DEXIncreaseLocked(ctx, acc, coinWithExtension)) + + // try with locked balance + requireT.ErrorIs( + ftKeeper.DEXCheckOrderAmounts( + simapp.CopyContextWithMultiStore(ctx), + types.DEXOrder{Creator: acc}, + coinWithExtension, + coinWithoutExtension, + ), + types.ErrDEXInsufficientSpendableBalance, + ) + + // check expected to send and receive with prohibited amounts + requireT.NoError(ftKeeper.DEXDecreaseLocked(ctx, acc, coinWithExtension)) + + // try with prohibited balances + + coinWithExtensionProhibitedToSpend := sdk.NewCoin(extensionDenom, sdkmath.NewInt(103)) + requireT.ErrorContains( + ftKeeper.DEXCheckOrderAmounts( + simapp.CopyContextWithMultiStore(ctx), + types.DEXOrder{Creator: acc}, + coinWithExtensionProhibitedToSpend, + coinWithoutExtension, + ), + "wasm error: DEX order placement is failed", + ) + + coinWithExtensionProhibitedToReceive := sdk.NewCoin(extensionDenom, sdkmath.NewInt(104)) + requireT.ErrorContains( + ftKeeper.DEXCheckOrderAmounts( + simapp.CopyContextWithMultiStore(ctx), + types.DEXOrder{Creator: acc}, + coinWithoutExtension, + coinWithExtensionProhibitedToReceive, + ), + "wasm error: DEX order placement is failed", + ) + + // update whitelisted balance, and check that asset FT doesn't control it, so the check passes + requireT.NoError(ftKeeper.SetWhitelistedBalance(ctx, issuer, acc, sdk.NewCoin(extensionDenom, sdkmath.NewInt(1)))) + requireT.NoError( + ftKeeper.DEXCheckOrderAmounts( + ctx, + types.DEXOrder{Creator: acc}, + coinWithoutExtension, + coinWithExtension, + ), + ) +} diff --git a/x/asset/ft/keeper/test-contracts/asset-extension/src/contract.rs b/x/asset/ft/keeper/test-contracts/asset-extension/src/contract.rs index 196d4de82..2e701cd53 100644 --- a/x/asset/ft/keeper/test-contracts/asset-extension/src/contract.rs +++ b/x/asset/ft/keeper/test-contracts/asset-extension/src/contract.rs @@ -1,8 +1,5 @@ use crate::error::ContractError; -use crate::msg::{ - ExecuteMsg, IBCPurpose, InstantiateMsg, QueryIssuanceMsgResponse, QueryMsg, SudoMsg, - TransferContext, -}; +use crate::msg::{DEXOrder, ExecuteMsg, IBCPurpose, InstantiateMsg, QueryIssuanceMsgResponse, QueryMsg, SudoMsg, TransferContext}; use crate::state::{DENOM, EXTRA_DATA}; use coreum_wasm_sdk::assetft::{FREEZING, WHITELISTING}; use coreum_wasm_sdk::core::{CoreumMsg, CoreumResult}; @@ -18,6 +15,7 @@ use cosmwasm_std::{entry_point, to_json_binary, CosmosMsg, StdError}; use cosmwasm_std::{Binary, Deps, DepsMut, Env, MessageInfo, Response, StdResult, Uint128}; use cw2::set_contract_version; use std::ops::Div; +use std::string::ToString; // version info for migration info const CONTRACT_NAME: &str = env!("CARGO_PKG_NAME"); @@ -32,6 +30,9 @@ const AMOUNT_IGNORE_BURN_RATE_TRIGGER: Uint128 = Uint128::new(108); const AMOUNT_IGNORE_SEND_COMMISSION_RATE_TRIGGER: Uint128 = Uint128::new(109); const AMOUNT_BLOCK_IBC_TRIGGER: Uint128 = Uint128::new(110); const AMOUNT_BLOCK_SMART_CONTRACT_TRIGGER: Uint128 = Uint128::new(111); +const ID_DEX_ORDER_TRIGGER: &str = "id-blocked"; +const AMOUNT_DEX_EXPECT_TO_SPEND_TRIGGER: Uint128 = Uint128::new(103); +const AMOUNT_DEX_EXPECT_TO_RECEIVE_TRIGGER: Uint128 = Uint128::new(104); #[cfg_attr(not(feature = "library"), entry_point)] pub fn instantiate( @@ -83,6 +84,15 @@ pub fn sudo(deps: DepsMut, env: Env, msg: SudoMsg) -> CoreumResult sudo_extension_place_order( + order, + expected_to_spend, + expected_to_receive, + ), } } @@ -150,7 +160,7 @@ pub fn sudo_extension_transfer( denom: token.denom.to_string(), amount: amount.to_string(), }] - .to_vec(), + .to_vec(), }; let mut response = rsp.add_message(CosmosMsg::Any(transfer_msg.to_any())); @@ -180,6 +190,19 @@ pub fn sudo_extension_transfer( Ok(response) } +pub fn sudo_extension_place_order( + order: DEXOrder, + expected_to_spend: Coin, + expected_to_receive: Coin, +) -> CoreumResult { + if order.id == ID_DEX_ORDER_TRIGGER || + expected_to_spend.amount == AMOUNT_DEX_EXPECT_TO_SPEND_TRIGGER.to_string() || + expected_to_receive.amount == AMOUNT_DEX_EXPECT_TO_RECEIVE_TRIGGER.to_string() { + return Err(ContractError::DEXOrderPlacementError {}); + } + Ok(Response::new().add_attribute("method", "extension_place_order")) +} + #[cfg_attr(not(feature = "library"), entry_point)] pub fn query(deps: Deps, _env: Env, msg: QueryMsg) -> StdResult { match msg { @@ -304,7 +327,7 @@ fn assert_minting( denom: token.denom.to_string(), amount: amount.to_string(), }] - .to_vec(), + .to_vec(), }; Ok(Response::new() @@ -363,7 +386,7 @@ fn assert_send_commission_rate( denom: token.denom.to_string(), amount: commission_amount.to_string(), }] - .to_vec(), + .to_vec(), }; return Ok(response @@ -386,7 +409,7 @@ fn assert_send_commission_rate( denom: token.denom.to_string(), amount: admin_commission_amount.to_string(), }] - .to_vec(), + .to_vec(), }; return Ok(response @@ -417,7 +440,7 @@ fn assert_burn_rate( denom: token.denom.to_string(), amount: burn_amount.to_string(), }] - .to_vec(), + .to_vec(), }; return Ok(response diff --git a/x/asset/ft/keeper/test-contracts/asset-extension/src/error.rs b/x/asset/ft/keeper/test-contracts/asset-extension/src/error.rs index 312f42007..10a2818c8 100644 --- a/x/asset/ft/keeper/test-contracts/asset-extension/src/error.rs +++ b/x/asset/ft/keeper/test-contracts/asset-extension/src/error.rs @@ -26,4 +26,7 @@ pub enum ContractError { #[error("Invalid amount.")] InvalidAmountError {}, + + #[error("DEX order placement is failed.")] + DEXOrderPlacementError {}, } diff --git a/x/asset/ft/keeper/test-contracts/asset-extension/src/msg.rs b/x/asset/ft/keeper/test-contracts/asset-extension/src/msg.rs index 3b1479bee..4bfb80524 100644 --- a/x/asset/ft/keeper/test-contracts/asset-extension/src/msg.rs +++ b/x/asset/ft/keeper/test-contracts/asset-extension/src/msg.rs @@ -1,5 +1,6 @@ use cosmwasm_schema::{cw_serde, QueryResponses}; -use cosmwasm_std::Uint128; +use cosmwasm_std::{Uint128}; +use coreum_wasm_sdk::types::cosmos::base::v1beta1::Coin; #[cw_serde] pub struct InstantiateMsg { @@ -15,6 +16,19 @@ pub struct IssuanceMsg { #[cw_serde] pub enum ExecuteMsg {} +#[cw_serde] +pub struct DEXOrder { + pub creator: String, + #[serde(rename = "type")] + pub order_type: String, + pub id: String, + pub base_denom: String, + pub quote_denom: String, + pub price: Option, + pub quantity: Uint128, + pub side: String, +} + #[cw_serde] pub enum SudoMsg { ExtensionTransfer { @@ -25,6 +39,11 @@ pub enum SudoMsg { burn_amount: Uint128, context: TransferContext, }, + ExtensionPlaceOrder { + order: DEXOrder, + expected_to_spend: Coin, + expected_to_receive: Coin, + }, } #[cw_serde] diff --git a/x/dex/keeper/keeper_ft_test.go b/x/dex/keeper/keeper_ft_test.go index d00ebee47..99644735c 100644 --- a/x/dex/keeper/keeper_ft_test.go +++ b/x/dex/keeper/keeper_ft_test.go @@ -31,7 +31,6 @@ func TestKeeper_PlaceOrderWithExtension(t *testing.T) { AppHash: []byte("some-hash"), }) - acc, _ := testApp.GenAccount(sdkCtx) issuer, _ := testApp.GenAccount(sdkCtx) // extension @@ -45,7 +44,9 @@ func TestKeeper_PlaceOrderWithExtension(t *testing.T) { Subunit: "defext", Precision: 6, InitialAmount: sdkmath.NewIntWithDecimal(1, 10), - Features: []assetfttypes.Feature{assetfttypes.Feature_extension}, + Features: []assetfttypes.Feature{ + assetfttypes.Feature_extension, + }, ExtensionSettings: &assetfttypes.ExtensionIssueSettings{ CodeId: codeID, }, @@ -53,28 +54,103 @@ func TestKeeper_PlaceOrderWithExtension(t *testing.T) { denomWithExtension, err := testApp.AssetFTKeeper.Issue(sdkCtx, settingsWithExtension) require.NoError(t, err) - order := types.Order{ - Creator: acc.String(), - Type: types.ORDER_TYPE_LIMIT, - ID: uuid.Generate().String(), - BaseDenom: denomWithExtension, - QuoteDenom: denom2, - Price: lo.ToPtr(types.MustNewPriceFromString("12e-1")), - Quantity: sdkmath.NewInt(10), - Side: types.SIDE_SELL, - GoodTil: &types.GoodTil{ - GoodTilBlockHeight: 390, + tests := []struct { + name string + order types.Order + wantDEXErr bool + }{ + { + name: "sell_positive", + order: types.Order{ + Creator: func() string { + creator, _ := testApp.GenAccount(sdkCtx) + return creator.String() + }(), + Type: types.ORDER_TYPE_LIMIT, + ID: uuid.Generate().String(), + BaseDenom: denomWithExtension, + QuoteDenom: denom2, + Price: lo.ToPtr(types.MustNewPriceFromString("1")), + Quantity: sdkmath.NewInt(10), + Side: types.SIDE_SELL, + TimeInForce: types.TIME_IN_FORCE_GTC, + }, + wantDEXErr: false, + }, + { + name: "sell_dex_error", + order: types.Order{ + Creator: func() string { + creator, _ := testApp.GenAccount(sdkCtx) + return creator.String() + }(), + Type: types.ORDER_TYPE_LIMIT, + ID: uuid.Generate().String(), + BaseDenom: denomWithExtension, + QuoteDenom: denom2, + Price: lo.ToPtr(types.MustNewPriceFromString("1")), + Quantity: sdkmath.NewInt(103), // 103 is prohibited by extensions smart contract + Side: types.SIDE_SELL, + TimeInForce: types.TIME_IN_FORCE_GTC, + }, + wantDEXErr: true, + }, + { + name: "buy_positive", + order: types.Order{ + Creator: func() string { + creator, _ := testApp.GenAccount(sdkCtx) + return creator.String() + }(), + Type: types.ORDER_TYPE_LIMIT, + ID: uuid.Generate().String(), + BaseDenom: denom2, + QuoteDenom: denomWithExtension, + Price: lo.ToPtr(types.MustNewPriceFromString("1")), + Quantity: sdkmath.NewInt(10), + Side: types.SIDE_BUY, + TimeInForce: types.TIME_IN_FORCE_GTC, + }, + wantDEXErr: false, + }, + { + name: "buy_dex_error", + order: types.Order{ + Creator: func() string { + creator, _ := testApp.GenAccount(sdkCtx) + return creator.String() + }(), + Type: types.ORDER_TYPE_LIMIT, + ID: uuid.Generate().String(), + BaseDenom: denom2, + QuoteDenom: denomWithExtension, + Price: lo.ToPtr(types.MustNewPriceFromString("1")), + Quantity: sdkmath.NewInt(104), // 104 is prohibited by extensions smart contract + Side: types.SIDE_BUY, + TimeInForce: types.TIME_IN_FORCE_GTC, + }, + wantDEXErr: true, }, - TimeInForce: types.TIME_IN_FORCE_GTC, } - lockedBalance, err := order.ComputeLimitOrderLockedBalance() - require.NoError(t, err) - require.NoError(t, testApp.BankKeeper.SendCoins(sdkCtx, issuer, acc, sdk.NewCoins(lockedBalance))) - fundOrderReserve(t, testApp, sdkCtx, acc) - - require.ErrorContains( - t, testApp.DEXKeeper.PlaceOrder(sdkCtx, order), "is not supported for DEX, the token has extensions", - ) + for _, tt := range tests { + tt := tt + t.Run(tt.name, func(t *testing.T) { + creator := sdk.MustAccAddressFromBech32(tt.order.Creator) + lockedBalance, err := tt.order.ComputeLimitOrderLockedBalance() + require.NoError(t, err) + testApp.MintAndSendCoin(t, sdkCtx, creator, sdk.NewCoins(lockedBalance)) + fundOrderReserve(t, testApp, sdkCtx, creator) + if !tt.wantDEXErr { + require.NoError(t, testApp.DEXKeeper.PlaceOrder(sdkCtx, tt.order)) + } else { + require.ErrorContains( + t, + testApp.DEXKeeper.PlaceOrder(sdkCtx, tt.order), + "wasm error: DEX order placement is failed", + ) + } + }) + } } func TestKeeper_PlaceOrderWithDEXBlockFeature(t *testing.T) { diff --git a/x/dex/keeper/keeper_matching.go b/x/dex/keeper/keeper_matching.go index 443b069d2..59f08e953 100644 --- a/x/dex/keeper/keeper_matching.go +++ b/x/dex/keeper/keeper_matching.go @@ -154,7 +154,10 @@ func (k Keeper) getInitialRemainingAmount( } case types.ORDER_TYPE_MARKET: if order.Side == types.SIDE_BUY { - remainingBalance = k.assetFTKeeper.GetSpendableBalance(ctx, creatorAddr, order.QuoteDenom) + remainingBalance, err = k.assetFTKeeper.GetSpendableBalance(ctx, creatorAddr, order.QuoteDenom) + if err != nil { + return sdkmath.Int{}, err + } } else { remainingBalance = sdk.NewCoin(order.BaseDenom, order.Quantity) } diff --git a/x/dex/keeper/keeper_matching_fuzz_test.go b/x/dex/keeper/keeper_matching_fuzz_test.go index e6b3cd247..fd67ebe75 100644 --- a/x/dex/keeper/keeper_matching_fuzz_test.go +++ b/x/dex/keeper/keeper_matching_fuzz_test.go @@ -376,7 +376,8 @@ func (fa *FuzzApp) FundAccountAndApplyFTFeatures( order.TimeInForce == types.TIME_IN_FORCE_GTC && randBoolWithPercent(orderRnd, fa.cfg.FundOrderReservePercent) { reserve := fa.testApp.DEXKeeper.GetParams(sdkCtx).OrderReserve - spendableBalance := fa.testApp.AssetFTKeeper.GetSpendableBalance(sdkCtx, creator, reserve.Denom) + spendableBalance, err := fa.testApp.AssetFTKeeper.GetSpendableBalance(sdkCtx, creator, reserve.Denom) + require.NoError(t, err) if spendableBalance.IsLT(reserve) { t.Logf("Funding order reserve, account: %s coin: %s", creator.String(), reserve.String()) fa.testApp.MintAndSendCoin(t, sdkCtx, creator, sdk.NewCoins(reserve)) @@ -449,9 +450,10 @@ func (fa *FuzzApp) PlaceOrder(t *testing.T, sdkCtx sdk.Context, order types.Orde } // check failed because of reserve reserve := fa.testApp.DEXKeeper.GetParams(sdkCtx).OrderReserve - reserveDenomSpendableBalance := fa.testApp.AssetFTKeeper.GetSpendableBalance( + reserveDenomSpendableBalance, err := fa.testApp.AssetFTKeeper.GetSpendableBalance( sdkCtx, creator, reserve.Denom, ) + require.NoError(t, err) if reserveDenomSpendableBalance.Amount.LT(reserve.Amount) { t.Logf("Placement is failed due to insufficient reserve, reserve: %s, reserveDenomSpendableBalance: %s", reserve.String(), reserveDenomSpendableBalance.String()) @@ -459,9 +461,10 @@ func (fa *FuzzApp) PlaceOrder(t *testing.T, sdkCtx sdk.Context, order types.Orde } // check spendable balance - spendableBalance := fa.testApp.AssetFTKeeper.GetSpendableBalance( + spendableBalance, err := fa.testApp.AssetFTKeeper.GetSpendableBalance( sdkCtx, creator, order.GetSpendDenom(), ) + require.NoError(t, err) orderLockedBalance, err := order.ComputeLimitOrderLockedBalance() require.NoError(t, err) require.True( diff --git a/x/dex/types/expected_keepers.go b/x/dex/types/expected_keepers.go index 59b7a04d9..cf701c6ff 100644 --- a/x/dex/types/expected_keepers.go +++ b/x/dex/types/expected_keepers.go @@ -27,7 +27,7 @@ type AccountQueryServer interface { type AssetFTKeeper interface { DEXExecuteActions(ctx sdk.Context, actions dextypes.DEXActions) error DEXDecreaseLimits(ctx sdk.Context, addr sdk.AccAddress, lockedCoin sdk.Coins, expectedToReceiveCoin sdk.Coin) error - GetSpendableBalance(ctx sdk.Context, addr sdk.AccAddress, denom string) sdk.Coin + GetSpendableBalance(ctx sdk.Context, addr sdk.AccAddress, denom string) (sdk.Coin, error) GetDEXSettings(ctx sdk.Context, denom string) (dextypes.DEXSettings, error) ValidateDEXCancelOrdersByDenomIsAllowed(ctx sdk.Context, addr sdk.AccAddress, denom string) error } diff --git a/x/wbank/keeper/keeper.go b/x/wbank/keeper/keeper.go index e3634f53f..488bfaa94 100644 --- a/x/wbank/keeper/keeper.go +++ b/x/wbank/keeper/keeper.go @@ -6,7 +6,6 @@ import ( "cosmossdk.io/core/store" sdkerrors "cosmossdk.io/errors" "cosmossdk.io/log" - sdkmath "cosmossdk.io/math" "github.com/cosmos/cosmos-sdk/codec" sdk "github.com/cosmos/cosmos-sdk/types" cosmoserrors "github.com/cosmos/cosmos-sdk/types/errors" @@ -192,12 +191,13 @@ func (k BaseKeeperWrapper) SpendableBalances( return nil, err } - bankLockedCoins := k.BaseKeeper.LockedCoins(ctx, addr) - balances := balancesRes.Balances for i := range balances { - bankLockedCoin := sdk.NewCoin(balances[i].Denom, bankLockedCoins.AmountOf(balances[i].Denom)) - balances[i] = k.getSpendableCoin(sdk.UnwrapSDKContext(ctx), addr, balances[i], bankLockedCoin) + spendableCoin, err := k.ftProvider.GetSpendableBalance(sdk.UnwrapSDKContext(ctx), addr, balances[i].Denom) + if err != nil { + return nil, err + } + balances[i] = spendableCoin } return &banktypes.QuerySpendableBalancesResponse{ @@ -228,34 +228,16 @@ func (k BaseKeeperWrapper) SpendableBalanceByDenom( return &banktypes.QuerySpendableBalanceByDenomResponse{}, nil } - bankLockedCoins := k.BaseKeeper.LockedCoins(ctx, addr) - bankLockedCoin := sdk.NewCoin(req.Denom, bankLockedCoins.AmountOf(req.Denom)) + spendableCoin, err := k.ftProvider.GetSpendableBalance(sdk.UnwrapSDKContext(ctx), addr, balanceRes.Balance.Denom) + if err != nil { + return nil, err + } return &banktypes.QuerySpendableBalanceByDenomResponse{ - Balance: lo.ToPtr(k.getSpendableCoin(sdk.UnwrapSDKContext(ctx), addr, *balanceRes.Balance, bankLockedCoin)), + Balance: lo.ToPtr(spendableCoin), }, nil } -func (k BaseKeeperWrapper) getSpendableCoin( - ctx sdk.Context, - addr sdk.AccAddress, - balance, bankLocked sdk.Coin, -) sdk.Coin { - denom := balance.Denom - notLockedAmt := balance.Amount. - Sub(bankLocked.Amount). - Sub(k.ftProvider.GetDEXLockedBalance(ctx, addr, denom).Amount) - - notFrozenAmt := balance.Amount.Sub(k.ftProvider.GetFrozenBalance(ctx, addr, denom).Amount) - - spendableAmount := sdkmath.MinInt(notLockedAmt, notFrozenAmt) - if !spendableAmount.IsPositive() { - return sdk.NewCoin(denom, sdkmath.ZeroInt()) - } - - return sdk.NewCoin(denom, spendableAmount) -} - func (k BaseKeeperWrapper) isSmartContract(ctx sdk.Context, addr sdk.AccAddress) bool { return wasm.IsSmartContract(ctx, addr, k.wasmKeeper) } diff --git a/x/wbank/types/expected_keepers.go b/x/wbank/types/expected_keepers.go index 5718bbcad..0c4acc2ad 100644 --- a/x/wbank/types/expected_keepers.go +++ b/x/wbank/types/expected_keepers.go @@ -9,6 +9,6 @@ import ( type FungibleTokenProvider interface { BeforeSendCoins(ctx sdk.Context, fromAddress, toAddress sdk.AccAddress, coins sdk.Coins) error BeforeInputOutputCoins(ctx sdk.Context, input banktypes.Input, outputs []banktypes.Output) error - GetFrozenBalance(ctx sdk.Context, addr sdk.AccAddress, denom string) sdk.Coin + GetSpendableBalance(ctx sdk.Context, addr sdk.AccAddress, denom string) (sdk.Coin, error) GetDEXLockedBalance(ctx sdk.Context, addr sdk.AccAddress, denom string) sdk.Coin } From 107cb872b3a50d5ba1e69109c4371850cc520270 Mon Sep 17 00:00:00 2001 From: Dzmitry Hil Date: Fri, 22 Nov 2024 11:26:44 +0300 Subject: [PATCH 2/5] Fix tests --- .../modules/assetft_extension_test.go | 28 +++++++++---------- x/dex/keeper/keeper_ft_test.go | 1 - 2 files changed, 14 insertions(+), 15 deletions(-) diff --git a/integration-tests/modules/assetft_extension_test.go b/integration-tests/modules/assetft_extension_test.go index 769d57700..4fff6c715 100644 --- a/integration-tests/modules/assetft_extension_test.go +++ b/integration-tests/modules/assetft_extension_test.go @@ -50,7 +50,7 @@ func TestAssetFTExtensionIssue(t *testing.T) { issuer := chain.GenAccount() chain.FundAccountWithOptions(ctx, t, issuer, integration.BalancesOptions{ Amount: chain.QueryAssetFTParams(ctx, t).IssueFee.Amount. - Add(sdkmath.NewInt(500_000)). + Add(sdkmath.NewInt(1_000_000)). Add(sdkmath.NewInt(3 * 500_000)), // give 500k gas for each message since extensions are nondeterministic }) @@ -206,7 +206,7 @@ func TestAssetFTExtensionWhitelist(t *testing.T) { &assetfttypes.MsgSetWhitelistedLimit{}, }, Amount: chain.QueryAssetFTParams(ctx, t).IssueFee.Amount.Mul(sdkmath.NewInt(2)). - Add(sdkmath.NewInt(500_000)). // added 1 million for smart contract upload + Add(sdkmath.NewInt(1_000_000)). // added 1 million for smart contract upload Add(sdkmath.NewInt(10 * 500_000)), // give 500k gas for each message since extensions are nondeterministic }) chain.FundAccountWithOptions(ctx, t, nonIssuer, integration.BalancesOptions{ @@ -464,7 +464,7 @@ func TestAssetFTExtensionFreeze(t *testing.T) { &assetfttypes.MsgUnfreeze{}, }, Amount: chain.QueryAssetFTParams(ctx, t).IssueFee.Amount. - Add(sdkmath.NewInt(500_000)). // added 1 million for smart contract upload + Add(sdkmath.NewInt(1_000_000)). // added 1 million for smart contract upload Add(sdkmath.NewInt(2 * 500_000)), // add 500k for each message with extenstion transfer }) chain.FundAccountWithOptions(ctx, t, recipient, integration.BalancesOptions{ @@ -617,7 +617,7 @@ func TestAssetFTExtensionBurn(t *testing.T) { chain.FundAccountWithOptions(ctx, t, issuer, integration.BalancesOptions{ Messages: []sdk.Msg{}, Amount: chain.QueryAssetFTParams(ctx, t).IssueFee.Amount.MulRaw(2). - Add(sdkmath.NewInt(500_000)). // added 1 million for smart contract upload + Add(sdkmath.NewInt(1_000_000)). // added 1 million for smart contract upload Add(sdkmath.NewInt(6 * 500_000)), // add 500k for each message with extenstion transfer }) @@ -804,7 +804,7 @@ func TestAssetFTExtensionMint(t *testing.T) { chain.FundAccountWithOptions(ctx, t, issuer, integration.BalancesOptions{ Amount: chain.QueryAssetFTParams(ctx, t).IssueFee.Amount.MulRaw(2). - Add(sdkmath.NewInt(500_000)). // added 1 million for smart contract upload + Add(sdkmath.NewInt(1_000_000)). // added 1 million for smart contract upload Add(sdkmath.NewInt(7 * 500_000)), // add 500k for each message with extenstion transfer }) @@ -813,7 +813,7 @@ func TestAssetFTExtensionMint(t *testing.T) { }) chain.FundAccountWithOptions(ctx, t, admin, integration.BalancesOptions{ - Amount: sdkmath.NewInt(500_000), + Amount: sdkmath.NewInt(1_000_000), }) codeID, err := chain.Wasm.DeployWASMContract( @@ -1011,7 +1011,7 @@ func TestAssetFTExtensionSendingToSmartContractIsDenied(t *testing.T) { chain.FundAccountWithOptions(ctx, t, issuer, integration.BalancesOptions{ Amount: chain.QueryAssetFTParams(ctx, t).IssueFee.Amount. - Add(sdkmath.NewInt(500_000)), + Add(sdkmath.NewInt(1_000_000)), }) clientCtx := chain.ClientContext @@ -1118,7 +1118,7 @@ func TestAssetFTExtensionAttachingToSmartContractCallIsDenied(t *testing.T) { requireT := require.New(t) chain.FundAccountWithOptions(ctx, t, issuer, integration.BalancesOptions{ Amount: chain.QueryAssetFTParams(ctx, t).IssueFee.Amount. - Add(sdkmath.NewInt(500_000)), + Add(sdkmath.NewInt(1_000_000)), }) codeID, err := chain.Wasm.DeployWASMContract( @@ -1199,7 +1199,7 @@ func TestAssetFTExtensionIssuingSmartContractIsAllowedToSendAndReceive(t *testin requireT := require.New(t) chain.FundAccountWithOptions(ctx, t, admin, integration.BalancesOptions{ Amount: chain.QueryAssetFTParams(ctx, t).IssueFee.Amount.MulRaw(2). - Add(sdkmath.NewInt(500_000)), + Add(sdkmath.NewInt(1_000_000)), }) chain.FundAccountWithOptions(ctx, t, recipient, integration.BalancesOptions{ Amount: sdkmath.NewInt(500_000), @@ -1313,7 +1313,7 @@ func TestAssetFTExtensionAttachingToSmartContractInstantiationIsDenied(t *testin requireT := require.New(t) chain.FundAccountWithOptions(ctx, t, issuer, integration.BalancesOptions{ Amount: chain.QueryAssetFTParams(ctx, t).IssueFee.Amount. - Add(sdkmath.NewInt(500_000)), + Add(sdkmath.NewInt(1_000_000)), }) codeID, err := chain.Wasm.DeployWASMContract( @@ -1390,7 +1390,7 @@ func TestAssetFTExtensionMintingAndSendingOnBehalfOfIssuingSmartContractIsPossib requireT := require.New(t) chain.FundAccountWithOptions(ctx, t, admin, integration.BalancesOptions{ Amount: chain.QueryAssetFTParams(ctx, t).IssueFee.Amount. - Add(sdkmath.NewInt(500_000)), + Add(sdkmath.NewInt(1_000_000)), }) codeID, err := chain.Wasm.DeployWASMContract( @@ -1531,7 +1531,7 @@ func TestAssetFTExtensionBurnRate(t *testing.T) { chain.FundAccountWithOptions(ctx, t, admin, integration.BalancesOptions{ Amount: chain.QueryAssetFTParams(ctx, t).IssueFee.Amount. - Add(sdkmath.NewInt(500_000)). // added 1 million for smart contract upload + Add(sdkmath.NewInt(1_000_000)). // added 1 million for smart contract upload Add(sdkmath.NewInt(2 * 500_000)), }) chain.FundAccountWithOptions(ctx, t, recipient1, integration.BalancesOptions{ @@ -1673,7 +1673,7 @@ func TestAssetFTExtensionSendCommissionRate(t *testing.T) { chain.FundAccountWithOptions(ctx, t, admin, integration.BalancesOptions{ Amount: chain.QueryAssetFTParams(ctx, t).IssueFee.Amount. - Add(sdkmath.NewInt(500_000)). // added 1 million for smart contract upload + Add(sdkmath.NewInt(1_000_000)). // added 1 million for smart contract upload Add(sdkmath.NewInt(2 * 500_000)), }) chain.FundAccountWithOptions(ctx, t, recipient1, integration.BalancesOptions{ @@ -1829,7 +1829,7 @@ func TestAssetFTExtensionDEX(t *testing.T) { chain.FundAccountWithOptions(ctx, t, admin, integration.BalancesOptions{ Amount: chain.QueryAssetFTParams(ctx, t).IssueFee.Amount. - AddRaw(500_000). // added 1 million for smart contract upload + AddRaw(1_000_000). // added 1 million for smart contract upload AddRaw(2 * 500_000), }) chain.FundAccountWithOptions(ctx, t, acc1, integration.BalancesOptions{ diff --git a/x/dex/keeper/keeper_ft_test.go b/x/dex/keeper/keeper_ft_test.go index 99644735c..d8107b844 100644 --- a/x/dex/keeper/keeper_ft_test.go +++ b/x/dex/keeper/keeper_ft_test.go @@ -133,7 +133,6 @@ func TestKeeper_PlaceOrderWithExtension(t *testing.T) { }, } for _, tt := range tests { - tt := tt t.Run(tt.name, func(t *testing.T) { creator := sdk.MustAccAddressFromBech32(tt.order.Creator) lockedBalance, err := tt.order.ComputeLimitOrderLockedBalance() From 37bcc418a3f9f6c8bf1bc05c344a42fb48cb044c Mon Sep 17 00:00:00 2001 From: Dzmitry Hil Date: Fri, 22 Nov 2024 15:29:27 +0300 Subject: [PATCH 3/5] Format rust --- .../asset-extension/src/contract.rs | 28 +++++++++---------- .../test-contracts/asset-extension/src/msg.rs | 4 +-- 2 files changed, 16 insertions(+), 16 deletions(-) diff --git a/x/asset/ft/keeper/test-contracts/asset-extension/src/contract.rs b/x/asset/ft/keeper/test-contracts/asset-extension/src/contract.rs index 2e701cd53..2dc4d2abf 100644 --- a/x/asset/ft/keeper/test-contracts/asset-extension/src/contract.rs +++ b/x/asset/ft/keeper/test-contracts/asset-extension/src/contract.rs @@ -1,5 +1,8 @@ use crate::error::ContractError; -use crate::msg::{DEXOrder, ExecuteMsg, IBCPurpose, InstantiateMsg, QueryIssuanceMsgResponse, QueryMsg, SudoMsg, TransferContext}; +use crate::msg::{ + DEXOrder, ExecuteMsg, IBCPurpose, InstantiateMsg, QueryIssuanceMsgResponse, QueryMsg, SudoMsg, + TransferContext, +}; use crate::state::{DENOM, EXTRA_DATA}; use coreum_wasm_sdk::assetft::{FREEZING, WHITELISTING}; use coreum_wasm_sdk::core::{CoreumMsg, CoreumResult}; @@ -88,11 +91,7 @@ pub fn sudo(deps: DepsMut, env: Env, msg: SudoMsg) -> CoreumResult sudo_extension_place_order( - order, - expected_to_spend, - expected_to_receive, - ), + } => sudo_extension_place_order(order, expected_to_spend, expected_to_receive), } } @@ -160,7 +159,7 @@ pub fn sudo_extension_transfer( denom: token.denom.to_string(), amount: amount.to_string(), }] - .to_vec(), + .to_vec(), }; let mut response = rsp.add_message(CosmosMsg::Any(transfer_msg.to_any())); @@ -195,9 +194,10 @@ pub fn sudo_extension_place_order( expected_to_spend: Coin, expected_to_receive: Coin, ) -> CoreumResult { - if order.id == ID_DEX_ORDER_TRIGGER || - expected_to_spend.amount == AMOUNT_DEX_EXPECT_TO_SPEND_TRIGGER.to_string() || - expected_to_receive.amount == AMOUNT_DEX_EXPECT_TO_RECEIVE_TRIGGER.to_string() { + if order.id == ID_DEX_ORDER_TRIGGER + || expected_to_spend.amount == AMOUNT_DEX_EXPECT_TO_SPEND_TRIGGER.to_string() + || expected_to_receive.amount == AMOUNT_DEX_EXPECT_TO_RECEIVE_TRIGGER.to_string() + { return Err(ContractError::DEXOrderPlacementError {}); } Ok(Response::new().add_attribute("method", "extension_place_order")) @@ -327,7 +327,7 @@ fn assert_minting( denom: token.denom.to_string(), amount: amount.to_string(), }] - .to_vec(), + .to_vec(), }; Ok(Response::new() @@ -386,7 +386,7 @@ fn assert_send_commission_rate( denom: token.denom.to_string(), amount: commission_amount.to_string(), }] - .to_vec(), + .to_vec(), }; return Ok(response @@ -409,7 +409,7 @@ fn assert_send_commission_rate( denom: token.denom.to_string(), amount: admin_commission_amount.to_string(), }] - .to_vec(), + .to_vec(), }; return Ok(response @@ -440,7 +440,7 @@ fn assert_burn_rate( denom: token.denom.to_string(), amount: burn_amount.to_string(), }] - .to_vec(), + .to_vec(), }; return Ok(response diff --git a/x/asset/ft/keeper/test-contracts/asset-extension/src/msg.rs b/x/asset/ft/keeper/test-contracts/asset-extension/src/msg.rs index 4bfb80524..b89121160 100644 --- a/x/asset/ft/keeper/test-contracts/asset-extension/src/msg.rs +++ b/x/asset/ft/keeper/test-contracts/asset-extension/src/msg.rs @@ -1,6 +1,6 @@ -use cosmwasm_schema::{cw_serde, QueryResponses}; -use cosmwasm_std::{Uint128}; use coreum_wasm_sdk::types::cosmos::base::v1beta1::Coin; +use cosmwasm_schema::{cw_serde, QueryResponses}; +use cosmwasm_std::Uint128; #[cw_serde] pub struct InstantiateMsg { From 4c212f48d47eccf6a8957887eb5ab016aa90cf1d Mon Sep 17 00:00:00 2001 From: Dzmitry Hil Date: Mon, 25 Nov 2024 15:58:32 +0300 Subject: [PATCH 4/5] Move some variable to the constants --- .../modules/assetft_extension_test.go | 66 ++++++++++--------- x/asset/ft/keeper/keeper_dex_test.go | 7 +- x/asset/ft/keeper/keeper_extension_test.go | 54 +++++++-------- x/dex/keeper/keeper_ft_test.go | 9 ++- 4 files changed, 72 insertions(+), 64 deletions(-) diff --git a/integration-tests/modules/assetft_extension_test.go b/integration-tests/modules/assetft_extension_test.go index 4fff6c715..6cc650637 100644 --- a/integration-tests/modules/assetft_extension_test.go +++ b/integration-tests/modules/assetft_extension_test.go @@ -27,15 +27,17 @@ import ( dextypes "github.com/CoreumFoundation/coreum/v5/x/dex/types" ) -const ( - AmountDisallowedTrigger = 7 - AmountIgnoreWhitelistingTrigger = 49 - AmountIgnoreFreezingTrigger = 79 - AmountBurningTrigger = 101 - AmountMintingTrigger = 105 - AmountIgnoreBurnRateTrigger = 108 - AmountIgnoreSendCommissionRateTrigger = 109 - AmountBlockSmartContractTrigger = 111 +var ( + AmountDisallowedTrigger = sdkmath.NewInt(7) + AmountIgnoreWhitelistingTrigger = sdkmath.NewInt(49) + AmountIgnoreFreezingTrigger = sdkmath.NewInt(79) + AmountBurningTrigger = sdkmath.NewInt(101) + AmountMintingTrigger = sdkmath.NewInt(105) + AmountIgnoreBurnRateTrigger = sdkmath.NewInt(108) + AmountIgnoreSendCommissionRateTrigger = sdkmath.NewInt(109) + AmountBlockSmartContractTrigger = sdkmath.NewInt(111) + AmountDEXExpectToSpendTrigger = sdkmath.NewInt(103) + AmountDEXExpectToReceiveTrigger = sdkmath.NewInt(104) ) // TestAssetFTExtensionIssue tests extension issue functionality of fungible tokens. @@ -142,7 +144,7 @@ func TestAssetFTExtensionIssue(t *testing.T) { requireT.EqualValues("12", balance.Balance.Amount.String()) // sending 7 will fail - sendMsg.Amount = sdk.NewCoins(sdk.NewCoin(denom, sdkmath.NewInt(AmountDisallowedTrigger))) + sendMsg.Amount = sdk.NewCoins(sdk.NewCoin(denom, AmountDisallowedTrigger)) _, err = client.BroadcastTx( ctx, chain.ClientContext.WithFromAddress(issuer), @@ -406,7 +408,7 @@ func TestAssetFTExtensionWhitelist(t *testing.T) { sendMsg = &banktypes.MsgSend{ FromAddress: issuer.String(), ToAddress: recipient.String(), - Amount: sdk.NewCoins(sdk.NewCoin(denom, sdkmath.NewInt(AmountIgnoreWhitelistingTrigger))), + Amount: sdk.NewCoins(sdk.NewCoin(denom, AmountIgnoreWhitelistingTrigger)), } _, err = client.BroadcastTx( ctx, @@ -420,12 +422,12 @@ func TestAssetFTExtensionWhitelist(t *testing.T) { // try to send trigger amount via Multisend multiSendMsg = &banktypes.MsgMultiSend{ Inputs: []banktypes.Input{{Address: issuer.String(), Coins: sdk.NewCoins( - sdk.NewCoin(denom, sdkmath.NewInt(AmountIgnoreWhitelistingTrigger)), + sdk.NewCoin(denom, AmountIgnoreWhitelistingTrigger), sdk.NewCoin(denomWithoutExtension, sdkmath.NewInt(10)), chain.NewCoin(sdkmath.NewInt(10)), )}}, Outputs: []banktypes.Output{{Address: recipient.String(), Coins: sdk.NewCoins( - sdk.NewCoin(denom, sdkmath.NewInt(AmountIgnoreWhitelistingTrigger)), + sdk.NewCoin(denom, AmountIgnoreWhitelistingTrigger), sdk.NewCoin(denomWithoutExtension, sdkmath.NewInt(10)), chain.NewCoin(sdkmath.NewInt(10)), )}}, @@ -590,7 +592,7 @@ func TestAssetFTExtensionFreeze(t *testing.T) { sendMsg = &banktypes.MsgSend{ FromAddress: recipient.String(), ToAddress: recipient2.String(), - Amount: sdk.NewCoins(sdk.NewCoin(denom, sdkmath.NewInt(AmountIgnoreFreezingTrigger))), + Amount: sdk.NewCoins(sdk.NewCoin(denom, AmountIgnoreFreezingTrigger)), } res, err = client.BroadcastTx( ctx, @@ -703,7 +705,7 @@ func TestAssetFTExtensionBurn(t *testing.T) { ToAddress: issuer.String(), Amount: sdk.NewCoins(sdk.Coin{ Denom: unburnable, - Amount: sdkmath.NewInt(AmountBurningTrigger), + Amount: AmountBurningTrigger, }), } @@ -764,7 +766,7 @@ func TestAssetFTExtensionBurn(t *testing.T) { // burn tokens and check balance and total supply oldSupply, err := bankClient.SupplyOf(ctx, &banktypes.QuerySupplyOfRequest{Denom: burnableDenom}) requireT.NoError(err) - burnCoin := sdk.NewCoin(burnableDenom, sdkmath.NewInt(AmountBurningTrigger)) + burnCoin := sdk.NewCoin(burnableDenom, AmountBurningTrigger) burnMsg = &banktypes.MsgSend{ FromAddress: issuer.String(), @@ -860,7 +862,7 @@ func TestAssetFTExtensionMint(t *testing.T) { ToAddress: issuer.String(), Amount: sdk.NewCoins(sdk.Coin{ Denom: unmintableDenom, - Amount: sdkmath.NewInt(AmountMintingTrigger), + Amount: AmountMintingTrigger, }), } @@ -920,7 +922,7 @@ func TestAssetFTExtensionMint(t *testing.T) { // mint tokens and check balance and total supply oldSupply, err := bankClient.SupplyOf(ctx, &banktypes.QuerySupplyOfRequest{Denom: mintableDenom}) requireT.NoError(err) - mintCoin := sdk.NewCoin(mintableDenom, sdkmath.NewInt(AmountMintingTrigger)) + mintCoin := sdk.NewCoin(mintableDenom, AmountMintingTrigger) mintMsg = &banktypes.MsgSend{ FromAddress: issuer.String(), ToAddress: issuer.String(), @@ -946,7 +948,7 @@ func TestAssetFTExtensionMint(t *testing.T) { assertT.EqualValues(mintCoin, newSupply.GetAmount().Sub(oldSupply.GetAmount())) // mint tokens to recipient - mintCoin = sdk.NewCoin(mintableDenom, sdkmath.NewInt(AmountMintingTrigger)) + mintCoin = sdk.NewCoin(mintableDenom, AmountMintingTrigger) mintMsg = &banktypes.MsgSend{ FromAddress: issuer.String(), ToAddress: recipient.String(), @@ -988,7 +990,7 @@ func TestAssetFTExtensionMint(t *testing.T) { mintMsg = &banktypes.MsgSend{ FromAddress: issuer.String(), ToAddress: contractAddr, - Amount: sdk.NewCoins(sdk.NewCoin(mintableDenom, sdkmath.NewInt(AmountBlockSmartContractTrigger))), + Amount: sdk.NewCoins(sdk.NewCoin(mintableDenom, AmountBlockSmartContractTrigger)), } _, err = client.BroadcastTx( ctx, @@ -1073,7 +1075,7 @@ func TestAssetFTExtensionSendingToSmartContractIsDenied(t *testing.T) { sendMsg := &banktypes.MsgSend{ FromAddress: issuer.String(), ToAddress: contractAddr, - Amount: sdk.NewCoins(sdk.NewInt64Coin(denom, AmountBlockSmartContractTrigger)), + Amount: sdk.NewCoins(sdk.NewCoin(denom, AmountBlockSmartContractTrigger)), } _, err = client.BroadcastTx( ctx, @@ -1087,13 +1089,13 @@ func TestAssetFTExtensionSendingToSmartContractIsDenied(t *testing.T) { Inputs: []banktypes.Input{ { Address: issuer.String(), - Coins: sdk.NewCoins(sdk.NewInt64Coin(denom, AmountBlockSmartContractTrigger)), + Coins: sdk.NewCoins(sdk.NewCoin(denom, AmountBlockSmartContractTrigger)), }, }, Outputs: []banktypes.Output{ { Address: contractAddr, - Coins: sdk.NewCoins(sdk.NewInt64Coin(denom, AmountBlockSmartContractTrigger)), + Coins: sdk.NewCoins(sdk.NewCoin(denom, AmountBlockSmartContractTrigger)), }, }, } @@ -1181,7 +1183,7 @@ func TestAssetFTExtensionAttachingToSmartContractCallIsDenied(t *testing.T) { incrementPayload, err := moduleswasm.MethodToEmptyBodyPayload(moduleswasm.SimpleIncrement) requireT.NoError(err) _, err = chain.Wasm.ExecuteWASMContract( - ctx, txf, issuer, contractAddr, incrementPayload, sdk.NewInt64Coin(denom, AmountBlockSmartContractTrigger), + ctx, txf, issuer, contractAddr, incrementPayload, sdk.NewCoin(denom, AmountBlockSmartContractTrigger), ) requireT.ErrorContains(err, "Transferring to or from smart contracts are prohibited.") } @@ -1274,7 +1276,7 @@ func TestAssetFTExtensionIssuingSmartContractIsAllowedToSendAndReceive(t *testin requireT.NoError(err) // mint to someone else - amountToMint = sdkmath.NewInt(AmountBlockSmartContractTrigger) + amountToMint = AmountBlockSmartContractTrigger mintPayload, err = json.Marshal(map[ftMethod]amountRecipientBodyFTRequest{ ftMethodMint: { Amount: amountToMint.String(), @@ -1366,7 +1368,7 @@ func TestAssetFTExtensionAttachingToSmartContractInstantiationIsDenied(t *testin integration.InstantiateConfig{ AccessType: wasmtypes.AccessTypeUnspecified, Payload: initialPayload, - Amount: sdk.NewInt64Coin(denom, AmountBlockSmartContractTrigger), + Amount: sdk.NewCoin(denom, AmountBlockSmartContractTrigger), Label: "simple_state", }, ) @@ -1488,12 +1490,12 @@ func TestAssetFTExtensionMintingAndSendingOnBehalfOfIssuingSmartContractIsPossib &banktypes.MsgSend{ FromAddress: contractAddr, ToAddress: recipient.String(), - Amount: sdk.NewCoins(sdk.NewInt64Coin(denom, AmountBlockSmartContractTrigger)), + Amount: sdk.NewCoins(sdk.NewCoin(denom, AmountBlockSmartContractTrigger)), }, &assetfttypes.MsgMint{ Sender: contractAddr, Recipient: recipient.String(), - Coin: sdk.NewInt64Coin(denom, AmountBlockSmartContractTrigger), + Coin: sdk.NewCoin(denom, AmountBlockSmartContractTrigger), }, }) @@ -1601,7 +1603,7 @@ func TestAssetFTExtensionBurnRate(t *testing.T) { sendMsg = &banktypes.MsgSend{ FromAddress: recipient1.String(), ToAddress: recipient2.String(), - Amount: sdk.NewCoins(sdk.NewCoin(denom, sdkmath.NewInt(AmountIgnoreBurnRateTrigger))), + Amount: sdk.NewCoins(sdk.NewCoin(denom, AmountIgnoreBurnRateTrigger)), } _, err = client.BroadcastTx( @@ -1748,7 +1750,7 @@ func TestAssetFTExtensionSendCommissionRate(t *testing.T) { sendMsg = &banktypes.MsgSend{ FromAddress: recipient1.String(), ToAddress: recipient2.String(), - Amount: sdk.NewCoins(sdk.NewCoin(denom, sdkmath.NewInt(AmountIgnoreSendCommissionRateTrigger))), + Amount: sdk.NewCoins(sdk.NewCoin(denom, AmountIgnoreSendCommissionRateTrigger)), } _, err = client.BroadcastTx( @@ -1894,7 +1896,7 @@ func TestAssetFTExtensionDEX(t *testing.T) { BaseDenom: denomWithExtension, QuoteDenom: chain.ChainSettings.Denom, Price: lo.ToPtr(dextypes.MustNewPriceFromString("1")), - Quantity: sdkmath.NewInt(104), // 104 is prohibited by extensions smart contract + Quantity: AmountDEXExpectToReceiveTrigger, Side: dextypes.SIDE_SELL, TimeInForce: dextypes.TIME_IN_FORCE_GTC, } @@ -1941,7 +1943,7 @@ func TestAssetFTExtensionDEX(t *testing.T) { BaseDenom: denomWithExtension, QuoteDenom: chain.ChainSettings.Denom, Price: lo.ToPtr(dextypes.MustNewPriceFromString("1")), - Quantity: sdkmath.NewInt(103), // 103 is prohibited by extensions smart contract + Quantity: AmountDEXExpectToSpendTrigger, Side: dextypes.SIDE_BUY, TimeInForce: dextypes.TIME_IN_FORCE_GTC, } diff --git a/x/asset/ft/keeper/keeper_dex_test.go b/x/asset/ft/keeper/keeper_dex_test.go index 942497612..270b94f9c 100644 --- a/x/asset/ft/keeper/keeper_dex_test.go +++ b/x/asset/ft/keeper/keeper_dex_test.go @@ -1064,7 +1064,7 @@ func TestKeeper_DEXExtensions(t *testing.T) { coinWithoutExtension := sdk.NewInt64Coin(denom1, 123) testApp.MintAndSendCoin(t, ctx, acc, sdk.NewCoins(coinWithoutExtension)) - // the extension controls all features, bot to locked, amount + // the extension controls all features, but to locked, amount requireT.ErrorIs( ftKeeper.DEXCheckOrderAmounts( simapp.CopyContextWithMultiStore(ctx), @@ -1104,8 +1104,7 @@ func TestKeeper_DEXExtensions(t *testing.T) { requireT.NoError(ftKeeper.DEXDecreaseLocked(ctx, acc, coinWithExtension)) // try with prohibited balances - - coinWithExtensionProhibitedToSpend := sdk.NewCoin(extensionDenom, sdkmath.NewInt(103)) + coinWithExtensionProhibitedToSpend := sdk.NewCoin(extensionDenom, AmountDEXExpectToSpendTrigger) requireT.ErrorContains( ftKeeper.DEXCheckOrderAmounts( simapp.CopyContextWithMultiStore(ctx), @@ -1116,7 +1115,7 @@ func TestKeeper_DEXExtensions(t *testing.T) { "wasm error: DEX order placement is failed", ) - coinWithExtensionProhibitedToReceive := sdk.NewCoin(extensionDenom, sdkmath.NewInt(104)) + coinWithExtensionProhibitedToReceive := sdk.NewCoin(extensionDenom, AmountDEXExpectToReceiveTrigger) requireT.ErrorContains( ftKeeper.DEXCheckOrderAmounts( simapp.CopyContextWithMultiStore(ctx), diff --git a/x/asset/ft/keeper/keeper_extension_test.go b/x/asset/ft/keeper/keeper_extension_test.go index 0373695cf..0ddf2be7b 100644 --- a/x/asset/ft/keeper/keeper_extension_test.go +++ b/x/asset/ft/keeper/keeper_extension_test.go @@ -24,14 +24,16 @@ import ( "github.com/CoreumFoundation/coreum/v5/x/asset/ft/types" ) -const ( - AmountDisallowedTrigger = 7 - AmountIgnoreWhitelistingTrigger = 49 - AmountIgnoreFreezingTrigger = 79 - AmountBurningTrigger = 101 - AmountMintingTrigger = 105 - AmountIgnoreBurnRateTrigger = 108 - AmountIgnoreSendCommissionRateTrigger = 109 +var ( + AmountDisallowedTrigger = sdkmath.NewInt(7) + AmountIgnoreWhitelistingTrigger = sdkmath.NewInt(49) + AmountIgnoreFreezingTrigger = sdkmath.NewInt(79) + AmountBurningTrigger = sdkmath.NewInt(101) + AmountMintingTrigger = sdkmath.NewInt(105) + AmountIgnoreBurnRateTrigger = sdkmath.NewInt(108) + AmountIgnoreSendCommissionRateTrigger = sdkmath.NewInt(109) + AmountDEXExpectToSpendTrigger = sdkmath.NewInt(103) + AmountDEXExpectToReceiveTrigger = sdkmath.NewInt(104) ) func TestKeeper_Extension_Issue(t *testing.T) { @@ -111,7 +113,7 @@ func TestKeeper_Extension_Issue(t *testing.T) { // send 7 coin will fail. // the test contract is written as such that sending 7 will fail. err = bankKeeper.SendCoins(ctx, settings.Issuer, receiver, sdk.NewCoins( - sdk.NewCoin(denom, sdkmath.NewInt(AmountDisallowedTrigger))), + sdk.NewCoin(denom, AmountDisallowedTrigger)), ) requireT.ErrorIs(err, types.ErrExtensionCallFailed) balance = bankKeeper.GetBalance(ctx, receiver, denom) @@ -309,7 +311,7 @@ func TestKeeper_Extension_Whitelist(t *testing.T) { // sending trigger amount will be transferred despite whitelisted amount being exceeded err = bankKeeper.SendCoins(ctx, issuer, recipient, sdk.NewCoins( - sdk.NewCoin(denom, sdkmath.NewInt(AmountIgnoreWhitelistingTrigger))), + sdk.NewCoin(denom, AmountIgnoreWhitelistingTrigger)), ) requireT.NoError(err) @@ -425,7 +427,7 @@ func TestKeeper_Extension_FreezeUnfreeze(t *testing.T) { // send trigger amount to transfer despite freezing err = bankKeeper.SendCoins(ctx, recipient, issuer, sdk.NewCoins( - sdk.NewCoin(denom, sdkmath.NewInt(AmountIgnoreFreezingTrigger))), + sdk.NewCoin(denom, AmountIgnoreFreezingTrigger)), ) requireT.NoError(err) @@ -486,7 +488,7 @@ func TestKeeper_Extension_Burn(t *testing.T) { err = bankKeeper.SendCoins(ctx, issuer, recipient, sdk.NewCoins(sdk.NewCoin(unburnableDenom, sdkmath.NewInt(102)))) requireT.NoError(err) - coinsToBurn := sdk.NewCoins(sdk.NewCoin(unburnableDenom, sdkmath.NewInt(AmountBurningTrigger))) + coinsToBurn := sdk.NewCoins(sdk.NewCoin(unburnableDenom, AmountBurningTrigger)) // try to burn unburnable token from the recipient account and make sure that extension can do it err = bankKeeper.SendCoins(ctx, recipient, issuer, coinsToBurn) @@ -511,12 +513,12 @@ func TestKeeper_Extension_Burn(t *testing.T) { // the amount should be burnt requireT.Equal( issuerBalanceBefore.String(), - issuerBalanceAfter.Add(sdk.NewCoin(unburnableDenom, sdkmath.NewInt(AmountBurningTrigger))).String(), + issuerBalanceAfter.Add(sdk.NewCoin(unburnableDenom, AmountBurningTrigger)).String(), ) requireT.Equal(cwExtensionBalanceBefore.String(), cwExtensionBalanceAfter.String()) requireT.Equal( totalSupplyBefore.Supply.String(), - totalSupplyAfter.Supply.Add(sdk.NewCoin(unburnableDenom, sdkmath.NewInt(AmountBurningTrigger))).String(), + totalSupplyAfter.Supply.Add(sdk.NewCoin(unburnableDenom, AmountBurningTrigger)).String(), ) // Issue a burnable fungible token @@ -557,7 +559,7 @@ func TestKeeper_Extension_Burn(t *testing.T) { // try to burn as non-issuer err = bankKeeper.SendCoins(ctx, recipient, issuer, sdk.NewCoins( - sdk.NewCoin(burnableDenom, sdkmath.NewInt(AmountBurningTrigger))), + sdk.NewCoin(burnableDenom, AmountBurningTrigger)), ) requireT.NoError(err) @@ -570,12 +572,12 @@ func TestKeeper_Extension_Burn(t *testing.T) { // the amount should be burnt requireT.Equal( recipientBalanceBefore.String(), - recipientBalanceAfter.Add(sdk.NewCoin(burnableDenom, sdkmath.NewInt(AmountBurningTrigger))).String(), + recipientBalanceAfter.Add(sdk.NewCoin(burnableDenom, AmountBurningTrigger)).String(), ) requireT.Equal(cwExtensionBalanceBefore.String(), cwExtensionBalanceAfter.String()) requireT.Equal( totalSupplyBefore.Supply.String(), - totalSupplyAfter.Supply.Add(sdk.NewCoin(burnableDenom, sdkmath.NewInt(AmountBurningTrigger))).String(), + totalSupplyAfter.Supply.Add(sdk.NewCoin(burnableDenom, AmountBurningTrigger)).String(), ) issuerBalanceBefore = bankKeeper.GetBalance(ctx, issuer, burnableDenom) @@ -586,7 +588,7 @@ func TestKeeper_Extension_Burn(t *testing.T) { // burn tokens and check balance and total supply err = bankKeeper.SendCoins(ctx, issuer, issuer, sdk.NewCoins( - sdk.NewCoin(burnableDenom, sdkmath.NewInt(AmountBurningTrigger))), + sdk.NewCoin(burnableDenom, AmountBurningTrigger)), ) requireT.NoError(err) @@ -599,12 +601,12 @@ func TestKeeper_Extension_Burn(t *testing.T) { // the amount should be burnt requireT.Equal( issuerBalanceBefore.String(), - issuerBalanceAfter.Add(sdk.NewCoin(burnableDenom, sdkmath.NewInt(AmountBurningTrigger))).String(), + issuerBalanceAfter.Add(sdk.NewCoin(burnableDenom, AmountBurningTrigger)).String(), ) requireT.Equal(cwExtensionBalanceBefore.String(), cwExtensionBalanceAfter.String()) requireT.Equal( totalSupplyBefore.Supply.String(), - totalSupplyAfter.Supply.Add(sdk.NewCoin(burnableDenom, sdkmath.NewInt(AmountBurningTrigger))).String(), + totalSupplyAfter.Supply.Add(sdk.NewCoin(burnableDenom, AmountBurningTrigger)).String(), ) balance := bankKeeper.GetBalance(ctx, issuer, burnableDenom) @@ -619,10 +621,10 @@ func TestKeeper_Extension_Burn(t *testing.T) { requireT.ErrorIs(err, cosmoserrors.ErrUnauthorized) // try to burn non-issuer frozen coins - err = ftKeeper.Freeze(ctx, issuer, recipient, sdk.NewCoin(burnableDenom, sdkmath.NewInt(AmountBurningTrigger))) + err = ftKeeper.Freeze(ctx, issuer, recipient, sdk.NewCoin(burnableDenom, AmountBurningTrigger)) requireT.NoError(err) err = bankKeeper.SendCoins(ctx, recipient, issuer, sdk.NewCoins( - sdk.NewCoin(burnableDenom, sdkmath.NewInt(AmountBurningTrigger))), + sdk.NewCoin(burnableDenom, AmountBurningTrigger)), ) requireT.ErrorContains(err, "Requested transfer token is frozen.") } @@ -669,7 +671,7 @@ func TestKeeper_Extension_Mint(t *testing.T) { // try to mint unmintable token err = bankKeeper.SendCoins(ctx, addr, addr, sdk.NewCoins( - sdk.NewCoin(unmintableDenom, sdkmath.NewInt(AmountMintingTrigger))), + sdk.NewCoin(unmintableDenom, AmountMintingTrigger)), ) requireT.ErrorContains(err, "feature minting is disabled") @@ -692,7 +694,7 @@ func TestKeeper_Extension_Mint(t *testing.T) { mintableDenom, err := ftKeeper.Issue(ctx, settings) requireT.NoError(err) - coinsToMint := sdk.NewCoins(sdk.NewCoin(mintableDenom, sdkmath.NewInt(AmountMintingTrigger))) + coinsToMint := sdk.NewCoins(sdk.NewCoin(mintableDenom, AmountMintingTrigger)) randomAddr := sdk.AccAddress(ed25519.GenPrivKey().PubKey().Address()) @@ -780,7 +782,7 @@ func TestKeeper_Extension_BurnRate_BankSend(t *testing.T) { // send trigger amount from recipient1 to recipient2 (burn must not apply if the extension decides) recipient2 := sdk.AccAddress(secp256k1.GenPrivKey().PubKey().Address()) err = bankKeeper.SendCoins(ctx, recipient, recipient2, sdk.NewCoins( - sdk.NewCoin(denom, sdkmath.NewInt(AmountIgnoreBurnRateTrigger)), + sdk.NewCoin(denom, AmountIgnoreBurnRateTrigger), )) requireT.NoError(err) @@ -1071,7 +1073,7 @@ func TestKeeper_Extension_SendCommissionRate_BankSend(t *testing.T) { // send trigger amount from recipient1 to recipient2 (send commission rate must not apply) recipient2 := sdk.AccAddress(secp256k1.GenPrivKey().PubKey().Address()) err = bankKeeper.SendCoins(ctx, recipient, recipient2, sdk.NewCoins( - sdk.NewCoin(denom, sdkmath.NewInt(AmountIgnoreSendCommissionRateTrigger)), + sdk.NewCoin(denom, AmountIgnoreSendCommissionRateTrigger), )) requireT.NoError(err) diff --git a/x/dex/keeper/keeper_ft_test.go b/x/dex/keeper/keeper_ft_test.go index d8107b844..1cace6abb 100644 --- a/x/dex/keeper/keeper_ft_test.go +++ b/x/dex/keeper/keeper_ft_test.go @@ -24,6 +24,11 @@ import ( "github.com/CoreumFoundation/coreum/v5/x/dex/types" ) +var ( + AmountDEXExpectToSpendTrigger = sdkmath.NewInt(103) + AmountDEXExpectToReceiveTrigger = sdkmath.NewInt(104) +) + func TestKeeper_PlaceOrderWithExtension(t *testing.T) { testApp := simapp.New() sdkCtx := testApp.BaseApp.NewContextLegacy(false, tmproto.Header{ @@ -89,7 +94,7 @@ func TestKeeper_PlaceOrderWithExtension(t *testing.T) { BaseDenom: denomWithExtension, QuoteDenom: denom2, Price: lo.ToPtr(types.MustNewPriceFromString("1")), - Quantity: sdkmath.NewInt(103), // 103 is prohibited by extensions smart contract + Quantity: AmountDEXExpectToSpendTrigger, Side: types.SIDE_SELL, TimeInForce: types.TIME_IN_FORCE_GTC, }, @@ -125,7 +130,7 @@ func TestKeeper_PlaceOrderWithExtension(t *testing.T) { BaseDenom: denom2, QuoteDenom: denomWithExtension, Price: lo.ToPtr(types.MustNewPriceFromString("1")), - Quantity: sdkmath.NewInt(104), // 104 is prohibited by extensions smart contract + Quantity: AmountDEXExpectToReceiveTrigger, Side: types.SIDE_BUY, TimeInForce: types.TIME_IN_FORCE_GTC, }, From ef22d920ce6823bd8f4a409fe4665a6e7192a197 Mon Sep 17 00:00:00 2001 From: Dzmitry Hil Date: Tue, 26 Nov 2024 10:06:49 +0300 Subject: [PATCH 5/5] Merge conflicts --- x/asset/ft/keeper/keeper.go | 16 ++++++++-------- x/asset/ft/keeper/keeper_dex.go | 6 +++++- 2 files changed, 13 insertions(+), 9 deletions(-) diff --git a/x/asset/ft/keeper/keeper.go b/x/asset/ft/keeper/keeper.go index 90ba81af8..2dd58f506 100644 --- a/x/asset/ft/keeper/keeper.go +++ b/x/asset/ft/keeper/keeper.go @@ -922,14 +922,14 @@ func (k Keeper) validateCoinSpendable( return nil } - isGloballyFrozen, err := k.isGloballyFrozen(ctx, def.Denom) - if err != nil { - return err - } - if def.IsFeatureEnabled(types.Feature_freezing) && - isGloballyFrozen && - !def.HasAdminPrivileges(addr) { - return sdkerrors.Wrapf(types.ErrGloballyFrozen, "%s is globally frozen", def.Denom) + if def.IsFeatureEnabled(types.Feature_freezing) { + isGloballyFrozen, err := k.isGloballyFrozen(ctx, def.Denom) + if err != nil { + return err + } + if isGloballyFrozen && !def.HasAdminPrivileges(addr) { + return sdkerrors.Wrapf(types.ErrGloballyFrozen, "%s is globally frozen", def.Denom) + } } // Checking for IBC-received transfer is done here (after call to k.isGloballyFrozen), because those transfers diff --git a/x/asset/ft/keeper/keeper_dex.go b/x/asset/ft/keeper/keeper_dex.go index 8bc822522..2ed0e7e50 100644 --- a/x/asset/ft/keeper/keeper_dex.go +++ b/x/asset/ft/keeper/keeper_dex.go @@ -424,7 +424,11 @@ func (k Keeper) dexCheckExpectedToSpend( } if spendDef.IsFeatureEnabled(types.Feature_freezing) && !spendDef.HasAdminPrivileges(order.Creator) { - frozenAmt := k.GetFrozenBalance(ctx, order.Creator, expectedToSpend.Denom).Amount + frozenCoin, err := k.GetFrozenBalance(ctx, order.Creator, expectedToSpend.Denom) + if err != nil { + return err + } + frozenAmt := frozenCoin.Amount notFrozenTotalAmt := balance.Amount.Sub(frozenAmt) if notFrozenTotalAmt.LT(expectedToSpend.Amount) { return sdkerrors.Wrapf(