package keeper import ( "github.com/0glabs/0g-chain/x/savings/types" sdk "github.com/cosmos/cosmos-sdk/types" ) // RegisterInvariants registers the savings module invariants func RegisterInvariants(ir sdk.InvariantRegistry, k Keeper) { ir.RegisterRoute(types.ModuleName, "deposits", DepositsInvariant(k)) ir.RegisterRoute(types.ModuleName, "solvency", SolvencyInvariant(k)) } // AllInvariants runs all invariants of the savings module func AllInvariants(k Keeper) sdk.Invariant { return func(ctx sdk.Context) (string, bool) { if res, stop := DepositsInvariant(k)(ctx); stop { return res, stop } res, stop := SolvencyInvariant(k)(ctx) return res, stop } } // DepositsInvariant iterates all deposits and asserts that they are valid func DepositsInvariant(k Keeper) sdk.Invariant { broken := false message := sdk.FormatInvariant(types.ModuleName, "validate deposits broken", "deposit invalid") return func(ctx sdk.Context) (string, bool) { k.IterateDeposits(ctx, func(deposit types.Deposit) bool { if err := deposit.Validate(); err != nil { broken = true return true } if !deposit.Amount.IsAllPositive() { broken = true return true } return false }) return message, broken } } // SolvencyInvariant iterates all deposits and ensures the total amount matches the module account coins func SolvencyInvariant(k Keeper) sdk.Invariant { message := sdk.FormatInvariant(types.ModuleName, "module solvency broken", "total deposited amount does not match module account") return func(ctx sdk.Context) (string, bool) { balance := k.GetSavingsModuleAccountBalances(ctx) deposited := sdk.Coins{} k.IterateDeposits(ctx, func(deposit types.Deposit) bool { for _, coin := range deposit.Amount { deposited = deposited.Add(coin) } return false }) broken := !deposited.IsEqual(balance) return message, broken } }