mirror of
https://github.com/0glabs/0g-chain.git
synced 2025-01-18 19:15:19 +00:00
4cf41d18c2
Implement GetBalance for extended balances which passes through to `x/bank` for non-extended denoms. This diverges from `x/evmutil` behavior which will panic on non-"akava" calls. Add bank / account keeper mocks for testing, with mockery config for [mockery package setup](https://vektra.github.io/mockery/latest/migrating_to_packages/)
216 lines
6.5 KiB
Go
216 lines
6.5 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/kava-labs/kava/x/precisebank/types/mocks"
|
|
"github.com/stretchr/testify/require"
|
|
)
|
|
|
|
func TestBalancedFractionalTotalInvariant(t *testing.T) {
|
|
tests := []struct {
|
|
name string
|
|
setupFn func(ctx sdk.Context, k keeper.Keeper)
|
|
wantBroken bool
|
|
wantMsg string
|
|
}{
|
|
{
|
|
"valid - empty state",
|
|
func(_ sdk.Context, _ keeper.Keeper) {},
|
|
false,
|
|
"",
|
|
},
|
|
{
|
|
"valid - balances, 0 remainder",
|
|
func(ctx sdk.Context, k keeper.Keeper) {
|
|
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(ctx sdk.Context, k keeper.Keeper) {
|
|
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(ctx sdk.Context, k keeper.Keeper) {
|
|
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(ctx sdk.Context, k keeper.Keeper) {
|
|
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(ctx sdk.Context, k keeper.Keeper) {
|
|
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
|
|
td := NewMockedTestData(t)
|
|
|
|
tt.setupFn(td.ctx, td.keeper)
|
|
|
|
invariantFn := keeper.BalancedFractionalTotalInvariant(td.keeper)
|
|
msg, broken := invariantFn(td.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) {
|
|
tests := []struct {
|
|
name string
|
|
setupFn func(ctx sdk.Context, k keeper.Keeper, storeKey storetypes.StoreKey)
|
|
wantBroken bool
|
|
wantMsg string
|
|
}{
|
|
{
|
|
"valid - empty state",
|
|
func(_ sdk.Context, _ keeper.Keeper, _ storetypes.StoreKey) {},
|
|
false,
|
|
"",
|
|
},
|
|
{
|
|
"valid - valid balances",
|
|
func(ctx sdk.Context, k keeper.Keeper, _ storetypes.StoreKey) {
|
|
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(ctx sdk.Context, _ keeper.Keeper, storeKey storetypes.StoreKey) {
|
|
// 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
|
|
td := NewMockedTestData(t)
|
|
|
|
tt.setupFn(td.ctx, td.keeper, td.storeKey)
|
|
|
|
invariantFn := keeper.ValidFractionalAmountsInvariant(td.keeper)
|
|
msg, broken := invariantFn(td.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 TestFractionalDenomNotInBankInvariant(t *testing.T) {
|
|
tests := []struct {
|
|
name string
|
|
setupFn func(ctx sdk.Context, bk *mocks.MockBankKeeper)
|
|
wantBroken bool
|
|
wantMsg string
|
|
}{
|
|
{
|
|
"valid - integer denom (ukava) supply",
|
|
func(ctx sdk.Context, bk *mocks.MockBankKeeper) {
|
|
// No fractional balance in x/bank
|
|
// This also enforces there is no GetSupply() call for IntegerCoinDenom / ukava
|
|
bk.EXPECT().
|
|
GetSupply(ctx, types.ExtendedCoinDenom).
|
|
Return(sdk.NewCoin(types.ExtendedCoinDenom, sdkmath.ZeroInt())).
|
|
Once()
|
|
},
|
|
false,
|
|
"",
|
|
},
|
|
{
|
|
"invalid - x/bank contains fractional denom (akava)",
|
|
func(ctx sdk.Context, bk *mocks.MockBankKeeper) {
|
|
bk.EXPECT().
|
|
GetSupply(ctx, types.ExtendedCoinDenom).
|
|
Return(sdk.NewCoin(types.ExtendedCoinDenom, sdk.NewInt(1000))).
|
|
Once()
|
|
},
|
|
true,
|
|
"precisebank: fractional-denom-not-in-bank invariant\nx/bank should not hold any akava but has supply of 1000akava\n",
|
|
},
|
|
}
|
|
|
|
for _, tt := range tests {
|
|
t.Run(tt.name, func(t *testing.T) {
|
|
// Reset each time
|
|
td := NewMockedTestData(t)
|
|
|
|
tt.setupFn(td.ctx, td.bk)
|
|
|
|
invariantFn := keeper.FractionalDenomNotInBankInvariant(td.keeper)
|
|
msg, broken := invariantFn(td.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")
|
|
}
|
|
})
|
|
}
|
|
}
|