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

fix: apply suggestions from the audit review #202

Merged
merged 11 commits into from
Nov 10, 2021
6 changes: 4 additions & 2 deletions x/farming/keeper/genesis.go
Original file line number Diff line number Diff line change
Expand Up @@ -20,8 +20,10 @@ func (k Keeper) InitGenesis(ctx sdk.Context, genState types.GenesisState) {
k.SetParams(ctx, genState.Params)
// TODO: what if CurrentEpochDays field was empty?
k.SetCurrentEpochDays(ctx, genState.CurrentEpochDays)
moduleAcc := k.accountKeeper.GetModuleAccount(ctx, types.ModuleName)
k.accountKeeper.SetModuleAccount(ctx, moduleAcc)
if addr := k.accountKeeper.GetModuleAddress(types.ModuleName); addr == nil {
panic(fmt.Sprintf("%s module account has not been set", types.ModuleName))
}


for i, record := range genState.PlanRecords {
plan, err := types.UnpackPlan(&record.Plan)
Expand Down
14 changes: 2 additions & 12 deletions x/farming/keeper/plan.go
Original file line number Diff line number Diff line change
Expand Up @@ -140,19 +140,14 @@ func (k Keeper) CreateFixedAmountPlan(ctx sdk.Context, msg *types.MsgCreateFixed
nextId := k.GetNextPlanIdWithUpdate(ctx)
if typ == types.PlanTypePrivate {
params := k.GetParams(ctx)
balances := k.bankKeeper.GetAllBalances(ctx, msg.GetCreator())
diffs, hasNeg := balances.SafeSub(params.PrivatePlanCreationFee)
if hasNeg {
return nil, sdkerrors.Wrapf(sdkerrors.ErrInsufficientFunds, "lack of %s coins to pay private plan creation fee", diffs.String())
}

farmingFeeCollectorAcc, err := sdk.AccAddressFromBech32(params.FarmingFeeCollector)
if err != nil {
return nil, err
}

if err := k.bankKeeper.SendCoins(ctx, msg.GetCreator(), farmingFeeCollectorAcc, params.PrivatePlanCreationFee); err != nil {
return nil, err
return nil, sdkerrors.Wrap(err, "failed to pay private plan creation fee")
}
}

Expand Down Expand Up @@ -191,19 +186,14 @@ func (k Keeper) CreateRatioPlan(ctx sdk.Context, msg *types.MsgCreateRatioPlan,
nextId := k.GetNextPlanIdWithUpdate(ctx)
if typ == types.PlanTypePrivate {
params := k.GetParams(ctx)
balances := k.bankKeeper.GetAllBalances(ctx, msg.GetCreator())
diffs, hasNeg := balances.SafeSub(params.PrivatePlanCreationFee)
if hasNeg {
return nil, sdkerrors.Wrapf(sdkerrors.ErrInsufficientFunds, "lack of %s coins to pay private plan createion fee", diffs.String())
}

farmingFeeCollectorAcc, err := sdk.AccAddressFromBech32(params.FarmingFeeCollector)
if err != nil {
return nil, err
}

if err := k.bankKeeper.SendCoins(ctx, msg.GetCreator(), farmingFeeCollectorAcc, params.PrivatePlanCreationFee); err != nil {
return nil, err
return nil, sdkerrors.Wrap(err, "failed to pay private plan creation fee")
}
}

Expand Down
69 changes: 49 additions & 20 deletions x/farming/keeper/reward.go
Original file line number Diff line number Diff line change
Expand Up @@ -283,6 +283,7 @@ func (k Keeper) Harvest(ctx sdk.Context, farmerAcc sdk.AccAddress, stakingCoinDe
types.EventTypeHarvest,
sdk.NewAttribute(types.AttributeKeyFarmer, farmerAcc.String()),
sdk.NewAttribute(types.AttributeKeyStakingCoinDenoms, strings.Join(stakingCoinDenoms, ",")),
sdk.NewAttribute(types.AttributeKeyRewardCoins, totalRewards.String()),
),
})

Expand All @@ -300,33 +301,50 @@ type AllocationInfo struct {
// When total allocated coins for a farming pool exceeds the pool's
// balance, then allocation will not happen.
func (k Keeper) AllocationInfos(ctx sdk.Context) []AllocationInfo {
farmingPoolBalances := make(map[string]sdk.Coins) // farmingPoolAddress => sdk.Coins
allocCoins := make(map[string]map[uint64]sdk.Coins) // farmingPoolAddress => (planId => sdk.Coins)

plans := make(map[uint64]types.PlanI)
// farmingPoolBalances is a cache for balances of each farming pool,
// to reduce number of BankKeeper.GetAllBalances calls.
// It maps farmingPoolAddress to the pool's balance.
farmingPoolBalances := map[string]sdk.Coins{}

// allocCoins is a table that records which farming pool allocates
// how many coins to which plan.
// It maps farmingPoolAddress to a map that maps planId to amount of
// coins to allocate.
allocCoins := map[string]map[uint64]sdk.Coins{}

plans := map[uint64]types.PlanI{} // it maps planId to plan.
for _, plan := range k.GetPlans(ctx) {
// Filter plans by their start time and end time.
// Add plans that are not terminated and active to the map.
if !plan.GetTerminated() && types.IsPlanActiveAt(plan, ctx.BlockTime()) {
plans[plan.GetId()] = plan
}
}

// Calculate how many coins the plans want to allocate rewards from farming pools.
// Note that in this step, we don't check if the farming pool has
// sufficient balance for all allocations. We'll do that check in the next step.
for _, plan := range plans {
farmingPoolAcc := plan.GetFarmingPoolAddress()
farmingPool := farmingPoolAcc.String()

// Lookup if we already have the farming pool's balance in the cache.
// If not, call BankKeeper.GetAllBalances and add the result to the cache.
balances, ok := farmingPoolBalances[farmingPool]
if !ok {
balances = k.bankKeeper.GetAllBalances(ctx, farmingPoolAcc)
farmingPoolBalances[farmingPool] = balances
}

// Lookup if we already have the farming pool's allocation map in the cache.
// If not, create new allocation map and add it to the cache.
ac, ok := allocCoins[farmingPool]
if !ok {
ac = make(map[uint64]sdk.Coins)
ac = map[uint64]sdk.Coins{}
allocCoins[farmingPool] = ac
}

// Based on the plan's type, record how many coins the plan wants to
// allocate in the allocation map.
switch plan := plan.(type) {
case *types.FixedAmountPlan:
ac[plan.GetId()] = plan.EpochAmount
Expand All @@ -335,10 +353,12 @@ func (k Keeper) AllocationInfos(ctx sdk.Context) []AllocationInfo {
}
}

// In this step, we check if farming pools have sufficient balance for allocations.
// If not, we don't allocate rewards from that farming pool for this epoch.
var allocInfos []AllocationInfo
for farmingPool, coins := range allocCoins {
for farmingPool, planCoins := range allocCoins {
totalCoins := sdk.NewCoins()
for _, amt := range coins {
for _, amt := range planCoins {
totalCoins = totalCoins.Add(amt...)
}

Expand All @@ -347,7 +367,7 @@ func (k Keeper) AllocationInfos(ctx sdk.Context) []AllocationInfo {
continue
}

for planID, amt := range coins {
for planID, amt := range planCoins {
allocInfos = append(allocInfos, AllocationInfo{
Plan: plans[planID],
Amount: amt,
Expand All @@ -361,35 +381,42 @@ func (k Keeper) AllocationInfos(ctx sdk.Context) []AllocationInfo {
// AllocateRewards updates historical rewards and current epoch info
// based on the allocation infos.
func (k Keeper) AllocateRewards(ctx sdk.Context) error {
unitRewardsByDenom := map[string]sdk.DecCoins{} // (staking coin denom) => (unit rewards)
// unitRewardsByDenom is a table that records how much unit rewards should
// be increased in this epoch, for each staking coin denom.
// It maps staking coin denom to unit rewards.
unitRewardsByDenom := map[string]sdk.DecCoins{}

for _, allocInfo := range k.AllocationInfos(ctx) {
totalWeight := sdk.ZeroDec()
for _, weight := range allocInfo.Plan.GetStakingCoinWeights() {
totalWeight = totalWeight.Add(weight.Amount)
}
// Get allocation information first.
allocInfos := k.AllocationInfos(ctx)

for _, allocInfo := range allocInfos {
totalAllocCoins := sdk.NewCoins()

// Calculate how many coins are allocated based on each staking coin weight.
// It is calculated with the following formula:
// (unit rewards for this epoch) = (weighted rewards for the denom) / (total staking amount for the denom)
for _, weight := range allocInfo.Plan.GetStakingCoinWeights() {
// Check if there are any coins staked for this denom.
// If not, skip this denom for rewards allocation.
totalStakings, found := k.GetTotalStakings(ctx, weight.Denom)
if !found {
continue
}
if !totalStakings.Amount.IsPositive() {
continue
}

weightProportion := weight.Amount.QuoTruncate(totalWeight)
allocCoins, _ := sdk.NewDecCoinsFromCoins(allocInfo.Amount...).MulDecTruncate(weightProportion).TruncateDecimal()
allocCoins, _ := sdk.NewDecCoinsFromCoins(allocInfo.Amount...).MulDecTruncate(weight.Amount).TruncateDecimal()
allocCoinsDec := sdk.NewDecCoinsFromCoins(allocCoins...)

// Multiple plans can have same denom in their staking coin weights,
// so we accumulate all unit rewards for this denom in the table.
unitRewardsByDenom[weight.Denom] = unitRewardsByDenom[weight.Denom].Add(allocCoinsDec.QuoDecTruncate(totalStakings.Amount.ToDec())...)

k.IncreaseOutstandingRewards(ctx, weight.Denom, allocCoinsDec)

totalAllocCoins = totalAllocCoins.Add(allocCoins...)
}

// If total allocated amount for this plan is zero, then skip allocation
// for this plan.
if totalAllocCoins.IsZero() {
continue
}
Expand All @@ -413,6 +440,8 @@ func (k Keeper) AllocateRewards(ctx sdk.Context) error {
})
}

// For each staking coin denom in the table, increase cumulative unit rewards
// and increment current epoch number by 1.
for stakingCoinDenom, unitRewards := range unitRewardsByDenom {
currentEpoch := k.GetCurrentEpoch(ctx, stakingCoinDenom)
historical, _ := k.GetHistoricalRewards(ctx, stakingCoinDenom, currentEpoch-1)
Expand Down
10 changes: 4 additions & 6 deletions x/farming/keeper/staking.go
Original file line number Diff line number Diff line change
Expand Up @@ -300,9 +300,8 @@ func (k Keeper) Unstake(ctx sdk.Context, farmerAcc sdk.AccAddress, amount sdk.Co
return err
}

staking.Amount = staking.Amount.Add(queuedStaking.Amount)
removedFromStaking := queuedStaking.Amount.Neg()
queuedStaking.Amount = sdk.ZeroInt()
removedFromStaking := queuedStaking.Amount.Neg() // Make negative a positive
staking.Amount = staking.Amount.Sub(removedFromStaking)
if staking.Amount.IsPositive() {
currentEpoch := k.GetCurrentEpoch(ctx, coin.Denom)
staking.StartingEpoch = currentEpoch
Expand All @@ -311,10 +310,9 @@ func (k Keeper) Unstake(ctx sdk.Context, farmerAcc sdk.AccAddress, amount sdk.Co
k.DeleteStaking(ctx, coin.Denom, farmerAcc)
}

k.DeleteQueuedStaking(ctx, coin.Denom, farmerAcc)
k.DecreaseTotalStakings(ctx, coin.Denom, removedFromStaking)
}

if queuedStaking.Amount.IsPositive() {
} else if queuedStaking.Amount.IsPositive() {
k.SetQueuedStaking(ctx, coin.Denom, farmerAcc, queuedStaking)
} else {
k.DeleteQueuedStaking(ctx, coin.Denom, farmerAcc)
Expand Down