cosmos-sdk/x/stake/tick.go

76 lines
2.3 KiB
Go
Raw Normal View History

2018-02-23 15:57:31 -08:00
package stake
import (
sdk "github.com/cosmos/cosmos-sdk/types"
abci "github.com/tendermint/abci/types"
2018-02-23 15:57:31 -08:00
)
const (
2018-03-29 05:27:35 -07:00
hrsPerYr = 8766 // as defined by a julian year of 365.25 days
precision = 1000000000
)
2018-03-29 05:27:35 -07:00
var hrsPerYrRat = sdk.NewRat(hrsPerYr) // as defined by a julian year of 365.25 days
2018-02-23 15:57:31 -08:00
// Tick - called at the end of every block
func (k Keeper) Tick(ctx sdk.Context) (change []abci.Validator) {
p := k.GetPool(ctx)
2018-02-23 15:57:31 -08:00
// Process Validator Provisions
2018-03-29 05:27:35 -07:00
blockTime := ctx.BlockHeader().Time // XXX assuming in seconds, confirm
if p.InflationLastTime+blockTime >= 3600 {
p.InflationLastTime = blockTime
p = k.processProvisions(ctx)
2018-02-23 15:57:31 -08:00
}
2018-03-29 05:27:35 -07:00
// save the params
k.setPool(ctx, p)
2018-03-29 05:27:35 -07:00
change = k.getAccUpdateValidators(ctx)
2018-03-29 05:27:35 -07:00
return
2018-02-23 15:57:31 -08:00
}
// process provisions for an hour period
2018-03-29 05:27:35 -07:00
func (k Keeper) processProvisions(ctx sdk.Context) Pool {
2018-02-23 15:57:31 -08:00
pool := k.GetPool(ctx)
2018-03-30 13:19:21 -07:00
pool.Inflation = k.nextInflation(ctx)
2018-02-23 15:57:31 -08:00
// Because the validators hold a relative bonded share (`GlobalStakeShare`), when
// more bonded tokens are added proportionally to all validators the only term
// which needs to be updated is the `BondedPool`. So for each previsions cycle:
provisions := pool.Inflation.Mul(sdk.NewRat(pool.TotalSupply)).Quo(hrsPerYrRat).Evaluate()
pool.BondedPool += provisions
pool.TotalSupply += provisions
2018-03-29 05:27:35 -07:00
return pool
2018-02-23 15:57:31 -08:00
}
// get the next inflation rate for the hour
func (k Keeper) nextInflation(ctx sdk.Context) (inflation sdk.Rat) {
2018-02-23 15:57:31 -08:00
params := k.GetParams(ctx)
pool := k.GetPool(ctx)
2018-02-23 15:57:31 -08:00
// The target annual inflation rate is recalculated for each previsions cycle. The
// inflation is also subject to a rate change (positive of negative) depending or
// the distance from the desired ratio (67%). The maximum rate change possible is
// defined to be 13% per year, however the annual inflation is capped as between
// 7% and 20%.
// (1 - bondedRatio/GoalBonded) * InflationRateChange
inflationRateChangePerYear := sdk.OneRat.Sub(pool.bondedRatio().Quo(params.GoalBonded)).Mul(params.InflationRateChange)
inflationRateChange := inflationRateChangePerYear.Quo(hrsPerYrRat)
2018-02-23 15:57:31 -08:00
// increase the new annual inflation for this next cycle
inflation = pool.Inflation.Add(inflationRateChange)
2018-02-23 15:57:31 -08:00
if inflation.GT(params.InflationMax) {
inflation = params.InflationMax
}
if inflation.LT(params.InflationMin) {
inflation = params.InflationMin
}
2018-03-30 13:19:21 -07:00
return inflation.Round(precision)
2018-02-23 15:57:31 -08:00
}