mirror of
https://github.com/0glabs/0g-chain.git
synced 2024-12-27 00:35:18 +00:00
4ff43eb270
- Add store methods to get/set/delete/etc account fractional balances & remainder amount - Add invariants to ensure stored state is correct
171 lines
4.7 KiB
Go
171 lines
4.7 KiB
Go
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")
|
|
}
|
|
})
|
|
}
|
|
}
|