mirror of
https://github.com/0glabs/0g-chain.git
synced 2024-12-27 00:35:18 +00:00
ffef832d45
- Upgrade cosmos-sdk to v0.44.5 from v0.39.2 - Add Legacy Tx Endpoint for backwards compatibility - Add IBC v1.2.3 Support Co-authored-by: DracoLi <draco@dracoli.com> Co-authored-by: drklee3 <derrick@dlee.dev> Co-authored-by: denalimarsh <denalimarsh@gmail.com> Co-authored-by: Draco Li <draco@kava.io> Co-authored-by: Nick DeLuca <nickdeluca08@gmail.com> Co-authored-by: Kevin Davis <karzak@users.noreply.github.com> Co-authored-by: Denali Marsh <denali@kava.io>
549 lines
21 KiB
Go
549 lines
21 KiB
Go
package simulation
|
|
|
|
// import (
|
|
// "errors"
|
|
// "fmt"
|
|
// "math/big"
|
|
// "math/rand"
|
|
// "time"
|
|
|
|
// "github.com/cosmos/cosmos-sdk/baseapp"
|
|
// "github.com/cosmos/cosmos-sdk/codec"
|
|
// "github.com/cosmos/cosmos-sdk/simapp/helpers"
|
|
// sdk "github.com/cosmos/cosmos-sdk/types"
|
|
// authexported "github.com/cosmos/cosmos-sdk/x/auth/exported"
|
|
// "github.com/cosmos/cosmos-sdk/x/simulation"
|
|
|
|
// appparams "github.com/kava-labs/kava/app/params"
|
|
// "github.com/kava-labs/kava/x/swap/keeper"
|
|
// "github.com/kava-labs/kava/x/swap/types"
|
|
// )
|
|
|
|
// var (
|
|
// //nolint
|
|
// noOpMsg = simulation.NoOpMsg(types.ModuleName)
|
|
// errorNotEnoughCoins = errors.New("account doesn't have enough coins")
|
|
// )
|
|
|
|
// // Simulation operation weights constants
|
|
// const (
|
|
// OpWeightMsgDeposit = "op_weight_msg_deposit"
|
|
// OpWeightMsgWithdraw = "op_weight_msg_withdraw"
|
|
// OpWeightMsgSwapExactForTokens = "op_weight_msg_swap_exact_for_tokens"
|
|
// OpWeightMsgSwapForExactTokens = "op_weight_msg_swap_for_exact_tokens"
|
|
// )
|
|
|
|
// // WeightedOperations returns all the operations from the module with their respective weights
|
|
// func WeightedOperations(
|
|
// appParams simulation.AppParams, cdc *codec.Codec, ak types.AccountKeeper, k keeper.Keeper,
|
|
// ) simulation.WeightedOperations {
|
|
// var weightMsgDeposit int
|
|
// var weightMsgWithdraw int
|
|
// var weightMsgSwapExactForTokens int
|
|
// var weightMsgSwapForExactTokens int
|
|
|
|
// appParams.GetOrGenerate(cdc, OpWeightMsgDeposit, &weightMsgDeposit, nil,
|
|
// func(_ *rand.Rand) {
|
|
// weightMsgDeposit = appparams.DefaultWeightMsgDeposit
|
|
// },
|
|
// )
|
|
|
|
// appParams.GetOrGenerate(cdc, OpWeightMsgWithdraw, &weightMsgWithdraw, nil,
|
|
// func(_ *rand.Rand) {
|
|
// weightMsgWithdraw = appparams.DefaultWeightMsgWithdraw
|
|
// },
|
|
// )
|
|
|
|
// appParams.GetOrGenerate(cdc, OpWeightMsgSwapExactForTokens, &weightMsgSwapExactForTokens, nil,
|
|
// func(_ *rand.Rand) {
|
|
// weightMsgSwapExactForTokens = appparams.DefaultWeightMsgSwapExactForTokens
|
|
// },
|
|
// )
|
|
|
|
// appParams.GetOrGenerate(cdc, OpWeightMsgSwapForExactTokens, &weightMsgSwapForExactTokens, nil,
|
|
// func(_ *rand.Rand) {
|
|
// weightMsgSwapForExactTokens = appparams.DefaultWeightMsgSwapForExactTokens
|
|
// },
|
|
// )
|
|
|
|
// return simulation.WeightedOperations{
|
|
// simulation.NewWeightedOperation(
|
|
// weightMsgDeposit,
|
|
// SimulateMsgDeposit(ak, k),
|
|
// ),
|
|
// simulation.NewWeightedOperation(
|
|
// weightMsgWithdraw,
|
|
// SimulateMsgWithdraw(ak, k),
|
|
// ),
|
|
// simulation.NewWeightedOperation(
|
|
// weightMsgSwapExactForTokens,
|
|
// SimulateMsgSwapExactForTokens(ak, k),
|
|
// ),
|
|
// simulation.NewWeightedOperation(
|
|
// weightMsgSwapForExactTokens,
|
|
// SimulateMsgSwapForExactTokens(ak, k),
|
|
// ),
|
|
// }
|
|
// }
|
|
|
|
// // SimulateMsgDeposit generates a MsgDeposit
|
|
// func SimulateMsgDeposit(ak types.AccountKeeper, k keeper.Keeper) simulation.Operation {
|
|
// return func(
|
|
// r *rand.Rand, app *baseapp.BaseApp, ctx sdk.Context, accs []simulation.Account, chainID string,
|
|
// ) (simulation.OperationMsg, []simulation.FutureOperation, error) {
|
|
// // Get possible pools and shuffle so that deposits are evenly distributed across pools
|
|
// params := k.GetParams(ctx)
|
|
// allowedPools := params.AllowedPools
|
|
// r.Shuffle(len(allowedPools), func(i, j int) {
|
|
// allowedPools[i], allowedPools[j] = allowedPools[j], allowedPools[i]
|
|
// })
|
|
|
|
// // Find an account-pool pair that is likely to result in a successful deposit
|
|
// blockTime := ctx.BlockHeader().Time
|
|
// depositor, allowedPool, found := findValidAccountAllowedPoolPair(accs, allowedPools, func(acc simulation.Account, pool types.AllowedPool) bool {
|
|
// account := ak.GetAccount(ctx, acc.Address)
|
|
|
|
// err := validateDepositor(ctx, k, pool, account, blockTime)
|
|
// if err == errorNotEnoughCoins {
|
|
// return false // keep searching
|
|
// } else if err != nil {
|
|
// panic(err) // raise errors
|
|
// }
|
|
// return true // found valid pair
|
|
// })
|
|
// if !found {
|
|
// return simulation.NewOperationMsgBasic(types.ModuleName, "no-operation (no valid allowed pool and depositor)", "", false, nil), nil, nil
|
|
// }
|
|
|
|
// // Get random slippage amount between 1-99%
|
|
// slippageRaw, err := RandIntInclusive(r, sdk.OneInt(), sdk.NewInt(99))
|
|
// if err != nil {
|
|
// panic(err)
|
|
// }
|
|
// slippage := slippageRaw.ToDec().Quo(sdk.NewDec(100))
|
|
|
|
// // Generate random deadline
|
|
// deadline := genRandDeadline(r, blockTime)
|
|
|
|
// depositorAcc := ak.GetAccount(ctx, depositor.Address)
|
|
// depositorCoins := depositorAcc.SpendableCoins(blockTime)
|
|
|
|
// // Construct initial msg (without coin amounts)
|
|
// msg := types.NewMsgDeposit(depositorAcc.GetAddress(), sdk.Coin{}, sdk.Coin{}, slippage, deadline)
|
|
|
|
// // Populate msg with randomized token amounts
|
|
// pool, found := k.GetPool(ctx, allowedPool.Name())
|
|
// if !found { // Pool doesn't exist: first deposit
|
|
// depositTokenA := randCoinFromCoins(r, depositorCoins, allowedPool.TokenA)
|
|
// msg.TokenA = depositTokenA
|
|
|
|
// depositTokenB := randCoinFromCoins(r, depositorCoins, allowedPool.TokenB)
|
|
// msg.TokenB = depositTokenB
|
|
// } else { // Pool exists: successive deposit
|
|
// var denomX string // Denom X is the token denom in the pool with the larger amount
|
|
// var denomY string // Denom Y is the token denom in the pool with the larger amount
|
|
// if pool.ReservesA.Amount.GTE(pool.ReservesB.Amount) {
|
|
// denomX = pool.ReservesA.Denom
|
|
// denomY = pool.ReservesB.Denom
|
|
// } else {
|
|
// denomX = pool.ReservesB.Denom
|
|
// denomY = pool.ReservesA.Denom
|
|
// }
|
|
// depositTokenY := randCoinFromCoins(r, depositorCoins, denomY)
|
|
// msg.TokenA = depositTokenY
|
|
|
|
// // Calculate the pool's slippage ratio and use it to build other coin
|
|
// ratio := pool.Reserves().AmountOf(denomX).ToDec().Quo(pool.Reserves().AmountOf(denomY).ToDec())
|
|
// amtTokenX := depositTokenY.Amount.ToDec().Mul(ratio).RoundInt()
|
|
// depositTokenX := sdk.NewCoin(denomX, amtTokenX)
|
|
// if depositorCoins.AmountOf(denomX).LT(amtTokenX) {
|
|
// return simulation.NewOperationMsgBasic(types.ModuleName, "no-operation (depositor has insufficient coins)", "", false, nil), nil, nil
|
|
// }
|
|
// msg.TokenB = depositTokenX
|
|
// }
|
|
|
|
// err = msg.ValidateBasic()
|
|
// if err != nil {
|
|
// return noOpMsg, nil, nil
|
|
// }
|
|
|
|
// tx := helpers.GenTx(
|
|
// []sdk.Msg{msg},
|
|
// sdk.NewCoins(),
|
|
// helpers.DefaultGenTxGas,
|
|
// chainID,
|
|
// []uint64{depositorAcc.GetAccountNumber()},
|
|
// []uint64{depositorAcc.GetSequence()},
|
|
// depositor.PrivKey,
|
|
// )
|
|
|
|
// _, result, err := app.Deliver(tx)
|
|
// if err != nil {
|
|
// // to aid debugging, add the stack trace to the comment field of the returned opMsg
|
|
// return simulation.NewOperationMsg(msg, false, fmt.Sprintf("%+v", err)), nil, err
|
|
// }
|
|
// return simulation.NewOperationMsg(msg, true, result.Log), nil, nil
|
|
// }
|
|
// }
|
|
|
|
// // SimulateMsgWithdraw generates a MsgWithdraw
|
|
// func SimulateMsgWithdraw(ak types.AccountKeeper, k keeper.Keeper) simulation.Operation {
|
|
// return func(
|
|
// r *rand.Rand, app *baseapp.BaseApp, ctx sdk.Context, accs []simulation.Account, chainID string,
|
|
// ) (simulation.OperationMsg, []simulation.FutureOperation, error) {
|
|
|
|
// poolRecords := k.GetAllPools(ctx)
|
|
// r.Shuffle(len(poolRecords), func(i, j int) {
|
|
// poolRecords[i], poolRecords[j] = poolRecords[j], poolRecords[i]
|
|
// })
|
|
|
|
// // Find an account-pool pair for which withdraw is possible
|
|
// withdrawer, poolRecord, found := findValidAccountPoolRecordPair(accs, poolRecords, func(acc simulation.Account, poolRecord types.PoolRecord) bool {
|
|
// _, found := k.GetDepositorShares(ctx, acc.Address, poolRecord.PoolID)
|
|
// return found
|
|
// })
|
|
// if !found {
|
|
// return simulation.NewOperationMsgBasic(types.ModuleName, "no-operation (no valid pool record and withdrawer)", "", false, nil), nil, nil
|
|
// }
|
|
|
|
// withdrawerAcc := ak.GetAccount(ctx, withdrawer.Address)
|
|
// shareRecord, _ := k.GetDepositorShares(ctx, withdrawerAcc.GetAddress(), poolRecord.PoolID)
|
|
// denominatedPool, err := types.NewDenominatedPoolWithExistingShares(poolRecord.Reserves(), poolRecord.TotalShares)
|
|
// if err != nil {
|
|
// return noOpMsg, nil, nil
|
|
// }
|
|
// coinsOwned := denominatedPool.ShareValue(shareRecord.SharesOwned)
|
|
|
|
// // Get random amount of shares between 2-10% of the total
|
|
// sharePercentage, err := RandIntInclusive(r, sdk.NewInt(2), sdk.NewInt(10))
|
|
// if err != nil {
|
|
// panic(err)
|
|
// }
|
|
// shares := shareRecord.SharesOwned.Mul(sharePercentage).Quo(sdk.NewInt(100))
|
|
|
|
// // Expect minimum token amounts relative to the % of shares owned and withdrawn
|
|
// oneLessThanSharePercentage := sharePercentage.Sub(sdk.OneInt())
|
|
|
|
// amtTokenAOwned := coinsOwned.AmountOf(poolRecord.ReservesA.Denom)
|
|
// minAmtTokenA := amtTokenAOwned.Mul(oneLessThanSharePercentage).Quo(sdk.NewInt(100))
|
|
// minTokenA := sdk.NewCoin(poolRecord.ReservesA.Denom, minAmtTokenA)
|
|
|
|
// amtTokenBOwned := coinsOwned.AmountOf(poolRecord.ReservesB.Denom)
|
|
// minTokenAmtB := amtTokenBOwned.Mul(oneLessThanSharePercentage).Quo(sdk.NewInt(100))
|
|
// minTokenB := sdk.NewCoin(poolRecord.ReservesB.Denom, minTokenAmtB)
|
|
|
|
// // Generate random deadline
|
|
// blockTime := ctx.BlockHeader().Time
|
|
// deadline := genRandDeadline(r, blockTime)
|
|
|
|
// // Construct MsgWithdraw
|
|
// msg := types.NewMsgWithdraw(withdrawerAcc.GetAddress(), shares, minTokenA, minTokenB, deadline)
|
|
// err = msg.ValidateBasic()
|
|
// if err != nil {
|
|
// return noOpMsg, nil, nil
|
|
// }
|
|
|
|
// tx := helpers.GenTx(
|
|
// []sdk.Msg{msg},
|
|
// sdk.NewCoins(),
|
|
// helpers.DefaultGenTxGas,
|
|
// chainID,
|
|
// []uint64{withdrawerAcc.GetAccountNumber()},
|
|
// []uint64{withdrawerAcc.GetSequence()},
|
|
// withdrawer.PrivKey,
|
|
// )
|
|
|
|
// _, result, err := app.Deliver(tx)
|
|
// if err != nil {
|
|
// // to aid debugging, add the stack trace to the comment field of the returned opMsg
|
|
// return simulation.NewOperationMsg(msg, false, fmt.Sprintf("%+v", err)), nil, err
|
|
// }
|
|
// return simulation.NewOperationMsg(msg, true, result.Log), nil, nil
|
|
|
|
// }
|
|
// }
|
|
|
|
// // SimulateMsgSwapExactForTokens generates a MsgSwapExactForTokens
|
|
// func SimulateMsgSwapExactForTokens(ak types.AccountKeeper, k keeper.Keeper) simulation.Operation {
|
|
// return func(
|
|
// r *rand.Rand, app *baseapp.BaseApp, ctx sdk.Context, accs []simulation.Account, chainID string,
|
|
// ) (simulation.OperationMsg, []simulation.FutureOperation, error) {
|
|
|
|
// poolRecords := k.GetAllPools(ctx)
|
|
// r.Shuffle(len(poolRecords), func(i, j int) {
|
|
// poolRecords[i], poolRecords[j] = poolRecords[j], poolRecords[i]
|
|
// })
|
|
|
|
// // Find an account-pool pair for which trade is possible
|
|
// trader, poolRecord, found := findValidAccountPoolRecordPair(accs, poolRecords, func(acc simulation.Account, poolRecord types.PoolRecord) bool {
|
|
// traderAcc := ak.GetAccount(ctx, acc.Address)
|
|
// balanceTokenA := traderAcc.GetCoins().AmountOf(poolRecord.ReservesA.Denom)
|
|
// balanceTokenB := traderAcc.GetCoins().AmountOf(poolRecord.ReservesB.Denom)
|
|
// if !balanceTokenA.IsPositive() || !balanceTokenB.IsPositive() {
|
|
// return false
|
|
// }
|
|
// return true
|
|
// })
|
|
// if !found {
|
|
// return simulation.NewOperationMsgBasic(types.ModuleName, "no-operation (no valid pool record and trader)", "", false, nil), nil, nil
|
|
// }
|
|
|
|
// // Select input token
|
|
// randInt, err := RandInt(r, sdk.OneInt(), sdk.NewInt(9))
|
|
// if err != nil {
|
|
// panic(err)
|
|
// }
|
|
// inputToken := poolRecord.ReservesA
|
|
// outputToken := poolRecord.ReservesB
|
|
// if randInt.Int64()%2 == 0 {
|
|
// inputToken = poolRecord.ReservesB
|
|
// outputToken = poolRecord.ReservesA
|
|
// }
|
|
|
|
// // Select entity (trader account or pool) with smaller token amount
|
|
// traderAcc := ak.GetAccount(ctx, trader.Address)
|
|
// maxTradeAmount := inputToken.Amount
|
|
// if traderAcc.GetCoins().AmountOf(inputToken.Denom).LT(inputToken.Amount) {
|
|
// maxTradeAmount = traderAcc.GetCoins().AmountOf(inputToken.Denom)
|
|
// }
|
|
|
|
// // Exact input token is between 2-10% of the max trade amount
|
|
// percentage, err := RandIntInclusive(r, sdk.NewInt(2), sdk.NewInt(10))
|
|
// if err != nil {
|
|
// panic(err)
|
|
// }
|
|
// tradeAmount := maxTradeAmount.Mul(percentage).Quo(sdk.NewInt(100))
|
|
// exactInputToken := sdk.NewCoin(inputToken.Denom, tradeAmount)
|
|
|
|
// // Calculate expected output coin
|
|
// globalSwapFee := k.GetSwapFee(ctx)
|
|
// tradeAmountAfterFee := exactInputToken.Amount.ToDec().Mul(sdk.OneDec().Sub(globalSwapFee)).TruncateInt()
|
|
|
|
// var outputAmt big.Int
|
|
// outputAmt.Mul(outputToken.Amount.BigInt(), tradeAmountAfterFee.BigInt())
|
|
// outputAmt.Quo(&outputAmt, inputToken.Amount.Add(tradeAmountAfterFee).BigInt())
|
|
// expectedOutTokenAmount := sdk.NewIntFromBigInt(&outputAmt)
|
|
// expectedOutputToken := sdk.NewCoin(outputToken.Denom, expectedOutTokenAmount)
|
|
|
|
// // Get random slippage amount between 50-100%
|
|
// slippageRaw, err := RandIntInclusive(r, sdk.NewInt(50), sdk.NewInt(99))
|
|
// if err != nil {
|
|
// panic(err)
|
|
// }
|
|
// slippage := slippageRaw.ToDec().Quo(sdk.NewDec(100))
|
|
|
|
// // Generate random deadline
|
|
// blockTime := ctx.BlockHeader().Time
|
|
// deadline := genRandDeadline(r, blockTime)
|
|
|
|
// // Construct MsgSwapExactForTokens
|
|
// msg := types.NewMsgSwapExactForTokens(traderAcc.GetAddress(), exactInputToken, expectedOutputToken, slippage, deadline)
|
|
// err = msg.ValidateBasic()
|
|
// if err != nil {
|
|
// return noOpMsg, nil, nil
|
|
// }
|
|
|
|
// tx := helpers.GenTx(
|
|
// []sdk.Msg{msg},
|
|
// sdk.NewCoins(),
|
|
// helpers.DefaultGenTxGas,
|
|
// chainID,
|
|
// []uint64{traderAcc.GetAccountNumber()},
|
|
// []uint64{traderAcc.GetSequence()},
|
|
// trader.PrivKey,
|
|
// )
|
|
|
|
// _, result, err := app.Deliver(tx)
|
|
// if err != nil {
|
|
// // to aid debugging, add the stack trace to the comment field of the returned opMsg
|
|
// return simulation.NewOperationMsg(msg, false, fmt.Sprintf("%+v", err)), nil, err
|
|
// }
|
|
// return simulation.NewOperationMsg(msg, true, result.Log), nil, nil
|
|
// }
|
|
// }
|
|
|
|
// // SimulateMsgSwapForExactTokens generates a MsgSwapForExactTokens
|
|
// func SimulateMsgSwapForExactTokens(ak types.AccountKeeper, k keeper.Keeper) simulation.Operation {
|
|
// return func(
|
|
// r *rand.Rand, app *baseapp.BaseApp, ctx sdk.Context, accs []simulation.Account, chainID string,
|
|
// ) (simulation.OperationMsg, []simulation.FutureOperation, error) {
|
|
|
|
// poolRecords := k.GetAllPools(ctx)
|
|
// r.Shuffle(len(poolRecords), func(i, j int) {
|
|
// poolRecords[i], poolRecords[j] = poolRecords[j], poolRecords[i]
|
|
// })
|
|
|
|
// // Find an account-pool pair for which trade is possible
|
|
// trader, poolRecord, found := findValidAccountPoolRecordPair(accs, poolRecords, func(acc simulation.Account, poolRecord types.PoolRecord) bool {
|
|
// traderAcc := ak.GetAccount(ctx, acc.Address)
|
|
// balanceTokenA := traderAcc.GetCoins().AmountOf(poolRecord.ReservesA.Denom)
|
|
// balanceTokenB := traderAcc.GetCoins().AmountOf(poolRecord.ReservesB.Denom)
|
|
// if !balanceTokenA.IsPositive() || !balanceTokenB.IsPositive() {
|
|
// return false
|
|
// }
|
|
// return true
|
|
// })
|
|
// if !found {
|
|
// return simulation.NewOperationMsgBasic(types.ModuleName, "no-operation (no valid pool record and trader)", "", false, nil), nil, nil
|
|
// }
|
|
|
|
// // Select input token
|
|
// randInt, err := RandInt(r, sdk.OneInt(), sdk.NewInt(9))
|
|
// if err != nil {
|
|
// panic(err)
|
|
// }
|
|
// inputToken := poolRecord.ReservesA
|
|
// outputToken := poolRecord.ReservesB
|
|
// if randInt.Int64()%2 == 0 {
|
|
// inputToken = poolRecord.ReservesB
|
|
// outputToken = poolRecord.ReservesA
|
|
// }
|
|
|
|
// // Select entity (trader account or pool) with smaller token amount
|
|
// traderAcc := ak.GetAccount(ctx, trader.Address)
|
|
// maxTradeAmount := inputToken.Amount
|
|
// if traderAcc.GetCoins().AmountOf(inputToken.Denom).LT(inputToken.Amount) {
|
|
// maxTradeAmount = traderAcc.GetCoins().AmountOf(inputToken.Denom)
|
|
// }
|
|
|
|
// // Expected input token is between 2-10% of the max trade amount
|
|
// percentage, err := RandIntInclusive(r, sdk.NewInt(2), sdk.NewInt(10))
|
|
// if err != nil {
|
|
// panic(err)
|
|
// }
|
|
// tradeAmount := maxTradeAmount.Mul(percentage).Quo(sdk.NewInt(100))
|
|
// expectedInputToken := sdk.NewCoin(inputToken.Denom, tradeAmount)
|
|
|
|
// // Calculate exact output coin
|
|
// globalSwapFee := k.GetSwapFee(ctx)
|
|
// tradeAmountAfterFee := expectedInputToken.Amount.ToDec().Mul(sdk.OneDec().Sub(globalSwapFee)).TruncateInt()
|
|
|
|
// var outputAmt big.Int
|
|
// outputAmt.Mul(outputToken.Amount.BigInt(), tradeAmountAfterFee.BigInt())
|
|
// outputAmt.Quo(&outputAmt, inputToken.Amount.Add(tradeAmountAfterFee).BigInt())
|
|
// outputTokenAmount := sdk.NewIntFromBigInt(&outputAmt)
|
|
// exactOutputToken := sdk.NewCoin(outputToken.Denom, outputTokenAmount)
|
|
|
|
// // Get random slippage amount between 50-100%
|
|
// slippageRaw, err := RandIntInclusive(r, sdk.NewInt(50), sdk.NewInt(99))
|
|
// if err != nil {
|
|
// panic(err)
|
|
// }
|
|
// slippage := slippageRaw.ToDec().Quo(sdk.NewDec(100))
|
|
|
|
// // Generate random deadline
|
|
// blockTime := ctx.BlockHeader().Time
|
|
// deadline := genRandDeadline(r, blockTime)
|
|
|
|
// // Construct MsgSwapForExactTokens
|
|
// msg := types.NewMsgSwapForExactTokens(traderAcc.GetAddress(), expectedInputToken, exactOutputToken, slippage, deadline)
|
|
// err = msg.ValidateBasic()
|
|
// if err != nil {
|
|
// return noOpMsg, nil, nil
|
|
// }
|
|
|
|
// tx := helpers.GenTx(
|
|
// []sdk.Msg{msg},
|
|
// sdk.NewCoins(),
|
|
// helpers.DefaultGenTxGas,
|
|
// chainID,
|
|
// []uint64{traderAcc.GetAccountNumber()},
|
|
// []uint64{traderAcc.GetSequence()},
|
|
// trader.PrivKey,
|
|
// )
|
|
|
|
// _, result, err := app.Deliver(tx)
|
|
// if err != nil {
|
|
// // to aid debugging, add the stack trace to the comment field of the returned opMsg
|
|
// return simulation.NewOperationMsg(msg, false, fmt.Sprintf("%+v", err)), nil, err
|
|
// }
|
|
// return simulation.NewOperationMsg(msg, true, result.Log), nil, nil
|
|
// }
|
|
// }
|
|
|
|
// // From a set of coins return a coin of the specified denom with 1-10% of the total amount
|
|
// func randCoinFromCoins(r *rand.Rand, coins sdk.Coins, denom string) sdk.Coin {
|
|
// percentOfBalance, err := RandIntInclusive(r, sdk.OneInt(), sdk.NewInt(10))
|
|
// if err != nil {
|
|
// panic(err)
|
|
// }
|
|
// balance := coins.AmountOf(denom)
|
|
// amtToken := balance.Mul(percentOfBalance).Quo(sdk.NewInt(100))
|
|
// return sdk.NewCoin(denom, amtToken)
|
|
// }
|
|
|
|
// func validateDepositor(ctx sdk.Context, k keeper.Keeper, allowedPool types.AllowedPool,
|
|
// depositor authexported.Account, blockTime time.Time) error {
|
|
// depositorCoins := depositor.SpendableCoins(blockTime)
|
|
// tokenABalance := depositorCoins.AmountOf(allowedPool.TokenA)
|
|
// tokenBBalance := depositorCoins.AmountOf(allowedPool.TokenB)
|
|
|
|
// oneThousand := sdk.NewInt(1000)
|
|
// if tokenABalance.LT(oneThousand) || tokenBBalance.LT(oneThousand) {
|
|
// return errorNotEnoughCoins
|
|
// }
|
|
|
|
// return nil
|
|
// }
|
|
|
|
// // findValidAccountAllowedPoolPair finds an account for which the callback func returns true
|
|
// func findValidAccountAllowedPoolPair(accounts []simulation.Account, pools types.AllowedPools,
|
|
// cb func(simulation.Account, types.AllowedPool) bool) (simulation.Account, types.AllowedPool, bool) {
|
|
// for _, pool := range pools {
|
|
// for _, acc := range accounts {
|
|
// if isValid := cb(acc, pool); isValid {
|
|
// return acc, pool, true
|
|
// }
|
|
// }
|
|
// }
|
|
// return simulation.Account{}, types.AllowedPool{}, false
|
|
// }
|
|
|
|
// // findValidAccountPoolRecordPair finds an account for which the callback func returns true
|
|
// func findValidAccountPoolRecordPair(accounts []simulation.Account, pools types.PoolRecords,
|
|
// cb func(simulation.Account, types.PoolRecord) bool) (simulation.Account, types.PoolRecord, bool) {
|
|
// for _, pool := range pools {
|
|
// for _, acc := range accounts {
|
|
// if isValid := cb(acc, pool); isValid {
|
|
// return acc, pool, true
|
|
// }
|
|
// }
|
|
// }
|
|
// return simulation.Account{}, types.PoolRecord{}, false
|
|
// }
|
|
|
|
// func genRandDeadline(r *rand.Rand, blockTime time.Time) int64 {
|
|
// // Set up deadline
|
|
// durationNanoseconds, err := RandIntInclusive(r,
|
|
// sdk.NewInt((time.Second * 10).Nanoseconds()), // ten seconds
|
|
// sdk.NewInt((time.Hour * 24).Nanoseconds()), // one day
|
|
// )
|
|
// if err != nil {
|
|
// panic(err)
|
|
// }
|
|
// extraTime := time.Duration(durationNanoseconds.Int64())
|
|
// return blockTime.Add(extraTime).Unix()
|
|
// }
|
|
|
|
// // RandIntInclusive randomly generates an sdk.Int in the range [inclusiveMin, inclusiveMax]. It works for negative and positive integers.
|
|
// func RandIntInclusive(r *rand.Rand, inclusiveMin, inclusiveMax sdk.Int) (sdk.Int, error) {
|
|
// if inclusiveMin.GT(inclusiveMax) {
|
|
// return sdk.Int{}, fmt.Errorf("min larger than max")
|
|
// }
|
|
// return RandInt(r, inclusiveMin, inclusiveMax.Add(sdk.OneInt()))
|
|
// }
|
|
|
|
// // RandInt randomly generates an sdk.Int in the range [inclusiveMin, exclusiveMax). It works for negative and positive integers.
|
|
// func RandInt(r *rand.Rand, inclusiveMin, exclusiveMax sdk.Int) (sdk.Int, error) {
|
|
// // validate input
|
|
// if inclusiveMin.GTE(exclusiveMax) {
|
|
// return sdk.Int{}, fmt.Errorf("min larger or equal to max")
|
|
// }
|
|
// // shift the range to start at 0
|
|
// shiftedRange := exclusiveMax.Sub(inclusiveMin) // should always be positive given the check above
|
|
// // randomly pick from the shifted range
|
|
// shiftedRandInt := sdk.NewIntFromBigInt(new(big.Int).Rand(r, shiftedRange.BigInt()))
|
|
// // shift back to the original range
|
|
// return shiftedRandInt.Add(inclusiveMin), nil
|
|
// }
|