mirror of
https://github.com/0glabs/0g-chain.git
synced 2025-01-04 04:35:18 +00:00
614d4e40fe
* Update cometbft, cosmos, ethermint, and ibc-go * Replace github.com/tendermint/tendermint by github.com/cometbft/cometbft * Replace github.com/tendermint/tm-db by github.com/cometbft/cometbft-db * Replace gogo/protobuf with cosmos/gogoproto & simapp replacement * Replace cosmos-sdk/simapp/helpers with cosmos-sdk/testutil/sims * Remove no longer used simulations * Replace ibchost with ibcexported See https://github.com/cosmos/ibc-go/blob/v7.2.2/docs/migrations/v6-to-v7.md#ibc-module-constants * Add new consensus params keeper * Add consensus keeper to blockers * Fix keeper and module issues in app.go * Add IsSendEnabledCoins and update SetParams interface changes * Fix protobuf build for cosmos 47 (#1800) * fix cp errors by using -f; fix lint by only linting our proto dir; and use proofs.proto directly from ics23 for ibc-go v7 * run proto-all; commit updated third party deps and swagger changes * regenerate proto files * use correct gocosmos build plugin for buf * re-gen all protobuf files to update paths for new gocosmos plugin * update protoc and buf to latest versions * fix staking keeper issues in app.go * update tally handler for gov changes * chain id fix and flag fixes * update deps for cometbft 47.7 upgrade * remove all module legacy queriers * update stakingKeeper to pointer * Replace ModuleCdc from govv1beta1 to govcodec * remove simulations * abci.LastCommitInfo → abci.CommitInfo * Remove unused code in keys.go * simapp.MakeTestEncodingConfig -> moduletestutil.MakeTestEncodingConfi * Fix chain id issues in tests * Fix remaining unit test issues * Update changelog for upgrade * Fix e2e tests using updated kvtool * Update protonet to v47 compatible genesis * Bump cometbft-db to v0.9.1-kava.1 * Update kvtool * Remove extra changelog * Fix merged rocksdb issues * go mod cleanup * Bump cometbft-db to v9 and go to 1.21 * Bump rocksdb version to v8.10.0 * Update kvtool to latest version * Update gin to v1.9.0 * Use ibctm.ModuleName in app_test * Fallback to genesis chain id instead of client toml * Remove all simulations * Fix cdp migrations issue with v47 * Update dependencies to correct tags --------- Co-authored-by: Nick DeLuca <nickdeluca08@gmail.com>
632 lines
34 KiB
Go
632 lines
34 KiB
Go
package keeper_test
|
|
|
|
import (
|
|
"errors"
|
|
"fmt"
|
|
|
|
sdkmath "cosmossdk.io/math"
|
|
tmproto "github.com/cometbft/cometbft/proto/tendermint/types"
|
|
tmtime "github.com/cometbft/cometbft/types/time"
|
|
sdk "github.com/cosmos/cosmos-sdk/types"
|
|
sdkerrors "github.com/cosmos/cosmos-sdk/types/errors"
|
|
"github.com/kava-labs/kava/x/swap/types"
|
|
)
|
|
|
|
func (suite *keeperTestSuite) TestSwapExactForTokens() {
|
|
suite.Keeper.SetParams(suite.Ctx, types.Params{
|
|
SwapFee: sdk.MustNewDecFromStr("0.0025"),
|
|
})
|
|
owner := suite.CreateAccount(sdk.Coins{})
|
|
reserves := sdk.NewCoins(
|
|
sdk.NewCoin("ukava", sdkmath.NewInt(1000e6)),
|
|
sdk.NewCoin("usdx", sdkmath.NewInt(5000e6)),
|
|
)
|
|
totalShares := sdkmath.NewInt(30e6)
|
|
poolID := suite.setupPool(reserves, totalShares, owner.GetAddress())
|
|
|
|
balance := sdk.NewCoins(
|
|
sdk.NewCoin("ukava", sdkmath.NewInt(10e6)),
|
|
)
|
|
requester := suite.NewAccountFromAddr(sdk.AccAddress("requester-----------"), balance)
|
|
coinA := sdk.NewCoin("ukava", sdkmath.NewInt(1e6))
|
|
coinB := sdk.NewCoin("usdx", sdkmath.NewInt(5e6))
|
|
|
|
err := suite.Keeper.SwapExactForTokens(suite.Ctx, requester.GetAddress(), coinA, coinB, sdk.MustNewDecFromStr("0.01"))
|
|
suite.Require().NoError(err)
|
|
|
|
expectedOutput := sdk.NewCoin("usdx", sdkmath.NewInt(4982529))
|
|
|
|
suite.AccountBalanceEqual(requester.GetAddress(), balance.Sub(coinA).Add(expectedOutput))
|
|
suite.ModuleAccountBalanceEqual(reserves.Add(coinA).Sub(expectedOutput))
|
|
suite.PoolLiquidityEqual(reserves.Add(coinA).Sub(expectedOutput))
|
|
|
|
suite.EventsContains(suite.Ctx.EventManager().Events(), sdk.NewEvent(
|
|
types.EventTypeSwapTrade,
|
|
sdk.NewAttribute(types.AttributeKeyPoolID, poolID),
|
|
sdk.NewAttribute(types.AttributeKeyRequester, requester.GetAddress().String()),
|
|
sdk.NewAttribute(types.AttributeKeySwapInput, coinA.String()),
|
|
sdk.NewAttribute(types.AttributeKeySwapOutput, expectedOutput.String()),
|
|
sdk.NewAttribute(types.AttributeKeyFeePaid, "2500ukava"),
|
|
sdk.NewAttribute(types.AttributeKeyExactDirection, "input"),
|
|
))
|
|
}
|
|
|
|
func (suite *keeperTestSuite) TestSwapExactForTokens_OutputGreaterThanZero() {
|
|
owner := suite.CreateAccount(sdk.Coins{})
|
|
reserves := sdk.NewCoins(
|
|
sdk.NewCoin("ukava", sdkmath.NewInt(10e6)),
|
|
sdk.NewCoin("usdx", sdkmath.NewInt(50e6)),
|
|
)
|
|
totalShares := sdkmath.NewInt(30e6)
|
|
suite.setupPool(reserves, totalShares, owner.GetAddress())
|
|
|
|
balance := sdk.NewCoins(
|
|
sdk.NewCoin("usdx", sdkmath.NewInt(10e6)),
|
|
)
|
|
requester := suite.NewAccountFromAddr(sdk.AccAddress("requester-----------"), balance)
|
|
coinA := sdk.NewCoin("usdx", sdkmath.NewInt(5))
|
|
coinB := sdk.NewCoin("ukava", sdkmath.NewInt(1))
|
|
|
|
err := suite.Keeper.SwapExactForTokens(suite.Ctx, requester.GetAddress(), coinA, coinB, sdk.MustNewDecFromStr("1"))
|
|
suite.EqualError(err, "swap output rounds to zero, increase input amount: insufficient liquidity")
|
|
}
|
|
|
|
func (suite *keeperTestSuite) TestSwapExactForTokens_Slippage() {
|
|
owner := suite.CreateAccount(sdk.Coins{})
|
|
reserves := sdk.NewCoins(
|
|
sdk.NewCoin("ukava", sdkmath.NewInt(100e6)),
|
|
sdk.NewCoin("usdx", sdkmath.NewInt(500e6)),
|
|
)
|
|
totalShares := sdkmath.NewInt(30e6)
|
|
suite.setupPool(reserves, totalShares, owner.GetAddress())
|
|
|
|
testCases := []struct {
|
|
coinA sdk.Coin
|
|
coinB sdk.Coin
|
|
slippage sdk.Dec
|
|
fee sdk.Dec
|
|
shouldFail bool
|
|
}{
|
|
// positive slippage OK
|
|
{sdk.NewCoin("ukava", sdkmath.NewInt(1e6)), sdk.NewCoin("usdx", sdkmath.NewInt(2e6)), sdk.MustNewDecFromStr("0.01"), sdk.MustNewDecFromStr("0.0025"), false},
|
|
{sdk.NewCoin("ukava", sdkmath.NewInt(1e6)), sdk.NewCoin("usdx", sdkmath.NewInt(4e6)), sdk.MustNewDecFromStr("0.01"), sdk.MustNewDecFromStr("0.0025"), false},
|
|
{sdk.NewCoin("usdx", sdkmath.NewInt(50e6)), sdk.NewCoin("ukava", sdkmath.NewInt(5e6)), sdk.MustNewDecFromStr("0.01"), sdk.MustNewDecFromStr("0.0025"), false},
|
|
{sdk.NewCoin("usdx", sdkmath.NewInt(50e6)), sdk.NewCoin("ukava", sdkmath.NewInt(1e6)), sdk.MustNewDecFromStr("0.01"), sdk.MustNewDecFromStr("0.0025"), false},
|
|
// positive slippage with zero slippage OK
|
|
{sdk.NewCoin("ukava", sdkmath.NewInt(1e6)), sdk.NewCoin("usdx", sdkmath.NewInt(2e6)), sdk.ZeroDec(), sdk.MustNewDecFromStr("0.0025"), false},
|
|
{sdk.NewCoin("ukava", sdkmath.NewInt(1e6)), sdk.NewCoin("usdx", sdkmath.NewInt(4e6)), sdk.ZeroDec(), sdk.MustNewDecFromStr("0.0025"), false},
|
|
{sdk.NewCoin("usdx", sdkmath.NewInt(50e6)), sdk.NewCoin("ukava", sdkmath.NewInt(5e6)), sdk.ZeroDec(), sdk.MustNewDecFromStr("0.0025"), false},
|
|
{sdk.NewCoin("usdx", sdkmath.NewInt(50e6)), sdk.NewCoin("ukava", sdkmath.NewInt(1e6)), sdk.ZeroDec(), sdk.MustNewDecFromStr("0.0025"), false},
|
|
// exact zero slippage OK
|
|
{sdk.NewCoin("ukava", sdkmath.NewInt(1e6)), sdk.NewCoin("usdx", sdkmath.NewInt(4950495)), sdk.ZeroDec(), sdk.MustNewDecFromStr("0"), false},
|
|
{sdk.NewCoin("ukava", sdkmath.NewInt(1e6)), sdk.NewCoin("usdx", sdkmath.NewInt(4935790)), sdk.ZeroDec(), sdk.MustNewDecFromStr("0.003"), false},
|
|
{sdk.NewCoin("ukava", sdkmath.NewInt(1e6)), sdk.NewCoin("usdx", sdkmath.NewInt(4705299)), sdk.ZeroDec(), sdk.MustNewDecFromStr("0.05"), false},
|
|
{sdk.NewCoin("usdx", sdkmath.NewInt(5e6)), sdk.NewCoin("ukava", sdkmath.NewInt(990099)), sdk.ZeroDec(), sdk.MustNewDecFromStr("0"), false},
|
|
{sdk.NewCoin("usdx", sdkmath.NewInt(5e6)), sdk.NewCoin("ukava", sdkmath.NewInt(987158)), sdk.ZeroDec(), sdk.MustNewDecFromStr("0.003"), false},
|
|
{sdk.NewCoin("usdx", sdkmath.NewInt(5e6)), sdk.NewCoin("ukava", sdkmath.NewInt(941059)), sdk.ZeroDec(), sdk.MustNewDecFromStr("0.05"), false},
|
|
// slippage failure, zero slippage tolerance
|
|
{sdk.NewCoin("ukava", sdkmath.NewInt(1e6)), sdk.NewCoin("usdx", sdkmath.NewInt(4950496)), sdk.ZeroDec(), sdk.MustNewDecFromStr("0"), true},
|
|
{sdk.NewCoin("ukava", sdkmath.NewInt(1e6)), sdk.NewCoin("usdx", sdkmath.NewInt(4935793)), sdk.ZeroDec(), sdk.MustNewDecFromStr("0.003"), true},
|
|
{sdk.NewCoin("ukava", sdkmath.NewInt(1e6)), sdk.NewCoin("usdx", sdkmath.NewInt(4705300)), sdk.ZeroDec(), sdk.MustNewDecFromStr("0.05"), true},
|
|
{sdk.NewCoin("usdx", sdkmath.NewInt(5e6)), sdk.NewCoin("ukava", sdkmath.NewInt(990100)), sdk.ZeroDec(), sdk.MustNewDecFromStr("0"), true},
|
|
{sdk.NewCoin("usdx", sdkmath.NewInt(5e6)), sdk.NewCoin("ukava", sdkmath.NewInt(987159)), sdk.ZeroDec(), sdk.MustNewDecFromStr("0.003"), true},
|
|
{sdk.NewCoin("usdx", sdkmath.NewInt(5e6)), sdk.NewCoin("ukava", sdkmath.NewInt(941060)), sdk.ZeroDec(), sdk.MustNewDecFromStr("0.05"), true},
|
|
// slippage failure, 1 percent slippage
|
|
{sdk.NewCoin("ukava", sdkmath.NewInt(1e6)), sdk.NewCoin("usdx", sdkmath.NewInt(5000501)), sdk.MustNewDecFromStr("0.01"), sdk.MustNewDecFromStr("0"), true},
|
|
{sdk.NewCoin("ukava", sdkmath.NewInt(1e6)), sdk.NewCoin("usdx", sdkmath.NewInt(4985647)), sdk.MustNewDecFromStr("0.01"), sdk.MustNewDecFromStr("0.003"), true},
|
|
{sdk.NewCoin("ukava", sdkmath.NewInt(1e6)), sdk.NewCoin("usdx", sdkmath.NewInt(4752828)), sdk.MustNewDecFromStr("0.01"), sdk.MustNewDecFromStr("0.05"), true},
|
|
{sdk.NewCoin("usdx", sdkmath.NewInt(5e6)), sdk.NewCoin("ukava", sdkmath.NewInt(1000101)), sdk.MustNewDecFromStr("0.01"), sdk.MustNewDecFromStr("0"), true},
|
|
{sdk.NewCoin("usdx", sdkmath.NewInt(5e6)), sdk.NewCoin("ukava", sdkmath.NewInt(997130)), sdk.MustNewDecFromStr("0.01"), sdk.MustNewDecFromStr("0.003"), true},
|
|
{sdk.NewCoin("usdx", sdkmath.NewInt(5e6)), sdk.NewCoin("ukava", sdkmath.NewInt(950565)), sdk.MustNewDecFromStr("0.01"), sdk.MustNewDecFromStr("0.05"), true},
|
|
// slippage OK, 1 percent slippage
|
|
{sdk.NewCoin("ukava", sdkmath.NewInt(1e6)), sdk.NewCoin("usdx", sdkmath.NewInt(5000500)), sdk.MustNewDecFromStr("0.01"), sdk.MustNewDecFromStr("0"), false},
|
|
{sdk.NewCoin("ukava", sdkmath.NewInt(1e6)), sdk.NewCoin("usdx", sdkmath.NewInt(4985646)), sdk.MustNewDecFromStr("0.01"), sdk.MustNewDecFromStr("0.003"), false},
|
|
{sdk.NewCoin("ukava", sdkmath.NewInt(1e6)), sdk.NewCoin("usdx", sdkmath.NewInt(4752827)), sdk.MustNewDecFromStr("0.01"), sdk.MustNewDecFromStr("0.05"), false},
|
|
{sdk.NewCoin("usdx", sdkmath.NewInt(5e6)), sdk.NewCoin("ukava", sdkmath.NewInt(1000100)), sdk.MustNewDecFromStr("0.01"), sdk.MustNewDecFromStr("0"), false},
|
|
{sdk.NewCoin("usdx", sdkmath.NewInt(5e6)), sdk.NewCoin("ukava", sdkmath.NewInt(997129)), sdk.MustNewDecFromStr("0.01"), sdk.MustNewDecFromStr("0.003"), false},
|
|
{sdk.NewCoin("usdx", sdkmath.NewInt(5e6)), sdk.NewCoin("ukava", sdkmath.NewInt(950564)), sdk.MustNewDecFromStr("0.01"), sdk.MustNewDecFromStr("0.05"), false},
|
|
}
|
|
|
|
for _, tc := range testCases {
|
|
suite.Run(fmt.Sprintf("coinA=%s coinB=%s slippage=%s fee=%s", tc.coinA, tc.coinB, tc.slippage, tc.fee), func() {
|
|
suite.SetupTest()
|
|
suite.Keeper.SetParams(suite.Ctx, types.Params{
|
|
SwapFee: tc.fee,
|
|
})
|
|
owner := suite.CreateAccount(sdk.Coins{})
|
|
reserves := sdk.NewCoins(
|
|
sdk.NewCoin("ukava", sdkmath.NewInt(100e6)),
|
|
sdk.NewCoin("usdx", sdkmath.NewInt(500e6)),
|
|
)
|
|
totalShares := sdkmath.NewInt(30e6)
|
|
suite.setupPool(reserves, totalShares, owner.GetAddress())
|
|
balance := sdk.NewCoins(
|
|
sdk.NewCoin("ukava", sdkmath.NewInt(100e6)),
|
|
sdk.NewCoin("usdx", sdkmath.NewInt(100e6)),
|
|
)
|
|
requester := suite.NewAccountFromAddr(sdk.AccAddress("requester-----------"), balance)
|
|
|
|
ctx := suite.App.NewContext(true, tmproto.Header{Height: 1, Time: tmtime.Now()})
|
|
err := suite.Keeper.SwapExactForTokens(ctx, requester.GetAddress(), tc.coinA, tc.coinB, tc.slippage)
|
|
|
|
if tc.shouldFail {
|
|
suite.Require().Error(err)
|
|
suite.Contains(err.Error(), "slippage exceeded")
|
|
} else {
|
|
suite.NoError(err)
|
|
}
|
|
})
|
|
}
|
|
}
|
|
|
|
func (suite *keeperTestSuite) TestSwapExactForTokens_InsufficientFunds() {
|
|
testCases := []struct {
|
|
name string
|
|
balanceA sdk.Coin
|
|
coinA sdk.Coin
|
|
coinB sdk.Coin
|
|
}{
|
|
{"no ukava balance", sdk.NewCoin("ukava", sdk.ZeroInt()), sdk.NewCoin("ukava", sdkmath.NewInt(100)), sdk.NewCoin("usdx", sdkmath.NewInt(500))},
|
|
{"no usdx balance", sdk.NewCoin("usdx", sdk.ZeroInt()), sdk.NewCoin("usdx", sdkmath.NewInt(500)), sdk.NewCoin("ukava", sdkmath.NewInt(100))},
|
|
{"low ukava balance", sdk.NewCoin("ukava", sdkmath.NewInt(1000000)), sdk.NewCoin("ukava", sdkmath.NewInt(1000001)), sdk.NewCoin("usdx", sdkmath.NewInt(5000000))},
|
|
{"low ukava balance", sdk.NewCoin("usdx", sdkmath.NewInt(5000000)), sdk.NewCoin("usdx", sdkmath.NewInt(5000001)), sdk.NewCoin("ukava", sdkmath.NewInt(1000000))},
|
|
{"large ukava balance difference", sdk.NewCoin("ukava", sdkmath.NewInt(100e6)), sdk.NewCoin("ukava", sdkmath.NewInt(1000e6)), sdk.NewCoin("usdx", sdkmath.NewInt(5000e6))},
|
|
{"large usdx balance difference", sdk.NewCoin("usdx", sdkmath.NewInt(500e6)), sdk.NewCoin("usdx", sdkmath.NewInt(5000e6)), sdk.NewCoin("ukava", sdkmath.NewInt(1000e6))},
|
|
}
|
|
|
|
for _, tc := range testCases {
|
|
suite.Run(tc.name, func() {
|
|
suite.SetupTest()
|
|
owner := suite.CreateAccount(sdk.Coins{})
|
|
reserves := sdk.NewCoins(
|
|
sdk.NewCoin("ukava", sdkmath.NewInt(100000e6)),
|
|
sdk.NewCoin("usdx", sdkmath.NewInt(500000e6)),
|
|
)
|
|
totalShares := sdkmath.NewInt(30000e6)
|
|
suite.setupPool(reserves, totalShares, owner.GetAddress())
|
|
balance := sdk.NewCoins(tc.balanceA)
|
|
requester := suite.NewAccountFromAddr(sdk.AccAddress("requester-----------"), balance)
|
|
|
|
ctx := suite.App.NewContext(true, tmproto.Header{Height: 1, Time: tmtime.Now()})
|
|
err := suite.Keeper.SwapExactForTokens(ctx, requester.GetAddress(), tc.coinA, tc.coinB, sdk.MustNewDecFromStr("0.1"))
|
|
suite.Require().True(errors.Is(err, sdkerrors.ErrInsufficientFunds), fmt.Sprintf("got err %s", err))
|
|
})
|
|
}
|
|
}
|
|
|
|
func (suite *keeperTestSuite) TestSwapExactForTokens_InsufficientFunds_Vesting() {
|
|
testCases := []struct {
|
|
name string
|
|
balanceA sdk.Coin
|
|
vestingA sdk.Coin
|
|
coinA sdk.Coin
|
|
coinB sdk.Coin
|
|
}{
|
|
{"no ukava balance, vesting only", sdk.NewCoin("ukava", sdk.ZeroInt()), sdk.NewCoin("ukava", sdkmath.NewInt(100)), sdk.NewCoin("ukava", sdkmath.NewInt(100)), sdk.NewCoin("usdx", sdkmath.NewInt(500))},
|
|
{"no usdx balance, vesting only", sdk.NewCoin("usdx", sdk.ZeroInt()), sdk.NewCoin("usdx", sdkmath.NewInt(500)), sdk.NewCoin("usdx", sdkmath.NewInt(500)), sdk.NewCoin("ukava", sdkmath.NewInt(100))},
|
|
{"low ukava balance, vesting matches exact", sdk.NewCoin("ukava", sdkmath.NewInt(1000000)), sdk.NewCoin("ukava", sdkmath.NewInt(1)), sdk.NewCoin("ukava", sdkmath.NewInt(1000001)), sdk.NewCoin("usdx", sdkmath.NewInt(5000000))},
|
|
{"low ukava balance, vesting matches exact", sdk.NewCoin("usdx", sdkmath.NewInt(5000000)), sdk.NewCoin("usdx", sdkmath.NewInt(1)), sdk.NewCoin("usdx", sdkmath.NewInt(5000001)), sdk.NewCoin("ukava", sdkmath.NewInt(1000000))},
|
|
{"large ukava balance difference, vesting covers difference", sdk.NewCoin("ukava", sdkmath.NewInt(100e6)), sdk.NewCoin("ukava", sdkmath.NewInt(1000e6)), sdk.NewCoin("ukava", sdkmath.NewInt(1000e6)), sdk.NewCoin("usdx", sdkmath.NewInt(5000e6))},
|
|
{"large usdx balance difference, vesting covers difference", sdk.NewCoin("usdx", sdkmath.NewInt(500e6)), sdk.NewCoin("usdx", sdkmath.NewInt(5000e6)), sdk.NewCoin("usdx", sdkmath.NewInt(5000e6)), sdk.NewCoin("ukava", sdkmath.NewInt(1000e6))},
|
|
}
|
|
|
|
for _, tc := range testCases {
|
|
suite.Run(tc.name, func() {
|
|
suite.SetupTest()
|
|
owner := suite.CreateAccount(sdk.Coins{})
|
|
reserves := sdk.NewCoins(
|
|
sdk.NewCoin("ukava", sdkmath.NewInt(100000e6)),
|
|
sdk.NewCoin("usdx", sdkmath.NewInt(500000e6)),
|
|
)
|
|
totalShares := sdkmath.NewInt(30000e6)
|
|
suite.setupPool(reserves, totalShares, owner.GetAddress())
|
|
balance := sdk.NewCoins(tc.balanceA)
|
|
vesting := sdk.NewCoins(tc.vestingA)
|
|
requester := suite.CreateVestingAccount(balance, vesting)
|
|
|
|
ctx := suite.App.NewContext(true, tmproto.Header{Height: 1, Time: tmtime.Now()})
|
|
err := suite.Keeper.SwapExactForTokens(ctx, requester.GetAddress(), tc.coinA, tc.coinB, sdk.MustNewDecFromStr("0.1"))
|
|
suite.Require().True(errors.Is(err, sdkerrors.ErrInsufficientFunds), fmt.Sprintf("got err %s", err))
|
|
})
|
|
}
|
|
}
|
|
|
|
func (suite *keeperTestSuite) TestSwapExactForTokens_PoolNotFound() {
|
|
owner := suite.CreateAccount(sdk.Coins{})
|
|
reserves := sdk.NewCoins(
|
|
sdk.NewCoin("ukava", sdkmath.NewInt(1000e6)),
|
|
sdk.NewCoin("usdx", sdkmath.NewInt(5000e6)),
|
|
)
|
|
totalShares := sdkmath.NewInt(3000e6)
|
|
poolID := suite.setupPool(reserves, totalShares, owner.GetAddress())
|
|
suite.Keeper.DeletePool(suite.Ctx, poolID)
|
|
|
|
balance := sdk.NewCoins(
|
|
sdk.NewCoin("ukava", sdkmath.NewInt(10e6)),
|
|
sdk.NewCoin("usdx", sdkmath.NewInt(10e6)),
|
|
)
|
|
requester := suite.NewAccountFromAddr(sdk.AccAddress("requester-----------"), balance)
|
|
coinA := sdk.NewCoin("ukava", sdkmath.NewInt(1e6))
|
|
coinB := sdk.NewCoin("usdx", sdkmath.NewInt(5e6))
|
|
|
|
err := suite.Keeper.SwapExactForTokens(suite.Ctx, requester.GetAddress(), coinA, coinB, sdk.MustNewDecFromStr("0.01"))
|
|
suite.EqualError(err, "pool ukava:usdx not found: invalid pool")
|
|
|
|
err = suite.Keeper.SwapExactForTokens(suite.Ctx, requester.GetAddress(), coinB, coinA, sdk.MustNewDecFromStr("0.01"))
|
|
suite.EqualError(err, "pool ukava:usdx not found: invalid pool")
|
|
}
|
|
|
|
func (suite *keeperTestSuite) TestSwapExactForTokens_PanicOnInvalidPool() {
|
|
owner := suite.CreateAccount(sdk.Coins{})
|
|
reserves := sdk.NewCoins(
|
|
sdk.NewCoin("ukava", sdkmath.NewInt(1000e6)),
|
|
sdk.NewCoin("usdx", sdkmath.NewInt(5000e6)),
|
|
)
|
|
totalShares := sdkmath.NewInt(3000e6)
|
|
poolID := suite.setupPool(reserves, totalShares, owner.GetAddress())
|
|
|
|
poolRecord, found := suite.Keeper.GetPool(suite.Ctx, poolID)
|
|
suite.Require().True(found, "expected pool record to exist")
|
|
|
|
poolRecord.TotalShares = sdk.ZeroInt()
|
|
suite.Keeper.SetPool_Raw(suite.Ctx, poolRecord)
|
|
|
|
balance := sdk.NewCoins(
|
|
sdk.NewCoin("ukava", sdkmath.NewInt(10e6)),
|
|
sdk.NewCoin("usdx", sdkmath.NewInt(10e6)),
|
|
)
|
|
requester := suite.NewAccountFromAddr(sdk.AccAddress("requester-----------"), balance)
|
|
coinA := sdk.NewCoin("ukava", sdkmath.NewInt(1e6))
|
|
coinB := sdk.NewCoin("usdx", sdkmath.NewInt(5e6))
|
|
|
|
suite.PanicsWithValue("invalid pool ukava:usdx: total shares must be greater than zero: invalid pool", func() {
|
|
_ = suite.Keeper.SwapExactForTokens(suite.Ctx, requester.GetAddress(), coinA, coinB, sdk.MustNewDecFromStr("0.01"))
|
|
}, "expected invalid pool record to panic")
|
|
|
|
suite.PanicsWithValue("invalid pool ukava:usdx: total shares must be greater than zero: invalid pool", func() {
|
|
_ = suite.Keeper.SwapExactForTokens(suite.Ctx, requester.GetAddress(), coinB, coinA, sdk.MustNewDecFromStr("0.01"))
|
|
}, "expected invalid pool record to panic")
|
|
}
|
|
|
|
func (suite *keeperTestSuite) TestSwapExactForTokens_PanicOnInsufficientModuleAccFunds() {
|
|
owner := suite.CreateAccount(sdk.Coins{})
|
|
reserves := sdk.NewCoins(
|
|
sdk.NewCoin("ukava", sdkmath.NewInt(1000e6)),
|
|
sdk.NewCoin("usdx", sdkmath.NewInt(5000e6)),
|
|
)
|
|
totalShares := sdkmath.NewInt(3000e6)
|
|
suite.setupPool(reserves, totalShares, owner.GetAddress())
|
|
|
|
suite.RemoveCoinsFromModule(sdk.NewCoins(
|
|
sdk.NewCoin("ukava", sdkmath.NewInt(1000e6)),
|
|
sdk.NewCoin("usdx", sdkmath.NewInt(5000e6)),
|
|
))
|
|
|
|
balance := sdk.NewCoins(
|
|
sdk.NewCoin("ukava", sdkmath.NewInt(10e6)),
|
|
sdk.NewCoin("usdx", sdkmath.NewInt(10e6)),
|
|
)
|
|
requester := suite.NewAccountFromAddr(sdk.AccAddress("requester-----------"), balance)
|
|
coinA := sdk.NewCoin("ukava", sdkmath.NewInt(1e6))
|
|
coinB := sdk.NewCoin("usdx", sdkmath.NewInt(5e6))
|
|
|
|
suite.Panics(func() {
|
|
_ = suite.Keeper.SwapExactForTokens(suite.Ctx, requester.GetAddress(), coinA, coinB, sdk.MustNewDecFromStr("0.01"))
|
|
}, "expected panic when module account does not have enough funds")
|
|
|
|
suite.Panics(func() {
|
|
_ = suite.Keeper.SwapExactForTokens(suite.Ctx, requester.GetAddress(), coinA, coinB, sdk.MustNewDecFromStr("0.01"))
|
|
}, "expected panic when module account does not have enough funds")
|
|
}
|
|
|
|
func (suite *keeperTestSuite) TestSwapForExactTokens() {
|
|
suite.Keeper.SetParams(suite.Ctx, types.Params{
|
|
SwapFee: sdk.MustNewDecFromStr("0.0025"),
|
|
})
|
|
owner := suite.CreateAccount(sdk.Coins{})
|
|
reserves := sdk.NewCoins(
|
|
sdk.NewCoin("ukava", sdkmath.NewInt(1000e6)),
|
|
sdk.NewCoin("usdx", sdkmath.NewInt(5000e6)),
|
|
)
|
|
totalShares := sdkmath.NewInt(30e6)
|
|
poolID := suite.setupPool(reserves, totalShares, owner.GetAddress())
|
|
|
|
balance := sdk.NewCoins(
|
|
sdk.NewCoin("ukava", sdkmath.NewInt(10e6)),
|
|
)
|
|
requester := suite.NewAccountFromAddr(sdk.AccAddress("requester-----------"), balance)
|
|
coinA := sdk.NewCoin("ukava", sdkmath.NewInt(1e6))
|
|
coinB := sdk.NewCoin("usdx", sdkmath.NewInt(5e6))
|
|
|
|
err := suite.Keeper.SwapForExactTokens(suite.Ctx, requester.GetAddress(), coinA, coinB, sdk.MustNewDecFromStr("0.01"))
|
|
suite.Require().NoError(err)
|
|
|
|
expectedInput := sdk.NewCoin("ukava", sdkmath.NewInt(1003511))
|
|
|
|
suite.AccountBalanceEqual(requester.GetAddress(), balance.Sub(expectedInput).Add(coinB))
|
|
suite.ModuleAccountBalanceEqual(reserves.Add(expectedInput).Sub(coinB))
|
|
suite.PoolLiquidityEqual(reserves.Add(expectedInput).Sub(coinB))
|
|
|
|
suite.EventsContains(suite.Ctx.EventManager().Events(), sdk.NewEvent(
|
|
types.EventTypeSwapTrade,
|
|
sdk.NewAttribute(types.AttributeKeyPoolID, poolID),
|
|
sdk.NewAttribute(types.AttributeKeyRequester, requester.GetAddress().String()),
|
|
sdk.NewAttribute(types.AttributeKeySwapInput, expectedInput.String()),
|
|
sdk.NewAttribute(types.AttributeKeySwapOutput, coinB.String()),
|
|
sdk.NewAttribute(types.AttributeKeyFeePaid, "2509ukava"),
|
|
sdk.NewAttribute(types.AttributeKeyExactDirection, "output"),
|
|
))
|
|
}
|
|
|
|
func (suite *keeperTestSuite) TestSwapForExactTokens_OutputLessThanPoolReserves() {
|
|
owner := suite.CreateAccount(sdk.Coins{})
|
|
reserves := sdk.NewCoins(
|
|
sdk.NewCoin("ukava", sdkmath.NewInt(100e6)),
|
|
sdk.NewCoin("usdx", sdkmath.NewInt(500e6)),
|
|
)
|
|
totalShares := sdkmath.NewInt(300e6)
|
|
suite.setupPool(reserves, totalShares, owner.GetAddress())
|
|
|
|
balance := sdk.NewCoins(
|
|
sdk.NewCoin("ukava", sdkmath.NewInt(1000e6)),
|
|
)
|
|
requester := suite.NewAccountFromAddr(sdk.AccAddress("requester-----------"), balance)
|
|
coinA := sdk.NewCoin("ukava", sdkmath.NewInt(1e6))
|
|
|
|
coinB := sdk.NewCoin("usdx", sdkmath.NewInt(500e6).Add(sdk.OneInt()))
|
|
err := suite.Keeper.SwapForExactTokens(suite.Ctx, requester.GetAddress(), coinA, coinB, sdk.MustNewDecFromStr("0.01"))
|
|
suite.EqualError(err, "output 500000001 >= pool reserves 500000000: insufficient liquidity")
|
|
|
|
coinB = sdk.NewCoin("usdx", sdkmath.NewInt(500e6))
|
|
err = suite.Keeper.SwapForExactTokens(suite.Ctx, requester.GetAddress(), coinA, coinB, sdk.MustNewDecFromStr("0.01"))
|
|
suite.EqualError(err, "output 500000000 >= pool reserves 500000000: insufficient liquidity")
|
|
}
|
|
|
|
func (suite *keeperTestSuite) TestSwapForExactTokens_Slippage() {
|
|
owner := suite.CreateAccount(sdk.Coins{})
|
|
reserves := sdk.NewCoins(
|
|
sdk.NewCoin("ukava", sdkmath.NewInt(100e6)),
|
|
sdk.NewCoin("usdx", sdkmath.NewInt(500e6)),
|
|
)
|
|
totalShares := sdkmath.NewInt(30e6)
|
|
suite.setupPool(reserves, totalShares, owner.GetAddress())
|
|
|
|
testCases := []struct {
|
|
coinA sdk.Coin
|
|
coinB sdk.Coin
|
|
slippage sdk.Dec
|
|
fee sdk.Dec
|
|
shouldFail bool
|
|
}{
|
|
// positive slippage OK
|
|
{sdk.NewCoin("ukava", sdkmath.NewInt(5e6)), sdk.NewCoin("usdx", sdkmath.NewInt(5e6)), sdk.MustNewDecFromStr("0.01"), sdk.MustNewDecFromStr("0.0025"), false},
|
|
{sdk.NewCoin("ukava", sdkmath.NewInt(5e6)), sdk.NewCoin("usdx", sdkmath.NewInt(5e6)), sdk.MustNewDecFromStr("0.01"), sdk.MustNewDecFromStr("0.0025"), false},
|
|
{sdk.NewCoin("usdx", sdkmath.NewInt(100e6)), sdk.NewCoin("ukava", sdkmath.NewInt(10e6)), sdk.MustNewDecFromStr("0.01"), sdk.MustNewDecFromStr("0.0025"), false},
|
|
{sdk.NewCoin("usdx", sdkmath.NewInt(100e6)), sdk.NewCoin("ukava", sdkmath.NewInt(10e6)), sdk.MustNewDecFromStr("0.01"), sdk.MustNewDecFromStr("0.0025"), false},
|
|
// positive slippage with zero slippage OK
|
|
{sdk.NewCoin("ukava", sdkmath.NewInt(5e6)), sdk.NewCoin("usdx", sdkmath.NewInt(5e6)), sdk.ZeroDec(), sdk.MustNewDecFromStr("0.0025"), false},
|
|
{sdk.NewCoin("ukava", sdkmath.NewInt(5e6)), sdk.NewCoin("usdx", sdkmath.NewInt(5e6)), sdk.ZeroDec(), sdk.MustNewDecFromStr("0.0025"), false},
|
|
{sdk.NewCoin("usdx", sdkmath.NewInt(100e6)), sdk.NewCoin("ukava", sdkmath.NewInt(10e6)), sdk.ZeroDec(), sdk.MustNewDecFromStr("0.0025"), false},
|
|
{sdk.NewCoin("usdx", sdkmath.NewInt(100e6)), sdk.NewCoin("ukava", sdkmath.NewInt(10e6)), sdk.ZeroDec(), sdk.MustNewDecFromStr("0.0025"), false},
|
|
// exact zero slippage OK
|
|
{sdk.NewCoin("ukava", sdkmath.NewInt(1010102)), sdk.NewCoin("usdx", sdkmath.NewInt(5e6)), sdk.ZeroDec(), sdk.MustNewDecFromStr("0"), false},
|
|
{sdk.NewCoin("ukava", sdkmath.NewInt(1010102)), sdk.NewCoin("usdx", sdkmath.NewInt(5e6)), sdk.ZeroDec(), sdk.MustNewDecFromStr("0.003"), false},
|
|
{sdk.NewCoin("ukava", sdkmath.NewInt(1010102)), sdk.NewCoin("usdx", sdkmath.NewInt(5e6)), sdk.ZeroDec(), sdk.MustNewDecFromStr("0.05"), false},
|
|
{sdk.NewCoin("usdx", sdkmath.NewInt(5050506)), sdk.NewCoin("ukava", sdkmath.NewInt(1e6)), sdk.ZeroDec(), sdk.MustNewDecFromStr("0"), false},
|
|
{sdk.NewCoin("usdx", sdkmath.NewInt(5050506)), sdk.NewCoin("ukava", sdkmath.NewInt(1e6)), sdk.ZeroDec(), sdk.MustNewDecFromStr("0.003"), false},
|
|
{sdk.NewCoin("usdx", sdkmath.NewInt(5050506)), sdk.NewCoin("ukava", sdkmath.NewInt(1e6)), sdk.ZeroDec(), sdk.MustNewDecFromStr("0.05"), false},
|
|
// slippage failure, zero slippage tolerance
|
|
{sdk.NewCoin("ukava", sdkmath.NewInt(1010101)), sdk.NewCoin("usdx", sdkmath.NewInt(5e6)), sdk.ZeroDec(), sdk.MustNewDecFromStr("0"), true},
|
|
{sdk.NewCoin("ukava", sdkmath.NewInt(1010101)), sdk.NewCoin("usdx", sdkmath.NewInt(5e6)), sdk.ZeroDec(), sdk.MustNewDecFromStr("0.003"), true},
|
|
{sdk.NewCoin("ukava", sdkmath.NewInt(1010101)), sdk.NewCoin("usdx", sdkmath.NewInt(5e6)), sdk.ZeroDec(), sdk.MustNewDecFromStr("0.05"), true},
|
|
{sdk.NewCoin("usdx", sdkmath.NewInt(5050505)), sdk.NewCoin("ukava", sdkmath.NewInt(1e6)), sdk.ZeroDec(), sdk.MustNewDecFromStr("0"), true},
|
|
{sdk.NewCoin("usdx", sdkmath.NewInt(5050505)), sdk.NewCoin("ukava", sdkmath.NewInt(1e6)), sdk.ZeroDec(), sdk.MustNewDecFromStr("0.003"), true},
|
|
{sdk.NewCoin("usdx", sdkmath.NewInt(5050505)), sdk.NewCoin("ukava", sdkmath.NewInt(1e6)), sdk.ZeroDec(), sdk.MustNewDecFromStr("0.05"), true},
|
|
// slippage failure, 1 percent slippage
|
|
{sdk.NewCoin("ukava", sdkmath.NewInt(1000000)), sdk.NewCoin("usdx", sdkmath.NewInt(5e6)), sdk.MustNewDecFromStr("0.01"), sdk.MustNewDecFromStr("0"), true},
|
|
{sdk.NewCoin("ukava", sdkmath.NewInt(1000000)), sdk.NewCoin("usdx", sdkmath.NewInt(5e6)), sdk.MustNewDecFromStr("0.01"), sdk.MustNewDecFromStr("0.003"), true},
|
|
{sdk.NewCoin("ukava", sdkmath.NewInt(1000000)), sdk.NewCoin("usdx", sdkmath.NewInt(5e6)), sdk.MustNewDecFromStr("0.01"), sdk.MustNewDecFromStr("0.05"), true},
|
|
{sdk.NewCoin("usdx", sdkmath.NewInt(5000000)), sdk.NewCoin("ukava", sdkmath.NewInt(1e6)), sdk.MustNewDecFromStr("0.01"), sdk.MustNewDecFromStr("0"), true},
|
|
{sdk.NewCoin("usdx", sdkmath.NewInt(5000000)), sdk.NewCoin("ukava", sdkmath.NewInt(1e6)), sdk.MustNewDecFromStr("0.01"), sdk.MustNewDecFromStr("0.003"), true},
|
|
{sdk.NewCoin("usdx", sdkmath.NewInt(5000000)), sdk.NewCoin("ukava", sdkmath.NewInt(1e6)), sdk.MustNewDecFromStr("0.01"), sdk.MustNewDecFromStr("0.05"), true},
|
|
// slippage OK, 1 percent slippage
|
|
{sdk.NewCoin("ukava", sdkmath.NewInt(1000001)), sdk.NewCoin("usdx", sdkmath.NewInt(5e6)), sdk.MustNewDecFromStr("0.01"), sdk.MustNewDecFromStr("0"), false},
|
|
{sdk.NewCoin("ukava", sdkmath.NewInt(1000001)), sdk.NewCoin("usdx", sdkmath.NewInt(5e6)), sdk.MustNewDecFromStr("0.01"), sdk.MustNewDecFromStr("0.003"), false},
|
|
{sdk.NewCoin("ukava", sdkmath.NewInt(1000001)), sdk.NewCoin("usdx", sdkmath.NewInt(5e6)), sdk.MustNewDecFromStr("0.01"), sdk.MustNewDecFromStr("0.05"), false},
|
|
{sdk.NewCoin("usdx", sdkmath.NewInt(5000001)), sdk.NewCoin("ukava", sdkmath.NewInt(1e6)), sdk.MustNewDecFromStr("0.01"), sdk.MustNewDecFromStr("0"), false},
|
|
{sdk.NewCoin("usdx", sdkmath.NewInt(5000001)), sdk.NewCoin("ukava", sdkmath.NewInt(1e6)), sdk.MustNewDecFromStr("0.01"), sdk.MustNewDecFromStr("0.003"), false},
|
|
{sdk.NewCoin("usdx", sdkmath.NewInt(5000001)), sdk.NewCoin("ukava", sdkmath.NewInt(1e6)), sdk.MustNewDecFromStr("0.01"), sdk.MustNewDecFromStr("0.05"), false},
|
|
}
|
|
|
|
for _, tc := range testCases {
|
|
suite.Run(fmt.Sprintf("coinA=%s coinB=%s slippage=%s fee=%s", tc.coinA, tc.coinB, tc.slippage, tc.fee), func() {
|
|
suite.SetupTest()
|
|
suite.Keeper.SetParams(suite.Ctx, types.Params{
|
|
SwapFee: tc.fee,
|
|
})
|
|
owner := suite.CreateAccount(sdk.Coins{})
|
|
reserves := sdk.NewCoins(
|
|
sdk.NewCoin("ukava", sdkmath.NewInt(100e6)),
|
|
sdk.NewCoin("usdx", sdkmath.NewInt(500e6)),
|
|
)
|
|
totalShares := sdkmath.NewInt(30e6)
|
|
suite.setupPool(reserves, totalShares, owner.GetAddress())
|
|
balance := sdk.NewCoins(
|
|
sdk.NewCoin("ukava", sdkmath.NewInt(100e6)),
|
|
sdk.NewCoin("usdx", sdkmath.NewInt(100e6)),
|
|
)
|
|
requester := suite.NewAccountFromAddr(sdk.AccAddress("requester-----------"), balance)
|
|
|
|
ctx := suite.App.NewContext(true, tmproto.Header{Height: 1, Time: tmtime.Now()})
|
|
err := suite.Keeper.SwapForExactTokens(ctx, requester.GetAddress(), tc.coinA, tc.coinB, tc.slippage)
|
|
|
|
if tc.shouldFail {
|
|
suite.Require().Error(err)
|
|
suite.Contains(err.Error(), "slippage exceeded")
|
|
} else {
|
|
suite.NoError(err)
|
|
}
|
|
})
|
|
}
|
|
}
|
|
|
|
func (suite *keeperTestSuite) TestSwapForExactTokens_InsufficientFunds() {
|
|
testCases := []struct {
|
|
name string
|
|
balanceA sdk.Coin
|
|
coinA sdk.Coin
|
|
coinB sdk.Coin
|
|
}{
|
|
{"no ukava balance", sdk.NewCoin("ukava", sdk.ZeroInt()), sdk.NewCoin("ukava", sdkmath.NewInt(100)), sdk.NewCoin("usdx", sdkmath.NewInt(500))},
|
|
{"no usdx balance", sdk.NewCoin("usdx", sdk.ZeroInt()), sdk.NewCoin("usdx", sdkmath.NewInt(500)), sdk.NewCoin("ukava", sdkmath.NewInt(100))},
|
|
{"low ukava balance", sdk.NewCoin("ukava", sdkmath.NewInt(1000000)), sdk.NewCoin("ukava", sdkmath.NewInt(1000000)), sdk.NewCoin("usdx", sdkmath.NewInt(5000000))},
|
|
{"low ukava balance", sdk.NewCoin("usdx", sdkmath.NewInt(5000000)), sdk.NewCoin("usdx", sdkmath.NewInt(5000000)), sdk.NewCoin("ukava", sdkmath.NewInt(1000000))},
|
|
{"large ukava balance difference", sdk.NewCoin("ukava", sdkmath.NewInt(100e6)), sdk.NewCoin("ukava", sdkmath.NewInt(1000e6)), sdk.NewCoin("usdx", sdkmath.NewInt(5000e6))},
|
|
{"large usdx balance difference", sdk.NewCoin("usdx", sdkmath.NewInt(500e6)), sdk.NewCoin("usdx", sdkmath.NewInt(5000e6)), sdk.NewCoin("ukava", sdkmath.NewInt(1000e6))},
|
|
}
|
|
|
|
for _, tc := range testCases {
|
|
suite.Run(tc.name, func() {
|
|
suite.SetupTest()
|
|
owner := suite.CreateAccount(sdk.Coins{})
|
|
reserves := sdk.NewCoins(
|
|
sdk.NewCoin("ukava", sdkmath.NewInt(100000e6)),
|
|
sdk.NewCoin("usdx", sdkmath.NewInt(500000e6)),
|
|
)
|
|
totalShares := sdkmath.NewInt(30000e6)
|
|
suite.setupPool(reserves, totalShares, owner.GetAddress())
|
|
balance := sdk.NewCoins(tc.balanceA)
|
|
requester := suite.NewAccountFromAddr(sdk.AccAddress("requester-----------"), balance)
|
|
|
|
ctx := suite.App.NewContext(true, tmproto.Header{Height: 1, Time: tmtime.Now()})
|
|
err := suite.Keeper.SwapForExactTokens(ctx, requester.GetAddress(), tc.coinA, tc.coinB, sdk.MustNewDecFromStr("0.1"))
|
|
suite.Require().True(errors.Is(err, sdkerrors.ErrInsufficientFunds), fmt.Sprintf("got err %s", err))
|
|
})
|
|
}
|
|
}
|
|
|
|
func (suite *keeperTestSuite) TestSwapForExactTokens_InsufficientFunds_Vesting() {
|
|
testCases := []struct {
|
|
name string
|
|
balanceA sdk.Coin
|
|
vestingA sdk.Coin
|
|
coinA sdk.Coin
|
|
coinB sdk.Coin
|
|
}{
|
|
{"no ukava balance, vesting only", sdk.NewCoin("ukava", sdk.ZeroInt()), sdk.NewCoin("ukava", sdkmath.NewInt(100)), sdk.NewCoin("ukava", sdkmath.NewInt(1000)), sdk.NewCoin("usdx", sdkmath.NewInt(500))},
|
|
{"no usdx balance, vesting only", sdk.NewCoin("usdx", sdk.ZeroInt()), sdk.NewCoin("usdx", sdkmath.NewInt(500)), sdk.NewCoin("usdx", sdkmath.NewInt(5000)), sdk.NewCoin("ukava", sdkmath.NewInt(100))},
|
|
{"low ukava balance, vesting matches exact", sdk.NewCoin("ukava", sdkmath.NewInt(1000000)), sdk.NewCoin("ukava", sdkmath.NewInt(100000)), sdk.NewCoin("ukava", sdkmath.NewInt(1000000)), sdk.NewCoin("usdx", sdkmath.NewInt(5000000))},
|
|
{"low ukava balance, vesting matches exact", sdk.NewCoin("usdx", sdkmath.NewInt(5000000)), sdk.NewCoin("usdx", sdkmath.NewInt(500000)), sdk.NewCoin("usdx", sdkmath.NewInt(5000000)), sdk.NewCoin("ukava", sdkmath.NewInt(1000000))},
|
|
{"large ukava balance difference, vesting covers difference", sdk.NewCoin("ukava", sdkmath.NewInt(100e6)), sdk.NewCoin("ukava", sdkmath.NewInt(10000e6)), sdk.NewCoin("ukava", sdkmath.NewInt(1000e6)), sdk.NewCoin("usdx", sdkmath.NewInt(5000e6))},
|
|
{"large usdx balance difference, vesting covers difference", sdk.NewCoin("usdx", sdkmath.NewInt(500e6)), sdk.NewCoin("usdx", sdkmath.NewInt(500000e6)), sdk.NewCoin("usdx", sdkmath.NewInt(5000e6)), sdk.NewCoin("ukava", sdkmath.NewInt(1000e6))},
|
|
}
|
|
|
|
for _, tc := range testCases {
|
|
suite.Run(tc.name, func() {
|
|
suite.SetupTest()
|
|
owner := suite.CreateAccount(sdk.Coins{})
|
|
reserves := sdk.NewCoins(
|
|
sdk.NewCoin("ukava", sdkmath.NewInt(100000e6)),
|
|
sdk.NewCoin("usdx", sdkmath.NewInt(500000e6)),
|
|
)
|
|
totalShares := sdkmath.NewInt(30000e6)
|
|
suite.setupPool(reserves, totalShares, owner.GetAddress())
|
|
balance := sdk.NewCoins(tc.balanceA)
|
|
vesting := sdk.NewCoins(tc.vestingA)
|
|
requester := suite.CreateVestingAccount(balance, vesting)
|
|
|
|
ctx := suite.App.NewContext(true, tmproto.Header{Height: 1, Time: tmtime.Now()})
|
|
err := suite.Keeper.SwapForExactTokens(ctx, requester.GetAddress(), tc.coinA, tc.coinB, sdk.MustNewDecFromStr("0.1"))
|
|
suite.Require().True(errors.Is(err, sdkerrors.ErrInsufficientFunds), fmt.Sprintf("got err %s", err))
|
|
})
|
|
}
|
|
}
|
|
|
|
func (suite *keeperTestSuite) TestSwapForExactTokens_PoolNotFound() {
|
|
owner := suite.CreateAccount(sdk.Coins{})
|
|
reserves := sdk.NewCoins(
|
|
sdk.NewCoin("ukava", sdkmath.NewInt(1000e6)),
|
|
sdk.NewCoin("usdx", sdkmath.NewInt(5000e6)),
|
|
)
|
|
totalShares := sdkmath.NewInt(3000e6)
|
|
poolID := suite.setupPool(reserves, totalShares, owner.GetAddress())
|
|
suite.Keeper.DeletePool(suite.Ctx, poolID)
|
|
|
|
balance := sdk.NewCoins(
|
|
sdk.NewCoin("ukava", sdkmath.NewInt(10e6)),
|
|
sdk.NewCoin("usdx", sdkmath.NewInt(10e6)),
|
|
)
|
|
requester := suite.NewAccountFromAddr(sdk.AccAddress("requester-----------"), balance)
|
|
coinA := sdk.NewCoin("ukava", sdkmath.NewInt(1e6))
|
|
coinB := sdk.NewCoin("usdx", sdkmath.NewInt(5e6))
|
|
|
|
err := suite.Keeper.SwapForExactTokens(suite.Ctx, requester.GetAddress(), coinA, coinB, sdk.MustNewDecFromStr("0.01"))
|
|
suite.EqualError(err, "pool ukava:usdx not found: invalid pool")
|
|
|
|
err = suite.Keeper.SwapForExactTokens(suite.Ctx, requester.GetAddress(), coinB, coinA, sdk.MustNewDecFromStr("0.01"))
|
|
suite.EqualError(err, "pool ukava:usdx not found: invalid pool")
|
|
}
|
|
|
|
func (suite *keeperTestSuite) TestSwapForExactTokens_PanicOnInvalidPool() {
|
|
owner := suite.CreateAccount(sdk.Coins{})
|
|
reserves := sdk.NewCoins(
|
|
sdk.NewCoin("ukava", sdkmath.NewInt(1000e6)),
|
|
sdk.NewCoin("usdx", sdkmath.NewInt(5000e6)),
|
|
)
|
|
totalShares := sdkmath.NewInt(3000e6)
|
|
poolID := suite.setupPool(reserves, totalShares, owner.GetAddress())
|
|
|
|
poolRecord, found := suite.Keeper.GetPool(suite.Ctx, poolID)
|
|
suite.Require().True(found, "expected pool record to exist")
|
|
|
|
poolRecord.TotalShares = sdk.ZeroInt()
|
|
suite.Keeper.SetPool_Raw(suite.Ctx, poolRecord)
|
|
|
|
balance := sdk.NewCoins(
|
|
sdk.NewCoin("ukava", sdkmath.NewInt(10e6)),
|
|
sdk.NewCoin("usdx", sdkmath.NewInt(10e6)),
|
|
)
|
|
requester := suite.NewAccountFromAddr(sdk.AccAddress("requester-----------"), balance)
|
|
coinA := sdk.NewCoin("ukava", sdkmath.NewInt(1e6))
|
|
coinB := sdk.NewCoin("usdx", sdkmath.NewInt(5e6))
|
|
|
|
suite.PanicsWithValue("invalid pool ukava:usdx: total shares must be greater than zero: invalid pool", func() {
|
|
_ = suite.Keeper.SwapForExactTokens(suite.Ctx, requester.GetAddress(), coinA, coinB, sdk.MustNewDecFromStr("0.01"))
|
|
}, "expected invalid pool record to panic")
|
|
|
|
suite.PanicsWithValue("invalid pool ukava:usdx: total shares must be greater than zero: invalid pool", func() {
|
|
_ = suite.Keeper.SwapForExactTokens(suite.Ctx, requester.GetAddress(), coinB, coinA, sdk.MustNewDecFromStr("0.01"))
|
|
}, "expected invalid pool record to panic")
|
|
}
|
|
|
|
func (suite *keeperTestSuite) TestSwapForExactTokens_PanicOnInsufficientModuleAccFunds() {
|
|
owner := suite.CreateAccount(sdk.Coins{})
|
|
reserves := sdk.NewCoins(
|
|
sdk.NewCoin("ukava", sdkmath.NewInt(1000e6)),
|
|
sdk.NewCoin("usdx", sdkmath.NewInt(5000e6)),
|
|
)
|
|
totalShares := sdkmath.NewInt(3000e6)
|
|
suite.setupPool(reserves, totalShares, owner.GetAddress())
|
|
|
|
suite.RemoveCoinsFromModule(sdk.NewCoins(
|
|
sdk.NewCoin("ukava", sdkmath.NewInt(1000e6)),
|
|
sdk.NewCoin("usdx", sdkmath.NewInt(5000e6)),
|
|
))
|
|
|
|
balance := sdk.NewCoins(
|
|
sdk.NewCoin("ukava", sdkmath.NewInt(10e6)),
|
|
sdk.NewCoin("usdx", sdkmath.NewInt(10e6)),
|
|
)
|
|
requester := suite.NewAccountFromAddr(sdk.AccAddress("requester-----------"), balance)
|
|
coinA := sdk.NewCoin("ukava", sdkmath.NewInt(1e6))
|
|
coinB := sdk.NewCoin("usdx", sdkmath.NewInt(5e6))
|
|
|
|
suite.Panics(func() {
|
|
_ = suite.Keeper.SwapForExactTokens(suite.Ctx, requester.GetAddress(), coinA, coinB, sdk.MustNewDecFromStr("0.01"))
|
|
}, "expected panic when module account does not have enough funds")
|
|
|
|
suite.Panics(func() {
|
|
_ = suite.Keeper.SwapForExactTokens(suite.Ctx, requester.GetAddress(), coinA, coinB, sdk.MustNewDecFromStr("0.01"))
|
|
}, "expected panic when module account does not have enough funds")
|
|
}
|