cosmos-sdk/x/distribution/keeper/allocation.go

86 lines
3.4 KiB
Go
Raw Normal View History

2018-09-04 23:41:17 -07:00
package keeper
2018-09-11 10:35:47 -07:00
import (
2019-01-16 13:38:05 -08:00
abci "github.com/tendermint/tendermint/abci/types"
2018-09-11 10:35:47 -07:00
sdk "github.com/cosmos/cosmos-sdk/types"
)
2018-09-04 23:41:17 -07:00
2019-01-16 13:38:05 -08:00
// allocate fees handles distribution of the collected fees
func (k Keeper) AllocateTokens(ctx sdk.Context, sumPrecommitPower, totalPower int64, proposer sdk.ConsAddress, votes []abci.VoteInfo) {
2019-01-16 13:38:05 -08:00
// fetch collected fees & fee pool
feesCollectedInt := k.feeCollectionKeeper.GetCollectedFees(ctx)
feesCollected := sdk.NewDecCoins(feesCollectedInt)
feePool := k.GetFeePool(ctx)
2018-09-05 15:29:20 -07:00
2019-01-16 13:38:05 -08:00
// clear collected fees, which will now be distributed
k.feeCollectionKeeper.ClearCollectedFees(ctx)
2018-09-11 10:35:47 -07:00
2019-01-16 13:38:05 -08:00
// temporary workaround to keep CanWithdrawInvariant happy
// general discussions here: https://github.com/cosmos/cosmos-sdk/issues/2906#issuecomment-441867634
if totalPower == 0 {
feePool.CommunityPool = feePool.CommunityPool.Add(feesCollected)
k.SetFeePool(ctx, feePool)
return
}
2019-01-16 13:38:05 -08:00
// calculate fraction votes
fractionVotes := sdk.NewDec(sumPrecommitPower).Quo(sdk.NewDec(totalPower))
// calculate proposer reward
2018-10-15 13:12:42 -07:00
baseProposerReward := k.GetBaseProposerReward(ctx)
bonusProposerReward := k.GetBonusProposerReward(ctx)
2019-01-16 13:38:05 -08:00
proposerMultiplier := baseProposerReward.Add(bonusProposerReward.Mul(fractionVotes))
proposerReward := feesCollected.MulDec(proposerMultiplier)
2018-09-04 23:41:17 -07:00
2019-01-16 13:38:05 -08:00
// pay proposer
proposerValidator := k.stakingKeeper.ValidatorByConsAddr(ctx, proposer)
k.AllocateTokensToValidator(ctx, proposerValidator, proposerReward)
remaining := feesCollected.Sub(proposerReward)
2018-09-04 23:41:17 -07:00
2019-01-16 13:38:05 -08:00
// calculate fraction allocated to validators
2018-09-18 21:42:05 -07:00
communityTax := k.GetCommunityTax(ctx)
2019-01-16 13:38:05 -08:00
voteMultiplier := sdk.OneDec().Sub(proposerMultiplier).Sub(communityTax)
2018-09-04 23:41:17 -07:00
2019-01-16 13:38:05 -08:00
// allocate tokens proportionally to voting power
// TODO consider parallelizing later, ref https://github.com/cosmos/cosmos-sdk/pull/3099#discussion_r246276376
for _, vote := range votes {
validator := k.stakingKeeper.ValidatorByConsAddr(ctx, vote.Validator.Address)
2018-09-04 23:41:17 -07:00
2019-01-16 13:38:05 -08:00
// TODO likely we should only reward validators who actually signed the block.
// ref https://github.com/cosmos/cosmos-sdk/issues/2525#issuecomment-430838701
powerFraction := sdk.NewDec(vote.Validator.Power).Quo(sdk.NewDec(totalPower))
reward := feesCollected.MulDec(voteMultiplier).MulDec(powerFraction)
k.AllocateTokensToValidator(ctx, validator, reward)
remaining = remaining.Sub(reward)
2019-01-16 13:38:05 -08:00
}
// allocate community funding
feePool.CommunityPool = feePool.CommunityPool.Add(remaining)
2018-09-19 19:13:12 -07:00
k.SetFeePool(ctx, feePool)
2018-09-11 10:35:47 -07:00
2019-01-16 13:38:05 -08:00
// update outstanding rewards
outstanding := k.GetOutstandingRewards(ctx)
outstanding = outstanding.Add(feesCollected.Sub(remaining))
2019-01-16 13:38:05 -08:00
k.SetOutstandingRewards(ctx, outstanding)
}
// allocate tokens to a particular validator, splitting according to commission
func (k Keeper) AllocateTokensToValidator(ctx sdk.Context, val sdk.Validator, tokens sdk.DecCoins) {
// split tokens between validator and delegators according to commission
commission := tokens.MulDec(val.GetCommission())
shared := tokens.Sub(commission)
2019-01-16 13:38:05 -08:00
// update current commission
currentCommission := k.GetValidatorAccumulatedCommission(ctx, val.GetOperator())
currentCommission = currentCommission.Add(commission)
2019-01-16 13:38:05 -08:00
k.SetValidatorAccumulatedCommission(ctx, val.GetOperator(), currentCommission)
// update current rewards
currentRewards := k.GetValidatorCurrentRewards(ctx, val.GetOperator())
currentRewards.Rewards = currentRewards.Rewards.Add(shared)
2019-01-16 13:38:05 -08:00
k.SetValidatorCurrentRewards(ctx, val.GetOperator(), currentRewards)
2018-09-04 23:41:17 -07:00
}