0g-chain/x/precisebank/keeper/invariants_test.go

171 lines
4.7 KiB
Go
Raw Normal View History

package keeper_test
import (
"testing"
sdkmath "cosmossdk.io/math"
"github.com/cosmos/cosmos-sdk/store/prefix"
storetypes "github.com/cosmos/cosmos-sdk/store/types"
sdk "github.com/cosmos/cosmos-sdk/types"
"github.com/kava-labs/kava/x/precisebank/keeper"
"github.com/kava-labs/kava/x/precisebank/types"
"github.com/stretchr/testify/require"
)
func TestBalancedFractionalTotalInvariant(t *testing.T) {
var ctx sdk.Context
var k keeper.Keeper
tests := []struct {
name string
setupFn func()
wantBroken bool
wantMsg string
}{
{
"valid - empty state",
func() {},
false,
"",
},
{
"valid - balances, 0 remainder",
func() {
k.SetFractionalBalance(ctx, sdk.AccAddress{1}, types.ConversionFactor().QuoRaw(2))
k.SetFractionalBalance(ctx, sdk.AccAddress{2}, types.ConversionFactor().QuoRaw(2))
},
false,
"",
},
{
"valid - balances, non-zero remainder",
func() {
k.SetFractionalBalance(ctx, sdk.AccAddress{1}, types.ConversionFactor().QuoRaw(2))
k.SetFractionalBalance(ctx, sdk.AccAddress{2}, types.ConversionFactor().QuoRaw(2).SubRaw(1))
k.SetRemainderAmount(ctx, sdkmath.OneInt())
},
false,
"",
},
{
"invalid - balances, 0 remainder",
func() {
k.SetFractionalBalance(ctx, sdk.AccAddress{1}, types.ConversionFactor().QuoRaw(2))
k.SetFractionalBalance(ctx, sdk.AccAddress{2}, types.ConversionFactor().QuoRaw(2).SubRaw(1))
},
true,
"precisebank: balance-remainder-total invariant\n(sum(FractionalBalances) + remainder) % conversionFactor should be 0 but got 999999999999\n",
},
{
"invalid - invalid balances, non-zero (insufficient) remainder",
func() {
k.SetFractionalBalance(ctx, sdk.AccAddress{1}, types.ConversionFactor().QuoRaw(2))
k.SetFractionalBalance(ctx, sdk.AccAddress{2}, types.ConversionFactor().QuoRaw(2).SubRaw(2))
k.SetRemainderAmount(ctx, sdkmath.OneInt())
},
true,
"precisebank: balance-remainder-total invariant\n(sum(FractionalBalances) + remainder) % conversionFactor should be 0 but got 999999999999\n",
},
{
"invalid - invalid balances, non-zero (excess) remainder",
func() {
k.SetFractionalBalance(ctx, sdk.AccAddress{1}, types.ConversionFactor().QuoRaw(2))
k.SetFractionalBalance(ctx, sdk.AccAddress{2}, types.ConversionFactor().QuoRaw(2).SubRaw(2))
k.SetRemainderAmount(ctx, sdkmath.NewInt(5))
},
true,
"precisebank: balance-remainder-total invariant\n(sum(FractionalBalances) + remainder) % conversionFactor should be 0 but got 3\n",
},
}
for _, tt := range tests {
t.Run(tt.name, func(t *testing.T) {
// Reset each time
tk := NewTestKeeper()
ctx, k = tk.ctx, tk.keeper
tt.setupFn()
invariantFn := keeper.BalancedFractionalTotalInvariant(k)
msg, broken := invariantFn(ctx)
if tt.wantBroken {
require.True(t, broken, "invariant should be broken but is not")
require.Equal(t, tt.wantMsg, msg)
} else {
require.False(t, broken, "invariant should not be broken but is")
}
})
}
}
func TestValidFractionalAmountsInvariant(t *testing.T) {
var ctx sdk.Context
var k keeper.Keeper
var storeKey storetypes.StoreKey
tests := []struct {
name string
setupFn func()
wantBroken bool
wantMsg string
}{
{
"valid - empty state",
func() {},
false,
"",
},
{
"valid - valid balances",
func() {
k.SetFractionalBalance(ctx, sdk.AccAddress{1}, types.ConversionFactor().QuoRaw(2))
k.SetFractionalBalance(ctx, sdk.AccAddress{2}, types.ConversionFactor().QuoRaw(2))
},
false,
"",
},
{
"invalid - exceeds max balance",
func() {
// Requires manual store manipulation so it is unlikely to have
// invalid state in practice. SetFractionalBalance will validate
// before setting.
addr := sdk.AccAddress{1}
amount := types.ConversionFactor()
store := prefix.NewStore(ctx.KVStore(storeKey), types.FractionalBalancePrefix)
amountBytes, err := amount.Marshal()
require.NoError(t, err)
store.Set(types.FractionalBalanceKey(addr), amountBytes)
},
true,
"precisebank: valid-fractional-balances invariant\namount of invalid fractional balances found 1\n\tkava1qy0xn7za has an invalid fractional amount of 1000000000000\n\n",
},
}
for _, tt := range tests {
t.Run(tt.name, func(t *testing.T) {
// Reset each time
tk := NewTestKeeper()
ctx, k, storeKey = tk.ctx, tk.keeper, tk.storeKey
tt.setupFn()
invariantFn := keeper.ValidFractionalAmountsInvariant(k)
msg, broken := invariantFn(ctx)
if tt.wantBroken {
require.True(t, broken, "invariant should be broken but is not")
require.Equal(t, tt.wantMsg, msg)
} else {
require.False(t, broken, "invariant should not be broken but is")
}
})
}
}