Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

feat: add and refactor invariant checks #186

Merged
merged 6 commits into from
Oct 28, 2021
Merged
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
2 changes: 1 addition & 1 deletion x/farming/keeper/genesis.go
Original file line number Diff line number Diff line change
Expand Up @@ -99,7 +99,7 @@ func (k Keeper) InitGenesis(ctx sdk.Context, genState types.GenesisState) {
stakingReserveCoins, genState.StakingReserveCoins))
}

if err := k.ValidateOutstandingRewards(ctx); err != nil {
if err := k.ValidateOutstandingRewardsAmount(ctx); err != nil {
panic(err)
}

Expand Down
201 changes: 191 additions & 10 deletions x/farming/keeper/invariants.go
Original file line number Diff line number Diff line change
@@ -1,27 +1,96 @@
package keeper

import (
"fmt"

sdk "github.com/cosmos/cosmos-sdk/types"

"github.com/tendermint/farming/x/farming/types"
)

// RegisterInvariants registers all farming invariants.
func RegisterInvariants(ir sdk.InvariantRegistry, k Keeper) {
ir.RegisterRoute(types.ModuleName, "staking-reserved",
ir.RegisterRoute(types.ModuleName, "positive-staking-amount",
PositiveStakingAmountInvariant(k))
ir.RegisterRoute(types.ModuleName, "positive-queued-staking-amount",
PositiveQueuedStakingAmountInvariant(k))
ir.RegisterRoute(types.ModuleName, "staking-reserved-amount",
StakingReservedAmountInvariant(k))
ir.RegisterRoute(types.ModuleName, "remaining-rewards",
ir.RegisterRoute(types.ModuleName, "remaining-rewards-amount",
RemainingRewardsAmountInvariant(k))
ir.RegisterRoute(types.ModuleName, "non-negative-outstanding-rewards",
NonNegativeOutstandingRewardsInvariant(k))
ir.RegisterRoute(types.ModuleName, "outstanding-rewards-amount",
OutstandingRewardsAmountInvariant(k))
ir.RegisterRoute(types.ModuleName, "non-negative-historical-rewards",
NonNegativeHistoricalRewardsInvariant(k))
ir.RegisterRoute(types.ModuleName, "positive-total-stakings-amount",
PositiveTotalStakingsAmountInvariant(k))
ir.RegisterRoute(types.ModuleName, "plan-termination-status",
PlanTerminationStatusInvariant(k))
}

// AllInvariants runs all invariants of the farming module.
func AllInvariants(k Keeper) sdk.Invariant {
return func(ctx sdk.Context) (string, bool) {
res, stop := StakingReservedAmountInvariant(k)(ctx)
if stop {
return res, stop
for _, inv := range []func(Keeper) sdk.Invariant{
PositiveStakingAmountInvariant,
StakingReservedAmountInvariant,
RemainingRewardsAmountInvariant,
NonNegativeOutstandingRewardsInvariant,
OutstandingRewardsAmountInvariant,
NonNegativeHistoricalRewardsInvariant,
PositiveTotalStakingsAmountInvariant,
PlanTerminationStatusInvariant,
} {
res, stop := inv(k)(ctx)
if stop {
return res, stop
}
}
return RemainingRewardsAmountInvariant(k)(ctx)
return "", false
}
}

// PositiveStakingAmountInvariant checks that the amount of staking coins is positive.
func PositiveStakingAmountInvariant(k Keeper) sdk.Invariant {
return func(ctx sdk.Context) (string, bool) {
msg := ""
count := 0
k.IterateStakings(ctx, func(stakingCoinDenom string, farmerAcc sdk.AccAddress, staking types.Staking) (stop bool) {
if !staking.Amount.IsPositive() {
msg += fmt.Sprintf("\t%v has non-positive staking amount: %v\n",
farmerAcc, sdk.Coin{Denom: stakingCoinDenom, Amount: staking.Amount})
count++
}
return false
})
broken := count != 0
return sdk.FormatInvariant(
types.ModuleName, "positive staking amount",
fmt.Sprintf("found %d staking coins with non-positive amount\n%s", count, msg),
), broken
}
}

// PositiveQueuedStakingAmountInvariant checks that the amount of queued staking coins is positive.
func PositiveQueuedStakingAmountInvariant(k Keeper) sdk.Invariant {
return func(ctx sdk.Context) (string, bool) {
msg := ""
count := 0
k.IterateQueuedStakings(ctx, func(stakingCoinDenom string, farmerAcc sdk.AccAddress, queuedStaking types.QueuedStaking) (stop bool) {
if !queuedStaking.Amount.IsPositive() {
msg += fmt.Sprintf("\t%v has non-positive queued staking amount: %v\n",
farmerAcc, sdk.Coin{Denom: stakingCoinDenom, Amount: queuedStaking.Amount})
count++
}
return false
})
broken := count != 0
return sdk.FormatInvariant(
types.ModuleName, "positive queued staking amount",
fmt.Sprintf("found %d queued staking coins with non-positive amount\n%s", count, msg),
), broken
}
}

Expand All @@ -30,8 +99,9 @@ func StakingReservedAmountInvariant(k Keeper) sdk.Invariant {
return func(ctx sdk.Context) (string, bool) {
err := k.ValidateStakingReservedAmount(ctx)
broken := err != nil
return sdk.FormatInvariant(types.ModuleName, "staking reserved amount invariant broken",
"the balance of StakingReserveAcc less than the amount of staked, Queued coins in all staking objects"), broken
return sdk.FormatInvariant(types.ModuleName, "staking reserved amount",
"the balance of StakingReserveAcc less than the amount of staked, queued coins in all staking objects",
), broken
}
}

Expand All @@ -40,7 +110,118 @@ func RemainingRewardsAmountInvariant(k Keeper) sdk.Invariant {
return func(ctx sdk.Context) (string, bool) {
err := k.ValidateRemainingRewardsAmount(ctx)
broken := err != nil
return sdk.FormatInvariant(types.ModuleName, "remaining rewards amount invariant broken",
"the balance of the RewardPoolAddresses of all plans less than the total amount of unwithdrawn reward coins in all reward objects"), broken
return sdk.FormatInvariant(types.ModuleName, "remaining rewards amount",
"the balance of the RewardPoolAddresses of all plans less than the total amount of unwithdrawn reward coins in all reward objects",
), broken
}
}

// NonNegativeOutstandingRewardsInvariant checks that all OutstandingRewards are
// non-negative.
func NonNegativeOutstandingRewardsInvariant(k Keeper) sdk.Invariant {
return func(ctx sdk.Context) (string, bool) {
msg := ""
count := 0
k.IterateOutstandingRewards(ctx, func(stakingCoinDenom string, rewards types.OutstandingRewards) (stop bool) {
if rewards.Rewards.IsAnyNegative() {
msg += fmt.Sprintf("\t%v has negative outstanding rewards: %v\n", stakingCoinDenom, rewards.Rewards)
count++
}
return false
})
broken := count != 0
return sdk.FormatInvariant(
types.ModuleName, "non-negative outstanding rewards",
fmt.Sprintf("found %d staking coin with negative outstanding rewards\n%s", count, msg),
), broken
}
}

// OutstandingRewardsAmountInvariant checks that OutstandingRewards are
// consistent with rewards that can be withdrawn.
func OutstandingRewardsAmountInvariant(k Keeper) sdk.Invariant {
return func(ctx sdk.Context) (string, bool) {
totalRewards := sdk.DecCoins{}
k.IterateOutstandingRewards(ctx, func(stakingCoinDenom string, rewards types.OutstandingRewards) (stop bool) {
totalRewards = totalRewards.Add(rewards.Rewards...)
return false
})
balances := k.bankKeeper.GetAllBalances(ctx, k.GetRewardsReservePoolAcc(ctx))
_, hasNeg := sdk.NewDecCoinsFromCoins(balances...).SafeSub(totalRewards)
broken := hasNeg
return sdk.FormatInvariant(
types.ModuleName, "wrong outstanding rewards",
fmt.Sprintf("balance of rewards reserve pool is less than outstanding rewards\n"+
"\texpected minimum amount of balance: %s\n"+
"\tbalance: %s", totalRewards, balances,
),
), broken
}
}

// NonNegativeHistoricalRewardsInvariant checks that all HistoricalRewards are
// non-negative.
func NonNegativeHistoricalRewardsInvariant(k Keeper) sdk.Invariant {
return func(ctx sdk.Context) (string, bool) {
msg := ""
count := 0
k.IterateHistoricalRewards(ctx, func(stakingCoinDenom string, epoch uint64, rewards types.HistoricalRewards) (stop bool) {
if rewards.CumulativeUnitRewards.IsAnyNegative() {
msg += fmt.Sprintf("\t%v has negative historical rewards at epoch %d: %v\n",
stakingCoinDenom, epoch, rewards.CumulativeUnitRewards)
count++
}
return false
})
broken := count != 0
return sdk.FormatInvariant(
types.ModuleName, "non-negative historical rewards",
fmt.Sprintf("found %d staking coin with negative historical rewards\n%s", count, msg),
), broken
}
}

// PositiveTotalStakingsAmountInvariant checks that all TotalStakings
// have positive amount.
func PositiveTotalStakingsAmountInvariant(k Keeper) sdk.Invariant {
return func(ctx sdk.Context) (string, bool) {
msg := ""
count := 0
k.IterateTotalStakings(ctx, func(stakingCoinDenom string, totalStakings types.TotalStakings) (stop bool) {
if !totalStakings.Amount.IsPositive() {
msg += fmt.Sprintf("\t%v has non-positive total staking amount: %v\n",
stakingCoinDenom, totalStakings.Amount)
count++
}
return false
})
broken := count != 0
return sdk.FormatInvariant(
types.ModuleName, "positive total stakings amount",
fmt.Sprintf("found %d total stakings with non-positive amount\n%s", count, msg),
), broken
}
}

// PlanTerminationStatusInvariant checks that all plans that should have been
// terminated have been terminated.
func PlanTerminationStatusInvariant(k Keeper) sdk.Invariant {
return func(ctx sdk.Context) (string, bool) {
msg := ""
count := 0
k.IteratePlans(ctx, func(plan types.PlanI) (stop bool) {
expected := ctx.BlockTime().After(plan.GetEndTime())
terminated := plan.GetTerminated()
if terminated != expected {
msg += fmt.Sprintf("\tplan %d should have been terminated but not\n", plan.GetId())
count++
}
return false
})
broken := count != 0
return sdk.FormatInvariant(
types.ModuleName, "plan termination status",
fmt.Sprintf("found %d plans have not been terminated\n%s", count, msg),
), broken
}
}
Loading