2021-08-06 01:43:55 +00:00
|
|
|
package keeper_test
|
|
|
|
|
|
|
|
import (
|
|
|
|
"testing"
|
|
|
|
|
2024-05-01 03:17:24 +00:00
|
|
|
"github.com/0glabs/0g-chain/x/swap/keeper"
|
|
|
|
"github.com/0glabs/0g-chain/x/swap/testutil"
|
|
|
|
"github.com/0glabs/0g-chain/x/swap/types"
|
2021-08-06 01:43:55 +00:00
|
|
|
|
2023-04-05 23:21:59 +00:00
|
|
|
sdkmath "cosmossdk.io/math"
|
2021-08-06 01:43:55 +00:00
|
|
|
sdk "github.com/cosmos/cosmos-sdk/types"
|
|
|
|
"github.com/stretchr/testify/suite"
|
|
|
|
)
|
|
|
|
|
|
|
|
type invariantTestSuite struct {
|
|
|
|
testutil.Suite
|
|
|
|
invariants map[string]map[string]sdk.Invariant
|
|
|
|
}
|
|
|
|
|
|
|
|
func (suite *invariantTestSuite) SetupTest() {
|
|
|
|
suite.Suite.SetupTest()
|
|
|
|
suite.invariants = make(map[string]map[string]sdk.Invariant)
|
|
|
|
keeper.RegisterInvariants(suite, suite.Keeper)
|
|
|
|
}
|
|
|
|
|
|
|
|
func (suite *invariantTestSuite) SetupValidState() {
|
|
|
|
suite.Keeper.SetPool(suite.Ctx, types.NewPoolRecord(
|
|
|
|
sdk.NewCoins(
|
2023-04-05 23:21:59 +00:00
|
|
|
sdk.NewCoin("ukava", sdkmath.NewInt(1e6)),
|
|
|
|
sdk.NewCoin("usdx", sdkmath.NewInt(5e6)),
|
2021-08-06 01:43:55 +00:00
|
|
|
),
|
2023-04-05 23:21:59 +00:00
|
|
|
sdkmath.NewInt(3e6),
|
2021-08-06 01:43:55 +00:00
|
|
|
))
|
|
|
|
suite.AddCoinsToModule(
|
|
|
|
sdk.NewCoins(
|
2023-04-05 23:21:59 +00:00
|
|
|
sdk.NewCoin("ukava", sdkmath.NewInt(1e6)),
|
|
|
|
sdk.NewCoin("usdx", sdkmath.NewInt(5e6)),
|
2021-08-06 01:43:55 +00:00
|
|
|
),
|
|
|
|
)
|
|
|
|
suite.Keeper.SetDepositorShares(suite.Ctx, types.NewShareRecord(
|
2022-01-08 00:39:27 +00:00
|
|
|
sdk.AccAddress("depositor 1---------"), // TODO these addresses are padded to get to the required length of 20 bytes. What is a nicer setup?
|
2021-08-06 01:43:55 +00:00
|
|
|
types.PoolID("ukava", "usdx"),
|
2023-04-05 23:21:59 +00:00
|
|
|
sdkmath.NewInt(2e6),
|
2021-08-06 01:43:55 +00:00
|
|
|
))
|
|
|
|
suite.Keeper.SetDepositorShares(suite.Ctx, types.NewShareRecord(
|
2022-01-08 00:39:27 +00:00
|
|
|
sdk.AccAddress("depositor 2---------"),
|
2021-08-06 01:43:55 +00:00
|
|
|
types.PoolID("ukava", "usdx"),
|
2023-04-05 23:21:59 +00:00
|
|
|
sdkmath.NewInt(1e6),
|
2021-08-06 01:43:55 +00:00
|
|
|
))
|
|
|
|
|
|
|
|
suite.Keeper.SetPool(suite.Ctx, types.NewPoolRecord(
|
|
|
|
sdk.NewCoins(
|
2023-04-05 23:21:59 +00:00
|
|
|
sdk.NewCoin("hard", sdkmath.NewInt(1e6)),
|
|
|
|
sdk.NewCoin("usdx", sdkmath.NewInt(2e6)),
|
2021-08-06 01:43:55 +00:00
|
|
|
),
|
2023-04-05 23:21:59 +00:00
|
|
|
sdkmath.NewInt(1e6),
|
2021-08-06 01:43:55 +00:00
|
|
|
))
|
|
|
|
suite.AddCoinsToModule(
|
|
|
|
sdk.NewCoins(
|
2023-04-05 23:21:59 +00:00
|
|
|
sdk.NewCoin("hard", sdkmath.NewInt(1e6)),
|
|
|
|
sdk.NewCoin("usdx", sdkmath.NewInt(2e6)),
|
2021-08-06 01:43:55 +00:00
|
|
|
),
|
|
|
|
)
|
|
|
|
suite.Keeper.SetDepositorShares(suite.Ctx, types.NewShareRecord(
|
2022-01-08 00:39:27 +00:00
|
|
|
sdk.AccAddress("depositor 1---------"),
|
2021-08-06 01:43:55 +00:00
|
|
|
types.PoolID("hard", "usdx"),
|
2023-04-05 23:21:59 +00:00
|
|
|
sdkmath.NewInt(1e6),
|
2021-08-06 01:43:55 +00:00
|
|
|
))
|
|
|
|
}
|
|
|
|
|
|
|
|
func (suite *invariantTestSuite) RegisterRoute(moduleName string, route string, invariant sdk.Invariant) {
|
|
|
|
_, exists := suite.invariants[moduleName]
|
|
|
|
|
|
|
|
if !exists {
|
|
|
|
suite.invariants[moduleName] = make(map[string]sdk.Invariant)
|
|
|
|
}
|
|
|
|
|
|
|
|
suite.invariants[moduleName][route] = invariant
|
|
|
|
}
|
|
|
|
|
|
|
|
func (suite *invariantTestSuite) runInvariant(route string, invariant func(k keeper.Keeper) sdk.Invariant) (string, bool) {
|
|
|
|
ctx := suite.Ctx
|
|
|
|
registeredInvariant := suite.invariants[types.ModuleName][route]
|
|
|
|
suite.Require().NotNil(registeredInvariant)
|
|
|
|
|
|
|
|
// direct call
|
|
|
|
dMessage, dBroken := invariant(suite.Keeper)(ctx)
|
|
|
|
// registered call
|
|
|
|
rMessage, rBroken := registeredInvariant(ctx)
|
|
|
|
// all call
|
|
|
|
aMessage, aBroken := keeper.AllInvariants(suite.Keeper)(ctx)
|
|
|
|
|
|
|
|
// require matching values for direct call and registered call
|
|
|
|
suite.Require().Equal(dMessage, rMessage, "expected registered invariant message to match")
|
|
|
|
suite.Require().Equal(dBroken, rBroken, "expected registered invariant broken to match")
|
|
|
|
// require matching values for direct call and all invariants call if broken
|
|
|
|
suite.Require().Equal(dBroken, aBroken, "expected all invariant broken to match")
|
|
|
|
if dBroken {
|
|
|
|
suite.Require().Equal(dMessage, aMessage, "expected all invariant message to match")
|
|
|
|
}
|
|
|
|
|
|
|
|
// return message, broken
|
|
|
|
return dMessage, dBroken
|
|
|
|
}
|
|
|
|
|
|
|
|
func (suite *invariantTestSuite) TestPoolRecordsInvariant() {
|
|
|
|
// default state is valid
|
|
|
|
message, broken := suite.runInvariant("pool-records", keeper.PoolRecordsInvariant)
|
|
|
|
suite.Equal("swap: validate pool records broken invariant\npool record invalid\n", message)
|
|
|
|
suite.Equal(false, broken)
|
|
|
|
|
|
|
|
suite.SetupValidState()
|
|
|
|
message, broken = suite.runInvariant("pool-records", keeper.PoolRecordsInvariant)
|
|
|
|
suite.Equal("swap: validate pool records broken invariant\npool record invalid\n", message)
|
|
|
|
suite.Equal(false, broken)
|
|
|
|
|
|
|
|
// broken with invalid pool record
|
|
|
|
suite.Keeper.SetPool_Raw(suite.Ctx, types.NewPoolRecord(
|
|
|
|
sdk.NewCoins(
|
2023-04-05 23:21:59 +00:00
|
|
|
sdk.NewCoin("ukava", sdkmath.NewInt(1e6)),
|
|
|
|
sdk.NewCoin("usdx", sdkmath.NewInt(5e6)),
|
2021-08-06 01:43:55 +00:00
|
|
|
),
|
2023-04-05 23:21:59 +00:00
|
|
|
sdkmath.NewInt(-1e6),
|
2021-08-06 01:43:55 +00:00
|
|
|
))
|
|
|
|
message, broken = suite.runInvariant("pool-records", keeper.PoolRecordsInvariant)
|
|
|
|
suite.Equal("swap: validate pool records broken invariant\npool record invalid\n", message)
|
|
|
|
suite.Equal(true, broken)
|
|
|
|
}
|
|
|
|
|
|
|
|
func (suite *invariantTestSuite) TestShareRecordsInvariant() {
|
|
|
|
message, broken := suite.runInvariant("share-records", keeper.ShareRecordsInvariant)
|
|
|
|
suite.Equal("swap: validate share records broken invariant\nshare record invalid\n", message)
|
|
|
|
suite.Equal(false, broken)
|
|
|
|
|
|
|
|
suite.SetupValidState()
|
|
|
|
message, broken = suite.runInvariant("share-records", keeper.ShareRecordsInvariant)
|
|
|
|
suite.Equal("swap: validate share records broken invariant\nshare record invalid\n", message)
|
|
|
|
suite.Equal(false, broken)
|
|
|
|
|
|
|
|
// broken with invalid share record
|
|
|
|
suite.Keeper.SetDepositorShares_Raw(suite.Ctx, types.NewShareRecord(
|
2022-01-08 00:39:27 +00:00
|
|
|
sdk.AccAddress("depositor 1---------"),
|
2021-08-06 01:43:55 +00:00
|
|
|
types.PoolID("ukava", "usdx"),
|
2023-04-05 23:21:59 +00:00
|
|
|
sdkmath.NewInt(-1e6),
|
2021-08-06 01:43:55 +00:00
|
|
|
))
|
|
|
|
message, broken = suite.runInvariant("share-records", keeper.ShareRecordsInvariant)
|
|
|
|
suite.Equal("swap: validate share records broken invariant\nshare record invalid\n", message)
|
|
|
|
suite.Equal(true, broken)
|
|
|
|
}
|
|
|
|
|
|
|
|
func (suite *invariantTestSuite) TestPoolReservesInvariant() {
|
|
|
|
message, broken := suite.runInvariant("pool-reserves", keeper.PoolReservesInvariant)
|
|
|
|
suite.Equal("swap: pool reserves broken invariant\npool reserves do not match module account\n", message)
|
|
|
|
suite.Equal(false, broken)
|
|
|
|
|
|
|
|
suite.SetupValidState()
|
|
|
|
message, broken = suite.runInvariant("pool-reserves", keeper.PoolReservesInvariant)
|
|
|
|
suite.Equal("swap: pool reserves broken invariant\npool reserves do not match module account\n", message)
|
|
|
|
suite.Equal(false, broken)
|
|
|
|
|
|
|
|
// broken when reserves are greater than module balance
|
|
|
|
suite.Keeper.SetPool(suite.Ctx, types.NewPoolRecord(
|
|
|
|
sdk.NewCoins(
|
2023-04-05 23:21:59 +00:00
|
|
|
sdk.NewCoin("ukava", sdkmath.NewInt(2e6)),
|
|
|
|
sdk.NewCoin("usdx", sdkmath.NewInt(10e6)),
|
2021-08-06 01:43:55 +00:00
|
|
|
),
|
2023-04-05 23:21:59 +00:00
|
|
|
sdkmath.NewInt(5e6),
|
2021-08-06 01:43:55 +00:00
|
|
|
))
|
|
|
|
message, broken = suite.runInvariant("pool-reserves", keeper.PoolReservesInvariant)
|
|
|
|
suite.Equal("swap: pool reserves broken invariant\npool reserves do not match module account\n", message)
|
|
|
|
suite.Equal(true, broken)
|
|
|
|
|
|
|
|
// broken when reserves are less than the module balance
|
|
|
|
suite.Keeper.SetPool(suite.Ctx, types.NewPoolRecord(
|
|
|
|
sdk.NewCoins(
|
2023-04-05 23:21:59 +00:00
|
|
|
sdk.NewCoin("ukava", sdkmath.NewInt(1e5)),
|
|
|
|
sdk.NewCoin("usdx", sdkmath.NewInt(5e5)),
|
2021-08-06 01:43:55 +00:00
|
|
|
),
|
2023-04-05 23:21:59 +00:00
|
|
|
sdkmath.NewInt(3e5),
|
2021-08-06 01:43:55 +00:00
|
|
|
))
|
|
|
|
message, broken = suite.runInvariant("pool-reserves", keeper.PoolReservesInvariant)
|
|
|
|
suite.Equal("swap: pool reserves broken invariant\npool reserves do not match module account\n", message)
|
|
|
|
suite.Equal(true, broken)
|
|
|
|
}
|
|
|
|
|
|
|
|
func (suite *invariantTestSuite) TestPoolSharesInvariant() {
|
|
|
|
message, broken := suite.runInvariant("pool-shares", keeper.PoolSharesInvariant)
|
|
|
|
suite.Equal("swap: pool shares broken invariant\npool shares do not match depositor shares\n", message)
|
|
|
|
suite.Equal(false, broken)
|
|
|
|
|
|
|
|
suite.SetupValidState()
|
|
|
|
message, broken = suite.runInvariant("pool-shares", keeper.PoolSharesInvariant)
|
|
|
|
suite.Equal("swap: pool shares broken invariant\npool shares do not match depositor shares\n", message)
|
|
|
|
suite.Equal(false, broken)
|
|
|
|
|
|
|
|
// broken when total shares are greater than depositor shares
|
|
|
|
suite.Keeper.SetPool(suite.Ctx, types.NewPoolRecord(
|
|
|
|
sdk.NewCoins(
|
2023-04-05 23:21:59 +00:00
|
|
|
sdk.NewCoin("ukava", sdkmath.NewInt(1e6)),
|
|
|
|
sdk.NewCoin("usdx", sdkmath.NewInt(5e6)),
|
2021-08-06 01:43:55 +00:00
|
|
|
),
|
2023-04-05 23:21:59 +00:00
|
|
|
sdkmath.NewInt(5e6),
|
2021-08-06 01:43:55 +00:00
|
|
|
))
|
|
|
|
message, broken = suite.runInvariant("pool-shares", keeper.PoolSharesInvariant)
|
|
|
|
suite.Equal("swap: pool shares broken invariant\npool shares do not match depositor shares\n", message)
|
|
|
|
suite.Equal(true, broken)
|
|
|
|
|
|
|
|
// broken when total shares are less than the depositor shares
|
|
|
|
suite.Keeper.SetPool(suite.Ctx, types.NewPoolRecord(
|
|
|
|
sdk.NewCoins(
|
2023-04-05 23:21:59 +00:00
|
|
|
sdk.NewCoin("ukava", sdkmath.NewInt(1e6)),
|
|
|
|
sdk.NewCoin("usdx", sdkmath.NewInt(5e6)),
|
2021-08-06 01:43:55 +00:00
|
|
|
),
|
2023-04-05 23:21:59 +00:00
|
|
|
sdkmath.NewInt(1e5),
|
2021-08-06 01:43:55 +00:00
|
|
|
))
|
|
|
|
message, broken = suite.runInvariant("pool-shares", keeper.PoolSharesInvariant)
|
|
|
|
suite.Equal("swap: pool shares broken invariant\npool shares do not match depositor shares\n", message)
|
|
|
|
suite.Equal(true, broken)
|
|
|
|
|
|
|
|
// broken when pool record is missing
|
|
|
|
suite.Keeper.DeletePool(suite.Ctx, types.PoolID("ukava", "usdx"))
|
|
|
|
suite.RemoveCoinsFromModule(
|
|
|
|
sdk.NewCoins(
|
2023-04-05 23:21:59 +00:00
|
|
|
sdk.NewCoin("ukava", sdkmath.NewInt(1e6)),
|
|
|
|
sdk.NewCoin("usdx", sdkmath.NewInt(5e6)),
|
2021-08-06 01:43:55 +00:00
|
|
|
),
|
|
|
|
)
|
|
|
|
message, broken = suite.runInvariant("pool-shares", keeper.PoolSharesInvariant)
|
|
|
|
suite.Equal("swap: pool shares broken invariant\npool shares do not match depositor shares\n", message)
|
|
|
|
suite.Equal(true, broken)
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestInvariantTestSuite(t *testing.T) {
|
|
|
|
suite.Run(t, new(invariantTestSuite))
|
|
|
|
}
|