Merge PR #3099: F1 fee distribution
This commit is contained in:
parent
26cb0a125a
commit
2942f83ff5
|
@ -57,6 +57,8 @@ FEATURES
|
||||||
* [\#3198](https://github.com/cosmos/cosmos-sdk/issues/3198) `add-genesis-account` can take both account addresses and key names
|
* [\#3198](https://github.com/cosmos/cosmos-sdk/issues/3198) `add-genesis-account` can take both account addresses and key names
|
||||||
|
|
||||||
* SDK
|
* SDK
|
||||||
|
- \#3099 Implement F1 fee distribution
|
||||||
|
- [\#2926](https://github.com/cosmos/cosmos-sdk/issues/2926) Add TxEncoder to client TxBuilder.
|
||||||
* \#2694 Vesting account implementation.
|
* \#2694 Vesting account implementation.
|
||||||
* \#2996 Update the `AccountKeeper` to contain params used in the context of
|
* \#2996 Update the `AccountKeeper` to contain params used in the context of
|
||||||
the ante handler.
|
the ante handler.
|
||||||
|
|
|
@ -5,11 +5,12 @@ import (
|
||||||
"errors"
|
"errors"
|
||||||
"testing"
|
"testing"
|
||||||
|
|
||||||
"github.com/cosmos/cosmos-sdk/codec"
|
|
||||||
"github.com/cosmos/cosmos-sdk/x/auth"
|
|
||||||
"github.com/stretchr/testify/require"
|
"github.com/stretchr/testify/require"
|
||||||
"github.com/tendermint/tendermint/crypto/ed25519"
|
"github.com/tendermint/tendermint/crypto/ed25519"
|
||||||
|
|
||||||
|
"github.com/cosmos/cosmos-sdk/codec"
|
||||||
|
"github.com/cosmos/cosmos-sdk/x/auth"
|
||||||
|
|
||||||
"github.com/stretchr/testify/assert"
|
"github.com/stretchr/testify/assert"
|
||||||
"github.com/tendermint/tendermint/libs/common"
|
"github.com/tendermint/tendermint/libs/common"
|
||||||
|
|
||||||
|
|
|
@ -238,6 +238,9 @@ func (app *GaiaApp) initFromGenesisState(ctx sdk.Context, genesisState GenesisSt
|
||||||
app.accountKeeper.SetAccount(ctx, acc)
|
app.accountKeeper.SetAccount(ctx, acc)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// initialize distribution (must happen before staking)
|
||||||
|
distr.InitGenesis(ctx, app.distrKeeper, genesisState.DistrData)
|
||||||
|
|
||||||
// load the initial staking information
|
// load the initial staking information
|
||||||
validators, err := staking.InitGenesis(ctx, app.stakingKeeper, genesisState.StakingData)
|
validators, err := staking.InitGenesis(ctx, app.stakingKeeper, genesisState.StakingData)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
|
@ -249,7 +252,6 @@ func (app *GaiaApp) initFromGenesisState(ctx sdk.Context, genesisState GenesisSt
|
||||||
slashing.InitGenesis(ctx, app.slashingKeeper, genesisState.SlashingData, genesisState.StakingData)
|
slashing.InitGenesis(ctx, app.slashingKeeper, genesisState.SlashingData, genesisState.StakingData)
|
||||||
gov.InitGenesis(ctx, app.govKeeper, genesisState.GovData)
|
gov.InitGenesis(ctx, app.govKeeper, genesisState.GovData)
|
||||||
mint.InitGenesis(ctx, app.mintKeeper, genesisState.MintData)
|
mint.InitGenesis(ctx, app.mintKeeper, genesisState.MintData)
|
||||||
distr.InitGenesis(ctx, app.distrKeeper, genesisState.DistrData)
|
|
||||||
|
|
||||||
// validate genesis state
|
// validate genesis state
|
||||||
err = GaiaValidateGenesisState(genesisState)
|
err = GaiaValidateGenesisState(genesisState)
|
||||||
|
@ -370,3 +372,11 @@ func (h StakingHooks) BeforeDelegationRemoved(ctx sdk.Context, delAddr sdk.AccAd
|
||||||
h.dh.BeforeDelegationRemoved(ctx, delAddr, valAddr)
|
h.dh.BeforeDelegationRemoved(ctx, delAddr, valAddr)
|
||||||
h.sh.BeforeDelegationRemoved(ctx, delAddr, valAddr)
|
h.sh.BeforeDelegationRemoved(ctx, delAddr, valAddr)
|
||||||
}
|
}
|
||||||
|
func (h StakingHooks) AfterDelegationModified(ctx sdk.Context, delAddr sdk.AccAddress, valAddr sdk.ValAddress) {
|
||||||
|
h.dh.AfterDelegationModified(ctx, delAddr, valAddr)
|
||||||
|
h.sh.AfterDelegationModified(ctx, delAddr, valAddr)
|
||||||
|
}
|
||||||
|
func (h StakingHooks) BeforeValidatorSlashed(ctx sdk.Context, valAddr sdk.ValAddress, fraction sdk.Dec) {
|
||||||
|
h.dh.BeforeValidatorSlashed(ctx, valAddr, fraction)
|
||||||
|
h.sh.BeforeValidatorSlashed(ctx, valAddr, fraction)
|
||||||
|
}
|
||||||
|
|
|
@ -2,7 +2,6 @@ package app
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"encoding/json"
|
"encoding/json"
|
||||||
"fmt"
|
|
||||||
|
|
||||||
abci "github.com/tendermint/tendermint/abci/types"
|
abci "github.com/tendermint/tendermint/abci/types"
|
||||||
tmtypes "github.com/tendermint/tendermint/types"
|
tmtypes "github.com/tendermint/tendermint/types"
|
||||||
|
@ -14,7 +13,7 @@ import (
|
||||||
"github.com/cosmos/cosmos-sdk/x/gov"
|
"github.com/cosmos/cosmos-sdk/x/gov"
|
||||||
"github.com/cosmos/cosmos-sdk/x/mint"
|
"github.com/cosmos/cosmos-sdk/x/mint"
|
||||||
"github.com/cosmos/cosmos-sdk/x/slashing"
|
"github.com/cosmos/cosmos-sdk/x/slashing"
|
||||||
staking "github.com/cosmos/cosmos-sdk/x/staking"
|
"github.com/cosmos/cosmos-sdk/x/staking"
|
||||||
)
|
)
|
||||||
|
|
||||||
// export the state of gaia for a genesis file
|
// export the state of gaia for a genesis file
|
||||||
|
@ -62,55 +61,41 @@ func (app *GaiaApp) prepForZeroHeightGenesis(ctx sdk.Context) {
|
||||||
|
|
||||||
/* Handle fee distribution state. */
|
/* Handle fee distribution state. */
|
||||||
|
|
||||||
// withdraw all delegator & validator rewards
|
// withdraw all validator commission
|
||||||
vdiIter := func(_ int64, valInfo distr.ValidatorDistInfo) (stop bool) {
|
app.stakingKeeper.IterateValidators(ctx, func(_ int64, val sdk.Validator) (stop bool) {
|
||||||
err := app.distrKeeper.WithdrawValidatorRewardsAll(ctx, valInfo.OperatorAddr)
|
_ = app.distrKeeper.WithdrawValidatorCommission(ctx, val.GetOperator())
|
||||||
if err != nil {
|
|
||||||
panic(err)
|
|
||||||
}
|
|
||||||
return false
|
|
||||||
}
|
|
||||||
app.distrKeeper.IterateValidatorDistInfos(ctx, vdiIter)
|
|
||||||
|
|
||||||
ddiIter := func(_ int64, distInfo distr.DelegationDistInfo) (stop bool) {
|
|
||||||
err := app.distrKeeper.WithdrawDelegationReward(
|
|
||||||
ctx, distInfo.DelegatorAddr, distInfo.ValOperatorAddr)
|
|
||||||
if err != nil {
|
|
||||||
panic(err)
|
|
||||||
}
|
|
||||||
return false
|
|
||||||
}
|
|
||||||
app.distrKeeper.IterateDelegationDistInfos(ctx, ddiIter)
|
|
||||||
|
|
||||||
app.assertRuntimeInvariantsOnContext(ctx)
|
|
||||||
|
|
||||||
// set distribution info withdrawal heights to 0
|
|
||||||
app.distrKeeper.IterateDelegationDistInfos(ctx, func(_ int64, delInfo distr.DelegationDistInfo) (stop bool) {
|
|
||||||
delInfo.DelPoolWithdrawalHeight = 0
|
|
||||||
app.distrKeeper.SetDelegationDistInfo(ctx, delInfo)
|
|
||||||
return false
|
|
||||||
})
|
|
||||||
app.distrKeeper.IterateValidatorDistInfos(ctx, func(_ int64, valInfo distr.ValidatorDistInfo) (stop bool) {
|
|
||||||
valInfo.FeePoolWithdrawalHeight = 0
|
|
||||||
valInfo.DelAccum.UpdateHeight = 0
|
|
||||||
app.distrKeeper.SetValidatorDistInfo(ctx, valInfo)
|
|
||||||
return false
|
return false
|
||||||
})
|
})
|
||||||
|
|
||||||
// assert that the fee pool is empty
|
// withdraw all delegator rewards
|
||||||
feePool := app.distrKeeper.GetFeePool(ctx)
|
dels := app.stakingKeeper.GetAllDelegations(ctx)
|
||||||
if !feePool.TotalValAccum.Accum.IsZero() {
|
for _, delegation := range dels {
|
||||||
panic("unexpected leftover validator accum")
|
_ = app.distrKeeper.WithdrawDelegationRewards(ctx, delegation.DelegatorAddr, delegation.ValidatorAddr)
|
||||||
}
|
|
||||||
bondDenom := app.stakingKeeper.GetParams(ctx).BondDenom
|
|
||||||
if !feePool.ValPool.AmountOf(bondDenom).IsZero() {
|
|
||||||
panic(fmt.Sprintf("unexpected leftover validator pool coins: %v",
|
|
||||||
feePool.ValPool.AmountOf(bondDenom).String()))
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// reset fee pool height, save fee pool
|
// clear validator slash events
|
||||||
feePool.TotalValAccum = distr.NewTotalAccum(0)
|
app.distrKeeper.DeleteValidatorSlashEvents(ctx)
|
||||||
app.distrKeeper.SetFeePool(ctx, feePool)
|
|
||||||
|
// clear validator historical rewards
|
||||||
|
app.distrKeeper.DeleteValidatorHistoricalRewards(ctx)
|
||||||
|
|
||||||
|
// set context height to zero
|
||||||
|
height := ctx.BlockHeight()
|
||||||
|
ctx = ctx.WithBlockHeight(0)
|
||||||
|
|
||||||
|
// reinitialize all validators
|
||||||
|
app.stakingKeeper.IterateValidators(ctx, func(_ int64, val sdk.Validator) (stop bool) {
|
||||||
|
app.distrKeeper.Hooks().AfterValidatorCreated(ctx, val.GetOperator())
|
||||||
|
return false
|
||||||
|
})
|
||||||
|
|
||||||
|
// reinitialize all delegations
|
||||||
|
for _, del := range dels {
|
||||||
|
app.distrKeeper.Hooks().BeforeDelegationCreated(ctx, del.DelegatorAddr, del.ValidatorAddr)
|
||||||
|
}
|
||||||
|
|
||||||
|
// reset context height
|
||||||
|
ctx = ctx.WithBlockHeight(height)
|
||||||
|
|
||||||
/* Handle staking state. */
|
/* Handle staking state. */
|
||||||
|
|
||||||
|
|
|
@ -16,7 +16,7 @@ import (
|
||||||
func (app *GaiaApp) runtimeInvariants() []simulation.Invariant {
|
func (app *GaiaApp) runtimeInvariants() []simulation.Invariant {
|
||||||
return []simulation.Invariant{
|
return []simulation.Invariant{
|
||||||
banksim.NonnegativeBalanceInvariant(app.accountKeeper),
|
banksim.NonnegativeBalanceInvariant(app.accountKeeper),
|
||||||
distrsim.ValAccumInvariants(app.distrKeeper, app.stakingKeeper),
|
distrsim.NonNegativeOutstandingInvariant(app.distrKeeper),
|
||||||
stakingsim.SupplyInvariants(app.bankKeeper, app.stakingKeeper,
|
stakingsim.SupplyInvariants(app.bankKeeper, app.stakingKeeper,
|
||||||
app.feeCollectionKeeper, app.distrKeeper, app.accountKeeper),
|
app.feeCollectionKeeper, app.distrKeeper, app.accountKeeper),
|
||||||
stakingsim.NonNegativePowerInvariant(app.stakingKeeper),
|
stakingsim.NonNegativePowerInvariant(app.stakingKeeper),
|
||||||
|
|
|
@ -29,7 +29,7 @@ import (
|
||||||
"github.com/cosmos/cosmos-sdk/x/mock/simulation"
|
"github.com/cosmos/cosmos-sdk/x/mock/simulation"
|
||||||
"github.com/cosmos/cosmos-sdk/x/slashing"
|
"github.com/cosmos/cosmos-sdk/x/slashing"
|
||||||
slashingsim "github.com/cosmos/cosmos-sdk/x/slashing/simulation"
|
slashingsim "github.com/cosmos/cosmos-sdk/x/slashing/simulation"
|
||||||
staking "github.com/cosmos/cosmos-sdk/x/staking"
|
"github.com/cosmos/cosmos-sdk/x/staking"
|
||||||
stakingsim "github.com/cosmos/cosmos-sdk/x/staking/simulation"
|
stakingsim "github.com/cosmos/cosmos-sdk/x/staking/simulation"
|
||||||
stakingTypes "github.com/cosmos/cosmos-sdk/x/staking/types"
|
stakingTypes "github.com/cosmos/cosmos-sdk/x/staking/types"
|
||||||
)
|
)
|
||||||
|
@ -120,8 +120,8 @@ func appStateFn(r *rand.Rand, accs []simulation.Account) json.RawMessage {
|
||||||
Params: slashing.Params{
|
Params: slashing.Params{
|
||||||
MaxEvidenceAge: stakingGenesis.Params.UnbondingTime,
|
MaxEvidenceAge: stakingGenesis.Params.UnbondingTime,
|
||||||
SignedBlocksWindow: int64(randIntBetween(r, 10, 1000)),
|
SignedBlocksWindow: int64(randIntBetween(r, 10, 1000)),
|
||||||
DowntimeJailDuration: time.Duration(randIntBetween(r, 60, 60*60*24)) * time.Second,
|
|
||||||
MinSignedPerWindow: sdk.NewDecWithPrec(int64(r.Intn(10)), 1),
|
MinSignedPerWindow: sdk.NewDecWithPrec(int64(r.Intn(10)), 1),
|
||||||
|
DowntimeJailDuration: time.Duration(randIntBetween(r, 60, 60*60*24)) * time.Second,
|
||||||
SlashFractionDoubleSign: sdk.NewDec(1).Quo(sdk.NewDec(int64(r.Intn(50) + 1))),
|
SlashFractionDoubleSign: sdk.NewDec(1).Quo(sdk.NewDec(int64(r.Intn(50) + 1))),
|
||||||
SlashFractionDowntime: sdk.NewDec(1).Quo(sdk.NewDec(int64(r.Intn(200) + 1))),
|
SlashFractionDowntime: sdk.NewDec(1).Quo(sdk.NewDec(int64(r.Intn(200) + 1))),
|
||||||
},
|
},
|
||||||
|
@ -160,12 +160,20 @@ func appStateFn(r *rand.Rand, accs []simulation.Account) json.RawMessage {
|
||||||
stakingGenesis.Validators = validators
|
stakingGenesis.Validators = validators
|
||||||
stakingGenesis.Bonds = delegations
|
stakingGenesis.Bonds = delegations
|
||||||
|
|
||||||
|
distrGenesis := distr.GenesisState{
|
||||||
|
FeePool: distr.InitialFeePool(),
|
||||||
|
CommunityTax: sdk.NewDecWithPrec(1, 2).Add(sdk.NewDecWithPrec(int64(r.Intn(30)), 2)),
|
||||||
|
BaseProposerReward: sdk.NewDecWithPrec(1, 2).Add(sdk.NewDecWithPrec(int64(r.Intn(30)), 2)),
|
||||||
|
BonusProposerReward: sdk.NewDecWithPrec(1, 2).Add(sdk.NewDecWithPrec(int64(r.Intn(30)), 2)),
|
||||||
|
}
|
||||||
|
fmt.Printf("Selected randomly generated distribution parameters:\n\t%+v\n", distrGenesis)
|
||||||
|
|
||||||
genesis := GenesisState{
|
genesis := GenesisState{
|
||||||
Accounts: genesisAccounts,
|
Accounts: genesisAccounts,
|
||||||
AuthData: authGenesis,
|
AuthData: authGenesis,
|
||||||
StakingData: stakingGenesis,
|
StakingData: stakingGenesis,
|
||||||
MintData: mintGenesis,
|
MintData: mintGenesis,
|
||||||
DistrData: distr.DefaultGenesisWithValidators(valAddrs),
|
DistrData: distrGenesis,
|
||||||
SlashingData: slashingGenesis,
|
SlashingData: slashingGenesis,
|
||||||
GovData: govGenesis,
|
GovData: govGenesis,
|
||||||
}
|
}
|
||||||
|
@ -188,9 +196,8 @@ func testAndRunTxs(app *GaiaApp) []simulation.WeightedOperation {
|
||||||
{5, authsim.SimulateDeductFee(app.accountKeeper, app.feeCollectionKeeper)},
|
{5, authsim.SimulateDeductFee(app.accountKeeper, app.feeCollectionKeeper)},
|
||||||
{100, banksim.SingleInputSendMsg(app.accountKeeper, app.bankKeeper)},
|
{100, banksim.SingleInputSendMsg(app.accountKeeper, app.bankKeeper)},
|
||||||
{50, distrsim.SimulateMsgSetWithdrawAddress(app.accountKeeper, app.distrKeeper)},
|
{50, distrsim.SimulateMsgSetWithdrawAddress(app.accountKeeper, app.distrKeeper)},
|
||||||
{50, distrsim.SimulateMsgWithdrawDelegatorRewardsAll(app.accountKeeper, app.distrKeeper)},
|
|
||||||
{50, distrsim.SimulateMsgWithdrawDelegatorReward(app.accountKeeper, app.distrKeeper)},
|
{50, distrsim.SimulateMsgWithdrawDelegatorReward(app.accountKeeper, app.distrKeeper)},
|
||||||
{50, distrsim.SimulateMsgWithdrawValidatorRewardsAll(app.accountKeeper, app.distrKeeper)},
|
{50, distrsim.SimulateMsgWithdrawValidatorCommission(app.accountKeeper, app.distrKeeper)},
|
||||||
{5, govsim.SimulateSubmittingVotingAndSlashingForProposal(app.govKeeper, app.stakingKeeper)},
|
{5, govsim.SimulateSubmittingVotingAndSlashingForProposal(app.govKeeper, app.stakingKeeper)},
|
||||||
{100, govsim.SimulateMsgDeposit(app.govKeeper)},
|
{100, govsim.SimulateMsgDeposit(app.govKeeper)},
|
||||||
{100, stakingsim.SimulateMsgCreateValidator(app.accountKeeper, app.stakingKeeper)},
|
{100, stakingsim.SimulateMsgCreateValidator(app.accountKeeper, app.stakingKeeper)},
|
||||||
|
|
|
@ -76,7 +76,7 @@ func TestGaiaCLIMinimumFees(t *testing.T) {
|
||||||
txFees = fmt.Sprintf("--fees=%s", sdk.NewInt64Coin(feeDenom, 23))
|
txFees = fmt.Sprintf("--fees=%s", sdk.NewInt64Coin(feeDenom, 23))
|
||||||
success, _, _ = f.TxSend(keyFoo, barAddr, sdk.NewInt64Coin(feeDenom, 10), txFees)
|
success, _, _ = f.TxSend(keyFoo, barAddr, sdk.NewInt64Coin(feeDenom, 10), txFees)
|
||||||
require.True(f.T, success)
|
require.True(f.T, success)
|
||||||
tests.WaitForNextNBlocksTM(1, f.Port)
|
tests.WaitForNextNBlocksTM(2, f.Port)
|
||||||
|
|
||||||
// Ensure tx w/ improper fees (footoken) fails
|
// Ensure tx w/ improper fees (footoken) fails
|
||||||
txFees = fmt.Sprintf("--fees=%s", sdk.NewInt64Coin(fooDenom, 23))
|
txFees = fmt.Sprintf("--fees=%s", sdk.NewInt64Coin(fooDenom, 23))
|
||||||
|
|
|
@ -4,6 +4,8 @@ import (
|
||||||
"encoding/json"
|
"encoding/json"
|
||||||
"io"
|
"io"
|
||||||
|
|
||||||
|
"github.com/cosmos/cosmos-sdk/store"
|
||||||
|
|
||||||
"github.com/cosmos/cosmos-sdk/baseapp"
|
"github.com/cosmos/cosmos-sdk/baseapp"
|
||||||
"github.com/cosmos/cosmos-sdk/store"
|
"github.com/cosmos/cosmos-sdk/store"
|
||||||
|
|
||||||
|
|
|
@ -69,6 +69,11 @@ func (v Validator) GetMoniker() string {
|
||||||
return ""
|
return ""
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Implements sdk.Validator
|
||||||
|
func (v Validator) GetDelegatorShareExRate() sdk.Dec {
|
||||||
|
return sdk.ZeroDec()
|
||||||
|
}
|
||||||
|
|
||||||
// Implements sdk.Validator
|
// Implements sdk.Validator
|
||||||
type ValidatorSet struct {
|
type ValidatorSet struct {
|
||||||
Validators []Validator
|
Validators []Validator
|
||||||
|
|
|
@ -3,34 +3,32 @@ package types
|
||||||
import (
|
import (
|
||||||
"fmt"
|
"fmt"
|
||||||
"strings"
|
"strings"
|
||||||
|
|
||||||
sdk "github.com/cosmos/cosmos-sdk/types"
|
|
||||||
)
|
)
|
||||||
|
|
||||||
// Coins which can have additional decimal points
|
// Coins which can have additional decimal points
|
||||||
type DecCoin struct {
|
type DecCoin struct {
|
||||||
Denom string `json:"denom"`
|
Denom string `json:"denom"`
|
||||||
Amount sdk.Dec `json:"amount"`
|
Amount Dec `json:"amount"`
|
||||||
}
|
}
|
||||||
|
|
||||||
func NewDecCoin(denom string, amount int64) DecCoin {
|
func NewDecCoin(denom string, amount int64) DecCoin {
|
||||||
return DecCoin{
|
return DecCoin{
|
||||||
Denom: denom,
|
Denom: denom,
|
||||||
Amount: sdk.NewDec(amount),
|
Amount: NewDec(amount),
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
func NewDecCoinFromDec(denom string, amount sdk.Dec) DecCoin {
|
func NewDecCoinFromDec(denom string, amount Dec) DecCoin {
|
||||||
return DecCoin{
|
return DecCoin{
|
||||||
Denom: denom,
|
Denom: denom,
|
||||||
Amount: amount,
|
Amount: amount,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
func NewDecCoinFromCoin(coin sdk.Coin) DecCoin {
|
func NewDecCoinFromCoin(coin Coin) DecCoin {
|
||||||
return DecCoin{
|
return DecCoin{
|
||||||
Denom: coin.Denom,
|
Denom: coin.Denom,
|
||||||
Amount: sdk.NewDecFromInt(coin.Amount),
|
Amount: NewDecFromInt(coin.Amount),
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -51,10 +49,10 @@ func (coin DecCoin) Minus(coinB DecCoin) DecCoin {
|
||||||
}
|
}
|
||||||
|
|
||||||
// return the decimal coins with trunctated decimals, and return the change
|
// return the decimal coins with trunctated decimals, and return the change
|
||||||
func (coin DecCoin) TruncateDecimal() (sdk.Coin, DecCoin) {
|
func (coin DecCoin) TruncateDecimal() (Coin, DecCoin) {
|
||||||
truncated := coin.Amount.TruncateInt()
|
truncated := coin.Amount.TruncateInt()
|
||||||
change := coin.Amount.Sub(sdk.NewDecFromInt(truncated))
|
change := coin.Amount.Sub(NewDecFromInt(truncated))
|
||||||
return sdk.NewCoin(coin.Denom, truncated), DecCoin{coin.Denom, change}
|
return NewCoin(coin.Denom, truncated), DecCoin{coin.Denom, change}
|
||||||
}
|
}
|
||||||
|
|
||||||
//_______________________________________________________________________
|
//_______________________________________________________________________
|
||||||
|
@ -62,7 +60,7 @@ func (coin DecCoin) TruncateDecimal() (sdk.Coin, DecCoin) {
|
||||||
// coins with decimal
|
// coins with decimal
|
||||||
type DecCoins []DecCoin
|
type DecCoins []DecCoin
|
||||||
|
|
||||||
func NewDecCoins(coins sdk.Coins) DecCoins {
|
func NewDecCoins(coins Coins) DecCoins {
|
||||||
dcs := make(DecCoins, len(coins))
|
dcs := make(DecCoins, len(coins))
|
||||||
for i, coin := range coins {
|
for i, coin := range coins {
|
||||||
dcs[i] = NewDecCoinFromCoin(coin)
|
dcs[i] = NewDecCoinFromCoin(coin)
|
||||||
|
@ -71,9 +69,9 @@ func NewDecCoins(coins sdk.Coins) DecCoins {
|
||||||
}
|
}
|
||||||
|
|
||||||
// return the coins with trunctated decimals, and return the change
|
// return the coins with trunctated decimals, and return the change
|
||||||
func (coins DecCoins) TruncateDecimal() (sdk.Coins, DecCoins) {
|
func (coins DecCoins) TruncateDecimal() (Coins, DecCoins) {
|
||||||
changeSum := DecCoins{}
|
changeSum := DecCoins{}
|
||||||
out := make(sdk.Coins, len(coins))
|
out := make(Coins, len(coins))
|
||||||
for i, coin := range coins {
|
for i, coin := range coins {
|
||||||
truncated, change := coin.TruncateDecimal()
|
truncated, change := coin.TruncateDecimal()
|
||||||
out[i] = truncated
|
out[i] = truncated
|
||||||
|
@ -135,7 +133,7 @@ func (coins DecCoins) Minus(coinsB DecCoins) DecCoins {
|
||||||
}
|
}
|
||||||
|
|
||||||
// multiply all the coins by a decimal
|
// multiply all the coins by a decimal
|
||||||
func (coins DecCoins) MulDec(d sdk.Dec) DecCoins {
|
func (coins DecCoins) MulDec(d Dec) DecCoins {
|
||||||
res := make([]DecCoin, len(coins))
|
res := make([]DecCoin, len(coins))
|
||||||
for i, coin := range coins {
|
for i, coin := range coins {
|
||||||
product := DecCoin{
|
product := DecCoin{
|
||||||
|
@ -148,7 +146,7 @@ func (coins DecCoins) MulDec(d sdk.Dec) DecCoins {
|
||||||
}
|
}
|
||||||
|
|
||||||
// divide all the coins by a decimal
|
// divide all the coins by a decimal
|
||||||
func (coins DecCoins) QuoDec(d sdk.Dec) DecCoins {
|
func (coins DecCoins) QuoDec(d Dec) DecCoins {
|
||||||
res := make([]DecCoin, len(coins))
|
res := make([]DecCoin, len(coins))
|
||||||
for i, coin := range coins {
|
for i, coin := range coins {
|
||||||
quotient := DecCoin{
|
quotient := DecCoin{
|
||||||
|
@ -161,16 +159,16 @@ func (coins DecCoins) QuoDec(d sdk.Dec) DecCoins {
|
||||||
}
|
}
|
||||||
|
|
||||||
// returns the amount of a denom from deccoins
|
// returns the amount of a denom from deccoins
|
||||||
func (coins DecCoins) AmountOf(denom string) sdk.Dec {
|
func (coins DecCoins) AmountOf(denom string) Dec {
|
||||||
switch len(coins) {
|
switch len(coins) {
|
||||||
case 0:
|
case 0:
|
||||||
return sdk.ZeroDec()
|
return ZeroDec()
|
||||||
case 1:
|
case 1:
|
||||||
coin := coins[0]
|
coin := coins[0]
|
||||||
if coin.Denom == denom {
|
if coin.Denom == denom {
|
||||||
return coin.Amount
|
return coin.Amount
|
||||||
}
|
}
|
||||||
return sdk.ZeroDec()
|
return ZeroDec()
|
||||||
default:
|
default:
|
||||||
midIdx := len(coins) / 2 // binary search
|
midIdx := len(coins) / 2 // binary search
|
||||||
coin := coins[midIdx]
|
coin := coins[midIdx]
|
|
@ -5,14 +5,12 @@ import (
|
||||||
|
|
||||||
"github.com/stretchr/testify/assert"
|
"github.com/stretchr/testify/assert"
|
||||||
"github.com/stretchr/testify/require"
|
"github.com/stretchr/testify/require"
|
||||||
|
|
||||||
sdk "github.com/cosmos/cosmos-sdk/types"
|
|
||||||
)
|
)
|
||||||
|
|
||||||
func TestPlusDecCoin(t *testing.T) {
|
func TestPlusDecCoin(t *testing.T) {
|
||||||
decCoinA1 := DecCoin{"A", sdk.NewDecWithPrec(11, 1)}
|
decCoinA1 := DecCoin{"A", NewDecWithPrec(11, 1)}
|
||||||
decCoinA2 := DecCoin{"A", sdk.NewDecWithPrec(22, 1)}
|
decCoinA2 := DecCoin{"A", NewDecWithPrec(22, 1)}
|
||||||
decCoinB1 := DecCoin{"B", sdk.NewDecWithPrec(11, 1)}
|
decCoinB1 := DecCoin{"B", NewDecWithPrec(11, 1)}
|
||||||
|
|
||||||
// regular add
|
// regular add
|
||||||
res := decCoinA1.Plus(decCoinA1)
|
res := decCoinA1.Plus(decCoinA1)
|
||||||
|
@ -25,11 +23,11 @@ func TestPlusDecCoin(t *testing.T) {
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
func TestPlusCoins(t *testing.T) {
|
func TestPlusDecCoins(t *testing.T) {
|
||||||
one := sdk.NewDec(1)
|
one := NewDec(1)
|
||||||
zero := sdk.NewDec(0)
|
zero := NewDec(0)
|
||||||
negone := sdk.NewDec(-1)
|
negone := NewDec(-1)
|
||||||
two := sdk.NewDec(2)
|
two := NewDec(2)
|
||||||
|
|
||||||
cases := []struct {
|
cases := []struct {
|
||||||
inputOne DecCoins
|
inputOne DecCoins
|
|
@ -45,8 +45,9 @@ type Validator interface {
|
||||||
GetPower() Int // validation power
|
GetPower() Int // validation power
|
||||||
GetTokens() Int // validation tokens
|
GetTokens() Int // validation tokens
|
||||||
GetCommission() Dec // validator commission rate
|
GetCommission() Dec // validator commission rate
|
||||||
GetDelegatorShares() Dec // Total out standing delegator shares
|
GetDelegatorShares() Dec // total outstanding delegator shares
|
||||||
GetBondHeight() int64 // height in which the validator became active
|
GetBondHeight() int64 // height in which the validator became active
|
||||||
|
GetDelegatorShareExRate() Dec // tokens per delegator share exchange rate
|
||||||
}
|
}
|
||||||
|
|
||||||
// validator which fulfills abci validator interface for use in Tendermint
|
// validator which fulfills abci validator interface for use in Tendermint
|
||||||
|
@ -126,4 +127,6 @@ type StakingHooks interface {
|
||||||
BeforeDelegationCreated(ctx Context, delAddr AccAddress, valAddr ValAddress) // Must be called when a delegation is created
|
BeforeDelegationCreated(ctx Context, delAddr AccAddress, valAddr ValAddress) // Must be called when a delegation is created
|
||||||
BeforeDelegationSharesModified(ctx Context, delAddr AccAddress, valAddr ValAddress) // Must be called when a delegation's shares are modified
|
BeforeDelegationSharesModified(ctx Context, delAddr AccAddress, valAddr ValAddress) // Must be called when a delegation's shares are modified
|
||||||
BeforeDelegationRemoved(ctx Context, delAddr AccAddress, valAddr ValAddress) // Must be called when a delegation is removed
|
BeforeDelegationRemoved(ctx Context, delAddr AccAddress, valAddr ValAddress) // Must be called when a delegation is removed
|
||||||
|
AfterDelegationModified(ctx Context, delAddr AccAddress, valAddr ValAddress)
|
||||||
|
BeforeValidatorSlashed(ctx Context, valAddr ValAddress, fraction Dec)
|
||||||
}
|
}
|
||||||
|
|
|
@ -10,21 +10,8 @@ import (
|
||||||
// set the proposer for determining distribution during endblock
|
// set the proposer for determining distribution during endblock
|
||||||
func BeginBlocker(ctx sdk.Context, req abci.RequestBeginBlock, k keeper.Keeper) {
|
func BeginBlocker(ctx sdk.Context, req abci.RequestBeginBlock, k keeper.Keeper) {
|
||||||
|
|
||||||
if ctx.BlockHeight() > 1 {
|
// determine the total power signing the block
|
||||||
previousPercentPrecommitVotes := getPreviousPercentPrecommitVotes(req)
|
var totalPower, sumPrecommitPower int64
|
||||||
previousProposer := k.GetPreviousProposerConsAddr(ctx)
|
|
||||||
k.AllocateTokens(ctx, previousPercentPrecommitVotes, previousProposer)
|
|
||||||
}
|
|
||||||
|
|
||||||
consAddr := sdk.ConsAddress(req.Header.ProposerAddress)
|
|
||||||
k.SetPreviousProposerConsAddr(ctx, consAddr)
|
|
||||||
}
|
|
||||||
|
|
||||||
// percent precommit votes for the previous block
|
|
||||||
func getPreviousPercentPrecommitVotes(req abci.RequestBeginBlock) sdk.Dec {
|
|
||||||
|
|
||||||
// determine the total number of signed power
|
|
||||||
totalPower, sumPrecommitPower := int64(0), int64(0)
|
|
||||||
for _, voteInfo := range req.LastCommitInfo.GetVotes() {
|
for _, voteInfo := range req.LastCommitInfo.GetVotes() {
|
||||||
totalPower += voteInfo.Validator.Power
|
totalPower += voteInfo.Validator.Power
|
||||||
if voteInfo.SignedLastBlock {
|
if voteInfo.SignedLastBlock {
|
||||||
|
@ -32,8 +19,15 @@ func getPreviousPercentPrecommitVotes(req abci.RequestBeginBlock) sdk.Dec {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if totalPower == 0 {
|
// TODO this is Tendermint-dependent
|
||||||
return sdk.ZeroDec()
|
// ref https://github.com/cosmos/cosmos-sdk/issues/3095
|
||||||
|
if ctx.BlockHeight() > 1 {
|
||||||
|
previousProposer := k.GetPreviousProposerConsAddr(ctx)
|
||||||
|
k.AllocateTokens(ctx, sumPrecommitPower, totalPower, previousProposer, req.LastCommitInfo.GetVotes())
|
||||||
}
|
}
|
||||||
return sdk.NewDec(sumPrecommitPower).Quo(sdk.NewDec(totalPower))
|
|
||||||
|
// record the proposer for when we payout on the next block
|
||||||
|
consAddr := sdk.ConsAddress(req.Header.ProposerAddress)
|
||||||
|
k.SetPreviousProposerConsAddr(ctx, consAddr)
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
|
@ -11,18 +11,9 @@ type (
|
||||||
Keeper = keeper.Keeper
|
Keeper = keeper.Keeper
|
||||||
Hooks = keeper.Hooks
|
Hooks = keeper.Hooks
|
||||||
|
|
||||||
DelegatorWithdrawInfo = types.DelegatorWithdrawInfo
|
|
||||||
DelegationDistInfo = types.DelegationDistInfo
|
|
||||||
ValidatorDistInfo = types.ValidatorDistInfo
|
|
||||||
TotalAccum = types.TotalAccum
|
|
||||||
FeePool = types.FeePool
|
|
||||||
DecCoin = types.DecCoin
|
|
||||||
DecCoins = types.DecCoins
|
|
||||||
|
|
||||||
MsgSetWithdrawAddress = types.MsgSetWithdrawAddress
|
MsgSetWithdrawAddress = types.MsgSetWithdrawAddress
|
||||||
MsgWithdrawDelegatorRewardsAll = types.MsgWithdrawDelegatorRewardsAll
|
|
||||||
MsgWithdrawDelegatorReward = types.MsgWithdrawDelegatorReward
|
MsgWithdrawDelegatorReward = types.MsgWithdrawDelegatorReward
|
||||||
MsgWithdrawValidatorRewardsAll = types.MsgWithdrawValidatorRewardsAll
|
MsgWithdrawValidatorCommission = types.MsgWithdrawValidatorCommission
|
||||||
|
|
||||||
GenesisState = types.GenesisState
|
GenesisState = types.GenesisState
|
||||||
|
|
||||||
|
@ -32,39 +23,6 @@ type (
|
||||||
FeeCollectionKeeper = types.FeeCollectionKeeper
|
FeeCollectionKeeper = types.FeeCollectionKeeper
|
||||||
)
|
)
|
||||||
|
|
||||||
var (
|
|
||||||
NewKeeper = keeper.NewKeeper
|
|
||||||
|
|
||||||
GetValidatorDistInfoKey = keeper.GetValidatorDistInfoKey
|
|
||||||
GetDelegationDistInfoKey = keeper.GetDelegationDistInfoKey
|
|
||||||
GetDelegationDistInfosKey = keeper.GetDelegationDistInfosKey
|
|
||||||
GetDelegatorWithdrawAddrKey = keeper.GetDelegatorWithdrawAddrKey
|
|
||||||
FeePoolKey = keeper.FeePoolKey
|
|
||||||
ValidatorDistInfoKey = keeper.ValidatorDistInfoKey
|
|
||||||
DelegationDistInfoKey = keeper.DelegationDistInfoKey
|
|
||||||
DelegatorWithdrawInfoKey = keeper.DelegatorWithdrawInfoKey
|
|
||||||
ProposerKey = keeper.ProposerKey
|
|
||||||
DefaultParamspace = keeper.DefaultParamspace
|
|
||||||
|
|
||||||
InitialFeePool = types.InitialFeePool
|
|
||||||
|
|
||||||
NewGenesisState = types.NewGenesisState
|
|
||||||
ValidateGenesis = types.ValidateGenesis
|
|
||||||
DefaultGenesisState = types.DefaultGenesisState
|
|
||||||
DefaultGenesisWithValidators = types.DefaultGenesisWithValidators
|
|
||||||
|
|
||||||
RegisterCodec = types.RegisterCodec
|
|
||||||
|
|
||||||
NewMsgSetWithdrawAddress = types.NewMsgSetWithdrawAddress
|
|
||||||
NewMsgWithdrawDelegatorRewardsAll = types.NewMsgWithdrawDelegatorRewardsAll
|
|
||||||
NewMsgWithdrawDelegatorReward = types.NewMsgWithdrawDelegatorReward
|
|
||||||
NewMsgWithdrawValidatorRewardsAll = types.NewMsgWithdrawValidatorRewardsAll
|
|
||||||
|
|
||||||
NewDecCoins = types.NewDecCoins
|
|
||||||
|
|
||||||
NewTotalAccum = types.NewTotalAccum
|
|
||||||
)
|
|
||||||
|
|
||||||
const (
|
const (
|
||||||
DefaultCodespace = types.DefaultCodespace
|
DefaultCodespace = types.DefaultCodespace
|
||||||
CodeInvalidInput = types.CodeInvalidInput
|
CodeInvalidInput = types.CodeInvalidInput
|
||||||
|
@ -81,4 +39,16 @@ var (
|
||||||
|
|
||||||
TagValidator = tags.Validator
|
TagValidator = tags.Validator
|
||||||
TagDelegator = tags.Delegator
|
TagDelegator = tags.Delegator
|
||||||
|
|
||||||
|
NewMsgSetWithdrawAddress = types.NewMsgSetWithdrawAddress
|
||||||
|
NewMsgWithdrawDelegatorReward = types.NewMsgWithdrawDelegatorReward
|
||||||
|
NewMsgWithdrawValidatorCommission = types.NewMsgWithdrawValidatorCommission
|
||||||
|
|
||||||
|
NewKeeper = keeper.NewKeeper
|
||||||
|
DefaultParamspace = keeper.DefaultParamspace
|
||||||
|
|
||||||
|
RegisterCodec = types.RegisterCodec
|
||||||
|
DefaultGenesisState = types.DefaultGenesisState
|
||||||
|
ValidateGenesis = types.ValidateGenesis
|
||||||
|
InitialFeePool = types.InitialFeePool
|
||||||
)
|
)
|
||||||
|
|
|
@ -43,7 +43,7 @@ func GetTxCmd(storeKey string, cdc *amino.Codec) *cobra.Command {
|
||||||
func GetCmdWithdrawRewards(cdc *codec.Codec) *cobra.Command {
|
func GetCmdWithdrawRewards(cdc *codec.Codec) *cobra.Command {
|
||||||
cmd := &cobra.Command{
|
cmd := &cobra.Command{
|
||||||
Use: "withdraw-rewards",
|
Use: "withdraw-rewards",
|
||||||
Short: "withdraw rewards for either: all-delegations, a delegation, or a validator",
|
Short: "withdraw rewards for either a delegation or a validator",
|
||||||
Args: cobra.NoArgs,
|
Args: cobra.NoArgs,
|
||||||
RunE: func(cmd *cobra.Command, args []string) error {
|
RunE: func(cmd *cobra.Command, args []string) error {
|
||||||
|
|
||||||
|
@ -68,8 +68,8 @@ func GetCmdWithdrawRewards(cdc *codec.Codec) *cobra.Command {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
valAddr := sdk.ValAddress(addr.Bytes())
|
valAddr := sdk.ValAddress(addr.Bytes())
|
||||||
msg = types.NewMsgWithdrawValidatorRewardsAll(valAddr)
|
msg = types.NewMsgWithdrawValidatorCommission(valAddr)
|
||||||
case onlyFromVal != "":
|
default:
|
||||||
delAddr, err := cliCtx.GetFromAddress()
|
delAddr, err := cliCtx.GetFromAddress()
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return err
|
return err
|
||||||
|
@ -81,12 +81,6 @@ func GetCmdWithdrawRewards(cdc *codec.Codec) *cobra.Command {
|
||||||
}
|
}
|
||||||
|
|
||||||
msg = types.NewMsgWithdrawDelegatorReward(delAddr, valAddr)
|
msg = types.NewMsgWithdrawDelegatorReward(delAddr, valAddr)
|
||||||
default:
|
|
||||||
delAddr, err := cliCtx.GetFromAddress()
|
|
||||||
if err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
msg = types.NewMsgWithdrawDelegatorRewardsAll(delAddr)
|
|
||||||
}
|
}
|
||||||
|
|
||||||
if cliCtx.GenerateOnly {
|
if cliCtx.GenerateOnly {
|
||||||
|
|
|
@ -11,30 +11,97 @@ func InitGenesis(ctx sdk.Context, keeper Keeper, data types.GenesisState) {
|
||||||
keeper.SetCommunityTax(ctx, data.CommunityTax)
|
keeper.SetCommunityTax(ctx, data.CommunityTax)
|
||||||
keeper.SetBaseProposerReward(ctx, data.BaseProposerReward)
|
keeper.SetBaseProposerReward(ctx, data.BaseProposerReward)
|
||||||
keeper.SetBonusProposerReward(ctx, data.BonusProposerReward)
|
keeper.SetBonusProposerReward(ctx, data.BonusProposerReward)
|
||||||
|
for _, dwi := range data.DelegatorWithdrawInfos {
|
||||||
for _, vdi := range data.ValidatorDistInfos {
|
keeper.SetDelegatorWithdrawAddr(ctx, dwi.DelegatorAddr, dwi.WithdrawAddr)
|
||||||
keeper.SetValidatorDistInfo(ctx, vdi)
|
|
||||||
}
|
|
||||||
for _, ddi := range data.DelegationDistInfos {
|
|
||||||
keeper.SetDelegationDistInfo(ctx, ddi)
|
|
||||||
}
|
|
||||||
for _, dw := range data.DelegatorWithdrawInfos {
|
|
||||||
keeper.SetDelegatorWithdrawAddr(ctx, dw.DelegatorAddr, dw.WithdrawAddr)
|
|
||||||
}
|
}
|
||||||
keeper.SetPreviousProposerConsAddr(ctx, data.PreviousProposer)
|
keeper.SetPreviousProposerConsAddr(ctx, data.PreviousProposer)
|
||||||
|
keeper.SetOutstandingRewards(ctx, data.OutstandingRewards)
|
||||||
|
for _, acc := range data.ValidatorAccumulatedCommissions {
|
||||||
|
keeper.SetValidatorAccumulatedCommission(ctx, acc.ValidatorAddr, acc.Accumulated)
|
||||||
|
}
|
||||||
|
for _, his := range data.ValidatorHistoricalRewards {
|
||||||
|
keeper.SetValidatorHistoricalRewards(ctx, his.ValidatorAddr, his.Period, his.Rewards)
|
||||||
|
}
|
||||||
|
for _, cur := range data.ValidatorCurrentRewards {
|
||||||
|
keeper.SetValidatorCurrentRewards(ctx, cur.ValidatorAddr, cur.Rewards)
|
||||||
|
}
|
||||||
|
for _, del := range data.DelegatorStartingInfos {
|
||||||
|
keeper.SetDelegatorStartingInfo(ctx, del.ValidatorAddr, del.DelegatorAddr, del.StartingInfo)
|
||||||
|
}
|
||||||
|
for _, evt := range data.ValidatorSlashEvents {
|
||||||
|
keeper.SetValidatorSlashEvent(ctx, evt.ValidatorAddr, evt.Height, evt.Event)
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// ExportGenesis returns a GenesisState for a given context and keeper. The
|
// ExportGenesis returns a GenesisState for a given context and keeper.
|
||||||
// GenesisState will contain the pool, and validator/delegator distribution info's
|
|
||||||
func ExportGenesis(ctx sdk.Context, keeper Keeper) types.GenesisState {
|
func ExportGenesis(ctx sdk.Context, keeper Keeper) types.GenesisState {
|
||||||
feePool := keeper.GetFeePool(ctx)
|
feePool := keeper.GetFeePool(ctx)
|
||||||
communityTax := keeper.GetCommunityTax(ctx)
|
communityTax := keeper.GetCommunityTax(ctx)
|
||||||
baseProposerRewards := keeper.GetBaseProposerReward(ctx)
|
baseProposerRewards := keeper.GetBaseProposerReward(ctx)
|
||||||
bonusProposerRewards := keeper.GetBonusProposerReward(ctx)
|
bonusProposerRewards := keeper.GetBonusProposerReward(ctx)
|
||||||
vdis := keeper.GetAllValidatorDistInfos(ctx)
|
dwi := make([]types.DelegatorWithdrawInfo, 0)
|
||||||
ddis := keeper.GetAllDelegationDistInfos(ctx)
|
keeper.IterateDelegatorWithdrawAddrs(ctx, func(del sdk.AccAddress, addr sdk.AccAddress) (stop bool) {
|
||||||
dwis := keeper.GetAllDelegatorWithdrawInfos(ctx)
|
dwi = append(dwi, types.DelegatorWithdrawInfo{
|
||||||
|
DelegatorAddr: del,
|
||||||
|
WithdrawAddr: addr,
|
||||||
|
})
|
||||||
|
return false
|
||||||
|
})
|
||||||
pp := keeper.GetPreviousProposerConsAddr(ctx)
|
pp := keeper.GetPreviousProposerConsAddr(ctx)
|
||||||
return NewGenesisState(feePool, communityTax, baseProposerRewards,
|
outstanding := keeper.GetOutstandingRewards(ctx)
|
||||||
bonusProposerRewards, vdis, ddis, dwis, pp)
|
acc := make([]types.ValidatorAccumulatedCommissionRecord, 0)
|
||||||
|
keeper.IterateValidatorAccumulatedCommissions(ctx,
|
||||||
|
func(addr sdk.ValAddress, commission types.ValidatorAccumulatedCommission) (stop bool) {
|
||||||
|
acc = append(acc, types.ValidatorAccumulatedCommissionRecord{
|
||||||
|
ValidatorAddr: addr,
|
||||||
|
Accumulated: commission,
|
||||||
|
})
|
||||||
|
return false
|
||||||
|
},
|
||||||
|
)
|
||||||
|
his := make([]types.ValidatorHistoricalRewardsRecord, 0)
|
||||||
|
keeper.IterateValidatorHistoricalRewards(ctx,
|
||||||
|
func(val sdk.ValAddress, period uint64, rewards types.ValidatorHistoricalRewards) (stop bool) {
|
||||||
|
his = append(his, types.ValidatorHistoricalRewardsRecord{
|
||||||
|
ValidatorAddr: val,
|
||||||
|
Period: period,
|
||||||
|
Rewards: rewards,
|
||||||
|
})
|
||||||
|
return false
|
||||||
|
},
|
||||||
|
)
|
||||||
|
cur := make([]types.ValidatorCurrentRewardsRecord, 0)
|
||||||
|
keeper.IterateValidatorCurrentRewards(ctx,
|
||||||
|
func(val sdk.ValAddress, rewards types.ValidatorCurrentRewards) (stop bool) {
|
||||||
|
cur = append(cur, types.ValidatorCurrentRewardsRecord{
|
||||||
|
ValidatorAddr: val,
|
||||||
|
Rewards: rewards,
|
||||||
|
})
|
||||||
|
return false
|
||||||
|
},
|
||||||
|
)
|
||||||
|
dels := make([]types.DelegatorStartingInfoRecord, 0)
|
||||||
|
keeper.IterateDelegatorStartingInfos(ctx,
|
||||||
|
func(val sdk.ValAddress, del sdk.AccAddress, info types.DelegatorStartingInfo) (stop bool) {
|
||||||
|
dels = append(dels, types.DelegatorStartingInfoRecord{
|
||||||
|
ValidatorAddr: val,
|
||||||
|
DelegatorAddr: del,
|
||||||
|
StartingInfo: info,
|
||||||
|
})
|
||||||
|
return false
|
||||||
|
},
|
||||||
|
)
|
||||||
|
slashes := make([]types.ValidatorSlashEventRecord, 0)
|
||||||
|
keeper.IterateValidatorSlashEvents(ctx,
|
||||||
|
func(val sdk.ValAddress, height uint64, event types.ValidatorSlashEvent) (stop bool) {
|
||||||
|
slashes = append(slashes, types.ValidatorSlashEventRecord{
|
||||||
|
ValidatorAddr: val,
|
||||||
|
Height: height,
|
||||||
|
Event: event,
|
||||||
|
})
|
||||||
|
return false
|
||||||
|
},
|
||||||
|
)
|
||||||
|
return types.NewGenesisState(feePool, communityTax, baseProposerRewards, bonusProposerRewards, dwi, pp, outstanding,
|
||||||
|
acc, his, cur, dels, slashes)
|
||||||
}
|
}
|
||||||
|
|
|
@ -13,24 +13,20 @@ func NewHandler(k keeper.Keeper) sdk.Handler {
|
||||||
switch msg := msg.(type) {
|
switch msg := msg.(type) {
|
||||||
case types.MsgSetWithdrawAddress:
|
case types.MsgSetWithdrawAddress:
|
||||||
return handleMsgModifyWithdrawAddress(ctx, msg, k)
|
return handleMsgModifyWithdrawAddress(ctx, msg, k)
|
||||||
case types.MsgWithdrawDelegatorRewardsAll:
|
|
||||||
return handleMsgWithdrawDelegatorRewardsAll(ctx, msg, k)
|
|
||||||
case types.MsgWithdrawDelegatorReward:
|
case types.MsgWithdrawDelegatorReward:
|
||||||
return handleMsgWithdrawDelegatorReward(ctx, msg, k)
|
return handleMsgWithdrawDelegatorReward(ctx, msg, k)
|
||||||
case types.MsgWithdrawValidatorRewardsAll:
|
case types.MsgWithdrawValidatorCommission:
|
||||||
return handleMsgWithdrawValidatorRewardsAll(ctx, msg, k)
|
return handleMsgWithdrawValidatorCommission(ctx, msg, k)
|
||||||
default:
|
default:
|
||||||
return sdk.ErrTxDecode("invalid message parse in distribution module").Result()
|
return sdk.ErrTxDecode("invalid message parse in distribution module").Result()
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
//_____________________________________________________________________
|
// These functions assume everything has been authenticated (ValidateBasic passed, and signatures checked)
|
||||||
|
|
||||||
// These functions assume everything has been authenticated,
|
|
||||||
// now we just perform action and save
|
|
||||||
|
|
||||||
func handleMsgModifyWithdrawAddress(ctx sdk.Context, msg types.MsgSetWithdrawAddress, k keeper.Keeper) sdk.Result {
|
func handleMsgModifyWithdrawAddress(ctx sdk.Context, msg types.MsgSetWithdrawAddress, k keeper.Keeper) sdk.Result {
|
||||||
|
|
||||||
k.SetDelegatorWithdrawAddr(ctx, msg.DelegatorAddr, msg.WithdrawAddr)
|
k.SetDelegatorWithdrawAddr(ctx, msg.DelegatorAddr, msg.WithdrawAddr)
|
||||||
|
|
||||||
tags := sdk.NewTags(
|
tags := sdk.NewTags(
|
||||||
|
@ -41,19 +37,9 @@ func handleMsgModifyWithdrawAddress(ctx sdk.Context, msg types.MsgSetWithdrawAdd
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
func handleMsgWithdrawDelegatorRewardsAll(ctx sdk.Context, msg types.MsgWithdrawDelegatorRewardsAll, k keeper.Keeper) sdk.Result {
|
|
||||||
k.WithdrawDelegationRewardsAll(ctx, msg.DelegatorAddr)
|
|
||||||
|
|
||||||
tags := sdk.NewTags(
|
|
||||||
tags.Delegator, []byte(msg.DelegatorAddr.String()),
|
|
||||||
)
|
|
||||||
return sdk.Result{
|
|
||||||
Tags: tags,
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
func handleMsgWithdrawDelegatorReward(ctx sdk.Context, msg types.MsgWithdrawDelegatorReward, k keeper.Keeper) sdk.Result {
|
func handleMsgWithdrawDelegatorReward(ctx sdk.Context, msg types.MsgWithdrawDelegatorReward, k keeper.Keeper) sdk.Result {
|
||||||
err := k.WithdrawDelegationReward(ctx, msg.DelegatorAddr, msg.ValidatorAddr)
|
|
||||||
|
err := k.WithdrawDelegationRewards(ctx, msg.DelegatorAddr, msg.ValidatorAddr)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return err.Result()
|
return err.Result()
|
||||||
}
|
}
|
||||||
|
@ -67,8 +53,9 @@ func handleMsgWithdrawDelegatorReward(ctx sdk.Context, msg types.MsgWithdrawDele
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
func handleMsgWithdrawValidatorRewardsAll(ctx sdk.Context, msg types.MsgWithdrawValidatorRewardsAll, k keeper.Keeper) sdk.Result {
|
func handleMsgWithdrawValidatorCommission(ctx sdk.Context, msg types.MsgWithdrawValidatorCommission, k keeper.Keeper) sdk.Result {
|
||||||
err := k.WithdrawValidatorRewardsAll(ctx, msg.ValidatorAddr)
|
|
||||||
|
err := k.WithdrawValidatorCommission(ctx, msg.ValidatorAddr)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return err.Result()
|
return err.Result()
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,57 +1,85 @@
|
||||||
package keeper
|
package keeper
|
||||||
|
|
||||||
import (
|
import (
|
||||||
|
abci "github.com/tendermint/tendermint/abci/types"
|
||||||
|
|
||||||
sdk "github.com/cosmos/cosmos-sdk/types"
|
sdk "github.com/cosmos/cosmos-sdk/types"
|
||||||
"github.com/cosmos/cosmos-sdk/x/distribution/types"
|
|
||||||
)
|
)
|
||||||
|
|
||||||
// Allocate fees handles distribution of the collected fees
|
// allocate fees handles distribution of the collected fees
|
||||||
func (k Keeper) AllocateTokens(ctx sdk.Context, percentVotes sdk.Dec, proposer sdk.ConsAddress) {
|
func (k Keeper) AllocateTokens(ctx sdk.Context, sumPrecommitPower, totalPower int64, proposer sdk.ConsAddress, votes []abci.VoteInfo) {
|
||||||
|
|
||||||
// get the proposer of this block
|
|
||||||
proposerValidator := k.stakingKeeper.ValidatorByConsAddr(ctx, proposer)
|
|
||||||
|
|
||||||
proposerDist := k.GetValidatorDistInfo(ctx, proposerValidator.GetOperator())
|
|
||||||
|
|
||||||
// get the fees which have been getting collected through all the
|
|
||||||
// transactions in the block
|
|
||||||
feesCollected := k.feeCollectionKeeper.GetCollectedFees(ctx)
|
|
||||||
feesCollectedDec := types.NewDecCoins(feesCollected)
|
|
||||||
|
|
||||||
|
// fetch collected fees & fee pool
|
||||||
|
feesCollectedInt := k.feeCollectionKeeper.GetCollectedFees(ctx)
|
||||||
|
feesCollected := sdk.NewDecCoins(feesCollectedInt)
|
||||||
feePool := k.GetFeePool(ctx)
|
feePool := k.GetFeePool(ctx)
|
||||||
// Temporary workaround to keep CanWithdrawInvariant happy.
|
|
||||||
// General discussions here: https://github.com/cosmos/cosmos-sdk/issues/2906#issuecomment-441867634
|
// clear collected fees, which will now be distributed
|
||||||
if k.stakingKeeper.GetLastTotalPower(ctx).IsZero() {
|
k.feeCollectionKeeper.ClearCollectedFees(ctx)
|
||||||
feePool.CommunityPool = feePool.CommunityPool.Plus(feesCollectedDec)
|
|
||||||
|
// 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.Plus(feesCollected)
|
||||||
k.SetFeePool(ctx, feePool)
|
k.SetFeePool(ctx, feePool)
|
||||||
k.feeCollectionKeeper.ClearCollectedFees(ctx)
|
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
// allocated rewards to proposer
|
// calculate fraction votes
|
||||||
|
fractionVotes := sdk.NewDec(sumPrecommitPower).Quo(sdk.NewDec(totalPower))
|
||||||
|
|
||||||
|
// calculate proposer reward
|
||||||
baseProposerReward := k.GetBaseProposerReward(ctx)
|
baseProposerReward := k.GetBaseProposerReward(ctx)
|
||||||
bonusProposerReward := k.GetBonusProposerReward(ctx)
|
bonusProposerReward := k.GetBonusProposerReward(ctx)
|
||||||
proposerMultiplier := baseProposerReward.Add(bonusProposerReward.Mul(percentVotes))
|
proposerMultiplier := baseProposerReward.Add(bonusProposerReward.Mul(fractionVotes))
|
||||||
proposerReward := feesCollectedDec.MulDec(proposerMultiplier)
|
proposerReward := feesCollected.MulDec(proposerMultiplier)
|
||||||
|
|
||||||
// apply commission
|
// pay proposer
|
||||||
commission := proposerReward.MulDec(proposerValidator.GetCommission())
|
proposerValidator := k.stakingKeeper.ValidatorByConsAddr(ctx, proposer)
|
||||||
remaining := proposerReward.Minus(commission)
|
k.AllocateTokensToValidator(ctx, proposerValidator, proposerReward)
|
||||||
proposerDist.ValCommission = proposerDist.ValCommission.Plus(commission)
|
remaining := feesCollected.Minus(proposerReward)
|
||||||
proposerDist.DelPool = proposerDist.DelPool.Plus(remaining)
|
|
||||||
|
// calculate fraction allocated to validators
|
||||||
|
communityTax := k.GetCommunityTax(ctx)
|
||||||
|
voteMultiplier := sdk.OneDec().Sub(proposerMultiplier).Sub(communityTax)
|
||||||
|
|
||||||
|
// 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)
|
||||||
|
|
||||||
|
// 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.Minus(reward)
|
||||||
|
}
|
||||||
|
|
||||||
// allocate community funding
|
// allocate community funding
|
||||||
communityTax := k.GetCommunityTax(ctx)
|
feePool.CommunityPool = feePool.CommunityPool.Plus(remaining)
|
||||||
communityFunding := feesCollectedDec.MulDec(communityTax)
|
|
||||||
feePool.CommunityPool = feePool.CommunityPool.Plus(communityFunding)
|
|
||||||
|
|
||||||
// set the global pool within the distribution module
|
|
||||||
poolReceived := feesCollectedDec.Minus(proposerReward).Minus(communityFunding)
|
|
||||||
feePool.ValPool = feePool.ValPool.Plus(poolReceived)
|
|
||||||
|
|
||||||
k.SetValidatorDistInfo(ctx, proposerDist)
|
|
||||||
k.SetFeePool(ctx, feePool)
|
k.SetFeePool(ctx, feePool)
|
||||||
|
|
||||||
// clear the now distributed fees
|
// update outstanding rewards
|
||||||
k.feeCollectionKeeper.ClearCollectedFees(ctx)
|
outstanding := k.GetOutstandingRewards(ctx)
|
||||||
|
outstanding = outstanding.Plus(feesCollected.Minus(remaining))
|
||||||
|
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.Minus(commission)
|
||||||
|
|
||||||
|
// update current commission
|
||||||
|
currentCommission := k.GetValidatorAccumulatedCommission(ctx, val.GetOperator())
|
||||||
|
currentCommission = currentCommission.Plus(commission)
|
||||||
|
k.SetValidatorAccumulatedCommission(ctx, val.GetOperator(), currentCommission)
|
||||||
|
|
||||||
|
// update current rewards
|
||||||
|
currentRewards := k.GetValidatorCurrentRewards(ctx, val.GetOperator())
|
||||||
|
currentRewards.Rewards = currentRewards.Rewards.Plus(shared)
|
||||||
|
k.SetValidatorCurrentRewards(ctx, val.GetOperator(), currentRewards)
|
||||||
}
|
}
|
||||||
|
|
|
@ -3,109 +3,106 @@ package keeper
|
||||||
import (
|
import (
|
||||||
"testing"
|
"testing"
|
||||||
|
|
||||||
"github.com/stretchr/testify/assert"
|
|
||||||
"github.com/stretchr/testify/require"
|
"github.com/stretchr/testify/require"
|
||||||
|
abci "github.com/tendermint/tendermint/abci/types"
|
||||||
|
|
||||||
sdk "github.com/cosmos/cosmos-sdk/types"
|
sdk "github.com/cosmos/cosmos-sdk/types"
|
||||||
"github.com/cosmos/cosmos-sdk/x/staking"
|
"github.com/cosmos/cosmos-sdk/x/staking"
|
||||||
)
|
)
|
||||||
|
|
||||||
func TestAllocateTokensBasic(t *testing.T) {
|
func TestAllocateTokensToValidatorWithCommission(t *testing.T) {
|
||||||
|
ctx, _, k, sk, _ := CreateTestInputDefault(t, false, 1000)
|
||||||
|
sh := staking.NewHandler(sk)
|
||||||
|
|
||||||
// no community tax on inputs
|
// initialize state
|
||||||
ctx, _, keeper, sk, fck := CreateTestInputAdvanced(t, false, 100, sdk.ZeroDec())
|
k.SetOutstandingRewards(ctx, sdk.DecCoins{})
|
||||||
stakingHandler := staking.NewHandler(sk)
|
|
||||||
denom := sk.GetParams(ctx).BondDenom
|
|
||||||
|
|
||||||
//first make a validator
|
// create validator with 50% commission
|
||||||
totalPower := int64(10)
|
commission := staking.NewCommissionMsg(sdk.NewDecWithPrec(5, 1), sdk.NewDecWithPrec(5, 1), sdk.NewDec(0))
|
||||||
totalPowerInt := sdk.NewInt(totalPower)
|
msg := staking.NewMsgCreateValidator(valOpAddr1, valConsPk1,
|
||||||
msgCreateValidator := staking.NewTestMsgCreateValidator(valOpAddr1, valConsPk1, totalPower)
|
sdk.NewCoin(staking.DefaultBondDenom, sdk.NewInt(100)), staking.Description{}, commission)
|
||||||
got := stakingHandler(ctx, msgCreateValidator)
|
require.True(t, sh(ctx, msg).IsOK())
|
||||||
require.True(t, got.IsOK(), "expected msg to be ok, got %v", got)
|
val := sk.Validator(ctx, valOpAddr1)
|
||||||
_ = sk.ApplyAndReturnValidatorSetUpdates(ctx)
|
|
||||||
|
|
||||||
// verify everything has been set in staking correctly
|
// allocate tokens
|
||||||
validator, found := sk.GetValidator(ctx, valOpAddr1)
|
tokens := sdk.DecCoins{
|
||||||
require.True(t, found)
|
{staking.DefaultBondDenom, sdk.NewDec(10)},
|
||||||
require.Equal(t, sdk.Bonded, validator.Status)
|
}
|
||||||
assert.True(sdk.IntEq(t, totalPowerInt, validator.Tokens))
|
k.AllocateTokensToValidator(ctx, val, tokens)
|
||||||
assert.True(sdk.DecEq(t, sdk.NewDec(totalPower), validator.DelegatorShares))
|
|
||||||
bondedTokens := sk.TotalPower(ctx)
|
|
||||||
assert.True(sdk.IntEq(t, totalPowerInt, bondedTokens))
|
|
||||||
|
|
||||||
// initial fee pool should be empty
|
// check commission
|
||||||
feePool := keeper.GetFeePool(ctx)
|
expected := sdk.DecCoins{
|
||||||
require.Nil(t, feePool.ValPool)
|
{staking.DefaultBondDenom, sdk.NewDec(5)},
|
||||||
|
}
|
||||||
|
require.Equal(t, expected, k.GetValidatorAccumulatedCommission(ctx, val.GetOperator()))
|
||||||
|
|
||||||
// allocate 100 denom of fees
|
// check current rewards
|
||||||
feeInputs := sdk.NewInt(100)
|
require.Equal(t, expected, k.GetValidatorCurrentRewards(ctx, val.GetOperator()).Rewards)
|
||||||
fck.SetCollectedFees(sdk.Coins{sdk.NewCoin(denom, feeInputs)})
|
|
||||||
require.Equal(t, feeInputs, fck.GetCollectedFees(ctx).AmountOf(denom))
|
|
||||||
keeper.AllocateTokens(ctx, sdk.OneDec(), valConsAddr1)
|
|
||||||
|
|
||||||
// verify that these fees have been received by the feePool
|
|
||||||
percentProposer := sdk.NewDecWithPrec(5, 2)
|
|
||||||
percentRemaining := sdk.OneDec().Sub(percentProposer)
|
|
||||||
feePool = keeper.GetFeePool(ctx)
|
|
||||||
expRes := sdk.NewDecFromInt(feeInputs).Mul(percentRemaining)
|
|
||||||
require.Equal(t, 1, len(feePool.ValPool))
|
|
||||||
require.True(sdk.DecEq(t, expRes, feePool.ValPool[0].Amount))
|
|
||||||
}
|
}
|
||||||
|
|
||||||
func TestAllocateTokensWithCommunityTax(t *testing.T) {
|
func TestAllocateTokensToManyValidators(t *testing.T) {
|
||||||
communityTax := sdk.NewDecWithPrec(1, 2) //1%
|
ctx, _, k, sk, fck := CreateTestInputDefault(t, false, 1000)
|
||||||
ctx, _, keeper, sk, fck := CreateTestInputAdvanced(t, false, 100, communityTax)
|
sh := staking.NewHandler(sk)
|
||||||
stakingHandler := staking.NewHandler(sk)
|
|
||||||
denom := sk.GetParams(ctx).BondDenom
|
|
||||||
|
|
||||||
//first make a validator
|
// initialize state
|
||||||
totalPower := int64(10)
|
k.SetOutstandingRewards(ctx, sdk.DecCoins{})
|
||||||
msgCreateValidator := staking.NewTestMsgCreateValidator(valOpAddr1, valConsPk1, totalPower)
|
|
||||||
got := stakingHandler(ctx, msgCreateValidator)
|
|
||||||
require.True(t, got.IsOK(), "expected msg to be ok, got %v", got)
|
|
||||||
_ = sk.ApplyAndReturnValidatorSetUpdates(ctx)
|
|
||||||
|
|
||||||
// allocate 100 denom of fees
|
// create validator with 50% commission
|
||||||
feeInputs := sdk.NewInt(100)
|
commission := staking.NewCommissionMsg(sdk.NewDecWithPrec(5, 1), sdk.NewDecWithPrec(5, 1), sdk.NewDec(0))
|
||||||
fck.SetCollectedFees(sdk.Coins{sdk.NewCoin(denom, feeInputs)})
|
msg := staking.NewMsgCreateValidator(valOpAddr1, valConsPk1,
|
||||||
keeper.AllocateTokens(ctx, sdk.OneDec(), valConsAddr1)
|
sdk.NewCoin(staking.DefaultBondDenom, sdk.NewInt(100)), staking.Description{}, commission)
|
||||||
|
require.True(t, sh(ctx, msg).IsOK())
|
||||||
|
|
||||||
// verify that these fees have been received by the feePool
|
// create second validator with 0% commission
|
||||||
feePool := keeper.GetFeePool(ctx)
|
commission = staking.NewCommissionMsg(sdk.NewDec(0), sdk.NewDec(0), sdk.NewDec(0))
|
||||||
// 5% goes to proposer, 1% community tax
|
msg = staking.NewMsgCreateValidator(valOpAddr2, valConsPk2,
|
||||||
percentProposer := sdk.NewDecWithPrec(5, 2)
|
sdk.NewCoin(staking.DefaultBondDenom, sdk.NewInt(100)), staking.Description{}, commission)
|
||||||
percentRemaining := sdk.OneDec().Sub(communityTax.Add(percentProposer))
|
require.True(t, sh(ctx, msg).IsOK())
|
||||||
expRes := sdk.NewDecFromInt(feeInputs).Mul(percentRemaining)
|
|
||||||
require.Equal(t, 1, len(feePool.ValPool))
|
abciValA := abci.Validator{
|
||||||
require.True(sdk.DecEq(t, expRes, feePool.ValPool[0].Amount))
|
Address: valConsPk1.Address(),
|
||||||
}
|
Power: 100,
|
||||||
|
}
|
||||||
func TestAllocateTokensWithPartialPrecommitPower(t *testing.T) {
|
abciValB := abci.Validator{
|
||||||
communityTax := sdk.NewDecWithPrec(1, 2)
|
Address: valConsPk2.Address(),
|
||||||
ctx, _, keeper, sk, fck := CreateTestInputAdvanced(t, false, 100, communityTax)
|
Power: 100,
|
||||||
stakingHandler := staking.NewHandler(sk)
|
}
|
||||||
denom := sk.GetParams(ctx).BondDenom
|
|
||||||
|
// assert initial state: zero outstanding rewards, zero community pool, zero commission, zero current rewards
|
||||||
//first make a validator
|
require.True(t, k.GetOutstandingRewards(ctx).IsZero())
|
||||||
totalPower := int64(100)
|
require.True(t, k.GetFeePool(ctx).CommunityPool.IsZero())
|
||||||
msgCreateValidator := staking.NewTestMsgCreateValidator(valOpAddr1, valConsPk1, totalPower)
|
require.True(t, k.GetValidatorAccumulatedCommission(ctx, valOpAddr1).IsZero())
|
||||||
got := stakingHandler(ctx, msgCreateValidator)
|
require.True(t, k.GetValidatorAccumulatedCommission(ctx, valOpAddr2).IsZero())
|
||||||
require.True(t, got.IsOK(), "expected msg to be ok, got %v", got)
|
require.True(t, k.GetValidatorCurrentRewards(ctx, valOpAddr1).Rewards.IsZero())
|
||||||
_ = sk.ApplyAndReturnValidatorSetUpdates(ctx)
|
require.True(t, k.GetValidatorCurrentRewards(ctx, valOpAddr2).Rewards.IsZero())
|
||||||
|
|
||||||
// allocate 100 denom of fees
|
// allocate tokens as if both had voted and second was proposer
|
||||||
feeInputs := sdk.NewInt(100)
|
fees := sdk.Coins{
|
||||||
fck.SetCollectedFees(sdk.Coins{sdk.NewCoin(denom, feeInputs)})
|
{staking.DefaultBondDenom, sdk.NewInt(100)},
|
||||||
percentPrecommitVotes := sdk.NewDecWithPrec(25, 2)
|
}
|
||||||
keeper.AllocateTokens(ctx, percentPrecommitVotes, valConsAddr1)
|
fck.SetCollectedFees(fees)
|
||||||
|
votes := []abci.VoteInfo{
|
||||||
// verify that these fees have been received by the feePool
|
{
|
||||||
feePool := keeper.GetFeePool(ctx)
|
Validator: abciValA,
|
||||||
// 1% + 4%*0.25 to proposer + 1% community tax = 97%
|
SignedLastBlock: true,
|
||||||
percentProposer := sdk.NewDecWithPrec(1, 2).Add(sdk.NewDecWithPrec(4, 2).Mul(percentPrecommitVotes))
|
},
|
||||||
percentRemaining := sdk.OneDec().Sub(communityTax.Add(percentProposer))
|
{
|
||||||
expRes := sdk.NewDecFromInt(feeInputs).Mul(percentRemaining)
|
Validator: abciValB,
|
||||||
require.Equal(t, 1, len(feePool.ValPool))
|
SignedLastBlock: true,
|
||||||
require.True(sdk.DecEq(t, expRes, feePool.ValPool[0].Amount))
|
},
|
||||||
|
}
|
||||||
|
k.AllocateTokens(ctx, 200, 200, valConsAddr2, votes)
|
||||||
|
|
||||||
|
// 98 outstanding rewards (100 less 2 to community pool)
|
||||||
|
require.Equal(t, sdk.DecCoins{{staking.DefaultBondDenom, sdk.NewDec(98)}}, k.GetOutstandingRewards(ctx))
|
||||||
|
// 2 community pool coins
|
||||||
|
require.Equal(t, sdk.DecCoins{{staking.DefaultBondDenom, sdk.NewDec(2)}}, k.GetFeePool(ctx).CommunityPool)
|
||||||
|
// 50% commission for first proposer, (0.5 * 93%) * 100 / 2 = 23.25
|
||||||
|
require.Equal(t, sdk.DecCoins{{staking.DefaultBondDenom, sdk.NewDecWithPrec(2325, 2)}}, k.GetValidatorAccumulatedCommission(ctx, valOpAddr1))
|
||||||
|
// zero commission for second proposer
|
||||||
|
require.True(t, k.GetValidatorAccumulatedCommission(ctx, valOpAddr2).IsZero())
|
||||||
|
// just staking.proportional for first proposer less commission = (0.5 * 93%) * 100 / 2 = 23.25
|
||||||
|
require.Equal(t, sdk.DecCoins{{staking.DefaultBondDenom, sdk.NewDecWithPrec(2325, 2)}}, k.GetValidatorCurrentRewards(ctx, valOpAddr1).Rewards)
|
||||||
|
// proposer reward + staking.proportional for second proposer = (5 % + 0.5 * (93%)) * 100 = 51.5
|
||||||
|
require.Equal(t, sdk.DecCoins{{staking.DefaultBondDenom, sdk.NewDecWithPrec(515, 1)}}, k.GetValidatorCurrentRewards(ctx, valOpAddr2).Rewards)
|
||||||
}
|
}
|
||||||
|
|
|
@ -2,229 +2,88 @@ package keeper
|
||||||
|
|
||||||
import (
|
import (
|
||||||
sdk "github.com/cosmos/cosmos-sdk/types"
|
sdk "github.com/cosmos/cosmos-sdk/types"
|
||||||
|
|
||||||
"github.com/cosmos/cosmos-sdk/x/distribution/types"
|
"github.com/cosmos/cosmos-sdk/x/distribution/types"
|
||||||
)
|
)
|
||||||
|
|
||||||
// check whether a delegator distribution info exists
|
// initialize starting info for a new delegation
|
||||||
func (k Keeper) HasDelegationDistInfo(ctx sdk.Context, delAddr sdk.AccAddress,
|
func (k Keeper) initializeDelegation(ctx sdk.Context, val sdk.ValAddress, del sdk.AccAddress) {
|
||||||
valOperatorAddr sdk.ValAddress) (has bool) {
|
// period has already been incremented
|
||||||
store := ctx.KVStore(k.storeKey)
|
period := k.GetValidatorCurrentRewards(ctx, val).Period
|
||||||
return store.Has(GetDelegationDistInfoKey(delAddr, valOperatorAddr))
|
validator := k.stakingKeeper.Validator(ctx, val)
|
||||||
|
delegation := k.stakingKeeper.Delegation(ctx, del, val)
|
||||||
|
// calculate delegation stake in tokens
|
||||||
|
// we don't store directly, so multiply delegation shares * (tokens per share)
|
||||||
|
stake := delegation.GetShares().Mul(validator.GetDelegatorShareExRate())
|
||||||
|
k.SetDelegatorStartingInfo(ctx, val, del, types.NewDelegatorStartingInfo(period-1, stake, uint64(ctx.BlockHeight())))
|
||||||
}
|
}
|
||||||
|
|
||||||
// get the delegator distribution info
|
// calculate the rewards accrued by a delegation between two periods
|
||||||
func (k Keeper) GetDelegationDistInfo(ctx sdk.Context, delAddr sdk.AccAddress,
|
func (k Keeper) calculateDelegationRewardsBetween(ctx sdk.Context, val sdk.Validator,
|
||||||
valOperatorAddr sdk.ValAddress) (ddi types.DelegationDistInfo) {
|
startingPeriod, endingPeriod uint64, staking sdk.Dec) (rewards sdk.DecCoins) {
|
||||||
|
// sanity check
|
||||||
store := ctx.KVStore(k.storeKey)
|
if startingPeriod > endingPeriod {
|
||||||
|
panic("startingPeriod cannot be greater than endingPeriod")
|
||||||
b := store.Get(GetDelegationDistInfoKey(delAddr, valOperatorAddr))
|
|
||||||
if b == nil {
|
|
||||||
panic("Stored delegation-distribution info should not have been nil")
|
|
||||||
}
|
}
|
||||||
|
|
||||||
k.cdc.MustUnmarshalBinaryLengthPrefixed(b, &ddi)
|
// return staking * (ending - starting)
|
||||||
|
starting := k.GetValidatorHistoricalRewards(ctx, val.GetOperator(), startingPeriod)
|
||||||
|
ending := k.GetValidatorHistoricalRewards(ctx, val.GetOperator(), endingPeriod)
|
||||||
|
difference := ending.Minus(starting)
|
||||||
|
rewards = difference.MulDec(staking)
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
// set the delegator distribution info
|
// calculate the total rewards accrued by a delegation
|
||||||
func (k Keeper) SetDelegationDistInfo(ctx sdk.Context, ddi types.DelegationDistInfo) {
|
func (k Keeper) calculateDelegationRewards(ctx sdk.Context, val sdk.Validator, del sdk.Delegation, endingPeriod uint64) (rewards sdk.DecCoins) {
|
||||||
store := ctx.KVStore(k.storeKey)
|
// fetch starting info for delegation
|
||||||
b := k.cdc.MustMarshalBinaryLengthPrefixed(ddi)
|
startingInfo := k.GetDelegatorStartingInfo(ctx, del.GetValidatorAddr(), del.GetDelegatorAddr())
|
||||||
store.Set(GetDelegationDistInfoKey(ddi.DelegatorAddr, ddi.ValOperatorAddr), b)
|
startingPeriod := startingInfo.PreviousPeriod
|
||||||
}
|
stake := startingInfo.Stake
|
||||||
|
|
||||||
// remove a delegator distribution info
|
// iterate through slashes and withdraw with calculated staking for sub-intervals
|
||||||
func (k Keeper) RemoveDelegationDistInfo(ctx sdk.Context, delAddr sdk.AccAddress,
|
// these offsets are dependent on *when* slashes happen - namely, in BeginBlock, after rewards are allocated...
|
||||||
valOperatorAddr sdk.ValAddress) {
|
// ... so we don't reduce stake for slashes which happened in the *first* block, because the delegation wouldn't have existed
|
||||||
|
startingHeight := startingInfo.Height + 1
|
||||||
store := ctx.KVStore(k.storeKey)
|
// ... or slashes which happened in *this* block, since they would have happened after reward allocation
|
||||||
store.Delete(GetDelegationDistInfoKey(delAddr, valOperatorAddr))
|
endingHeight := uint64(ctx.BlockHeight()) - 1
|
||||||
}
|
if endingHeight >= startingHeight {
|
||||||
|
k.IterateValidatorSlashEventsBetween(ctx, del.GetValidatorAddr(), startingHeight, endingHeight,
|
||||||
// remove all delegation distribution infos
|
func(height uint64, event types.ValidatorSlashEvent) (stop bool) {
|
||||||
func (k Keeper) RemoveDelegationDistInfos(ctx sdk.Context) {
|
endingPeriod := event.ValidatorPeriod - 1
|
||||||
store := ctx.KVStore(k.storeKey)
|
rewards = rewards.Plus(k.calculateDelegationRewardsBetween(ctx, val, startingPeriod, endingPeriod, stake))
|
||||||
iter := sdk.KVStorePrefixIterator(store, DelegationDistInfoKey)
|
stake = stake.Mul(sdk.OneDec().Sub(event.Fraction))
|
||||||
defer iter.Close()
|
startingPeriod = endingPeriod
|
||||||
for ; iter.Valid(); iter.Next() {
|
return false
|
||||||
store.Delete(iter.Key())
|
},
|
||||||
|
)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// calculate rewards for final period
|
||||||
|
rewards = rewards.Plus(k.calculateDelegationRewardsBetween(ctx, val, startingPeriod, endingPeriod, stake))
|
||||||
|
|
||||||
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
// iterate over all the validator distribution infos
|
func (k Keeper) withdrawDelegationRewards(ctx sdk.Context, val sdk.Validator, del sdk.Delegation) sdk.Error {
|
||||||
func (k Keeper) IterateDelegationDistInfos(ctx sdk.Context,
|
|
||||||
fn func(index int64, distInfo types.DelegationDistInfo) (stop bool)) {
|
|
||||||
|
|
||||||
store := ctx.KVStore(k.storeKey)
|
// end current period and calculate rewards
|
||||||
iter := sdk.KVStorePrefixIterator(store, DelegationDistInfoKey)
|
endingPeriod := k.incrementValidatorPeriod(ctx, val)
|
||||||
defer iter.Close()
|
rewards := k.calculateDelegationRewards(ctx, val, del, endingPeriod)
|
||||||
index := int64(0)
|
|
||||||
for ; iter.Valid(); iter.Next() {
|
|
||||||
var ddi types.DelegationDistInfo
|
|
||||||
k.cdc.MustUnmarshalBinaryLengthPrefixed(iter.Value(), &ddi)
|
|
||||||
if fn(index, ddi) {
|
|
||||||
return
|
|
||||||
}
|
|
||||||
index++
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
//___________________________________________________________________________________________
|
// truncate coins, return remainder to community pool
|
||||||
|
coins, remainder := rewards.TruncateDecimal()
|
||||||
// get the delegator withdraw address, return the delegator address if not set
|
outstanding := k.GetOutstandingRewards(ctx)
|
||||||
func (k Keeper) GetDelegatorWithdrawAddr(ctx sdk.Context, delAddr sdk.AccAddress) sdk.AccAddress {
|
k.SetOutstandingRewards(ctx, outstanding.Minus(rewards))
|
||||||
store := ctx.KVStore(k.storeKey)
|
feePool := k.GetFeePool(ctx)
|
||||||
|
feePool.CommunityPool = feePool.CommunityPool.Plus(remainder)
|
||||||
b := store.Get(GetDelegatorWithdrawAddrKey(delAddr))
|
|
||||||
if b == nil {
|
|
||||||
return delAddr
|
|
||||||
}
|
|
||||||
return sdk.AccAddress(b)
|
|
||||||
}
|
|
||||||
|
|
||||||
// set the delegator withdraw address
|
|
||||||
func (k Keeper) SetDelegatorWithdrawAddr(ctx sdk.Context, delAddr, withdrawAddr sdk.AccAddress) {
|
|
||||||
store := ctx.KVStore(k.storeKey)
|
|
||||||
store.Set(GetDelegatorWithdrawAddrKey(delAddr), withdrawAddr.Bytes())
|
|
||||||
}
|
|
||||||
|
|
||||||
// remove a delegator withdraw info
|
|
||||||
func (k Keeper) RemoveDelegatorWithdrawAddr(ctx sdk.Context, delAddr, withdrawAddr sdk.AccAddress) {
|
|
||||||
store := ctx.KVStore(k.storeKey)
|
|
||||||
store.Delete(GetDelegatorWithdrawAddrKey(delAddr))
|
|
||||||
}
|
|
||||||
|
|
||||||
//___________________________________________________________________________________________
|
|
||||||
|
|
||||||
// return all rewards for a delegation
|
|
||||||
func (k Keeper) withdrawDelegationReward(ctx sdk.Context,
|
|
||||||
delAddr sdk.AccAddress, valAddr sdk.ValAddress) (types.FeePool,
|
|
||||||
types.ValidatorDistInfo, types.DelegationDistInfo, types.DecCoins) {
|
|
||||||
|
|
||||||
wc := k.GetWithdrawContext(ctx, valAddr)
|
|
||||||
valInfo := k.GetValidatorDistInfo(ctx, valAddr)
|
|
||||||
delInfo := k.GetDelegationDistInfo(ctx, delAddr, valAddr)
|
|
||||||
validator := k.stakingKeeper.Validator(ctx, valAddr)
|
|
||||||
delegation := k.stakingKeeper.Delegation(ctx, delAddr, valAddr)
|
|
||||||
|
|
||||||
delInfo, valInfo, feePool, withdraw := delInfo.WithdrawRewards(wc, valInfo,
|
|
||||||
validator.GetDelegatorShares(), delegation.GetShares())
|
|
||||||
|
|
||||||
return feePool, valInfo, delInfo, withdraw
|
|
||||||
}
|
|
||||||
|
|
||||||
// get all rewards for all delegations of a delegator
|
|
||||||
func (k Keeper) currentDelegationReward(ctx sdk.Context, delAddr sdk.AccAddress,
|
|
||||||
valAddr sdk.ValAddress) types.DecCoins {
|
|
||||||
|
|
||||||
wc := k.GetWithdrawContext(ctx, valAddr)
|
|
||||||
|
|
||||||
valInfo := k.GetValidatorDistInfo(ctx, valAddr)
|
|
||||||
delInfo := k.GetDelegationDistInfo(ctx, delAddr, valAddr)
|
|
||||||
validator := k.stakingKeeper.Validator(ctx, valAddr)
|
|
||||||
delegation := k.stakingKeeper.Delegation(ctx, delAddr, valAddr)
|
|
||||||
|
|
||||||
estimation := delInfo.CurrentRewards(wc, valInfo,
|
|
||||||
validator.GetDelegatorShares(), delegation.GetShares())
|
|
||||||
|
|
||||||
return estimation
|
|
||||||
}
|
|
||||||
|
|
||||||
//___________________________________________________________________________________________
|
|
||||||
|
|
||||||
// withdraw all rewards for a single delegation
|
|
||||||
// NOTE: This gets called "onDelegationSharesModified",
|
|
||||||
// meaning any changes to bonded coins
|
|
||||||
func (k Keeper) WithdrawToDelegator(ctx sdk.Context, feePool types.FeePool,
|
|
||||||
delAddr sdk.AccAddress, amount types.DecCoins) {
|
|
||||||
|
|
||||||
withdrawAddr := k.GetDelegatorWithdrawAddr(ctx, delAddr)
|
|
||||||
coinsToAdd, change := amount.TruncateDecimal()
|
|
||||||
feePool.CommunityPool = feePool.CommunityPool.Plus(change)
|
|
||||||
k.SetFeePool(ctx, feePool)
|
k.SetFeePool(ctx, feePool)
|
||||||
_, _, err := k.bankKeeper.AddCoins(ctx, withdrawAddr, coinsToAdd)
|
|
||||||
if err != nil {
|
// add coins to user account
|
||||||
panic(err)
|
withdrawAddr := k.GetDelegatorWithdrawAddr(ctx, del.GetDelegatorAddr())
|
||||||
|
if _, _, err := k.bankKeeper.AddCoins(ctx, withdrawAddr, coins); err != nil {
|
||||||
|
return err
|
||||||
}
|
}
|
||||||
}
|
|
||||||
|
|
||||||
//___________________________________________________________________________________________
|
|
||||||
|
|
||||||
// withdraw all rewards for a single delegation
|
|
||||||
// NOTE: This gets called "onDelegationSharesModified",
|
|
||||||
// meaning any changes to bonded coins
|
|
||||||
func (k Keeper) WithdrawDelegationReward(ctx sdk.Context, delAddr sdk.AccAddress,
|
|
||||||
valAddr sdk.ValAddress) sdk.Error {
|
|
||||||
|
|
||||||
if !k.HasDelegationDistInfo(ctx, delAddr, valAddr) {
|
|
||||||
return types.ErrNoDelegationDistInfo(k.codespace)
|
|
||||||
}
|
|
||||||
|
|
||||||
feePool, valInfo, delInfo, withdraw := k.withdrawDelegationReward(ctx, delAddr, valAddr)
|
|
||||||
|
|
||||||
k.SetValidatorDistInfo(ctx, valInfo)
|
|
||||||
k.SetDelegationDistInfo(ctx, delInfo)
|
|
||||||
k.WithdrawToDelegator(ctx, feePool, delAddr, withdraw)
|
|
||||||
|
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
// current rewards for a single delegation
|
|
||||||
func (k Keeper) CurrentDelegationReward(ctx sdk.Context, delAddr sdk.AccAddress,
|
|
||||||
valAddr sdk.ValAddress) (sdk.Coins, sdk.Error) {
|
|
||||||
|
|
||||||
if !k.HasDelegationDistInfo(ctx, delAddr, valAddr) {
|
|
||||||
return sdk.Coins{}, types.ErrNoDelegationDistInfo(k.codespace)
|
|
||||||
}
|
|
||||||
estCoins := k.currentDelegationReward(ctx, delAddr, valAddr)
|
|
||||||
trucate, _ := estCoins.TruncateDecimal()
|
|
||||||
return trucate, nil
|
|
||||||
}
|
|
||||||
|
|
||||||
//___________________________________________________________________________________________
|
|
||||||
|
|
||||||
// return all rewards for all delegations of a delegator
|
|
||||||
func (k Keeper) WithdrawDelegationRewardsAll(ctx sdk.Context, delAddr sdk.AccAddress) {
|
|
||||||
withdraw := k.withdrawDelegationRewardsAll(ctx, delAddr)
|
|
||||||
feePool := k.GetFeePool(ctx)
|
|
||||||
k.WithdrawToDelegator(ctx, feePool, delAddr, withdraw)
|
|
||||||
}
|
|
||||||
|
|
||||||
func (k Keeper) withdrawDelegationRewardsAll(ctx sdk.Context,
|
|
||||||
delAddr sdk.AccAddress) types.DecCoins {
|
|
||||||
|
|
||||||
withdraw := types.DecCoins{}
|
|
||||||
|
|
||||||
// iterate over all the delegations
|
|
||||||
operationAtDelegation := func(_ int64, del sdk.Delegation) (stop bool) {
|
|
||||||
valAddr := del.GetValidatorAddr()
|
|
||||||
feePool, valInfo, delInfo, diWithdraw := k.withdrawDelegationReward(ctx, delAddr, valAddr)
|
|
||||||
withdraw = withdraw.Plus(diWithdraw)
|
|
||||||
|
|
||||||
k.SetFeePool(ctx, feePool)
|
|
||||||
k.SetValidatorDistInfo(ctx, valInfo)
|
|
||||||
k.SetDelegationDistInfo(ctx, delInfo)
|
|
||||||
|
|
||||||
return false
|
|
||||||
}
|
|
||||||
|
|
||||||
k.stakingKeeper.IterateDelegations(ctx, delAddr, operationAtDelegation)
|
|
||||||
return withdraw
|
|
||||||
}
|
|
||||||
|
|
||||||
// get all rewards for all delegations of a delegator
|
|
||||||
func (k Keeper) CurrentDelegationRewardsAll(ctx sdk.Context,
|
|
||||||
delAddr sdk.AccAddress) types.DecCoins {
|
|
||||||
|
|
||||||
// iterate over all the delegations
|
|
||||||
total := types.DecCoins{}
|
|
||||||
operationAtDelegation := func(_ int64, del sdk.Delegation) (stop bool) {
|
|
||||||
valAddr := del.GetValidatorAddr()
|
|
||||||
est := k.currentDelegationReward(ctx, delAddr, valAddr)
|
|
||||||
total = total.Plus(est)
|
|
||||||
return false
|
|
||||||
}
|
|
||||||
k.stakingKeeper.IterateDelegations(ctx, delAddr, operationAtDelegation)
|
|
||||||
return total
|
|
||||||
}
|
|
||||||
|
|
|
@ -9,246 +9,530 @@ import (
|
||||||
"github.com/cosmos/cosmos-sdk/x/staking"
|
"github.com/cosmos/cosmos-sdk/x/staking"
|
||||||
)
|
)
|
||||||
|
|
||||||
func TestWithdrawDelegationRewardBasic(t *testing.T) {
|
func TestCalculateRewardsBasic(t *testing.T) {
|
||||||
ctx, accMapper, keeper, sk, fck := CreateTestInputAdvanced(t, false, 100, sdk.ZeroDec())
|
ctx, _, k, sk, _ := CreateTestInputDefault(t, false, 1000)
|
||||||
stakingHandler := staking.NewHandler(sk)
|
sh := staking.NewHandler(sk)
|
||||||
denom := sk.GetParams(ctx).BondDenom
|
|
||||||
|
|
||||||
//first make a validator
|
// initialize state
|
||||||
msgCreateValidator := staking.NewTestMsgCreateValidator(valOpAddr1, valConsPk1, 10)
|
k.SetOutstandingRewards(ctx, sdk.DecCoins{})
|
||||||
got := stakingHandler(ctx, msgCreateValidator)
|
|
||||||
require.True(t, got.IsOK(), "expected msg to be ok, got %v", got)
|
|
||||||
_ = sk.ApplyAndReturnValidatorSetUpdates(ctx)
|
|
||||||
|
|
||||||
// delegate
|
// create validator with 50% commission
|
||||||
msgDelegate := staking.NewTestMsgDelegate(delAddr1, valOpAddr1, 10)
|
commission := staking.NewCommissionMsg(sdk.NewDecWithPrec(5, 1), sdk.NewDecWithPrec(5, 1), sdk.NewDec(0))
|
||||||
got = stakingHandler(ctx, msgDelegate)
|
msg := staking.NewMsgCreateValidator(valOpAddr1, valConsPk1,
|
||||||
require.True(t, got.IsOK())
|
sdk.NewCoin(staking.DefaultBondDenom, sdk.NewInt(100)), staking.Description{}, commission)
|
||||||
amt := accMapper.GetAccount(ctx, delAddr1).GetCoins().AmountOf(denom)
|
require.True(t, sh(ctx, msg).IsOK())
|
||||||
require.Equal(t, int64(90), amt.Int64())
|
|
||||||
|
|
||||||
// allocate 100 denom of fees
|
// end block to bond validator
|
||||||
feeInputs := sdk.NewInt(100)
|
staking.EndBlocker(ctx, sk)
|
||||||
fck.SetCollectedFees(sdk.Coins{sdk.NewCoin(denom, feeInputs)})
|
|
||||||
require.Equal(t, feeInputs, fck.GetCollectedFees(ctx).AmountOf(denom))
|
|
||||||
keeper.AllocateTokens(ctx, sdk.OneDec(), valConsAddr1)
|
|
||||||
|
|
||||||
// withdraw delegation
|
// fetch validator and delegation
|
||||||
ctx = ctx.WithBlockHeight(1)
|
val := sk.Validator(ctx, valOpAddr1)
|
||||||
sk.SetLastTotalPower(ctx, sdk.NewInt(10))
|
del := sk.Delegation(ctx, sdk.AccAddress(valOpAddr1), valOpAddr1)
|
||||||
sk.SetLastValidatorPower(ctx, valOpAddr1, sdk.NewInt(10))
|
|
||||||
keeper.WithdrawDelegationReward(ctx, delAddr1, valOpAddr1)
|
|
||||||
amt = accMapper.GetAccount(ctx, delAddr1).GetCoins().AmountOf(denom)
|
|
||||||
|
|
||||||
expRes := sdk.NewDec(90).Add(sdk.NewDec(100).Quo(sdk.NewDec(2))).TruncateInt() // 90 + 100 tokens * 10/20
|
// end period
|
||||||
require.True(sdk.IntEq(t, expRes, amt))
|
endingPeriod := k.incrementValidatorPeriod(ctx, val)
|
||||||
|
|
||||||
|
// calculate delegation rewards
|
||||||
|
rewards := k.calculateDelegationRewards(ctx, val, del, endingPeriod)
|
||||||
|
|
||||||
|
// rewards should be zero
|
||||||
|
require.True(t, rewards.IsZero())
|
||||||
|
|
||||||
|
// allocate some rewards
|
||||||
|
initial := int64(10)
|
||||||
|
tokens := sdk.DecCoins{{staking.DefaultBondDenom, sdk.NewDec(initial)}}
|
||||||
|
k.AllocateTokensToValidator(ctx, val, tokens)
|
||||||
|
|
||||||
|
// end period
|
||||||
|
endingPeriod = k.incrementValidatorPeriod(ctx, val)
|
||||||
|
|
||||||
|
// calculate delegation rewards
|
||||||
|
rewards = k.calculateDelegationRewards(ctx, val, del, endingPeriod)
|
||||||
|
|
||||||
|
// rewards should be half the tokens
|
||||||
|
require.Equal(t, sdk.DecCoins{{staking.DefaultBondDenom, sdk.NewDec(initial / 2)}}, rewards)
|
||||||
|
|
||||||
|
// commission should be the other half
|
||||||
|
require.Equal(t, sdk.DecCoins{{staking.DefaultBondDenom, sdk.NewDec(initial / 2)}}, k.GetValidatorAccumulatedCommission(ctx, valOpAddr1))
|
||||||
}
|
}
|
||||||
|
|
||||||
func TestWithdrawDelegationRewardWithCommission(t *testing.T) {
|
func TestCalculateRewardsAfterSlash(t *testing.T) {
|
||||||
ctx, accMapper, keeper, sk, fck := CreateTestInputAdvanced(t, false, 100, sdk.ZeroDec())
|
ctx, _, k, sk, _ := CreateTestInputDefault(t, false, 1000)
|
||||||
stakingHandler := staking.NewHandler(sk)
|
sh := staking.NewHandler(sk)
|
||||||
denom := sk.GetParams(ctx).BondDenom
|
|
||||||
|
|
||||||
//first make a validator with 10% commission
|
// initialize state
|
||||||
msgCreateValidator := staking.NewTestMsgCreateValidatorWithCommission(
|
k.SetOutstandingRewards(ctx, sdk.DecCoins{})
|
||||||
valOpAddr1, valConsPk1, 10, sdk.NewDecWithPrec(1, 1))
|
|
||||||
got := stakingHandler(ctx, msgCreateValidator)
|
|
||||||
require.True(t, got.IsOK(), "expected msg to be ok, got %v", got)
|
|
||||||
_ = sk.ApplyAndReturnValidatorSetUpdates(ctx)
|
|
||||||
|
|
||||||
// delegate
|
// create validator with 50% commission
|
||||||
msgDelegate := staking.NewTestMsgDelegate(delAddr1, valOpAddr1, 10)
|
commission := staking.NewCommissionMsg(sdk.NewDecWithPrec(5, 1), sdk.NewDecWithPrec(5, 1), sdk.NewDec(0))
|
||||||
got = stakingHandler(ctx, msgDelegate)
|
msg := staking.NewMsgCreateValidator(valOpAddr1, valConsPk1,
|
||||||
require.True(t, got.IsOK())
|
sdk.NewCoin(staking.DefaultBondDenom, sdk.NewInt(100)), staking.Description{}, commission)
|
||||||
amt := accMapper.GetAccount(ctx, delAddr1).GetCoins().AmountOf(denom)
|
require.True(t, sh(ctx, msg).IsOK())
|
||||||
require.Equal(t, int64(90), amt.Int64())
|
|
||||||
|
|
||||||
// allocate 100 denom of fees
|
// end block to bond validator
|
||||||
feeInputs := sdk.NewInt(100)
|
staking.EndBlocker(ctx, sk)
|
||||||
fck.SetCollectedFees(sdk.Coins{sdk.NewCoin(denom, feeInputs)})
|
|
||||||
require.Equal(t, feeInputs, fck.GetCollectedFees(ctx).AmountOf(denom))
|
|
||||||
keeper.AllocateTokens(ctx, sdk.OneDec(), valConsAddr1)
|
|
||||||
|
|
||||||
// withdraw delegation
|
// fetch validator and delegation
|
||||||
ctx = ctx.WithBlockHeight(1)
|
val := sk.Validator(ctx, valOpAddr1)
|
||||||
keeper.WithdrawDelegationReward(ctx, delAddr1, valOpAddr1)
|
del := sk.Delegation(ctx, sdk.AccAddress(valOpAddr1), valOpAddr1)
|
||||||
amt = accMapper.GetAccount(ctx, delAddr1).GetCoins().AmountOf(denom)
|
|
||||||
|
|
||||||
expRes := sdk.NewDec(90).Add(sdk.NewDec(90).Quo(sdk.NewDec(2))).TruncateInt() // 90 + 100*90% tokens * 10/20
|
// end period
|
||||||
require.True(sdk.IntEq(t, expRes, amt))
|
endingPeriod := k.incrementValidatorPeriod(ctx, val)
|
||||||
|
|
||||||
|
// calculate delegation rewards
|
||||||
|
rewards := k.calculateDelegationRewards(ctx, val, del, endingPeriod)
|
||||||
|
|
||||||
|
// rewards should be zero
|
||||||
|
require.True(t, rewards.IsZero())
|
||||||
|
|
||||||
|
// start out block height
|
||||||
|
ctx = ctx.WithBlockHeight(ctx.BlockHeight() + 3)
|
||||||
|
|
||||||
|
// slash the validator by 50%
|
||||||
|
sk.Slash(ctx, valConsAddr1, ctx.BlockHeight(), 100, sdk.NewDecWithPrec(5, 1))
|
||||||
|
|
||||||
|
// retrieve validator
|
||||||
|
val = sk.Validator(ctx, valOpAddr1)
|
||||||
|
|
||||||
|
// increase block height
|
||||||
|
ctx = ctx.WithBlockHeight(ctx.BlockHeight() + 3)
|
||||||
|
|
||||||
|
// allocate some rewards
|
||||||
|
initial := int64(10)
|
||||||
|
tokens := sdk.DecCoins{{staking.DefaultBondDenom, sdk.NewDec(initial)}}
|
||||||
|
k.AllocateTokensToValidator(ctx, val, tokens)
|
||||||
|
|
||||||
|
// end period
|
||||||
|
endingPeriod = k.incrementValidatorPeriod(ctx, val)
|
||||||
|
|
||||||
|
// calculate delegation rewards
|
||||||
|
rewards = k.calculateDelegationRewards(ctx, val, del, endingPeriod)
|
||||||
|
|
||||||
|
// rewards should be half the tokens
|
||||||
|
require.Equal(t, sdk.DecCoins{{staking.DefaultBondDenom, sdk.NewDec(initial / 2)}}, rewards)
|
||||||
|
|
||||||
|
// commission should be the other half
|
||||||
|
require.Equal(t, sdk.DecCoins{{staking.DefaultBondDenom, sdk.NewDec(initial / 2)}}, k.GetValidatorAccumulatedCommission(ctx, valOpAddr1))
|
||||||
}
|
}
|
||||||
|
|
||||||
func TestWithdrawDelegationRewardTwoDelegators(t *testing.T) {
|
func TestCalculateRewardsAfterManySlashes(t *testing.T) {
|
||||||
ctx, accMapper, keeper, sk, fck := CreateTestInputAdvanced(t, false, 100, sdk.ZeroDec())
|
ctx, _, k, sk, _ := CreateTestInputDefault(t, false, 1000)
|
||||||
stakingHandler := staking.NewHandler(sk)
|
sh := staking.NewHandler(sk)
|
||||||
denom := sk.GetParams(ctx).BondDenom
|
|
||||||
|
|
||||||
//first make a validator with 10% commission
|
// initialize state
|
||||||
msgCreateValidator := staking.NewTestMsgCreateValidatorWithCommission(
|
k.SetOutstandingRewards(ctx, sdk.DecCoins{})
|
||||||
valOpAddr1, valConsPk1, 10, sdk.NewDecWithPrec(1, 1))
|
|
||||||
got := stakingHandler(ctx, msgCreateValidator)
|
|
||||||
require.True(t, got.IsOK(), "expected msg to be ok, got %v", got)
|
|
||||||
_ = sk.ApplyAndReturnValidatorSetUpdates(ctx)
|
|
||||||
|
|
||||||
// delegate
|
// create validator with 50% commission
|
||||||
msgDelegate := staking.NewTestMsgDelegate(delAddr1, valOpAddr1, 10)
|
commission := staking.NewCommissionMsg(sdk.NewDecWithPrec(5, 1), sdk.NewDecWithPrec(5, 1), sdk.NewDec(0))
|
||||||
got = stakingHandler(ctx, msgDelegate)
|
msg := staking.NewMsgCreateValidator(valOpAddr1, valConsPk1,
|
||||||
require.True(t, got.IsOK())
|
sdk.NewCoin(staking.DefaultBondDenom, sdk.NewInt(100)), staking.Description{}, commission)
|
||||||
amt := accMapper.GetAccount(ctx, delAddr1).GetCoins().AmountOf(denom)
|
require.True(t, sh(ctx, msg).IsOK())
|
||||||
require.Equal(t, int64(90), amt.Int64())
|
|
||||||
|
|
||||||
msgDelegate = staking.NewTestMsgDelegate(delAddr2, valOpAddr1, 20)
|
// end block to bond validator
|
||||||
got = stakingHandler(ctx, msgDelegate)
|
staking.EndBlocker(ctx, sk)
|
||||||
require.True(t, got.IsOK())
|
|
||||||
amt = accMapper.GetAccount(ctx, delAddr2).GetCoins().AmountOf(denom)
|
|
||||||
require.Equal(t, int64(80), amt.Int64())
|
|
||||||
|
|
||||||
// allocate 100 denom of fees
|
// fetch validator and delegation
|
||||||
feeInputs := sdk.NewInt(100)
|
val := sk.Validator(ctx, valOpAddr1)
|
||||||
fck.SetCollectedFees(sdk.Coins{sdk.NewCoin(denom, feeInputs)})
|
del := sk.Delegation(ctx, sdk.AccAddress(valOpAddr1), valOpAddr1)
|
||||||
require.Equal(t, feeInputs, fck.GetCollectedFees(ctx).AmountOf(denom))
|
|
||||||
keeper.AllocateTokens(ctx, sdk.OneDec(), valConsAddr1)
|
|
||||||
|
|
||||||
// delegator 1 withdraw delegation
|
// end period
|
||||||
ctx = ctx.WithBlockHeight(1)
|
endingPeriod := k.incrementValidatorPeriod(ctx, val)
|
||||||
keeper.WithdrawDelegationReward(ctx, delAddr1, valOpAddr1)
|
|
||||||
amt = accMapper.GetAccount(ctx, delAddr1).GetCoins().AmountOf(denom)
|
|
||||||
|
|
||||||
expRes := sdk.NewDec(90).Add(sdk.NewDec(90).Quo(sdk.NewDec(4))).TruncateInt() // 90 + 100*90% tokens * 10/40
|
// calculate delegation rewards
|
||||||
require.True(sdk.IntEq(t, expRes, amt))
|
rewards := k.calculateDelegationRewards(ctx, val, del, endingPeriod)
|
||||||
|
|
||||||
|
// rewards should be zero
|
||||||
|
require.True(t, rewards.IsZero())
|
||||||
|
|
||||||
|
// start out block height
|
||||||
|
ctx = ctx.WithBlockHeight(ctx.BlockHeight() + 3)
|
||||||
|
|
||||||
|
// slash the validator by 50%
|
||||||
|
sk.Slash(ctx, valConsAddr1, ctx.BlockHeight(), 100, sdk.NewDecWithPrec(5, 1))
|
||||||
|
|
||||||
|
// fetch the validator again
|
||||||
|
val = sk.Validator(ctx, valOpAddr1)
|
||||||
|
|
||||||
|
// increase block height
|
||||||
|
ctx = ctx.WithBlockHeight(ctx.BlockHeight() + 3)
|
||||||
|
|
||||||
|
// allocate some rewards
|
||||||
|
initial := int64(10)
|
||||||
|
tokens := sdk.DecCoins{{staking.DefaultBondDenom, sdk.NewDec(initial)}}
|
||||||
|
k.AllocateTokensToValidator(ctx, val, tokens)
|
||||||
|
|
||||||
|
// slash the validator by 50% again
|
||||||
|
sk.Slash(ctx, valConsAddr1, ctx.BlockHeight(), 50, sdk.NewDecWithPrec(5, 1))
|
||||||
|
|
||||||
|
// fetch the validator again
|
||||||
|
val = sk.Validator(ctx, valOpAddr1)
|
||||||
|
|
||||||
|
// increase block height
|
||||||
|
ctx = ctx.WithBlockHeight(ctx.BlockHeight() + 3)
|
||||||
|
|
||||||
|
// allocate some more rewards
|
||||||
|
k.AllocateTokensToValidator(ctx, val, tokens)
|
||||||
|
|
||||||
|
// end period
|
||||||
|
endingPeriod = k.incrementValidatorPeriod(ctx, val)
|
||||||
|
|
||||||
|
// calculate delegation rewards
|
||||||
|
rewards = k.calculateDelegationRewards(ctx, val, del, endingPeriod)
|
||||||
|
|
||||||
|
// rewards should be half the tokens
|
||||||
|
require.Equal(t, sdk.DecCoins{{staking.DefaultBondDenom, sdk.NewDec(initial)}}, rewards)
|
||||||
|
|
||||||
|
// commission should be the other half
|
||||||
|
require.Equal(t, sdk.DecCoins{{staking.DefaultBondDenom, sdk.NewDec(initial)}}, k.GetValidatorAccumulatedCommission(ctx, valOpAddr1))
|
||||||
}
|
}
|
||||||
|
|
||||||
// this test demonstrates how two delegators with the same power can end up
|
func TestCalculateRewardsMultiDelegator(t *testing.T) {
|
||||||
// with different rewards in the end
|
ctx, _, k, sk, _ := CreateTestInputDefault(t, false, 1000)
|
||||||
func TestWithdrawDelegationRewardTwoDelegatorsUneven(t *testing.T) {
|
sh := staking.NewHandler(sk)
|
||||||
ctx, accMapper, keeper, sk, fck := CreateTestInputAdvanced(t, false, 100, sdk.ZeroDec())
|
|
||||||
stakingHandler := staking.NewHandler(sk)
|
|
||||||
denom := sk.GetParams(ctx).BondDenom
|
|
||||||
|
|
||||||
//first make a validator with no commission
|
// initialize state
|
||||||
msgCreateValidator := staking.NewTestMsgCreateValidatorWithCommission(
|
k.SetOutstandingRewards(ctx, sdk.DecCoins{})
|
||||||
valOpAddr1, valConsPk1, 10, sdk.ZeroDec())
|
|
||||||
got := stakingHandler(ctx, msgCreateValidator)
|
|
||||||
require.True(t, got.IsOK(), "expected msg to be ok, got %v", got)
|
|
||||||
_ = sk.ApplyAndReturnValidatorSetUpdates(ctx)
|
|
||||||
|
|
||||||
// delegate
|
// create validator with 50% commission
|
||||||
msgDelegate := staking.NewTestMsgDelegate(delAddr1, valOpAddr1, 10)
|
commission := staking.NewCommissionMsg(sdk.NewDecWithPrec(5, 1), sdk.NewDecWithPrec(5, 1), sdk.NewDec(0))
|
||||||
got = stakingHandler(ctx, msgDelegate)
|
msg := staking.NewMsgCreateValidator(valOpAddr1, valConsPk1,
|
||||||
require.True(t, got.IsOK())
|
sdk.NewCoin(staking.DefaultBondDenom, sdk.NewInt(100)), staking.Description{}, commission)
|
||||||
amt := accMapper.GetAccount(ctx, delAddr1).GetCoins().AmountOf(denom)
|
require.True(t, sh(ctx, msg).IsOK())
|
||||||
require.Equal(t, int64(90), amt.Int64())
|
|
||||||
|
|
||||||
msgDelegate = staking.NewTestMsgDelegate(delAddr2, valOpAddr1, 10)
|
// end block to bond validator
|
||||||
got = stakingHandler(ctx, msgDelegate)
|
staking.EndBlocker(ctx, sk)
|
||||||
require.True(t, got.IsOK())
|
|
||||||
amt = accMapper.GetAccount(ctx, delAddr2).GetCoins().AmountOf(denom)
|
|
||||||
require.Equal(t, int64(90), amt.Int64())
|
|
||||||
|
|
||||||
// allocate 100 denom of fees
|
// fetch validator and delegation
|
||||||
feeInputs := sdk.NewInt(90)
|
val := sk.Validator(ctx, valOpAddr1)
|
||||||
fck.SetCollectedFees(sdk.Coins{sdk.NewCoin(denom, feeInputs)})
|
del1 := sk.Delegation(ctx, sdk.AccAddress(valOpAddr1), valOpAddr1)
|
||||||
require.Equal(t, feeInputs, fck.GetCollectedFees(ctx).AmountOf(denom))
|
|
||||||
keeper.AllocateTokens(ctx, sdk.OneDec(), valConsAddr1)
|
|
||||||
ctx = ctx.WithBlockHeight(1)
|
|
||||||
|
|
||||||
// delegator 1 withdraw delegation early, delegator 2 just keeps it's accum
|
// allocate some rewards
|
||||||
keeper.WithdrawDelegationReward(ctx, delAddr1, valOpAddr1)
|
initial := int64(20)
|
||||||
amt = accMapper.GetAccount(ctx, delAddr1).GetCoins().AmountOf(denom)
|
tokens := sdk.DecCoins{{staking.DefaultBondDenom, sdk.NewDec(initial)}}
|
||||||
|
k.AllocateTokensToValidator(ctx, val, tokens)
|
||||||
|
|
||||||
expRes1 := sdk.NewDec(90).Add(sdk.NewDec(90).Quo(sdk.NewDec(3))).TruncateInt() // 90 + 100 * 10/30
|
// second delegation
|
||||||
require.True(sdk.IntEq(t, expRes1, amt))
|
msg2 := staking.NewMsgDelegate(sdk.AccAddress(valOpAddr2), valOpAddr1, sdk.NewCoin(staking.DefaultBondDenom, sdk.NewInt(100)))
|
||||||
|
require.True(t, sh(ctx, msg2).IsOK())
|
||||||
|
del2 := sk.Delegation(ctx, sdk.AccAddress(valOpAddr2), valOpAddr1)
|
||||||
|
|
||||||
// allocate 200 denom of fees
|
// fetch updated validator
|
||||||
feeInputs = sdk.NewInt(180)
|
val = sk.Validator(ctx, valOpAddr1)
|
||||||
fck.SetCollectedFees(sdk.Coins{sdk.NewCoin(denom, feeInputs)})
|
|
||||||
require.Equal(t, feeInputs, fck.GetCollectedFees(ctx).AmountOf(denom))
|
|
||||||
keeper.AllocateTokens(ctx, sdk.OneDec(), valConsAddr1)
|
|
||||||
ctx = ctx.WithBlockHeight(2)
|
|
||||||
|
|
||||||
// delegator 2 now withdraws everything it's entitled to
|
// end block
|
||||||
keeper.WithdrawDelegationReward(ctx, delAddr2, valOpAddr1)
|
staking.EndBlocker(ctx, sk)
|
||||||
amt = accMapper.GetAccount(ctx, delAddr2).GetCoins().AmountOf(denom)
|
|
||||||
// existingTokens + (100+200 * (10/(20+30))
|
|
||||||
withdrawnFromVal := sdk.NewDec(60 + 180).Mul(sdk.NewDec(2)).Quo(sdk.NewDec(5))
|
|
||||||
expRes2 := sdk.NewDec(90).Add(withdrawnFromVal).TruncateInt()
|
|
||||||
require.True(sdk.IntEq(t, expRes2, amt))
|
|
||||||
|
|
||||||
// finally delegator 1 withdraws the remainder of its reward
|
// allocate some more rewards
|
||||||
keeper.WithdrawDelegationReward(ctx, delAddr1, valOpAddr1)
|
k.AllocateTokensToValidator(ctx, val, tokens)
|
||||||
amt = accMapper.GetAccount(ctx, delAddr1).GetCoins().AmountOf(denom)
|
|
||||||
|
|
||||||
remainingInVal := sdk.NewDec(60 + 180).Sub(withdrawnFromVal)
|
// end period
|
||||||
expRes3 := sdk.NewDecFromInt(expRes1).Add(remainingInVal.Mul(sdk.NewDec(1)).Quo(sdk.NewDec(3))).TruncateInt()
|
endingPeriod := k.incrementValidatorPeriod(ctx, val)
|
||||||
require.True(sdk.IntEq(t, expRes3, amt))
|
|
||||||
|
|
||||||
// verify the final withdraw amounts are different
|
// calculate delegation rewards for del1
|
||||||
require.True(t, expRes2.GT(expRes3))
|
rewards := k.calculateDelegationRewards(ctx, val, del1, endingPeriod)
|
||||||
|
|
||||||
|
// rewards for del1 should be 3/4 initial
|
||||||
|
require.Equal(t, sdk.DecCoins{{staking.DefaultBondDenom, sdk.NewDec(initial * 3 / 4)}}, rewards)
|
||||||
|
|
||||||
|
// calculate delegation rewards for del2
|
||||||
|
rewards = k.calculateDelegationRewards(ctx, val, del2, endingPeriod)
|
||||||
|
|
||||||
|
// rewards for del2 should be 1/4 initial
|
||||||
|
require.Equal(t, sdk.DecCoins{{staking.DefaultBondDenom, sdk.NewDec(initial * 1 / 4)}}, rewards)
|
||||||
|
|
||||||
|
// commission should be equal to initial (50% twice)
|
||||||
|
require.Equal(t, sdk.DecCoins{{staking.DefaultBondDenom, sdk.NewDec(initial)}}, k.GetValidatorAccumulatedCommission(ctx, valOpAddr1))
|
||||||
}
|
}
|
||||||
|
|
||||||
func TestWithdrawDelegationRewardsAll(t *testing.T) {
|
func TestWithdrawDelegationRewardsBasic(t *testing.T) {
|
||||||
ctx, accMapper, keeper, sk, fck := CreateTestInputAdvanced(t, false, 100, sdk.ZeroDec())
|
balance := int64(1000)
|
||||||
stakingHandler := staking.NewHandler(sk)
|
ctx, ak, k, sk, _ := CreateTestInputDefault(t, false, balance)
|
||||||
denom := sk.GetParams(ctx).BondDenom
|
sh := staking.NewHandler(sk)
|
||||||
|
|
||||||
//make some validators with different commissions
|
// initialize state
|
||||||
msgCreateValidator := staking.NewTestMsgCreateValidatorWithCommission(
|
k.SetOutstandingRewards(ctx, sdk.DecCoins{})
|
||||||
valOpAddr1, valConsPk1, 10, sdk.NewDecWithPrec(1, 1))
|
|
||||||
got := stakingHandler(ctx, msgCreateValidator)
|
|
||||||
require.True(t, got.IsOK(), "expected msg to be ok, got %v", got)
|
|
||||||
|
|
||||||
msgCreateValidator = staking.NewTestMsgCreateValidatorWithCommission(
|
// create validator with 50% commission
|
||||||
valOpAddr2, valConsPk2, 50, sdk.NewDecWithPrec(2, 1))
|
bond := int64(100)
|
||||||
got = stakingHandler(ctx, msgCreateValidator)
|
commission := staking.NewCommissionMsg(sdk.NewDecWithPrec(5, 1), sdk.NewDecWithPrec(5, 1), sdk.NewDec(0))
|
||||||
require.True(t, got.IsOK(), "expected msg to be ok, got %v", got)
|
msg := staking.NewMsgCreateValidator(valOpAddr1, valConsPk1,
|
||||||
|
sdk.NewCoin(staking.DefaultBondDenom, sdk.NewInt(bond)), staking.Description{}, commission)
|
||||||
|
require.True(t, sh(ctx, msg).IsOK())
|
||||||
|
|
||||||
msgCreateValidator = staking.NewTestMsgCreateValidatorWithCommission(
|
// assert correct initial balance
|
||||||
valOpAddr3, valConsPk3, 40, sdk.NewDecWithPrec(3, 1))
|
require.Equal(t, sdk.Coins{{staking.DefaultBondDenom, sdk.NewInt(balance - bond)}}, ak.GetAccount(ctx, sdk.AccAddress(valOpAddr1)).GetCoins())
|
||||||
got = stakingHandler(ctx, msgCreateValidator)
|
|
||||||
require.True(t, got.IsOK(), "expected msg to be ok, got %v", got)
|
|
||||||
|
|
||||||
// delegate to all the validators
|
// end block to bond validator
|
||||||
msgDelegate := staking.NewTestMsgDelegate(delAddr1, valOpAddr1, 10)
|
staking.EndBlocker(ctx, sk)
|
||||||
require.True(t, stakingHandler(ctx, msgDelegate).IsOK())
|
|
||||||
msgDelegate = staking.NewTestMsgDelegate(delAddr1, valOpAddr2, 20)
|
|
||||||
require.True(t, stakingHandler(ctx, msgDelegate).IsOK())
|
|
||||||
msgDelegate = staking.NewTestMsgDelegate(delAddr1, valOpAddr3, 30)
|
|
||||||
require.True(t, stakingHandler(ctx, msgDelegate).IsOK())
|
|
||||||
|
|
||||||
// Update sk's LastValidatorPower/LastTotalPowers.
|
// set zero outstanding rewards
|
||||||
_ = sk.ApplyAndReturnValidatorSetUpdates(ctx)
|
k.SetOutstandingRewards(ctx, sdk.DecCoins{})
|
||||||
|
|
||||||
// 40 tokens left after delegating 60 of them
|
// fetch validator and delegation
|
||||||
amt := accMapper.GetAccount(ctx, delAddr1).GetCoins().AmountOf(denom)
|
val := sk.Validator(ctx, valOpAddr1)
|
||||||
require.Equal(t, int64(40), amt.Int64())
|
|
||||||
|
|
||||||
// total power of each validator:
|
// allocate some rewards
|
||||||
// validator 1: 10 (self) + 10 (delegator) = 20
|
initial := int64(10)
|
||||||
// validator 2: 50 (self) + 20 (delegator) = 70
|
tokens := sdk.DecCoins{{staking.DefaultBondDenom, sdk.NewDec(initial)}}
|
||||||
// validator 3: 40 (self) + 30 (delegator) = 70
|
k.AllocateTokensToValidator(ctx, val, tokens)
|
||||||
// grand total: 160
|
|
||||||
|
|
||||||
// allocate 100 denom of fees
|
// withdraw rewards
|
||||||
feeInputs := sdk.NewInt(1000)
|
require.Nil(t, k.WithdrawDelegationRewards(ctx, sdk.AccAddress(valOpAddr1), valOpAddr1))
|
||||||
fck.SetCollectedFees(sdk.Coins{sdk.NewCoin(denom, feeInputs)})
|
|
||||||
require.Equal(t, feeInputs, fck.GetCollectedFees(ctx).AmountOf(denom))
|
|
||||||
keeper.AllocateTokens(ctx, sdk.OneDec(), valConsAddr1)
|
|
||||||
|
|
||||||
// withdraw delegation
|
// assert correct balance
|
||||||
ctx = ctx.WithBlockHeight(1)
|
require.Equal(t, sdk.Coins{{staking.DefaultBondDenom, sdk.NewInt(balance - bond + (initial / 2))}}, ak.GetAccount(ctx, sdk.AccAddress(valOpAddr1)).GetCoins())
|
||||||
keeper.WithdrawDelegationRewardsAll(ctx, delAddr1)
|
|
||||||
amt = accMapper.GetAccount(ctx, delAddr1).GetCoins().AmountOf(denom)
|
|
||||||
|
|
||||||
// orig-amount + fees *(1-proposerReward)* (val1Portion * delegatorPotion * (1-val1Commission) ... etc)
|
// withdraw commission
|
||||||
// + fees *(proposerReward) * (delegatorPotion * (1-val1Commission))
|
require.Nil(t, k.WithdrawValidatorCommission(ctx, valOpAddr1))
|
||||||
// 40 + 1000 *(1- 0.95)* (20/160 * 10/20 * 0.9 + 70/160 * 20/70 * 0.8 + 70/160 * 30/70 * 0.7)
|
|
||||||
// 40 + 1000 *( 0.05) * (10/20 * 0.9)
|
// assert correct balance
|
||||||
feesInNonProposer := sdk.NewDecFromInt(feeInputs).Mul(sdk.NewDecWithPrec(95, 2))
|
require.Equal(t, sdk.Coins{{staking.DefaultBondDenom, sdk.NewInt(balance - bond + initial)}}, ak.GetAccount(ctx, sdk.AccAddress(valOpAddr1)).GetCoins())
|
||||||
feesInProposer := sdk.NewDecFromInt(feeInputs).Mul(sdk.NewDecWithPrec(5, 2))
|
}
|
||||||
feesInVal1 := feesInNonProposer.Mul(sdk.NewDec(10).Quo(sdk.NewDec(160))).Mul(sdk.NewDecWithPrec(9, 1))
|
|
||||||
feesInVal2 := feesInNonProposer.Mul(sdk.NewDec(20).Quo(sdk.NewDec(160))).Mul(sdk.NewDecWithPrec(8, 1))
|
func TestCalculateRewardsAfterManySlashesInSameBlock(t *testing.T) {
|
||||||
feesInVal3 := feesInNonProposer.Mul(sdk.NewDec(30).Quo(sdk.NewDec(160))).Mul(sdk.NewDecWithPrec(7, 1))
|
ctx, _, k, sk, _ := CreateTestInputDefault(t, false, 1000)
|
||||||
feesInVal1Proposer := feesInProposer.Mul(sdk.NewDec(10).Quo(sdk.NewDec(20))).Mul(sdk.NewDecWithPrec(9, 1))
|
sh := staking.NewHandler(sk)
|
||||||
expRes := sdk.NewDec(40).Add(feesInVal1).Add(feesInVal2).Add(feesInVal3).Add(feesInVal1Proposer).TruncateInt()
|
|
||||||
require.True(sdk.IntEq(t, expRes, amt))
|
// initialize state
|
||||||
|
k.SetOutstandingRewards(ctx, sdk.DecCoins{})
|
||||||
|
|
||||||
|
// create validator with 50% commission
|
||||||
|
commission := staking.NewCommissionMsg(sdk.NewDecWithPrec(5, 1), sdk.NewDecWithPrec(5, 1), sdk.NewDec(0))
|
||||||
|
msg := staking.NewMsgCreateValidator(valOpAddr1, valConsPk1,
|
||||||
|
sdk.NewCoin(staking.DefaultBondDenom, sdk.NewInt(100)), staking.Description{}, commission)
|
||||||
|
require.True(t, sh(ctx, msg).IsOK())
|
||||||
|
|
||||||
|
// end block to bond validator
|
||||||
|
staking.EndBlocker(ctx, sk)
|
||||||
|
|
||||||
|
// fetch validator and delegation
|
||||||
|
val := sk.Validator(ctx, valOpAddr1)
|
||||||
|
del := sk.Delegation(ctx, sdk.AccAddress(valOpAddr1), valOpAddr1)
|
||||||
|
|
||||||
|
// end period
|
||||||
|
endingPeriod := k.incrementValidatorPeriod(ctx, val)
|
||||||
|
|
||||||
|
// calculate delegation rewards
|
||||||
|
rewards := k.calculateDelegationRewards(ctx, val, del, endingPeriod)
|
||||||
|
|
||||||
|
// rewards should be zero
|
||||||
|
require.True(t, rewards.IsZero())
|
||||||
|
|
||||||
|
// start out block height
|
||||||
|
ctx = ctx.WithBlockHeight(ctx.BlockHeight() + 3)
|
||||||
|
|
||||||
|
// allocate some rewards
|
||||||
|
initial := int64(10)
|
||||||
|
tokens := sdk.DecCoins{{staking.DefaultBondDenom, sdk.NewDec(initial)}}
|
||||||
|
k.AllocateTokensToValidator(ctx, val, tokens)
|
||||||
|
|
||||||
|
// slash the validator by 50%
|
||||||
|
sk.Slash(ctx, valConsAddr1, ctx.BlockHeight(), 100, sdk.NewDecWithPrec(5, 1))
|
||||||
|
|
||||||
|
// slash the validator by 50% again
|
||||||
|
sk.Slash(ctx, valConsAddr1, ctx.BlockHeight(), 50, sdk.NewDecWithPrec(5, 1))
|
||||||
|
|
||||||
|
// fetch the validator again
|
||||||
|
val = sk.Validator(ctx, valOpAddr1)
|
||||||
|
|
||||||
|
// increase block height
|
||||||
|
ctx = ctx.WithBlockHeight(ctx.BlockHeight() + 3)
|
||||||
|
|
||||||
|
// allocate some more rewards
|
||||||
|
k.AllocateTokensToValidator(ctx, val, tokens)
|
||||||
|
|
||||||
|
// end period
|
||||||
|
endingPeriod = k.incrementValidatorPeriod(ctx, val)
|
||||||
|
|
||||||
|
// calculate delegation rewards
|
||||||
|
rewards = k.calculateDelegationRewards(ctx, val, del, endingPeriod)
|
||||||
|
|
||||||
|
// rewards should be half the tokens
|
||||||
|
require.Equal(t, sdk.DecCoins{{staking.DefaultBondDenom, sdk.NewDec(initial)}}, rewards)
|
||||||
|
|
||||||
|
// commission should be the other half
|
||||||
|
require.Equal(t, sdk.DecCoins{{staking.DefaultBondDenom, sdk.NewDec(initial)}}, k.GetValidatorAccumulatedCommission(ctx, valOpAddr1))
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestCalculateRewardsMultiDelegatorMultiSlash(t *testing.T) {
|
||||||
|
ctx, _, k, sk, _ := CreateTestInputDefault(t, false, 1000)
|
||||||
|
sh := staking.NewHandler(sk)
|
||||||
|
|
||||||
|
// initialize state
|
||||||
|
k.SetOutstandingRewards(ctx, sdk.DecCoins{})
|
||||||
|
|
||||||
|
// create validator with 50% commission
|
||||||
|
commission := staking.NewCommissionMsg(sdk.NewDecWithPrec(5, 1), sdk.NewDecWithPrec(5, 1), sdk.NewDec(0))
|
||||||
|
msg := staking.NewMsgCreateValidator(valOpAddr1, valConsPk1,
|
||||||
|
sdk.NewCoin(staking.DefaultBondDenom, sdk.NewInt(100)), staking.Description{}, commission)
|
||||||
|
require.True(t, sh(ctx, msg).IsOK())
|
||||||
|
|
||||||
|
// end block to bond validator
|
||||||
|
staking.EndBlocker(ctx, sk)
|
||||||
|
|
||||||
|
// fetch validator and delegation
|
||||||
|
val := sk.Validator(ctx, valOpAddr1)
|
||||||
|
del1 := sk.Delegation(ctx, sdk.AccAddress(valOpAddr1), valOpAddr1)
|
||||||
|
|
||||||
|
// allocate some rewards
|
||||||
|
initial := int64(30)
|
||||||
|
tokens := sdk.DecCoins{{staking.DefaultBondDenom, sdk.NewDec(initial)}}
|
||||||
|
k.AllocateTokensToValidator(ctx, val, tokens)
|
||||||
|
|
||||||
|
// slash the validator
|
||||||
|
ctx = ctx.WithBlockHeight(ctx.BlockHeight() + 3)
|
||||||
|
sk.Slash(ctx, valConsAddr1, ctx.BlockHeight(), 100, sdk.NewDecWithPrec(5, 1))
|
||||||
|
ctx = ctx.WithBlockHeight(ctx.BlockHeight() + 3)
|
||||||
|
|
||||||
|
// second delegation
|
||||||
|
msg2 := staking.NewMsgDelegate(sdk.AccAddress(valOpAddr2), valOpAddr1, sdk.NewCoin(staking.DefaultBondDenom, sdk.NewInt(100)))
|
||||||
|
require.True(t, sh(ctx, msg2).IsOK())
|
||||||
|
del2 := sk.Delegation(ctx, sdk.AccAddress(valOpAddr2), valOpAddr1)
|
||||||
|
|
||||||
|
// end block
|
||||||
|
staking.EndBlocker(ctx, sk)
|
||||||
|
|
||||||
|
// allocate some more rewards
|
||||||
|
k.AllocateTokensToValidator(ctx, val, tokens)
|
||||||
|
|
||||||
|
// slash the validator again
|
||||||
|
ctx = ctx.WithBlockHeight(ctx.BlockHeight() + 3)
|
||||||
|
sk.Slash(ctx, valConsAddr1, ctx.BlockHeight(), 100, sdk.NewDecWithPrec(5, 1))
|
||||||
|
ctx = ctx.WithBlockHeight(ctx.BlockHeight() + 3)
|
||||||
|
|
||||||
|
// fetch updated validator
|
||||||
|
val = sk.Validator(ctx, valOpAddr1)
|
||||||
|
|
||||||
|
// end period
|
||||||
|
endingPeriod := k.incrementValidatorPeriod(ctx, val)
|
||||||
|
|
||||||
|
// calculate delegation rewards for del1
|
||||||
|
rewards := k.calculateDelegationRewards(ctx, val, del1, endingPeriod)
|
||||||
|
|
||||||
|
// rewards for del1 should be 2/3 initial (half initial first period, 1/6 initial second period)
|
||||||
|
require.Equal(t, sdk.DecCoins{{staking.DefaultBondDenom, sdk.NewDec((initial / 2) + (initial / 6))}}, rewards)
|
||||||
|
|
||||||
|
// calculate delegation rewards for del2
|
||||||
|
rewards = k.calculateDelegationRewards(ctx, val, del2, endingPeriod)
|
||||||
|
|
||||||
|
// rewards for del2 should be initial / 3
|
||||||
|
require.Equal(t, sdk.DecCoins{{staking.DefaultBondDenom, sdk.NewDec(initial / 3)}}, rewards)
|
||||||
|
|
||||||
|
// commission should be equal to initial (twice 50% commission, unaffected by slashing)
|
||||||
|
require.Equal(t, sdk.DecCoins{{staking.DefaultBondDenom, sdk.NewDec(initial)}}, k.GetValidatorAccumulatedCommission(ctx, valOpAddr1))
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestCalculateRewardsMultiDelegatorMultWithdraw(t *testing.T) {
|
||||||
|
ctx, _, k, sk, _ := CreateTestInputDefault(t, false, 1000)
|
||||||
|
sh := staking.NewHandler(sk)
|
||||||
|
|
||||||
|
// initialize state
|
||||||
|
k.SetOutstandingRewards(ctx, sdk.DecCoins{})
|
||||||
|
|
||||||
|
// create validator with 50% commission
|
||||||
|
commission := staking.NewCommissionMsg(sdk.NewDecWithPrec(5, 1), sdk.NewDecWithPrec(5, 1), sdk.NewDec(0))
|
||||||
|
msg := staking.NewMsgCreateValidator(valOpAddr1, valConsPk1,
|
||||||
|
sdk.NewCoin(staking.DefaultBondDenom, sdk.NewInt(100)), staking.Description{}, commission)
|
||||||
|
require.True(t, sh(ctx, msg).IsOK())
|
||||||
|
|
||||||
|
// end block to bond validator
|
||||||
|
staking.EndBlocker(ctx, sk)
|
||||||
|
|
||||||
|
// fetch validator and delegation
|
||||||
|
val := sk.Validator(ctx, valOpAddr1)
|
||||||
|
del1 := sk.Delegation(ctx, sdk.AccAddress(valOpAddr1), valOpAddr1)
|
||||||
|
|
||||||
|
// allocate some rewards
|
||||||
|
initial := int64(20)
|
||||||
|
tokens := sdk.DecCoins{{staking.DefaultBondDenom, sdk.NewDec(initial)}}
|
||||||
|
k.AllocateTokensToValidator(ctx, val, tokens)
|
||||||
|
|
||||||
|
// second delegation
|
||||||
|
msg2 := staking.NewMsgDelegate(sdk.AccAddress(valOpAddr2), valOpAddr1, sdk.NewCoin(staking.DefaultBondDenom, sdk.NewInt(100)))
|
||||||
|
require.True(t, sh(ctx, msg2).IsOK())
|
||||||
|
|
||||||
|
// fetch updated validator
|
||||||
|
val = sk.Validator(ctx, valOpAddr1)
|
||||||
|
del2 := sk.Delegation(ctx, sdk.AccAddress(valOpAddr2), valOpAddr1)
|
||||||
|
|
||||||
|
// end block
|
||||||
|
staking.EndBlocker(ctx, sk)
|
||||||
|
|
||||||
|
// allocate some more rewards
|
||||||
|
k.AllocateTokensToValidator(ctx, val, tokens)
|
||||||
|
|
||||||
|
// first delegator withdraws
|
||||||
|
k.WithdrawDelegationRewards(ctx, sdk.AccAddress(valOpAddr1), valOpAddr1)
|
||||||
|
|
||||||
|
// second delegator withdraws
|
||||||
|
k.WithdrawDelegationRewards(ctx, sdk.AccAddress(valOpAddr2), valOpAddr1)
|
||||||
|
|
||||||
|
// validator withdraws commission
|
||||||
|
k.WithdrawValidatorCommission(ctx, valOpAddr1)
|
||||||
|
|
||||||
|
// end period
|
||||||
|
endingPeriod := k.incrementValidatorPeriod(ctx, val)
|
||||||
|
|
||||||
|
// calculate delegation rewards for del1
|
||||||
|
rewards := k.calculateDelegationRewards(ctx, val, del1, endingPeriod)
|
||||||
|
|
||||||
|
// rewards for del1 should be zero
|
||||||
|
require.True(t, rewards.IsZero())
|
||||||
|
|
||||||
|
// calculate delegation rewards for del2
|
||||||
|
rewards = k.calculateDelegationRewards(ctx, val, del2, endingPeriod)
|
||||||
|
|
||||||
|
// rewards for del2 should be zero
|
||||||
|
require.True(t, rewards.IsZero())
|
||||||
|
|
||||||
|
// commission should be zero
|
||||||
|
require.True(t, k.GetValidatorAccumulatedCommission(ctx, valOpAddr1).IsZero())
|
||||||
|
|
||||||
|
// allocate some more rewards
|
||||||
|
k.AllocateTokensToValidator(ctx, val, tokens)
|
||||||
|
|
||||||
|
// first delegator withdraws again
|
||||||
|
k.WithdrawDelegationRewards(ctx, sdk.AccAddress(valOpAddr1), valOpAddr1)
|
||||||
|
|
||||||
|
// end period
|
||||||
|
endingPeriod = k.incrementValidatorPeriod(ctx, val)
|
||||||
|
|
||||||
|
// calculate delegation rewards for del1
|
||||||
|
rewards = k.calculateDelegationRewards(ctx, val, del1, endingPeriod)
|
||||||
|
|
||||||
|
// rewards for del1 should be zero
|
||||||
|
require.True(t, rewards.IsZero())
|
||||||
|
|
||||||
|
// calculate delegation rewards for del2
|
||||||
|
rewards = k.calculateDelegationRewards(ctx, val, del2, endingPeriod)
|
||||||
|
|
||||||
|
// rewards for del2 should be 1/4 initial
|
||||||
|
require.Equal(t, sdk.DecCoins{{staking.DefaultBondDenom, sdk.NewDec(initial / 4)}}, rewards)
|
||||||
|
|
||||||
|
// commission should be half initial
|
||||||
|
require.Equal(t, sdk.DecCoins{{staking.DefaultBondDenom, sdk.NewDec(initial / 2)}}, k.GetValidatorAccumulatedCommission(ctx, valOpAddr1))
|
||||||
|
|
||||||
|
// allocate some more rewards
|
||||||
|
k.AllocateTokensToValidator(ctx, val, tokens)
|
||||||
|
|
||||||
|
// withdraw commission
|
||||||
|
k.WithdrawValidatorCommission(ctx, valOpAddr1)
|
||||||
|
|
||||||
|
// end period
|
||||||
|
endingPeriod = k.incrementValidatorPeriod(ctx, val)
|
||||||
|
|
||||||
|
// calculate delegation rewards for del1
|
||||||
|
rewards = k.calculateDelegationRewards(ctx, val, del1, endingPeriod)
|
||||||
|
|
||||||
|
// rewards for del1 should be 1/4 initial
|
||||||
|
require.Equal(t, sdk.DecCoins{{staking.DefaultBondDenom, sdk.NewDec(initial / 4)}}, rewards)
|
||||||
|
|
||||||
|
// calculate delegation rewards for del2
|
||||||
|
rewards = k.calculateDelegationRewards(ctx, val, del2, endingPeriod)
|
||||||
|
|
||||||
|
// rewards for del2 should be 1/2 initial
|
||||||
|
require.Equal(t, sdk.DecCoins{{staking.DefaultBondDenom, sdk.NewDec(initial / 2)}}, rewards)
|
||||||
|
|
||||||
|
// commission should be zero
|
||||||
|
require.True(t, k.GetValidatorAccumulatedCommission(ctx, valOpAddr1).IsZero())
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,50 +0,0 @@
|
||||||
package keeper
|
|
||||||
|
|
||||||
import (
|
|
||||||
sdk "github.com/cosmos/cosmos-sdk/types"
|
|
||||||
"github.com/cosmos/cosmos-sdk/x/distribution/types"
|
|
||||||
)
|
|
||||||
|
|
||||||
// Get the set of all validator-distribution-info's with no limits, used during genesis dump
|
|
||||||
func (k Keeper) GetAllValidatorDistInfos(ctx sdk.Context) (vdis []types.ValidatorDistInfo) {
|
|
||||||
store := ctx.KVStore(k.storeKey)
|
|
||||||
iterator := sdk.KVStorePrefixIterator(store, ValidatorDistInfoKey)
|
|
||||||
defer iterator.Close()
|
|
||||||
|
|
||||||
for ; iterator.Valid(); iterator.Next() {
|
|
||||||
var vdi types.ValidatorDistInfo
|
|
||||||
k.cdc.MustUnmarshalBinaryLengthPrefixed(iterator.Value(), &vdi)
|
|
||||||
vdis = append(vdis, vdi)
|
|
||||||
}
|
|
||||||
return vdis
|
|
||||||
}
|
|
||||||
|
|
||||||
// Get the set of all delegator-distribution-info's with no limits, used during genesis dump
|
|
||||||
func (k Keeper) GetAllDelegationDistInfos(ctx sdk.Context) (ddis []types.DelegationDistInfo) {
|
|
||||||
store := ctx.KVStore(k.storeKey)
|
|
||||||
iterator := sdk.KVStorePrefixIterator(store, DelegationDistInfoKey)
|
|
||||||
defer iterator.Close()
|
|
||||||
|
|
||||||
for ; iterator.Valid(); iterator.Next() {
|
|
||||||
var ddi types.DelegationDistInfo
|
|
||||||
k.cdc.MustUnmarshalBinaryLengthPrefixed(iterator.Value(), &ddi)
|
|
||||||
ddis = append(ddis, ddi)
|
|
||||||
}
|
|
||||||
return ddis
|
|
||||||
}
|
|
||||||
|
|
||||||
// Get the set of all delegator-withdraw addresses with no limits, used during genesis dump
|
|
||||||
func (k Keeper) GetAllDelegatorWithdrawInfos(ctx sdk.Context) (dwis []types.DelegatorWithdrawInfo) {
|
|
||||||
store := ctx.KVStore(k.storeKey)
|
|
||||||
iterator := sdk.KVStorePrefixIterator(store, DelegatorWithdrawInfoKey)
|
|
||||||
defer iterator.Close()
|
|
||||||
|
|
||||||
for ; iterator.Valid(); iterator.Next() {
|
|
||||||
dw := types.DelegatorWithdrawInfo{
|
|
||||||
DelegatorAddr: GetDelegatorWithdrawInfoAddress(iterator.Key()),
|
|
||||||
WithdrawAddr: sdk.AccAddress(iterator.Value()),
|
|
||||||
}
|
|
||||||
dwis = append(dwis, dw)
|
|
||||||
}
|
|
||||||
return dwis
|
|
||||||
}
|
|
|
@ -1,109 +1,9 @@
|
||||||
package keeper
|
package keeper
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"fmt"
|
|
||||||
|
|
||||||
sdk "github.com/cosmos/cosmos-sdk/types"
|
sdk "github.com/cosmos/cosmos-sdk/types"
|
||||||
"github.com/cosmos/cosmos-sdk/x/distribution/types"
|
|
||||||
)
|
)
|
||||||
|
|
||||||
// Create a new validator distribution record
|
|
||||||
func (k Keeper) AfterValidatorCreated(ctx sdk.Context, valAddr sdk.ValAddress) {
|
|
||||||
|
|
||||||
// defensive check for existence
|
|
||||||
if k.HasValidatorDistInfo(ctx, valAddr) {
|
|
||||||
panic("validator dist info already exists (not cleaned up properly)")
|
|
||||||
}
|
|
||||||
|
|
||||||
height := ctx.BlockHeight()
|
|
||||||
vdi := types.ValidatorDistInfo{
|
|
||||||
OperatorAddr: valAddr,
|
|
||||||
FeePoolWithdrawalHeight: height,
|
|
||||||
DelAccum: types.NewTotalAccum(height),
|
|
||||||
DelPool: types.DecCoins{},
|
|
||||||
ValCommission: types.DecCoins{},
|
|
||||||
}
|
|
||||||
k.SetValidatorDistInfo(ctx, vdi)
|
|
||||||
}
|
|
||||||
|
|
||||||
// Withdraw all validator rewards
|
|
||||||
func (k Keeper) BeforeValidatorModified(ctx sdk.Context, valAddr sdk.ValAddress) {
|
|
||||||
// Move the validator's rewards from the global pool to the validator's pools
|
|
||||||
// (dist info), but without actually withdrawing the rewards. This does not
|
|
||||||
// need to happen during the genesis block.
|
|
||||||
if ctx.BlockHeight() > 0 {
|
|
||||||
if err := k.takeValidatorFeePoolRewards(ctx, valAddr); err != nil {
|
|
||||||
panic(err)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// Withdraw all validator rewards
|
|
||||||
func (k Keeper) AfterValidatorBonded(ctx sdk.Context, valAddr sdk.ValAddress) {
|
|
||||||
lastPower := k.stakingKeeper.GetLastValidatorPower(ctx, valAddr)
|
|
||||||
if !lastPower.Equal(sdk.ZeroInt()) {
|
|
||||||
panic("expected last power to be 0 for validator entering bonded state")
|
|
||||||
}
|
|
||||||
k.BeforeValidatorModified(ctx, valAddr)
|
|
||||||
}
|
|
||||||
|
|
||||||
// Sanity check, very useful!
|
|
||||||
func (k Keeper) AfterValidatorPowerDidChange(ctx sdk.Context, valAddr sdk.ValAddress) {
|
|
||||||
vi := k.GetValidatorDistInfo(ctx, valAddr)
|
|
||||||
if vi.FeePoolWithdrawalHeight != ctx.BlockHeight() {
|
|
||||||
panic(fmt.Sprintf("expected validator (%v) dist info FeePoolWithdrawalHeight to be updated to %v, but was %v.",
|
|
||||||
valAddr.String(), ctx.BlockHeight(), vi.FeePoolWithdrawalHeight))
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// Withdrawal all validator distribution rewards and cleanup the distribution record
|
|
||||||
func (k Keeper) AfterValidatorRemoved(ctx sdk.Context, valAddr sdk.ValAddress) {
|
|
||||||
k.RemoveValidatorDistInfo(ctx, valAddr)
|
|
||||||
}
|
|
||||||
|
|
||||||
//_________________________________________________________________________________________
|
|
||||||
|
|
||||||
// Create a new delegator distribution record
|
|
||||||
func (k Keeper) BeforeDelegationCreated(ctx sdk.Context, delAddr sdk.AccAddress,
|
|
||||||
valAddr sdk.ValAddress) {
|
|
||||||
|
|
||||||
ddi := types.DelegationDistInfo{
|
|
||||||
DelegatorAddr: delAddr,
|
|
||||||
ValOperatorAddr: valAddr,
|
|
||||||
DelPoolWithdrawalHeight: ctx.BlockHeight(),
|
|
||||||
}
|
|
||||||
k.SetDelegationDistInfo(ctx, ddi)
|
|
||||||
}
|
|
||||||
|
|
||||||
// Withdrawal all validator rewards
|
|
||||||
func (k Keeper) BeforeDelegationSharesModified(ctx sdk.Context, delAddr sdk.AccAddress,
|
|
||||||
valAddr sdk.ValAddress) {
|
|
||||||
|
|
||||||
if err := k.WithdrawDelegationReward(ctx, delAddr, valAddr); err != nil {
|
|
||||||
panic(err)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// Withdrawal all validator distribution rewards and cleanup the distribution record
|
|
||||||
func (k Keeper) BeforeDelegationRemoved(ctx sdk.Context, delAddr sdk.AccAddress,
|
|
||||||
valAddr sdk.ValAddress) {
|
|
||||||
// Withdraw validator commission when validator self-bond is removed.
|
|
||||||
// Because we maintain the invariant that all delegations must be removed
|
|
||||||
// before a validator is deleted, this ensures that commission will be withdrawn
|
|
||||||
// before the validator is deleted (and the corresponding ValidatorDistInfo removed).
|
|
||||||
// If we change other parts of the code such that a self-delegation might remain after
|
|
||||||
// a validator is deleted, this logic will no longer be safe.
|
|
||||||
// TODO: Consider instead implementing this in a "BeforeValidatorRemoved" hook.
|
|
||||||
if valAddr.Equals(sdk.ValAddress(delAddr)) {
|
|
||||||
feePool, commission := k.withdrawValidatorCommission(ctx, valAddr)
|
|
||||||
k.WithdrawToDelegator(ctx, feePool, delAddr, commission)
|
|
||||||
}
|
|
||||||
|
|
||||||
k.RemoveDelegationDistInfo(ctx, delAddr, valAddr)
|
|
||||||
}
|
|
||||||
|
|
||||||
//_________________________________________________________________________________________
|
|
||||||
|
|
||||||
// Wrapper struct
|
// Wrapper struct
|
||||||
type Hooks struct {
|
type Hooks struct {
|
||||||
k Keeper
|
k Keeper
|
||||||
|
@ -111,36 +11,50 @@ type Hooks struct {
|
||||||
|
|
||||||
var _ sdk.StakingHooks = Hooks{}
|
var _ sdk.StakingHooks = Hooks{}
|
||||||
|
|
||||||
// New Validator Hooks
|
// Create new distribution hooks
|
||||||
func (k Keeper) Hooks() Hooks { return Hooks{k} }
|
func (k Keeper) Hooks() Hooks { return Hooks{k} }
|
||||||
|
|
||||||
// nolint
|
// nolint
|
||||||
func (h Hooks) AfterValidatorCreated(ctx sdk.Context, valAddr sdk.ValAddress) {
|
func (h Hooks) AfterValidatorCreated(ctx sdk.Context, valAddr sdk.ValAddress) {
|
||||||
h.k.AfterValidatorCreated(ctx, valAddr)
|
val := h.k.stakingKeeper.Validator(ctx, valAddr)
|
||||||
|
h.k.initializeValidator(ctx, val)
|
||||||
}
|
}
|
||||||
func (h Hooks) BeforeValidatorModified(ctx sdk.Context, valAddr sdk.ValAddress) {
|
func (h Hooks) BeforeValidatorModified(ctx sdk.Context, valAddr sdk.ValAddress) {
|
||||||
h.k.BeforeValidatorModified(ctx, valAddr)
|
val := h.k.stakingKeeper.Validator(ctx, valAddr)
|
||||||
|
// increment period
|
||||||
|
h.k.incrementValidatorPeriod(ctx, val)
|
||||||
}
|
}
|
||||||
func (h Hooks) AfterValidatorRemoved(ctx sdk.Context, _ sdk.ConsAddress, valAddr sdk.ValAddress) {
|
func (h Hooks) AfterValidatorRemoved(ctx sdk.Context, _ sdk.ConsAddress, valAddr sdk.ValAddress) {
|
||||||
h.k.AfterValidatorRemoved(ctx, valAddr)
|
|
||||||
}
|
}
|
||||||
func (h Hooks) BeforeDelegationCreated(ctx sdk.Context, delAddr sdk.AccAddress, valAddr sdk.ValAddress) {
|
func (h Hooks) BeforeDelegationCreated(ctx sdk.Context, delAddr sdk.AccAddress, valAddr sdk.ValAddress) {
|
||||||
h.k.BeforeValidatorModified(ctx, valAddr)
|
val := h.k.stakingKeeper.Validator(ctx, valAddr)
|
||||||
h.k.BeforeDelegationCreated(ctx, delAddr, valAddr)
|
|
||||||
|
// increment period
|
||||||
|
h.k.incrementValidatorPeriod(ctx, val)
|
||||||
}
|
}
|
||||||
func (h Hooks) BeforeDelegationSharesModified(ctx sdk.Context, delAddr sdk.AccAddress, valAddr sdk.ValAddress) {
|
func (h Hooks) BeforeDelegationSharesModified(ctx sdk.Context, delAddr sdk.AccAddress, valAddr sdk.ValAddress) {
|
||||||
h.k.BeforeValidatorModified(ctx, valAddr)
|
val := h.k.stakingKeeper.Validator(ctx, valAddr)
|
||||||
h.k.BeforeDelegationSharesModified(ctx, delAddr, valAddr)
|
del := h.k.stakingKeeper.Delegation(ctx, delAddr, valAddr)
|
||||||
|
|
||||||
|
// withdraw delegation rewards (which also increments period)
|
||||||
|
if err := h.k.withdrawDelegationRewards(ctx, val, del); err != nil {
|
||||||
|
panic(err)
|
||||||
|
}
|
||||||
}
|
}
|
||||||
func (h Hooks) BeforeDelegationRemoved(ctx sdk.Context, delAddr sdk.AccAddress, valAddr sdk.ValAddress) {
|
func (h Hooks) BeforeDelegationRemoved(ctx sdk.Context, delAddr sdk.AccAddress, valAddr sdk.ValAddress) {
|
||||||
h.k.BeforeDelegationRemoved(ctx, delAddr, valAddr)
|
// nothing needed here since OnDelegationSharesModified will always also be called
|
||||||
|
}
|
||||||
|
func (h Hooks) AfterDelegationModified(ctx sdk.Context, delAddr sdk.AccAddress, valAddr sdk.ValAddress) {
|
||||||
|
// create new delegation period record
|
||||||
|
h.k.initializeDelegation(ctx, valAddr, delAddr)
|
||||||
}
|
}
|
||||||
func (h Hooks) AfterValidatorBeginUnbonding(ctx sdk.Context, _ sdk.ConsAddress, valAddr sdk.ValAddress) {
|
func (h Hooks) AfterValidatorBeginUnbonding(ctx sdk.Context, _ sdk.ConsAddress, valAddr sdk.ValAddress) {
|
||||||
h.k.BeforeValidatorModified(ctx, valAddr)
|
|
||||||
}
|
}
|
||||||
func (h Hooks) AfterValidatorBonded(ctx sdk.Context, _ sdk.ConsAddress, valAddr sdk.ValAddress) {
|
func (h Hooks) AfterValidatorBonded(ctx sdk.Context, _ sdk.ConsAddress, valAddr sdk.ValAddress) {
|
||||||
h.k.AfterValidatorBonded(ctx, valAddr)
|
|
||||||
}
|
}
|
||||||
func (h Hooks) AfterValidatorPowerDidChange(ctx sdk.Context, _ sdk.ConsAddress, valAddr sdk.ValAddress) {
|
func (h Hooks) AfterValidatorPowerDidChange(ctx sdk.Context, _ sdk.ConsAddress, valAddr sdk.ValAddress) {
|
||||||
h.k.AfterValidatorPowerDidChange(ctx, valAddr)
|
}
|
||||||
|
func (h Hooks) BeforeValidatorSlashed(ctx sdk.Context, valAddr sdk.ValAddress, fraction sdk.Dec) {
|
||||||
|
// record the slash event
|
||||||
|
h.k.updateValidatorSlashFraction(ctx, valAddr, fraction)
|
||||||
}
|
}
|
||||||
|
|
|
@ -20,9 +20,9 @@ type Keeper struct {
|
||||||
codespace sdk.CodespaceType
|
codespace sdk.CodespaceType
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// create a new keeper
|
||||||
func NewKeeper(cdc *codec.Codec, key sdk.StoreKey, paramSpace params.Subspace, ck types.BankKeeper,
|
func NewKeeper(cdc *codec.Codec, key sdk.StoreKey, paramSpace params.Subspace, ck types.BankKeeper,
|
||||||
sk types.StakingKeeper, fck types.FeeCollectionKeeper, codespace sdk.CodespaceType) Keeper {
|
sk types.StakingKeeper, fck types.FeeCollectionKeeper, codespace sdk.CodespaceType) Keeper {
|
||||||
|
|
||||||
keeper := Keeper{
|
keeper := Keeper{
|
||||||
storeKey: key,
|
storeKey: key,
|
||||||
cdc: cdc,
|
cdc: cdc,
|
||||||
|
@ -35,123 +35,53 @@ func NewKeeper(cdc *codec.Codec, key sdk.StoreKey, paramSpace params.Subspace, c
|
||||||
return keeper
|
return keeper
|
||||||
}
|
}
|
||||||
|
|
||||||
//______________________________________________________________________
|
// withdraw rewards from a delegation
|
||||||
|
func (k Keeper) WithdrawDelegationRewards(ctx sdk.Context, delAddr sdk.AccAddress, valAddr sdk.ValAddress) sdk.Error {
|
||||||
// get the global fee pool distribution info
|
val := k.stakingKeeper.Validator(ctx, valAddr)
|
||||||
func (k Keeper) GetFeePool(ctx sdk.Context) (feePool types.FeePool) {
|
if val == nil {
|
||||||
store := ctx.KVStore(k.storeKey)
|
return types.ErrNoValidatorDistInfo(k.codespace)
|
||||||
b := store.Get(FeePoolKey)
|
|
||||||
if b == nil {
|
|
||||||
panic("Stored fee pool should not have been nil")
|
|
||||||
}
|
|
||||||
k.cdc.MustUnmarshalBinaryLengthPrefixed(b, &feePool)
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
// set the global fee pool distribution info
|
|
||||||
func (k Keeper) SetFeePool(ctx sdk.Context, feePool types.FeePool) {
|
|
||||||
store := ctx.KVStore(k.storeKey)
|
|
||||||
b := k.cdc.MustMarshalBinaryLengthPrefixed(feePool)
|
|
||||||
store.Set(FeePoolKey, b)
|
|
||||||
}
|
|
||||||
|
|
||||||
// get the total validator accum for the ctx height
|
|
||||||
// in the fee pool
|
|
||||||
func (k Keeper) GetFeePoolValAccum(ctx sdk.Context) sdk.Dec {
|
|
||||||
|
|
||||||
// withdraw self-delegation
|
|
||||||
height := ctx.BlockHeight()
|
|
||||||
totalPower := sdk.NewDecFromInt(k.stakingKeeper.GetLastTotalPower(ctx))
|
|
||||||
fp := k.GetFeePool(ctx)
|
|
||||||
return fp.GetTotalValAccum(height, totalPower)
|
|
||||||
}
|
|
||||||
|
|
||||||
//______________________________________________________________________
|
|
||||||
|
|
||||||
// set the proposer public key for this block
|
|
||||||
func (k Keeper) GetPreviousProposerConsAddr(ctx sdk.Context) (consAddr sdk.ConsAddress) {
|
|
||||||
store := ctx.KVStore(k.storeKey)
|
|
||||||
|
|
||||||
b := store.Get(ProposerKey)
|
|
||||||
if b == nil {
|
|
||||||
panic("Previous proposer not set")
|
|
||||||
}
|
}
|
||||||
|
|
||||||
k.cdc.MustUnmarshalBinaryLengthPrefixed(b, &consAddr)
|
del := k.stakingKeeper.Delegation(ctx, delAddr, valAddr)
|
||||||
return
|
if del == nil {
|
||||||
|
return types.ErrNoDelegationDistInfo(k.codespace)
|
||||||
|
}
|
||||||
|
|
||||||
|
// withdraw rewards
|
||||||
|
if err := k.withdrawDelegationRewards(ctx, val, del); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
// reinitialize the delegation
|
||||||
|
k.initializeDelegation(ctx, valAddr, delAddr)
|
||||||
|
|
||||||
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
// get the proposer public key for this block
|
// withdraw validator commission
|
||||||
func (k Keeper) SetPreviousProposerConsAddr(ctx sdk.Context, consAddr sdk.ConsAddress) {
|
func (k Keeper) WithdrawValidatorCommission(ctx sdk.Context, valAddr sdk.ValAddress) sdk.Error {
|
||||||
store := ctx.KVStore(k.storeKey)
|
|
||||||
b := k.cdc.MustMarshalBinaryLengthPrefixed(consAddr)
|
// fetch validator accumulated commission
|
||||||
store.Set(ProposerKey, b)
|
commission := k.GetValidatorAccumulatedCommission(ctx, valAddr)
|
||||||
}
|
if commission.IsZero() {
|
||||||
|
return types.ErrNoValidatorCommission(k.codespace)
|
||||||
//______________________________________________________________________
|
}
|
||||||
|
|
||||||
// get context required for withdraw operations
|
coins, remainder := commission.TruncateDecimal()
|
||||||
func (k Keeper) GetWithdrawContext(ctx sdk.Context,
|
|
||||||
valOperatorAddr sdk.ValAddress) types.WithdrawContext {
|
// leave remainder to withdraw later
|
||||||
|
k.SetValidatorAccumulatedCommission(ctx, valAddr, remainder)
|
||||||
feePool := k.GetFeePool(ctx)
|
|
||||||
height := ctx.BlockHeight()
|
// update outstanding
|
||||||
validator := k.stakingKeeper.Validator(ctx, valOperatorAddr)
|
outstanding := k.GetOutstandingRewards(ctx)
|
||||||
lastValPower := k.stakingKeeper.GetLastValidatorPower(ctx, valOperatorAddr)
|
k.SetOutstandingRewards(ctx, outstanding.Minus(sdk.NewDecCoins(coins)))
|
||||||
lastTotalPower := sdk.NewDecFromInt(k.stakingKeeper.GetLastTotalPower(ctx))
|
|
||||||
|
accAddr := sdk.AccAddress(valAddr)
|
||||||
return types.NewWithdrawContext(
|
withdrawAddr := k.GetDelegatorWithdrawAddr(ctx, accAddr)
|
||||||
feePool, height, lastTotalPower, sdk.NewDecFromInt(lastValPower),
|
|
||||||
validator.GetCommission())
|
if _, _, err := k.bankKeeper.AddCoins(ctx, withdrawAddr, coins); err != nil {
|
||||||
}
|
return err
|
||||||
|
}
|
||||||
//______________________________________________________________________
|
|
||||||
// PARAM STORE
|
return nil
|
||||||
|
|
||||||
// Type declaration for parameters
|
|
||||||
func ParamTypeTable() params.TypeTable {
|
|
||||||
return params.NewTypeTable(
|
|
||||||
ParamStoreKeyCommunityTax, sdk.Dec{},
|
|
||||||
ParamStoreKeyBaseProposerReward, sdk.Dec{},
|
|
||||||
ParamStoreKeyBonusProposerReward, sdk.Dec{},
|
|
||||||
)
|
|
||||||
}
|
|
||||||
|
|
||||||
// Returns the current CommunityTax rate from the global param store
|
|
||||||
// nolint: errcheck
|
|
||||||
func (k Keeper) GetCommunityTax(ctx sdk.Context) sdk.Dec {
|
|
||||||
var percent sdk.Dec
|
|
||||||
k.paramSpace.Get(ctx, ParamStoreKeyCommunityTax, &percent)
|
|
||||||
return percent
|
|
||||||
}
|
|
||||||
|
|
||||||
// nolint: errcheck
|
|
||||||
func (k Keeper) SetCommunityTax(ctx sdk.Context, percent sdk.Dec) {
|
|
||||||
k.paramSpace.Set(ctx, ParamStoreKeyCommunityTax, &percent)
|
|
||||||
}
|
|
||||||
|
|
||||||
// Returns the current BaseProposerReward rate from the global param store
|
|
||||||
// nolint: errcheck
|
|
||||||
func (k Keeper) GetBaseProposerReward(ctx sdk.Context) sdk.Dec {
|
|
||||||
var percent sdk.Dec
|
|
||||||
k.paramSpace.Get(ctx, ParamStoreKeyBaseProposerReward, &percent)
|
|
||||||
return percent
|
|
||||||
}
|
|
||||||
|
|
||||||
// nolint: errcheck
|
|
||||||
func (k Keeper) SetBaseProposerReward(ctx sdk.Context, percent sdk.Dec) {
|
|
||||||
k.paramSpace.Set(ctx, ParamStoreKeyBaseProposerReward, &percent)
|
|
||||||
}
|
|
||||||
|
|
||||||
// Returns the current BaseProposerReward rate from the global param store
|
|
||||||
// nolint: errcheck
|
|
||||||
func (k Keeper) GetBonusProposerReward(ctx sdk.Context) sdk.Dec {
|
|
||||||
var percent sdk.Dec
|
|
||||||
k.paramSpace.Get(ctx, ParamStoreKeyBonusProposerReward, &percent)
|
|
||||||
return percent
|
|
||||||
}
|
|
||||||
|
|
||||||
// nolint: errcheck
|
|
||||||
func (k Keeper) SetBonusProposerReward(ctx sdk.Context, percent sdk.Dec) {
|
|
||||||
k.paramSpace.Set(ctx, ParamStoreKeyBonusProposerReward, &percent)
|
|
||||||
}
|
}
|
||||||
|
|
|
@ -9,30 +9,40 @@ import (
|
||||||
"github.com/cosmos/cosmos-sdk/x/distribution/types"
|
"github.com/cosmos/cosmos-sdk/x/distribution/types"
|
||||||
)
|
)
|
||||||
|
|
||||||
func TestSetGetPreviousProposerConsAddr(t *testing.T) {
|
func TestWithdrawValidatorCommission(t *testing.T) {
|
||||||
ctx, _, keeper, _, _ := CreateTestInputDefault(t, false, 0)
|
ctx, ak, keeper, _, _ := CreateTestInputDefault(t, false, 1000)
|
||||||
|
|
||||||
keeper.SetPreviousProposerConsAddr(ctx, valConsAddr1)
|
// set zero outstanding rewards
|
||||||
res := keeper.GetPreviousProposerConsAddr(ctx)
|
keeper.SetOutstandingRewards(ctx, types.OutstandingRewards{})
|
||||||
require.True(t, res.Equals(valConsAddr1), "expected: %v got: %v", valConsAddr1.String(), res.String())
|
|
||||||
}
|
// check initial balance
|
||||||
|
balance := ak.GetAccount(ctx, sdk.AccAddress(valOpAddr3)).GetCoins()
|
||||||
func TestSetGetCommunityTax(t *testing.T) {
|
require.Equal(t, balance, sdk.Coins{
|
||||||
ctx, _, keeper, _, _ := CreateTestInputDefault(t, false, 0)
|
{"stake", sdk.NewInt(1000)},
|
||||||
|
})
|
||||||
someDec := sdk.NewDec(333)
|
|
||||||
keeper.SetCommunityTax(ctx, someDec)
|
// set commission
|
||||||
res := keeper.GetCommunityTax(ctx)
|
keeper.SetValidatorAccumulatedCommission(ctx, valOpAddr3, sdk.DecCoins{
|
||||||
require.True(sdk.DecEq(t, someDec, res))
|
{"mytoken", sdk.NewDec(5).Quo(sdk.NewDec(4))},
|
||||||
}
|
{"stake", sdk.NewDec(3).Quo(sdk.NewDec(2))},
|
||||||
|
})
|
||||||
func TestSetGetFeePool(t *testing.T) {
|
|
||||||
ctx, _, keeper, _, _ := CreateTestInputDefault(t, false, 0)
|
// withdraw commission
|
||||||
|
keeper.WithdrawValidatorCommission(ctx, valOpAddr3)
|
||||||
fp := types.InitialFeePool()
|
|
||||||
fp.TotalValAccum.UpdateHeight = 777
|
// check balance increase
|
||||||
|
balance = ak.GetAccount(ctx, sdk.AccAddress(valOpAddr3)).GetCoins()
|
||||||
keeper.SetFeePool(ctx, fp)
|
require.Equal(t, balance, sdk.Coins{
|
||||||
res := keeper.GetFeePool(ctx)
|
{"mytoken", sdk.NewInt(1)},
|
||||||
require.Equal(t, fp.TotalValAccum, res.TotalValAccum)
|
{"stake", sdk.NewInt(1001)},
|
||||||
|
})
|
||||||
|
|
||||||
|
// check remainder
|
||||||
|
remainder := keeper.GetValidatorAccumulatedCommission(ctx, valOpAddr3)
|
||||||
|
require.Equal(t, remainder, sdk.DecCoins{
|
||||||
|
{"mytoken", sdk.NewDec(1).Quo(sdk.NewDec(4))},
|
||||||
|
{"stake", sdk.NewDec(1).Quo(sdk.NewDec(2))},
|
||||||
|
})
|
||||||
|
|
||||||
|
require.True(t, true)
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,49 +1,33 @@
|
||||||
package keeper
|
package keeper
|
||||||
|
|
||||||
import (
|
import (
|
||||||
|
"encoding/binary"
|
||||||
|
|
||||||
sdk "github.com/cosmos/cosmos-sdk/types"
|
sdk "github.com/cosmos/cosmos-sdk/types"
|
||||||
)
|
)
|
||||||
|
|
||||||
// keys/key-prefixes
|
|
||||||
var (
|
|
||||||
FeePoolKey = []byte{0x00} // key for global distribution state
|
|
||||||
ValidatorDistInfoKey = []byte{0x01} // prefix for each key to a validator distribution
|
|
||||||
DelegationDistInfoKey = []byte{0x02} // prefix for each key to a delegation distribution
|
|
||||||
DelegatorWithdrawInfoKey = []byte{0x03} // prefix for each key to a delegator withdraw info
|
|
||||||
ProposerKey = []byte{0x04} // key for storing the proposer operator address
|
|
||||||
|
|
||||||
// params store
|
|
||||||
ParamStoreKeyCommunityTax = []byte("communitytax")
|
|
||||||
ParamStoreKeyBaseProposerReward = []byte("baseproposerreward")
|
|
||||||
ParamStoreKeyBonusProposerReward = []byte("bonusproposerreward")
|
|
||||||
)
|
|
||||||
|
|
||||||
const (
|
const (
|
||||||
// default paramspace for params keeper
|
// default paramspace for params keeper
|
||||||
DefaultParamspace = "distr"
|
DefaultParamspace = "distr"
|
||||||
)
|
)
|
||||||
|
|
||||||
// gets the key for the validator distribution info from address
|
// keys
|
||||||
// VALUE: distribution/types.ValidatorDistInfo
|
var (
|
||||||
func GetValidatorDistInfoKey(operatorAddr sdk.ValAddress) []byte {
|
FeePoolKey = []byte{0x00} // key for global distribution state
|
||||||
return append(ValidatorDistInfoKey, operatorAddr.Bytes()...)
|
ProposerKey = []byte{0x01} // key for the proposer operator address
|
||||||
}
|
OutstandingRewardsKey = []byte{0x02} // key for outstanding rewards
|
||||||
|
|
||||||
// gets the key for delegator distribution for a validator
|
DelegatorWithdrawAddrPrefix = []byte{0x03} // key for delegator withdraw address
|
||||||
// VALUE: distribution/types.DelegationDistInfo
|
DelegatorStartingInfoPrefix = []byte{0x04} // key for delegator starting info
|
||||||
func GetDelegationDistInfoKey(delAddr sdk.AccAddress, valAddr sdk.ValAddress) []byte {
|
ValidatorHistoricalRewardsPrefix = []byte{0x05} // key for historical validators rewards / stake
|
||||||
return append(GetDelegationDistInfosKey(delAddr), valAddr.Bytes()...)
|
ValidatorCurrentRewardsPrefix = []byte{0x06} // key for current validator rewards
|
||||||
}
|
ValidatorAccumulatedCommissionPrefix = []byte{0x07} // key for accumulated validator commission
|
||||||
|
ValidatorSlashEventPrefix = []byte{0x08} // key for validator slash fraction
|
||||||
|
|
||||||
// gets the prefix for a delegator's distributions across all validators
|
ParamStoreKeyCommunityTax = []byte("communitytax")
|
||||||
func GetDelegationDistInfosKey(delAddr sdk.AccAddress) []byte {
|
ParamStoreKeyBaseProposerReward = []byte("baseproposerreward")
|
||||||
return append(DelegationDistInfoKey, delAddr.Bytes()...)
|
ParamStoreKeyBonusProposerReward = []byte("bonusproposerreward")
|
||||||
}
|
)
|
||||||
|
|
||||||
// gets the prefix for a delegator's withdraw info
|
|
||||||
func GetDelegatorWithdrawAddrKey(delAddr sdk.AccAddress) []byte {
|
|
||||||
return append(DelegatorWithdrawInfoKey, delAddr.Bytes()...)
|
|
||||||
}
|
|
||||||
|
|
||||||
// gets an address from a delegator's withdraw info key
|
// gets an address from a delegator's withdraw info key
|
||||||
func GetDelegatorWithdrawInfoAddress(key []byte) (delAddr sdk.AccAddress) {
|
func GetDelegatorWithdrawInfoAddress(key []byte) (delAddr sdk.AccAddress) {
|
||||||
|
@ -53,3 +37,100 @@ func GetDelegatorWithdrawInfoAddress(key []byte) (delAddr sdk.AccAddress) {
|
||||||
}
|
}
|
||||||
return sdk.AccAddress(addr)
|
return sdk.AccAddress(addr)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// gets the addresses from a delegator starting info key
|
||||||
|
func GetDelegatorStartingInfoAddresses(key []byte) (valAddr sdk.ValAddress, delAddr sdk.AccAddress) {
|
||||||
|
addr := key[1 : 1+sdk.AddrLen]
|
||||||
|
if len(addr) != sdk.AddrLen {
|
||||||
|
panic("unexpected key length")
|
||||||
|
}
|
||||||
|
valAddr = sdk.ValAddress(addr)
|
||||||
|
addr = key[1+sdk.AddrLen:]
|
||||||
|
if len(addr) != sdk.AddrLen {
|
||||||
|
panic("unexpected key length")
|
||||||
|
}
|
||||||
|
delAddr = sdk.AccAddress(addr)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
// gets the address & period from a validator's historical rewards key
|
||||||
|
func GetValidatorHistoricalRewardsAddressPeriod(key []byte) (valAddr sdk.ValAddress, period uint64) {
|
||||||
|
addr := key[1 : 1+sdk.AddrLen]
|
||||||
|
if len(addr) != sdk.AddrLen {
|
||||||
|
panic("unexpected key length")
|
||||||
|
}
|
||||||
|
valAddr = sdk.ValAddress(addr)
|
||||||
|
b := key[1+sdk.AddrLen:]
|
||||||
|
if len(b) != 8 {
|
||||||
|
panic("unexpected key length")
|
||||||
|
}
|
||||||
|
period = binary.LittleEndian.Uint64(b)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
// gets the address from a validator's current rewards key
|
||||||
|
func GetValidatorCurrentRewardsAddress(key []byte) (valAddr sdk.ValAddress) {
|
||||||
|
addr := key[1:]
|
||||||
|
if len(addr) != sdk.AddrLen {
|
||||||
|
panic("unexpected key length")
|
||||||
|
}
|
||||||
|
return sdk.ValAddress(addr)
|
||||||
|
}
|
||||||
|
|
||||||
|
// gets the address from a validator's accumulated commission key
|
||||||
|
func GetValidatorAccumulatedCommissionAddress(key []byte) (valAddr sdk.ValAddress) {
|
||||||
|
addr := key[1:]
|
||||||
|
if len(addr) != sdk.AddrLen {
|
||||||
|
panic("unexpected key length")
|
||||||
|
}
|
||||||
|
return sdk.ValAddress(addr)
|
||||||
|
}
|
||||||
|
|
||||||
|
// gets the height from a validator's slash event key
|
||||||
|
func GetValidatorSlashEventAddressHeight(key []byte) (valAddr sdk.ValAddress, height uint64) {
|
||||||
|
addr := key[1 : 1+sdk.AddrLen]
|
||||||
|
if len(addr) != sdk.AddrLen {
|
||||||
|
panic("unexpected key length")
|
||||||
|
}
|
||||||
|
valAddr = sdk.ValAddress(addr)
|
||||||
|
b := key[1+sdk.AddrLen:]
|
||||||
|
if len(b) != 8 {
|
||||||
|
panic("unexpected key length")
|
||||||
|
}
|
||||||
|
height = binary.BigEndian.Uint64(b)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
// gets the key for a delegator's withdraw addr
|
||||||
|
func GetDelegatorWithdrawAddrKey(delAddr sdk.AccAddress) []byte {
|
||||||
|
return append(DelegatorWithdrawAddrPrefix, delAddr.Bytes()...)
|
||||||
|
}
|
||||||
|
|
||||||
|
// gets the key for a delegator's starting info
|
||||||
|
func GetDelegatorStartingInfoKey(v sdk.ValAddress, d sdk.AccAddress) []byte {
|
||||||
|
return append(append(DelegatorStartingInfoPrefix, v.Bytes()...), d.Bytes()...)
|
||||||
|
}
|
||||||
|
|
||||||
|
// gets the key for a validator's historical rewards
|
||||||
|
func GetValidatorHistoricalRewardsKey(v sdk.ValAddress, k uint64) []byte {
|
||||||
|
b := make([]byte, 8)
|
||||||
|
binary.LittleEndian.PutUint64(b, k)
|
||||||
|
return append(append(ValidatorHistoricalRewardsPrefix, v.Bytes()...), b...)
|
||||||
|
}
|
||||||
|
|
||||||
|
// gets the key for a validator's current rewards
|
||||||
|
func GetValidatorCurrentRewardsKey(v sdk.ValAddress) []byte {
|
||||||
|
return append(ValidatorCurrentRewardsPrefix, v.Bytes()...)
|
||||||
|
}
|
||||||
|
|
||||||
|
// gets the key for a validator's current commission
|
||||||
|
func GetValidatorAccumulatedCommissionKey(v sdk.ValAddress) []byte {
|
||||||
|
return append(ValidatorAccumulatedCommissionPrefix, v.Bytes()...)
|
||||||
|
}
|
||||||
|
|
||||||
|
// gets the key for a validator's slash fraction
|
||||||
|
func GetValidatorSlashEventKey(v sdk.ValAddress, height uint64) []byte {
|
||||||
|
b := make([]byte, 8)
|
||||||
|
binary.BigEndian.PutUint64(b, height)
|
||||||
|
return append(append(ValidatorSlashEventPrefix, v.Bytes()...), b...)
|
||||||
|
}
|
||||||
|
|
|
@ -0,0 +1,54 @@
|
||||||
|
package keeper
|
||||||
|
|
||||||
|
import (
|
||||||
|
sdk "github.com/cosmos/cosmos-sdk/types"
|
||||||
|
"github.com/cosmos/cosmos-sdk/x/params"
|
||||||
|
)
|
||||||
|
|
||||||
|
// type declaration for parameters
|
||||||
|
func ParamTypeTable() params.TypeTable {
|
||||||
|
return params.NewTypeTable(
|
||||||
|
ParamStoreKeyCommunityTax, sdk.Dec{},
|
||||||
|
ParamStoreKeyBaseProposerReward, sdk.Dec{},
|
||||||
|
ParamStoreKeyBonusProposerReward, sdk.Dec{},
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
// returns the current CommunityTax rate from the global param store
|
||||||
|
// nolint: errcheck
|
||||||
|
func (k Keeper) GetCommunityTax(ctx sdk.Context) sdk.Dec {
|
||||||
|
var percent sdk.Dec
|
||||||
|
k.paramSpace.Get(ctx, ParamStoreKeyCommunityTax, &percent)
|
||||||
|
return percent
|
||||||
|
}
|
||||||
|
|
||||||
|
// nolint: errcheck
|
||||||
|
func (k Keeper) SetCommunityTax(ctx sdk.Context, percent sdk.Dec) {
|
||||||
|
k.paramSpace.Set(ctx, ParamStoreKeyCommunityTax, &percent)
|
||||||
|
}
|
||||||
|
|
||||||
|
// returns the current BaseProposerReward rate from the global param store
|
||||||
|
// nolint: errcheck
|
||||||
|
func (k Keeper) GetBaseProposerReward(ctx sdk.Context) sdk.Dec {
|
||||||
|
var percent sdk.Dec
|
||||||
|
k.paramSpace.Get(ctx, ParamStoreKeyBaseProposerReward, &percent)
|
||||||
|
return percent
|
||||||
|
}
|
||||||
|
|
||||||
|
// nolint: errcheck
|
||||||
|
func (k Keeper) SetBaseProposerReward(ctx sdk.Context, percent sdk.Dec) {
|
||||||
|
k.paramSpace.Set(ctx, ParamStoreKeyBaseProposerReward, &percent)
|
||||||
|
}
|
||||||
|
|
||||||
|
// returns the current BaseProposerReward rate from the global param store
|
||||||
|
// nolint: errcheck
|
||||||
|
func (k Keeper) GetBonusProposerReward(ctx sdk.Context) sdk.Dec {
|
||||||
|
var percent sdk.Dec
|
||||||
|
k.paramSpace.Get(ctx, ParamStoreKeyBonusProposerReward, &percent)
|
||||||
|
return percent
|
||||||
|
}
|
||||||
|
|
||||||
|
// nolint: errcheck
|
||||||
|
func (k Keeper) SetBonusProposerReward(ctx sdk.Context, percent sdk.Dec) {
|
||||||
|
k.paramSpace.Set(ctx, ParamStoreKeyBonusProposerReward, &percent)
|
||||||
|
}
|
|
@ -0,0 +1,288 @@
|
||||||
|
package keeper
|
||||||
|
|
||||||
|
import (
|
||||||
|
sdk "github.com/cosmos/cosmos-sdk/types"
|
||||||
|
"github.com/cosmos/cosmos-sdk/x/distribution/types"
|
||||||
|
)
|
||||||
|
|
||||||
|
// get the delegator withdraw address, defaulting to the delegator address
|
||||||
|
func (k Keeper) GetDelegatorWithdrawAddr(ctx sdk.Context, delAddr sdk.AccAddress) sdk.AccAddress {
|
||||||
|
store := ctx.KVStore(k.storeKey)
|
||||||
|
b := store.Get(GetDelegatorWithdrawAddrKey(delAddr))
|
||||||
|
if b == nil {
|
||||||
|
return delAddr
|
||||||
|
}
|
||||||
|
return sdk.AccAddress(b)
|
||||||
|
}
|
||||||
|
|
||||||
|
// set the delegator withdraw address
|
||||||
|
func (k Keeper) SetDelegatorWithdrawAddr(ctx sdk.Context, delAddr, withdrawAddr sdk.AccAddress) {
|
||||||
|
store := ctx.KVStore(k.storeKey)
|
||||||
|
store.Set(GetDelegatorWithdrawAddrKey(delAddr), withdrawAddr.Bytes())
|
||||||
|
}
|
||||||
|
|
||||||
|
// remove a delegator withdraw addr
|
||||||
|
func (k Keeper) RemoveDelegatorWithdrawAddr(ctx sdk.Context, delAddr, withdrawAddr sdk.AccAddress) {
|
||||||
|
store := ctx.KVStore(k.storeKey)
|
||||||
|
store.Delete(GetDelegatorWithdrawAddrKey(delAddr))
|
||||||
|
}
|
||||||
|
|
||||||
|
// iterate over delegator withdraw addrs
|
||||||
|
func (k Keeper) IterateDelegatorWithdrawAddrs(ctx sdk.Context, handler func(del sdk.AccAddress, addr sdk.AccAddress) (stop bool)) {
|
||||||
|
store := ctx.KVStore(k.storeKey)
|
||||||
|
iter := sdk.KVStorePrefixIterator(store, DelegatorWithdrawAddrPrefix)
|
||||||
|
defer iter.Close()
|
||||||
|
for ; iter.Valid(); iter.Next() {
|
||||||
|
addr := sdk.AccAddress(iter.Value())
|
||||||
|
del := GetDelegatorWithdrawInfoAddress(iter.Key())
|
||||||
|
if handler(del, addr) {
|
||||||
|
break
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// get the global fee pool distribution info
|
||||||
|
func (k Keeper) GetFeePool(ctx sdk.Context) (feePool types.FeePool) {
|
||||||
|
store := ctx.KVStore(k.storeKey)
|
||||||
|
b := store.Get(FeePoolKey)
|
||||||
|
if b == nil {
|
||||||
|
panic("Stored fee pool should not have been nil")
|
||||||
|
}
|
||||||
|
k.cdc.MustUnmarshalBinaryLengthPrefixed(b, &feePool)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
// set the global fee pool distribution info
|
||||||
|
func (k Keeper) SetFeePool(ctx sdk.Context, feePool types.FeePool) {
|
||||||
|
store := ctx.KVStore(k.storeKey)
|
||||||
|
b := k.cdc.MustMarshalBinaryLengthPrefixed(feePool)
|
||||||
|
store.Set(FeePoolKey, b)
|
||||||
|
}
|
||||||
|
|
||||||
|
// get the proposer public key for this block
|
||||||
|
func (k Keeper) GetPreviousProposerConsAddr(ctx sdk.Context) (consAddr sdk.ConsAddress) {
|
||||||
|
store := ctx.KVStore(k.storeKey)
|
||||||
|
b := store.Get(ProposerKey)
|
||||||
|
if b == nil {
|
||||||
|
panic("Previous proposer not set")
|
||||||
|
}
|
||||||
|
k.cdc.MustUnmarshalBinaryLengthPrefixed(b, &consAddr)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
// set the proposer public key for this block
|
||||||
|
func (k Keeper) SetPreviousProposerConsAddr(ctx sdk.Context, consAddr sdk.ConsAddress) {
|
||||||
|
store := ctx.KVStore(k.storeKey)
|
||||||
|
b := k.cdc.MustMarshalBinaryLengthPrefixed(consAddr)
|
||||||
|
store.Set(ProposerKey, b)
|
||||||
|
}
|
||||||
|
|
||||||
|
// get the starting period associated with a delegator
|
||||||
|
func (k Keeper) GetDelegatorStartingInfo(ctx sdk.Context, val sdk.ValAddress, del sdk.AccAddress) (period types.DelegatorStartingInfo) {
|
||||||
|
store := ctx.KVStore(k.storeKey)
|
||||||
|
b := store.Get(GetDelegatorStartingInfoKey(val, del))
|
||||||
|
k.cdc.MustUnmarshalBinaryLengthPrefixed(b, &period)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
// set the starting period associated with a delegator
|
||||||
|
func (k Keeper) SetDelegatorStartingInfo(ctx sdk.Context, val sdk.ValAddress, del sdk.AccAddress, period types.DelegatorStartingInfo) {
|
||||||
|
store := ctx.KVStore(k.storeKey)
|
||||||
|
b := k.cdc.MustMarshalBinaryLengthPrefixed(period)
|
||||||
|
store.Set(GetDelegatorStartingInfoKey(val, del), b)
|
||||||
|
}
|
||||||
|
|
||||||
|
// iterate over delegator starting infos
|
||||||
|
func (k Keeper) IterateDelegatorStartingInfos(ctx sdk.Context, handler func(val sdk.ValAddress, del sdk.AccAddress, info types.DelegatorStartingInfo) (stop bool)) {
|
||||||
|
store := ctx.KVStore(k.storeKey)
|
||||||
|
iter := sdk.KVStorePrefixIterator(store, DelegatorStartingInfoPrefix)
|
||||||
|
defer iter.Close()
|
||||||
|
for ; iter.Valid(); iter.Next() {
|
||||||
|
var info types.DelegatorStartingInfo
|
||||||
|
k.cdc.MustUnmarshalBinaryLengthPrefixed(iter.Value(), &info)
|
||||||
|
val, del := GetDelegatorStartingInfoAddresses(iter.Key())
|
||||||
|
if handler(val, del, info) {
|
||||||
|
break
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// get historical rewards for a particular period
|
||||||
|
func (k Keeper) GetValidatorHistoricalRewards(ctx sdk.Context, val sdk.ValAddress, period uint64) (rewards types.ValidatorHistoricalRewards) {
|
||||||
|
store := ctx.KVStore(k.storeKey)
|
||||||
|
b := store.Get(GetValidatorHistoricalRewardsKey(val, period))
|
||||||
|
k.cdc.MustUnmarshalBinaryLengthPrefixed(b, &rewards)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
// set historical rewards for a particular period
|
||||||
|
func (k Keeper) SetValidatorHistoricalRewards(ctx sdk.Context, val sdk.ValAddress, period uint64, rewards types.ValidatorHistoricalRewards) {
|
||||||
|
store := ctx.KVStore(k.storeKey)
|
||||||
|
b := k.cdc.MustMarshalBinaryLengthPrefixed(rewards)
|
||||||
|
store.Set(GetValidatorHistoricalRewardsKey(val, period), b)
|
||||||
|
}
|
||||||
|
|
||||||
|
// iterate over historical rewards
|
||||||
|
func (k Keeper) IterateValidatorHistoricalRewards(ctx sdk.Context, handler func(val sdk.ValAddress, period uint64, rewards types.ValidatorHistoricalRewards) (stop bool)) {
|
||||||
|
store := ctx.KVStore(k.storeKey)
|
||||||
|
iter := sdk.KVStorePrefixIterator(store, ValidatorHistoricalRewardsPrefix)
|
||||||
|
defer iter.Close()
|
||||||
|
for ; iter.Valid(); iter.Next() {
|
||||||
|
var rewards types.ValidatorHistoricalRewards
|
||||||
|
k.cdc.MustUnmarshalBinaryLengthPrefixed(iter.Value(), &rewards)
|
||||||
|
addr, period := GetValidatorHistoricalRewardsAddressPeriod(iter.Key())
|
||||||
|
if handler(addr, period, rewards) {
|
||||||
|
break
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// delete historical rewards
|
||||||
|
func (k Keeper) DeleteValidatorHistoricalRewards(ctx sdk.Context) {
|
||||||
|
store := ctx.KVStore(k.storeKey)
|
||||||
|
iter := sdk.KVStorePrefixIterator(store, ValidatorHistoricalRewardsPrefix)
|
||||||
|
defer iter.Close()
|
||||||
|
for ; iter.Valid(); iter.Next() {
|
||||||
|
store.Delete(iter.Key())
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// get current rewards for a validator
|
||||||
|
func (k Keeper) GetValidatorCurrentRewards(ctx sdk.Context, val sdk.ValAddress) (rewards types.ValidatorCurrentRewards) {
|
||||||
|
store := ctx.KVStore(k.storeKey)
|
||||||
|
b := store.Get(GetValidatorCurrentRewardsKey(val))
|
||||||
|
k.cdc.MustUnmarshalBinaryLengthPrefixed(b, &rewards)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
// set current rewards for a validator
|
||||||
|
func (k Keeper) SetValidatorCurrentRewards(ctx sdk.Context, val sdk.ValAddress, rewards types.ValidatorCurrentRewards) {
|
||||||
|
store := ctx.KVStore(k.storeKey)
|
||||||
|
b := k.cdc.MustMarshalBinaryLengthPrefixed(rewards)
|
||||||
|
store.Set(GetValidatorCurrentRewardsKey(val), b)
|
||||||
|
}
|
||||||
|
|
||||||
|
// iterate over current rewards
|
||||||
|
func (k Keeper) IterateValidatorCurrentRewards(ctx sdk.Context, handler func(val sdk.ValAddress, rewards types.ValidatorCurrentRewards) (stop bool)) {
|
||||||
|
store := ctx.KVStore(k.storeKey)
|
||||||
|
iter := sdk.KVStorePrefixIterator(store, ValidatorCurrentRewardsPrefix)
|
||||||
|
defer iter.Close()
|
||||||
|
for ; iter.Valid(); iter.Next() {
|
||||||
|
var rewards types.ValidatorCurrentRewards
|
||||||
|
k.cdc.MustUnmarshalBinaryLengthPrefixed(iter.Value(), &rewards)
|
||||||
|
addr := GetValidatorCurrentRewardsAddress(iter.Key())
|
||||||
|
if handler(addr, rewards) {
|
||||||
|
break
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// get accumulated commission for a validator
|
||||||
|
func (k Keeper) GetValidatorAccumulatedCommission(ctx sdk.Context, val sdk.ValAddress) (commission types.ValidatorAccumulatedCommission) {
|
||||||
|
store := ctx.KVStore(k.storeKey)
|
||||||
|
b := store.Get(GetValidatorAccumulatedCommissionKey(val))
|
||||||
|
if b == nil {
|
||||||
|
return types.ValidatorAccumulatedCommission{}
|
||||||
|
}
|
||||||
|
k.cdc.MustUnmarshalBinaryLengthPrefixed(b, &commission)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
// set accumulated commission for a validator
|
||||||
|
func (k Keeper) SetValidatorAccumulatedCommission(ctx sdk.Context, val sdk.ValAddress, commission types.ValidatorAccumulatedCommission) {
|
||||||
|
store := ctx.KVStore(k.storeKey)
|
||||||
|
b := k.cdc.MustMarshalBinaryLengthPrefixed(commission)
|
||||||
|
store.Set(GetValidatorAccumulatedCommissionKey(val), b)
|
||||||
|
}
|
||||||
|
|
||||||
|
// iterate over accumulated commissions
|
||||||
|
func (k Keeper) IterateValidatorAccumulatedCommissions(ctx sdk.Context, handler func(val sdk.ValAddress, commission types.ValidatorAccumulatedCommission) (stop bool)) {
|
||||||
|
store := ctx.KVStore(k.storeKey)
|
||||||
|
iter := sdk.KVStorePrefixIterator(store, ValidatorAccumulatedCommissionPrefix)
|
||||||
|
defer iter.Close()
|
||||||
|
for ; iter.Valid(); iter.Next() {
|
||||||
|
var commission types.ValidatorAccumulatedCommission
|
||||||
|
k.cdc.MustUnmarshalBinaryLengthPrefixed(iter.Value(), &commission)
|
||||||
|
addr := GetValidatorAccumulatedCommissionAddress(iter.Key())
|
||||||
|
if handler(addr, commission) {
|
||||||
|
break
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// get outstanding rewards
|
||||||
|
func (k Keeper) GetOutstandingRewards(ctx sdk.Context) (rewards types.OutstandingRewards) {
|
||||||
|
store := ctx.KVStore(k.storeKey)
|
||||||
|
b := store.Get(OutstandingRewardsKey)
|
||||||
|
k.cdc.MustUnmarshalBinaryLengthPrefixed(b, &rewards)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
// set outstanding rewards
|
||||||
|
func (k Keeper) SetOutstandingRewards(ctx sdk.Context, rewards types.OutstandingRewards) {
|
||||||
|
store := ctx.KVStore(k.storeKey)
|
||||||
|
b := k.cdc.MustMarshalBinaryLengthPrefixed(rewards)
|
||||||
|
store.Set(OutstandingRewardsKey, b)
|
||||||
|
}
|
||||||
|
|
||||||
|
// get slash event for height
|
||||||
|
func (k Keeper) GetValidatorSlashEvent(ctx sdk.Context, val sdk.ValAddress, height uint64) (event types.ValidatorSlashEvent, found bool) {
|
||||||
|
store := ctx.KVStore(k.storeKey)
|
||||||
|
b := store.Get(GetValidatorSlashEventKey(val, height))
|
||||||
|
if b == nil {
|
||||||
|
return types.ValidatorSlashEvent{}, false
|
||||||
|
}
|
||||||
|
k.cdc.MustUnmarshalBinaryLengthPrefixed(b, &event)
|
||||||
|
return event, true
|
||||||
|
}
|
||||||
|
|
||||||
|
// set slash event for height
|
||||||
|
func (k Keeper) SetValidatorSlashEvent(ctx sdk.Context, val sdk.ValAddress, height uint64, event types.ValidatorSlashEvent) {
|
||||||
|
store := ctx.KVStore(k.storeKey)
|
||||||
|
b := k.cdc.MustMarshalBinaryLengthPrefixed(event)
|
||||||
|
store.Set(GetValidatorSlashEventKey(val, height), b)
|
||||||
|
}
|
||||||
|
|
||||||
|
// iterate over slash events between heights, inclusive
|
||||||
|
func (k Keeper) IterateValidatorSlashEventsBetween(ctx sdk.Context, val sdk.ValAddress, startingHeight uint64, endingHeight uint64,
|
||||||
|
handler func(height uint64, event types.ValidatorSlashEvent) (stop bool)) {
|
||||||
|
store := ctx.KVStore(k.storeKey)
|
||||||
|
iter := store.Iterator(
|
||||||
|
GetValidatorSlashEventKey(val, startingHeight),
|
||||||
|
GetValidatorSlashEventKey(val, endingHeight+1),
|
||||||
|
)
|
||||||
|
defer iter.Close()
|
||||||
|
for ; iter.Valid(); iter.Next() {
|
||||||
|
var event types.ValidatorSlashEvent
|
||||||
|
k.cdc.MustUnmarshalBinaryLengthPrefixed(iter.Value(), &event)
|
||||||
|
_, height := GetValidatorSlashEventAddressHeight(iter.Key())
|
||||||
|
if handler(height, event) {
|
||||||
|
break
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// iterate over all slash events
|
||||||
|
func (k Keeper) IterateValidatorSlashEvents(ctx sdk.Context, handler func(val sdk.ValAddress, height uint64, event types.ValidatorSlashEvent) (stop bool)) {
|
||||||
|
store := ctx.KVStore(k.storeKey)
|
||||||
|
iter := sdk.KVStorePrefixIterator(store, ValidatorSlashEventPrefix)
|
||||||
|
defer iter.Close()
|
||||||
|
for ; iter.Valid(); iter.Next() {
|
||||||
|
var event types.ValidatorSlashEvent
|
||||||
|
k.cdc.MustUnmarshalBinaryLengthPrefixed(iter.Value(), &event)
|
||||||
|
val, height := GetValidatorSlashEventAddressHeight(iter.Key())
|
||||||
|
if handler(val, height, event) {
|
||||||
|
break
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// delete all slash events
|
||||||
|
func (k Keeper) DeleteValidatorSlashEvents(ctx sdk.Context) {
|
||||||
|
store := ctx.KVStore(k.storeKey)
|
||||||
|
iter := sdk.KVStorePrefixIterator(store, ValidatorSlashEventPrefix)
|
||||||
|
defer iter.Close()
|
||||||
|
for ; iter.Valid(); iter.Next() {
|
||||||
|
store.Delete(iter.Key())
|
||||||
|
}
|
||||||
|
}
|
|
@ -2,170 +2,70 @@ package keeper
|
||||||
|
|
||||||
import (
|
import (
|
||||||
sdk "github.com/cosmos/cosmos-sdk/types"
|
sdk "github.com/cosmos/cosmos-sdk/types"
|
||||||
|
|
||||||
"github.com/cosmos/cosmos-sdk/x/distribution/types"
|
"github.com/cosmos/cosmos-sdk/x/distribution/types"
|
||||||
)
|
)
|
||||||
|
|
||||||
// check whether a validator has distribution info
|
// initialize rewards for a new validator
|
||||||
func (k Keeper) HasValidatorDistInfo(ctx sdk.Context,
|
func (k Keeper) initializeValidator(ctx sdk.Context, val sdk.Validator) {
|
||||||
operatorAddr sdk.ValAddress) (exists bool) {
|
// set initial historical rewards (period 0)
|
||||||
store := ctx.KVStore(k.storeKey)
|
k.SetValidatorHistoricalRewards(ctx, val.GetOperator(), 0, types.ValidatorHistoricalRewards{})
|
||||||
return store.Has(GetValidatorDistInfoKey(operatorAddr))
|
|
||||||
|
// set current rewards (starting at period 1)
|
||||||
|
k.SetValidatorCurrentRewards(ctx, val.GetOperator(), types.NewValidatorCurrentRewards(sdk.DecCoins{}, 1))
|
||||||
|
|
||||||
|
// set accumulated commission
|
||||||
|
k.SetValidatorAccumulatedCommission(ctx, val.GetOperator(), types.InitialValidatorAccumulatedCommission())
|
||||||
}
|
}
|
||||||
|
|
||||||
// get the validator distribution info
|
// increment validator period, returning the period just ended
|
||||||
func (k Keeper) GetValidatorDistInfo(ctx sdk.Context,
|
func (k Keeper) incrementValidatorPeriod(ctx sdk.Context, val sdk.Validator) uint64 {
|
||||||
operatorAddr sdk.ValAddress) (vdi types.ValidatorDistInfo) {
|
// fetch current rewards
|
||||||
|
rewards := k.GetValidatorCurrentRewards(ctx, val.GetOperator())
|
||||||
|
|
||||||
store := ctx.KVStore(k.storeKey)
|
// calculate current ratio
|
||||||
|
var current sdk.DecCoins
|
||||||
|
if val.GetTokens().IsZero() {
|
||||||
|
|
||||||
b := store.Get(GetValidatorDistInfoKey(operatorAddr))
|
// can't calculate ratio for zero-token validators
|
||||||
if b == nil {
|
// ergo we instead add to the community pool
|
||||||
panic("Stored validator-distribution info should not have been nil")
|
feePool := k.GetFeePool(ctx)
|
||||||
|
outstanding := k.GetOutstandingRewards(ctx)
|
||||||
|
feePool.CommunityPool = feePool.CommunityPool.Plus(rewards.Rewards)
|
||||||
|
outstanding = outstanding.Minus(rewards.Rewards)
|
||||||
|
k.SetFeePool(ctx, feePool)
|
||||||
|
k.SetOutstandingRewards(ctx, outstanding)
|
||||||
|
|
||||||
|
current = sdk.DecCoins{}
|
||||||
|
} else {
|
||||||
|
current = rewards.Rewards.QuoDec(sdk.NewDecFromInt(val.GetTokens()))
|
||||||
}
|
}
|
||||||
|
|
||||||
k.cdc.MustUnmarshalBinaryLengthPrefixed(b, &vdi)
|
// fetch historical rewards for last period
|
||||||
return
|
historical := k.GetValidatorHistoricalRewards(ctx, val.GetOperator(), rewards.Period-1)
|
||||||
|
|
||||||
|
// fet new historical rewards
|
||||||
|
k.SetValidatorHistoricalRewards(ctx, val.GetOperator(), rewards.Period, historical.Plus(current))
|
||||||
|
|
||||||
|
// set current rewards, incrementing period by 1
|
||||||
|
k.SetValidatorCurrentRewards(ctx, val.GetOperator(), types.NewValidatorCurrentRewards(sdk.DecCoins{}, rewards.Period+1))
|
||||||
|
|
||||||
|
return rewards.Period
|
||||||
}
|
}
|
||||||
|
|
||||||
// set the validator distribution info
|
func (k Keeper) updateValidatorSlashFraction(ctx sdk.Context, valAddr sdk.ValAddress, fraction sdk.Dec) {
|
||||||
func (k Keeper) SetValidatorDistInfo(ctx sdk.Context, vdi types.ValidatorDistInfo) {
|
height := uint64(ctx.BlockHeight())
|
||||||
store := ctx.KVStore(k.storeKey)
|
currentFraction := sdk.ZeroDec()
|
||||||
b := k.cdc.MustMarshalBinaryLengthPrefixed(vdi)
|
currentPeriod := k.GetValidatorCurrentRewards(ctx, valAddr).Period
|
||||||
store.Set(GetValidatorDistInfoKey(vdi.OperatorAddr), b)
|
current, found := k.GetValidatorSlashEvent(ctx, valAddr, height)
|
||||||
}
|
if found {
|
||||||
|
// there has already been a slash event this height,
|
||||||
// remove a validator distribution info
|
// and we don't need to store more than one,
|
||||||
func (k Keeper) RemoveValidatorDistInfo(ctx sdk.Context, valAddr sdk.ValAddress) {
|
// so just update the current slash fraction
|
||||||
|
currentFraction = current.Fraction
|
||||||
// defensive check
|
|
||||||
vdi := k.GetValidatorDistInfo(ctx, valAddr)
|
|
||||||
if vdi.DelAccum.Accum.IsPositive() {
|
|
||||||
panic("Should not delete validator with unwithdrawn delegator accum")
|
|
||||||
}
|
}
|
||||||
if !vdi.ValCommission.IsZero() {
|
currentMultiplicand := sdk.OneDec().Sub(currentFraction)
|
||||||
panic("Should not delete validator with unwithdrawn validator commission")
|
newMultiplicand := sdk.OneDec().Sub(fraction)
|
||||||
}
|
updatedFraction := sdk.OneDec().Sub(currentMultiplicand.Mul(newMultiplicand))
|
||||||
|
k.SetValidatorSlashEvent(ctx, valAddr, height, types.NewValidatorSlashEvent(currentPeriod, updatedFraction))
|
||||||
store := ctx.KVStore(k.storeKey)
|
|
||||||
store.Delete(GetValidatorDistInfoKey(valAddr))
|
|
||||||
}
|
|
||||||
|
|
||||||
// remove all validator distribution infos
|
|
||||||
func (k Keeper) RemoveValidatorDistInfos(ctx sdk.Context) {
|
|
||||||
store := ctx.KVStore(k.storeKey)
|
|
||||||
iter := sdk.KVStorePrefixIterator(store, ValidatorDistInfoKey)
|
|
||||||
defer iter.Close()
|
|
||||||
for ; iter.Valid(); iter.Next() {
|
|
||||||
store.Delete(iter.Key())
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// iterate over all the validator distribution infos
|
|
||||||
func (k Keeper) IterateValidatorDistInfos(ctx sdk.Context,
|
|
||||||
fn func(index int64, distInfo types.ValidatorDistInfo) (stop bool)) {
|
|
||||||
|
|
||||||
store := ctx.KVStore(k.storeKey)
|
|
||||||
iter := sdk.KVStorePrefixIterator(store, ValidatorDistInfoKey)
|
|
||||||
defer iter.Close()
|
|
||||||
index := int64(0)
|
|
||||||
for ; iter.Valid(); iter.Next() {
|
|
||||||
var vdi types.ValidatorDistInfo
|
|
||||||
k.cdc.MustUnmarshalBinaryLengthPrefixed(iter.Value(), &vdi)
|
|
||||||
if fn(index, vdi) {
|
|
||||||
return
|
|
||||||
}
|
|
||||||
index++
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// Get the calculated accum of a validator at the current block
|
|
||||||
// without affecting the state.
|
|
||||||
func (k Keeper) GetValidatorAccum(ctx sdk.Context, operatorAddr sdk.ValAddress) (sdk.Dec, sdk.Error) {
|
|
||||||
if !k.HasValidatorDistInfo(ctx, operatorAddr) {
|
|
||||||
return sdk.Dec{}, types.ErrNoValidatorDistInfo(k.codespace)
|
|
||||||
}
|
|
||||||
|
|
||||||
// withdraw self-delegation
|
|
||||||
height := ctx.BlockHeight()
|
|
||||||
lastValPower := k.stakingKeeper.GetLastValidatorPower(ctx, operatorAddr)
|
|
||||||
valInfo := k.GetValidatorDistInfo(ctx, operatorAddr)
|
|
||||||
accum := valInfo.GetValAccum(height, sdk.NewDecFromInt(lastValPower))
|
|
||||||
|
|
||||||
return accum, nil
|
|
||||||
}
|
|
||||||
|
|
||||||
// takeValidatorFeePoolRewards updates the validator's distribution info
|
|
||||||
// from the global fee pool without withdrawing any rewards. This will be called
|
|
||||||
// from a onValidatorModified hook.
|
|
||||||
func (k Keeper) takeValidatorFeePoolRewards(ctx sdk.Context, operatorAddr sdk.ValAddress) sdk.Error {
|
|
||||||
if !k.HasValidatorDistInfo(ctx, operatorAddr) {
|
|
||||||
return types.ErrNoValidatorDistInfo(k.codespace)
|
|
||||||
}
|
|
||||||
accAddr := sdk.AccAddress(operatorAddr.Bytes())
|
|
||||||
|
|
||||||
// withdraw reward for self-delegation
|
|
||||||
if k.HasDelegationDistInfo(ctx, accAddr, operatorAddr) {
|
|
||||||
fp, vi, di, withdraw :=
|
|
||||||
k.withdrawDelegationReward(ctx, accAddr, operatorAddr)
|
|
||||||
k.SetFeePool(ctx, fp)
|
|
||||||
k.SetValidatorDistInfo(ctx, vi)
|
|
||||||
k.SetDelegationDistInfo(ctx, di)
|
|
||||||
k.WithdrawToDelegator(ctx, fp, accAddr, withdraw)
|
|
||||||
}
|
|
||||||
|
|
||||||
// withdrawal validator commission rewards
|
|
||||||
valInfo := k.GetValidatorDistInfo(ctx, operatorAddr)
|
|
||||||
wc := k.GetWithdrawContext(ctx, operatorAddr)
|
|
||||||
valInfo, feePool := valInfo.TakeFeePoolRewards(wc)
|
|
||||||
k.SetFeePool(ctx, feePool)
|
|
||||||
k.SetValidatorDistInfo(ctx, valInfo)
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
|
|
||||||
func (k Keeper) withdrawValidatorCommission(ctx sdk.Context, operatorAddr sdk.ValAddress) (types.FeePool, types.DecCoins) {
|
|
||||||
valInfo := k.GetValidatorDistInfo(ctx, operatorAddr)
|
|
||||||
wc := k.GetWithdrawContext(ctx, operatorAddr)
|
|
||||||
valInfo, feePool, commission := valInfo.WithdrawCommission(wc)
|
|
||||||
k.SetValidatorDistInfo(ctx, valInfo)
|
|
||||||
|
|
||||||
return feePool, commission
|
|
||||||
}
|
|
||||||
|
|
||||||
// withdrawal all the validator rewards including the commission
|
|
||||||
func (k Keeper) WithdrawValidatorRewardsAll(ctx sdk.Context, operatorAddr sdk.ValAddress) sdk.Error {
|
|
||||||
if !k.HasValidatorDistInfo(ctx, operatorAddr) {
|
|
||||||
return types.ErrNoValidatorDistInfo(k.codespace)
|
|
||||||
}
|
|
||||||
|
|
||||||
// withdraw self-delegation
|
|
||||||
accAddr := sdk.AccAddress(operatorAddr.Bytes())
|
|
||||||
withdraw := k.withdrawDelegationRewardsAll(ctx, accAddr)
|
|
||||||
|
|
||||||
// withdrawal validator commission rewards
|
|
||||||
feePool, commission := k.withdrawValidatorCommission(ctx, operatorAddr)
|
|
||||||
withdraw = withdraw.Plus(commission)
|
|
||||||
|
|
||||||
k.WithdrawToDelegator(ctx, feePool, accAddr, withdraw)
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
|
|
||||||
// get all the validator rewards including the commission
|
|
||||||
func (k Keeper) CurrentValidatorRewardsAll(ctx sdk.Context, operatorAddr sdk.ValAddress) (sdk.Coins, sdk.Error) {
|
|
||||||
|
|
||||||
if !k.HasValidatorDistInfo(ctx, operatorAddr) {
|
|
||||||
return sdk.Coins{}, types.ErrNoValidatorDistInfo(k.codespace)
|
|
||||||
}
|
|
||||||
|
|
||||||
// withdraw self-delegation
|
|
||||||
accAddr := sdk.AccAddress(operatorAddr.Bytes())
|
|
||||||
withdraw := k.CurrentDelegationRewardsAll(ctx, accAddr)
|
|
||||||
|
|
||||||
// withdrawal validator commission rewards
|
|
||||||
valInfo := k.GetValidatorDistInfo(ctx, operatorAddr)
|
|
||||||
|
|
||||||
wc := k.GetWithdrawContext(ctx, operatorAddr)
|
|
||||||
commission := valInfo.CurrentCommissionRewards(wc)
|
|
||||||
withdraw = withdraw.Plus(commission)
|
|
||||||
truncated, _ := withdraw.TruncateDecimal()
|
|
||||||
return truncated, nil
|
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,201 +0,0 @@
|
||||||
package keeper
|
|
||||||
|
|
||||||
import (
|
|
||||||
"testing"
|
|
||||||
|
|
||||||
"github.com/stretchr/testify/require"
|
|
||||||
|
|
||||||
sdk "github.com/cosmos/cosmos-sdk/types"
|
|
||||||
"github.com/cosmos/cosmos-sdk/x/staking"
|
|
||||||
)
|
|
||||||
|
|
||||||
func TestWithdrawValidatorRewardsAllNoDelegator(t *testing.T) {
|
|
||||||
ctx, accMapper, keeper, sk, fck := CreateTestInputAdvanced(t, false, 100, sdk.ZeroDec())
|
|
||||||
stakingHandler := staking.NewHandler(sk)
|
|
||||||
denom := sk.GetParams(ctx).BondDenom
|
|
||||||
|
|
||||||
// first make a validator
|
|
||||||
msgCreateValidator := staking.NewTestMsgCreateValidator(valOpAddr1, valConsPk1, 10)
|
|
||||||
got := stakingHandler(ctx, msgCreateValidator)
|
|
||||||
require.True(t, got.IsOK(), "expected msg to be ok, got %v", got)
|
|
||||||
_ = sk.ApplyAndReturnValidatorSetUpdates(ctx)
|
|
||||||
|
|
||||||
// allocate 100 denom of fees
|
|
||||||
feeInputs := sdk.NewInt(100)
|
|
||||||
fck.SetCollectedFees(sdk.Coins{sdk.NewCoin(denom, feeInputs)})
|
|
||||||
require.Equal(t, feeInputs, fck.GetCollectedFees(ctx).AmountOf(denom))
|
|
||||||
keeper.AllocateTokens(ctx, sdk.OneDec(), valConsAddr1)
|
|
||||||
|
|
||||||
// withdraw self-delegation reward
|
|
||||||
ctx = ctx.WithBlockHeight(1)
|
|
||||||
keeper.WithdrawValidatorRewardsAll(ctx, valOpAddr1)
|
|
||||||
amt := accMapper.GetAccount(ctx, valAccAddr1).GetCoins().AmountOf(denom)
|
|
||||||
expRes := sdk.NewDec(90).Add(sdk.NewDec(100)).TruncateInt()
|
|
||||||
require.True(sdk.IntEq(t, expRes, amt))
|
|
||||||
}
|
|
||||||
|
|
||||||
func TestWithdrawValidatorRewardsAllDelegatorNoCommission(t *testing.T) {
|
|
||||||
ctx, accMapper, keeper, sk, fck := CreateTestInputAdvanced(t, false, 100, sdk.ZeroDec())
|
|
||||||
stakingHandler := staking.NewHandler(sk)
|
|
||||||
denom := sk.GetParams(ctx).BondDenom
|
|
||||||
|
|
||||||
//first make a validator
|
|
||||||
msgCreateValidator := staking.NewTestMsgCreateValidator(valOpAddr1, valConsPk1, 10)
|
|
||||||
got := stakingHandler(ctx, msgCreateValidator)
|
|
||||||
require.True(t, got.IsOK(), "expected msg to be ok, got %v", got)
|
|
||||||
_ = sk.ApplyAndReturnValidatorSetUpdates(ctx)
|
|
||||||
|
|
||||||
// delegate
|
|
||||||
msgDelegate := staking.NewTestMsgDelegate(delAddr1, valOpAddr1, 10)
|
|
||||||
got = stakingHandler(ctx, msgDelegate)
|
|
||||||
require.True(t, got.IsOK())
|
|
||||||
amt := accMapper.GetAccount(ctx, delAddr1).GetCoins().AmountOf(denom)
|
|
||||||
require.Equal(t, int64(90), amt.Int64())
|
|
||||||
|
|
||||||
// allocate 100 denom of fees
|
|
||||||
feeInputs := sdk.NewInt(100)
|
|
||||||
fck.SetCollectedFees(sdk.Coins{sdk.NewCoin(denom, feeInputs)})
|
|
||||||
require.Equal(t, feeInputs, fck.GetCollectedFees(ctx).AmountOf(denom))
|
|
||||||
keeper.AllocateTokens(ctx, sdk.OneDec(), valConsAddr1)
|
|
||||||
|
|
||||||
// withdraw self-delegation reward
|
|
||||||
ctx = ctx.WithBlockHeight(1)
|
|
||||||
keeper.WithdrawValidatorRewardsAll(ctx, valOpAddr1)
|
|
||||||
amt = accMapper.GetAccount(ctx, valAccAddr1).GetCoins().AmountOf(denom)
|
|
||||||
expRes := sdk.NewDec(90).Add(sdk.NewDec(100).Quo(sdk.NewDec(2))).TruncateInt() // 90 + 100 tokens * 10/20
|
|
||||||
require.True(sdk.IntEq(t, expRes, amt))
|
|
||||||
}
|
|
||||||
|
|
||||||
func TestWithdrawValidatorRewardsAllDelegatorWithCommission(t *testing.T) {
|
|
||||||
ctx, accMapper, keeper, sk, fck := CreateTestInputAdvanced(t, false, 100, sdk.ZeroDec())
|
|
||||||
stakingHandler := staking.NewHandler(sk)
|
|
||||||
denom := sk.GetParams(ctx).BondDenom
|
|
||||||
|
|
||||||
//first make a validator
|
|
||||||
commissionRate := sdk.NewDecWithPrec(1, 1)
|
|
||||||
msgCreateValidator := staking.NewTestMsgCreateValidatorWithCommission(
|
|
||||||
valOpAddr1, valConsPk1, 10, commissionRate)
|
|
||||||
got := stakingHandler(ctx, msgCreateValidator)
|
|
||||||
require.True(t, got.IsOK(), "expected msg to be ok, got %v", got)
|
|
||||||
_ = sk.ApplyAndReturnValidatorSetUpdates(ctx)
|
|
||||||
|
|
||||||
// delegate
|
|
||||||
msgDelegate := staking.NewTestMsgDelegate(delAddr1, valOpAddr1, 10)
|
|
||||||
got = stakingHandler(ctx, msgDelegate)
|
|
||||||
require.True(t, got.IsOK())
|
|
||||||
amt := accMapper.GetAccount(ctx, delAddr1).GetCoins().AmountOf(denom)
|
|
||||||
require.Equal(t, int64(90), amt.Int64())
|
|
||||||
|
|
||||||
// allocate 100 denom of fees
|
|
||||||
feeInputs := sdk.NewInt(100)
|
|
||||||
fck.SetCollectedFees(sdk.Coins{sdk.NewCoin(denom, feeInputs)})
|
|
||||||
require.Equal(t, feeInputs, fck.GetCollectedFees(ctx).AmountOf(denom))
|
|
||||||
keeper.AllocateTokens(ctx, sdk.OneDec(), valConsAddr1)
|
|
||||||
|
|
||||||
// withdraw validator reward
|
|
||||||
ctx = ctx.WithBlockHeight(1)
|
|
||||||
keeper.WithdrawValidatorRewardsAll(ctx, valOpAddr1)
|
|
||||||
amt = accMapper.GetAccount(ctx, valAccAddr1).GetCoins().AmountOf(denom)
|
|
||||||
commissionTaken := sdk.NewDec(100).Mul(commissionRate)
|
|
||||||
afterCommission := sdk.NewDec(100).Sub(commissionTaken)
|
|
||||||
selfDelegationReward := afterCommission.Quo(sdk.NewDec(2))
|
|
||||||
expRes := sdk.NewDec(90).Add(commissionTaken).Add(selfDelegationReward).TruncateInt() // 90 + 100 tokens * 10/20
|
|
||||||
require.True(sdk.IntEq(t, expRes, amt))
|
|
||||||
}
|
|
||||||
|
|
||||||
func TestWithdrawValidatorRewardsAllMultipleValidator(t *testing.T) {
|
|
||||||
ctx, accMapper, keeper, sk, fck := CreateTestInputAdvanced(t, false, 100, sdk.ZeroDec())
|
|
||||||
stakingHandler := staking.NewHandler(sk)
|
|
||||||
denom := sk.GetParams(ctx).BondDenom
|
|
||||||
|
|
||||||
// Make some validators with different commissions.
|
|
||||||
// Bond 10 of 100 with 0.1 commission.
|
|
||||||
msgCreateValidator := staking.NewTestMsgCreateValidatorWithCommission(
|
|
||||||
valOpAddr1, valConsPk1, 10, sdk.NewDecWithPrec(1, 1))
|
|
||||||
got := stakingHandler(ctx, msgCreateValidator)
|
|
||||||
require.True(t, got.IsOK(), "expected msg to be ok, got %v", got)
|
|
||||||
|
|
||||||
// Bond 50 of 100 with 0.2 commission.
|
|
||||||
msgCreateValidator = staking.NewTestMsgCreateValidatorWithCommission(
|
|
||||||
valOpAddr2, valConsPk2, 50, sdk.NewDecWithPrec(2, 1))
|
|
||||||
got = stakingHandler(ctx, msgCreateValidator)
|
|
||||||
require.True(t, got.IsOK(), "expected msg to be ok, got %v", got)
|
|
||||||
|
|
||||||
// Bond 40 of 100 with 0.3 commission.
|
|
||||||
msgCreateValidator = staking.NewTestMsgCreateValidatorWithCommission(
|
|
||||||
valOpAddr3, valConsPk3, 40, sdk.NewDecWithPrec(3, 1))
|
|
||||||
got = stakingHandler(ctx, msgCreateValidator)
|
|
||||||
require.True(t, got.IsOK(), "expected msg to be ok, got %v", got)
|
|
||||||
|
|
||||||
_ = sk.ApplyAndReturnValidatorSetUpdates(ctx)
|
|
||||||
|
|
||||||
// Allocate 1000 denom of fees.
|
|
||||||
feeInputs := sdk.NewInt(1000)
|
|
||||||
fck.SetCollectedFees(sdk.Coins{sdk.NewCoin(denom, feeInputs)})
|
|
||||||
require.Equal(t, feeInputs, fck.GetCollectedFees(ctx).AmountOf(denom))
|
|
||||||
// Collect proposer reward for 100% of votes.
|
|
||||||
keeper.AllocateTokens(ctx, sdk.OneDec(), valConsAddr1)
|
|
||||||
|
|
||||||
// Withdraw validator reward.
|
|
||||||
ctx = ctx.WithBlockHeight(1)
|
|
||||||
keeper.WithdrawValidatorRewardsAll(ctx, valOpAddr1)
|
|
||||||
amt := accMapper.GetAccount(ctx, valAccAddr1).GetCoins().AmountOf(denom)
|
|
||||||
|
|
||||||
feesInNonProposer := sdk.NewDecFromInt(feeInputs).Mul(sdk.NewDecWithPrec(95, 2))
|
|
||||||
feesInProposer := sdk.NewDecFromInt(feeInputs).Mul(sdk.NewDecWithPrec(5, 2))
|
|
||||||
// NOTE: the non-proposer rewards (95) and proposer rewards (50) add up to
|
|
||||||
// 145. During computation, this is further split into 130.5 and 14.5,
|
|
||||||
// which is the non-commission and commission respectively, but the
|
|
||||||
// commission is for self so the result is just 145.
|
|
||||||
expRes := sdk.NewDec(90). // orig tokens (100) - bonded (10)
|
|
||||||
Add(feesInNonProposer.Quo(sdk.NewDec(10))). // validator 1 has 1/10 total power (non-proposer rewards = 95)
|
|
||||||
Add(feesInProposer). // (proposer rewards = 50)
|
|
||||||
TruncateInt()
|
|
||||||
require.True(sdk.IntEq(t, expRes, amt))
|
|
||||||
}
|
|
||||||
|
|
||||||
func TestWithdrawValidatorRewardsAllMultipleDelegator(t *testing.T) {
|
|
||||||
ctx, accMapper, keeper, sk, fck := CreateTestInputAdvanced(t, false, 100, sdk.ZeroDec())
|
|
||||||
stakingHandler := staking.NewHandler(sk)
|
|
||||||
denom := sk.GetParams(ctx).BondDenom
|
|
||||||
|
|
||||||
//first make a validator with 10% commission
|
|
||||||
commissionRate := sdk.NewDecWithPrec(1, 1)
|
|
||||||
msgCreateValidator := staking.NewTestMsgCreateValidatorWithCommission(
|
|
||||||
valOpAddr1, valConsPk1, 10, sdk.NewDecWithPrec(1, 1))
|
|
||||||
got := stakingHandler(ctx, msgCreateValidator)
|
|
||||||
require.True(t, got.IsOK(), "expected msg to be ok, got %v", got)
|
|
||||||
_ = sk.ApplyAndReturnValidatorSetUpdates(ctx)
|
|
||||||
|
|
||||||
// delegate
|
|
||||||
msgDelegate := staking.NewTestMsgDelegate(delAddr1, valOpAddr1, 10)
|
|
||||||
got = stakingHandler(ctx, msgDelegate)
|
|
||||||
require.True(t, got.IsOK())
|
|
||||||
amt := accMapper.GetAccount(ctx, delAddr1).GetCoins().AmountOf(denom)
|
|
||||||
require.Equal(t, int64(90), amt.Int64())
|
|
||||||
|
|
||||||
msgDelegate = staking.NewTestMsgDelegate(delAddr2, valOpAddr1, 20)
|
|
||||||
got = stakingHandler(ctx, msgDelegate)
|
|
||||||
require.True(t, got.IsOK())
|
|
||||||
amt = accMapper.GetAccount(ctx, delAddr2).GetCoins().AmountOf(denom)
|
|
||||||
require.Equal(t, int64(80), amt.Int64())
|
|
||||||
|
|
||||||
// allocate 100 denom of fees
|
|
||||||
feeInputs := sdk.NewInt(100)
|
|
||||||
fck.SetCollectedFees(sdk.Coins{sdk.NewCoin(denom, feeInputs)})
|
|
||||||
require.Equal(t, feeInputs, fck.GetCollectedFees(ctx).AmountOf(denom))
|
|
||||||
keeper.AllocateTokens(ctx, sdk.OneDec(), valConsAddr1)
|
|
||||||
|
|
||||||
// withdraw validator reward
|
|
||||||
ctx = ctx.WithBlockHeight(1)
|
|
||||||
keeper.WithdrawValidatorRewardsAll(ctx, valOpAddr1)
|
|
||||||
amt = accMapper.GetAccount(ctx, valAccAddr1).GetCoins().AmountOf(denom)
|
|
||||||
|
|
||||||
commissionTaken := sdk.NewDec(100).Mul(commissionRate)
|
|
||||||
afterCommission := sdk.NewDec(100).Sub(commissionTaken)
|
|
||||||
expRes := sdk.NewDec(90).
|
|
||||||
Add(afterCommission.Quo(sdk.NewDec(4))).
|
|
||||||
Add(commissionTaken).
|
|
||||||
TruncateInt() // 90 + 100*90% tokens * 10/40
|
|
||||||
require.True(sdk.IntEq(t, expRes, amt))
|
|
||||||
}
|
|
|
@ -10,19 +10,13 @@ import (
|
||||||
)
|
)
|
||||||
|
|
||||||
// AllInvariants runs all invariants of the distribution module
|
// AllInvariants runs all invariants of the distribution module
|
||||||
// Currently: total supply, positive power
|
|
||||||
func AllInvariants(d distr.Keeper, stk staking.Keeper) simulation.Invariant {
|
func AllInvariants(d distr.Keeper, stk staking.Keeper) simulation.Invariant {
|
||||||
sk := distr.StakingKeeper(stk)
|
|
||||||
return func(ctx sdk.Context) error {
|
return func(ctx sdk.Context) error {
|
||||||
err := ValAccumInvariants(d, sk)(ctx)
|
err := CanWithdrawInvariant(d, stk)(ctx)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
err = DelAccumInvariants(d, sk)(ctx)
|
err = NonNegativeOutstandingInvariant(d)(ctx)
|
||||||
if err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
err = CanWithdrawInvariant(d, stk)(ctx)
|
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
|
@ -30,104 +24,13 @@ func AllInvariants(d distr.Keeper, stk staking.Keeper) simulation.Invariant {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// ValAccumInvariants checks that the fee pool accum == sum all validators' accum
|
// NonNegativeOutstandingInvariant checks that outstanding unwithdrawn fees are never negative
|
||||||
func ValAccumInvariants(k distr.Keeper, sk distr.StakingKeeper) simulation.Invariant {
|
func NonNegativeOutstandingInvariant(k distr.Keeper) simulation.Invariant {
|
||||||
|
|
||||||
return func(ctx sdk.Context) error {
|
return func(ctx sdk.Context) error {
|
||||||
height := ctx.BlockHeight()
|
outstanding := k.GetOutstandingRewards(ctx)
|
||||||
|
if outstanding.HasNegative() {
|
||||||
valAccum := sdk.ZeroDec()
|
return fmt.Errorf("Negative outstanding coins: %v", outstanding)
|
||||||
k.IterateValidatorDistInfos(ctx, func(_ int64, vdi distr.ValidatorDistInfo) bool {
|
|
||||||
lastValPower := sk.GetLastValidatorPower(ctx, vdi.OperatorAddr)
|
|
||||||
valAccum = valAccum.Add(vdi.GetValAccum(height, sdk.NewDecFromInt(lastValPower)))
|
|
||||||
return false
|
|
||||||
})
|
|
||||||
|
|
||||||
lastTotalPower := sdk.NewDecFromInt(sk.GetLastTotalPower(ctx))
|
|
||||||
totalAccum := k.GetFeePool(ctx).GetTotalValAccum(height, lastTotalPower)
|
|
||||||
|
|
||||||
if !totalAccum.Equal(valAccum) {
|
|
||||||
return fmt.Errorf("validator accum invariance: \n\tfee pool totalAccum: %v"+
|
|
||||||
"\n\tvalidator accum \t%v\n", totalAccum.String(), valAccum.String())
|
|
||||||
}
|
}
|
||||||
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// DelAccumInvariants checks that each validator del accum == sum all delegators' accum
|
|
||||||
func DelAccumInvariants(k distr.Keeper, sk distr.StakingKeeper) simulation.Invariant {
|
|
||||||
|
|
||||||
return func(ctx sdk.Context) error {
|
|
||||||
height := ctx.BlockHeight()
|
|
||||||
|
|
||||||
totalDelAccumFromVal := make(map[string]sdk.Dec) // key is the valOpAddr string
|
|
||||||
totalDelAccum := make(map[string]sdk.Dec)
|
|
||||||
|
|
||||||
// iterate the validators
|
|
||||||
iterVal := func(_ int64, vdi distr.ValidatorDistInfo) bool {
|
|
||||||
key := vdi.OperatorAddr.String()
|
|
||||||
validator := sk.Validator(ctx, vdi.OperatorAddr)
|
|
||||||
totalDelAccumFromVal[key] = vdi.GetTotalDelAccum(height,
|
|
||||||
validator.GetDelegatorShares())
|
|
||||||
|
|
||||||
// also initialize the delegation map
|
|
||||||
totalDelAccum[key] = sdk.ZeroDec()
|
|
||||||
|
|
||||||
return false
|
|
||||||
}
|
|
||||||
k.IterateValidatorDistInfos(ctx, iterVal)
|
|
||||||
|
|
||||||
// iterate the delegations
|
|
||||||
iterDel := func(_ int64, ddi distr.DelegationDistInfo) bool {
|
|
||||||
key := ddi.ValOperatorAddr.String()
|
|
||||||
delegation := sk.Delegation(ctx, ddi.DelegatorAddr, ddi.ValOperatorAddr)
|
|
||||||
totalDelAccum[key] = totalDelAccum[key].Add(
|
|
||||||
ddi.GetDelAccum(height, delegation.GetShares()))
|
|
||||||
return false
|
|
||||||
}
|
|
||||||
k.IterateDelegationDistInfos(ctx, iterDel)
|
|
||||||
|
|
||||||
// compare
|
|
||||||
for key, delAccumFromVal := range totalDelAccumFromVal {
|
|
||||||
sumDelAccum := totalDelAccum[key]
|
|
||||||
|
|
||||||
if !sumDelAccum.Equal(delAccumFromVal) {
|
|
||||||
|
|
||||||
logDelAccums := ""
|
|
||||||
iterDel := func(_ int64, ddi distr.DelegationDistInfo) bool {
|
|
||||||
keyLog := ddi.ValOperatorAddr.String()
|
|
||||||
if keyLog == key {
|
|
||||||
delegation := sk.Delegation(ctx, ddi.DelegatorAddr, ddi.ValOperatorAddr)
|
|
||||||
accum := ddi.GetDelAccum(height, delegation.GetShares())
|
|
||||||
if accum.IsPositive() {
|
|
||||||
logDelAccums += fmt.Sprintf("\n\t\tdel: %v, accum: %v, power: %v",
|
|
||||||
ddi.DelegatorAddr.String(),
|
|
||||||
accum.String(),
|
|
||||||
delegation.GetShares())
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return false
|
|
||||||
}
|
|
||||||
k.IterateDelegationDistInfos(ctx, iterDel)
|
|
||||||
|
|
||||||
operAddr, err := sdk.ValAddressFromBech32(key)
|
|
||||||
if err != nil {
|
|
||||||
panic(err)
|
|
||||||
}
|
|
||||||
validator := sk.Validator(ctx, operAddr)
|
|
||||||
|
|
||||||
return fmt.Errorf("delegator accum invariance: \n"+
|
|
||||||
"\tvalidator key: %v\n"+
|
|
||||||
"\tvalidator: %+v\n"+
|
|
||||||
"\tsum delegator accum: %v\n"+
|
|
||||||
"\tvalidator's total delegator accum: %v\n"+
|
|
||||||
"\tlog of delegations with accum: %v\n",
|
|
||||||
key, validator, sumDelAccum.String(),
|
|
||||||
delAccumFromVal.String(), logDelAccums)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -135,41 +38,28 @@ func DelAccumInvariants(k distr.Keeper, sk distr.StakingKeeper) simulation.Invar
|
||||||
// CanWithdrawInvariant checks that current rewards can be completely withdrawn
|
// CanWithdrawInvariant checks that current rewards can be completely withdrawn
|
||||||
func CanWithdrawInvariant(k distr.Keeper, sk staking.Keeper) simulation.Invariant {
|
func CanWithdrawInvariant(k distr.Keeper, sk staking.Keeper) simulation.Invariant {
|
||||||
return func(ctx sdk.Context) error {
|
return func(ctx sdk.Context) error {
|
||||||
// we don't want to write the changes
|
|
||||||
|
// cache, we don't want to write changes
|
||||||
ctx, _ = ctx.CacheContext()
|
ctx, _ = ctx.CacheContext()
|
||||||
|
|
||||||
// withdraw all delegator & validator rewards
|
// iterate over all bonded validators, withdraw commission
|
||||||
vdiIter := func(_ int64, valInfo distr.ValidatorDistInfo) (stop bool) {
|
sk.IterateValidators(ctx, func(_ int64, val sdk.Validator) (stop bool) {
|
||||||
err := k.WithdrawValidatorRewardsAll(ctx, valInfo.OperatorAddr)
|
_ = k.WithdrawValidatorCommission(ctx, val.GetOperator())
|
||||||
if err != nil {
|
|
||||||
panic(err)
|
|
||||||
}
|
|
||||||
return false
|
return false
|
||||||
}
|
})
|
||||||
k.IterateValidatorDistInfos(ctx, vdiIter)
|
|
||||||
|
|
||||||
ddiIter := func(_ int64, distInfo distr.DelegationDistInfo) (stop bool) {
|
// iterate over all current delegations, withdraw rewards
|
||||||
err := k.WithdrawDelegationReward(
|
dels := sk.GetAllDelegations(ctx)
|
||||||
ctx, distInfo.DelegatorAddr, distInfo.ValOperatorAddr)
|
for _, delegation := range dels {
|
||||||
if err != nil {
|
_ = k.WithdrawDelegationRewards(ctx, delegation.DelegatorAddr, delegation.ValidatorAddr)
|
||||||
panic(err)
|
}
|
||||||
}
|
|
||||||
return false
|
remaining := k.GetOutstandingRewards(ctx)
|
||||||
}
|
|
||||||
k.IterateDelegationDistInfos(ctx, ddiIter)
|
if len(remaining) > 0 && remaining[0].Amount.LT(sdk.ZeroDec()) {
|
||||||
|
return fmt.Errorf("Negative remaining coins: %v", remaining)
|
||||||
// assert that the fee pool is empty
|
|
||||||
feePool := k.GetFeePool(ctx)
|
|
||||||
if !feePool.TotalValAccum.Accum.IsZero() {
|
|
||||||
return fmt.Errorf("unexpected leftover validator accum")
|
|
||||||
}
|
|
||||||
bondDenom := sk.GetParams(ctx).BondDenom
|
|
||||||
if !feePool.ValPool.AmountOf(bondDenom).IsZero() {
|
|
||||||
return fmt.Errorf("unexpected leftover validator pool coins: %v",
|
|
||||||
feePool.ValPool.AmountOf(bondDenom).String())
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// all ok
|
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -39,33 +39,6 @@ func SimulateMsgSetWithdrawAddress(m auth.AccountKeeper, k distribution.Keeper)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// SimulateMsgWithdrawDelegatorRewardsAll
|
|
||||||
func SimulateMsgWithdrawDelegatorRewardsAll(m auth.AccountKeeper, k distribution.Keeper) simulation.Operation {
|
|
||||||
handler := distribution.NewHandler(k)
|
|
||||||
return func(r *rand.Rand, app *baseapp.BaseApp, ctx sdk.Context,
|
|
||||||
accs []simulation.Account, event func(string)) (
|
|
||||||
action string, fOp []simulation.FutureOperation, err error) {
|
|
||||||
|
|
||||||
account := simulation.RandomAcc(r, accs)
|
|
||||||
msg := distribution.NewMsgWithdrawDelegatorRewardsAll(account.Address)
|
|
||||||
|
|
||||||
if msg.ValidateBasic() != nil {
|
|
||||||
return "", nil, fmt.Errorf("expected msg to pass ValidateBasic: %s", msg.GetSignBytes())
|
|
||||||
}
|
|
||||||
|
|
||||||
ctx, write := ctx.CacheContext()
|
|
||||||
result := handler(ctx, msg)
|
|
||||||
if result.IsOK() {
|
|
||||||
write()
|
|
||||||
}
|
|
||||||
|
|
||||||
event(fmt.Sprintf("distribution/MsgWithdrawDelegatorRewardsAll/%v", result.IsOK()))
|
|
||||||
|
|
||||||
action = fmt.Sprintf("TestMsgWithdrawDelegatorRewardsAll: ok %v, msg %s", result.IsOK(), msg.GetSignBytes())
|
|
||||||
return action, nil, nil
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// SimulateMsgWithdrawDelegatorReward
|
// SimulateMsgWithdrawDelegatorReward
|
||||||
func SimulateMsgWithdrawDelegatorReward(m auth.AccountKeeper, k distribution.Keeper) simulation.Operation {
|
func SimulateMsgWithdrawDelegatorReward(m auth.AccountKeeper, k distribution.Keeper) simulation.Operation {
|
||||||
handler := distribution.NewHandler(k)
|
handler := distribution.NewHandler(k)
|
||||||
|
@ -94,15 +67,15 @@ func SimulateMsgWithdrawDelegatorReward(m auth.AccountKeeper, k distribution.Kee
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// SimulateMsgWithdrawValidatorRewardsAll
|
// SimulateMsgWithdrawValidatorCommission
|
||||||
func SimulateMsgWithdrawValidatorRewardsAll(m auth.AccountKeeper, k distribution.Keeper) simulation.Operation {
|
func SimulateMsgWithdrawValidatorCommission(m auth.AccountKeeper, k distribution.Keeper) simulation.Operation {
|
||||||
handler := distribution.NewHandler(k)
|
handler := distribution.NewHandler(k)
|
||||||
return func(r *rand.Rand, app *baseapp.BaseApp, ctx sdk.Context,
|
return func(r *rand.Rand, app *baseapp.BaseApp, ctx sdk.Context,
|
||||||
accs []simulation.Account, event func(string)) (
|
accs []simulation.Account, event func(string)) (
|
||||||
action string, fOp []simulation.FutureOperation, err error) {
|
action string, fOp []simulation.FutureOperation, err error) {
|
||||||
|
|
||||||
account := simulation.RandomAcc(r, accs)
|
account := simulation.RandomAcc(r, accs)
|
||||||
msg := distribution.NewMsgWithdrawValidatorRewardsAll(sdk.ValAddress(account.Address))
|
msg := distribution.NewMsgWithdrawValidatorCommission(sdk.ValAddress(account.Address))
|
||||||
|
|
||||||
if msg.ValidateBasic() != nil {
|
if msg.ValidateBasic() != nil {
|
||||||
return "", nil, fmt.Errorf("expected msg to pass ValidateBasic: %s", msg.GetSignBytes())
|
return "", nil, fmt.Errorf("expected msg to pass ValidateBasic: %s", msg.GetSignBytes())
|
||||||
|
@ -114,9 +87,9 @@ func SimulateMsgWithdrawValidatorRewardsAll(m auth.AccountKeeper, k distribution
|
||||||
write()
|
write()
|
||||||
}
|
}
|
||||||
|
|
||||||
event(fmt.Sprintf("distribution/MsgWithdrawValidatorRewardsAll/%v", result.IsOK()))
|
event(fmt.Sprintf("distribution/MsgWithdrawValidatorCommission/%v", result.IsOK()))
|
||||||
|
|
||||||
action = fmt.Sprintf("TestMsgWithdrawValidatorRewardsAll: ok %v, msg %s", result.IsOK(), msg.GetSignBytes())
|
action = fmt.Sprintf("TestMsgWithdrawValidatorCommission: ok %v, msg %s", result.IsOK(), msg.GetSignBytes())
|
||||||
return action, nil, nil
|
return action, nil, nil
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -6,9 +6,8 @@ import (
|
||||||
|
|
||||||
// Register concrete types on codec codec
|
// Register concrete types on codec codec
|
||||||
func RegisterCodec(cdc *codec.Codec) {
|
func RegisterCodec(cdc *codec.Codec) {
|
||||||
cdc.RegisterConcrete(MsgWithdrawDelegatorRewardsAll{}, "cosmos-sdk/MsgWithdrawDelegationRewardsAll", nil)
|
|
||||||
cdc.RegisterConcrete(MsgWithdrawDelegatorReward{}, "cosmos-sdk/MsgWithdrawDelegationReward", nil)
|
cdc.RegisterConcrete(MsgWithdrawDelegatorReward{}, "cosmos-sdk/MsgWithdrawDelegationReward", nil)
|
||||||
cdc.RegisterConcrete(MsgWithdrawValidatorRewardsAll{}, "cosmos-sdk/MsgWithdrawValidatorRewardsAll", nil)
|
cdc.RegisterConcrete(MsgWithdrawValidatorCommission{}, "cosmos-sdk/MsgWithdrawValidatorCommission", nil)
|
||||||
cdc.RegisterConcrete(MsgSetWithdrawAddress{}, "cosmos-sdk/MsgModifyWithdrawAddress", nil)
|
cdc.RegisterConcrete(MsgSetWithdrawAddress{}, "cosmos-sdk/MsgModifyWithdrawAddress", nil)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -1,122 +0,0 @@
|
||||||
package types
|
|
||||||
|
|
||||||
import (
|
|
||||||
"fmt"
|
|
||||||
|
|
||||||
sdk "github.com/cosmos/cosmos-sdk/types"
|
|
||||||
)
|
|
||||||
|
|
||||||
// distribution info for a delegation - used to determine entitled rewards
|
|
||||||
type DelegationDistInfo struct {
|
|
||||||
DelegatorAddr sdk.AccAddress `json:"delegator_addr"`
|
|
||||||
ValOperatorAddr sdk.ValAddress `json:"val_operator_addr"`
|
|
||||||
DelPoolWithdrawalHeight int64 `json:"del_pool_withdrawal_height"` // last time this delegation withdrew rewards
|
|
||||||
}
|
|
||||||
|
|
||||||
func NewDelegationDistInfo(delegatorAddr sdk.AccAddress, valOperatorAddr sdk.ValAddress,
|
|
||||||
currentHeight int64) DelegationDistInfo {
|
|
||||||
|
|
||||||
return DelegationDistInfo{
|
|
||||||
DelegatorAddr: delegatorAddr,
|
|
||||||
ValOperatorAddr: valOperatorAddr,
|
|
||||||
DelPoolWithdrawalHeight: currentHeight,
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// Get the calculated accum of this delegator at the provided height
|
|
||||||
func (di DelegationDistInfo) GetDelAccum(height int64, delegatorShares sdk.Dec) sdk.Dec {
|
|
||||||
blocks := height - di.DelPoolWithdrawalHeight
|
|
||||||
accum := delegatorShares.MulInt(sdk.NewInt(blocks))
|
|
||||||
|
|
||||||
// defensive check
|
|
||||||
if accum.IsNegative() {
|
|
||||||
panic(fmt.Sprintf("negative accum: %v\n"+
|
|
||||||
"\theight: %v\n"+
|
|
||||||
"\tdelegation_dist_info: %v\n"+
|
|
||||||
"\tdelegator_shares: %v\n",
|
|
||||||
accum.String(), height, di, delegatorShares))
|
|
||||||
}
|
|
||||||
return accum
|
|
||||||
}
|
|
||||||
|
|
||||||
// Withdraw rewards from delegator.
|
|
||||||
// Among many things, it does:
|
|
||||||
// * updates validator info's total del accum
|
|
||||||
// * calls vi.TakeFeePoolRewards, which:
|
|
||||||
// * updates validator info's FeePoolWithdrawalHeight, thus setting accum to 0
|
|
||||||
// * updates fee pool to latest height and total val accum w/ given totalBonded
|
|
||||||
// (see comment on TakeFeePoolRewards for more info)
|
|
||||||
func (di DelegationDistInfo) WithdrawRewards(wc WithdrawContext, vi ValidatorDistInfo,
|
|
||||||
totalDelShares, delegatorShares sdk.Dec) (
|
|
||||||
DelegationDistInfo, ValidatorDistInfo, FeePool, DecCoins) {
|
|
||||||
|
|
||||||
fp := wc.FeePool
|
|
||||||
vi = vi.UpdateTotalDelAccum(wc.Height, totalDelShares)
|
|
||||||
|
|
||||||
// Break out to prevent a divide by zero.
|
|
||||||
if vi.DelAccum.Accum.IsZero() {
|
|
||||||
di.DelPoolWithdrawalHeight = wc.Height
|
|
||||||
return di, vi, fp, DecCoins{}
|
|
||||||
}
|
|
||||||
|
|
||||||
vi, fp = vi.TakeFeePoolRewards(wc)
|
|
||||||
|
|
||||||
accum := di.GetDelAccum(wc.Height, delegatorShares)
|
|
||||||
di.DelPoolWithdrawalHeight = wc.Height
|
|
||||||
|
|
||||||
withdrawalTokens := vi.DelPool.MulDec(accum).QuoDec(vi.DelAccum.Accum)
|
|
||||||
|
|
||||||
// Clip withdrawal tokens by pool, due to possible rounding errors.
|
|
||||||
// This rounding error may be introduced upon multiplication since
|
|
||||||
// we're clipping decimal digits, and then when we divide by a number ~1 or
|
|
||||||
// < 1, the error doesn't get "buried", and if << 1 it'll get amplified.
|
|
||||||
// more: https://github.com/cosmos/cosmos-sdk/issues/2888#issuecomment-441387987
|
|
||||||
for i, decCoin := range withdrawalTokens {
|
|
||||||
poolDenomAmount := vi.DelPool.AmountOf(decCoin.Denom)
|
|
||||||
if decCoin.Amount.GT(poolDenomAmount) {
|
|
||||||
withdrawalTokens[i] = NewDecCoinFromDec(decCoin.Denom, poolDenomAmount)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// defensive check for impossible accum ratios
|
|
||||||
if accum.GT(vi.DelAccum.Accum) {
|
|
||||||
panic(fmt.Sprintf("accum > vi.DelAccum.Accum:\n"+
|
|
||||||
"\taccum\t\t\t%v\n"+
|
|
||||||
"\tvi.DelAccum.Accum\t%v\n",
|
|
||||||
accum, vi.DelAccum.Accum))
|
|
||||||
}
|
|
||||||
|
|
||||||
remDelPool := vi.DelPool.Minus(withdrawalTokens)
|
|
||||||
|
|
||||||
// defensive check
|
|
||||||
if remDelPool.HasNegative() {
|
|
||||||
panic(fmt.Sprintf("negative remDelPool: %v\n"+
|
|
||||||
"\tvi.DelPool\t\t%v\n"+
|
|
||||||
"\taccum\t\t\t%v\n"+
|
|
||||||
"\tvi.DelAccum.Accum\t%v\n"+
|
|
||||||
"\twithdrawalTokens\t%v\n",
|
|
||||||
remDelPool, vi.DelPool, accum,
|
|
||||||
vi.DelAccum.Accum, withdrawalTokens))
|
|
||||||
}
|
|
||||||
|
|
||||||
vi.DelPool = remDelPool
|
|
||||||
vi.DelAccum.Accum = vi.DelAccum.Accum.Sub(accum)
|
|
||||||
|
|
||||||
return di, vi, fp, withdrawalTokens
|
|
||||||
}
|
|
||||||
|
|
||||||
// get the delegators rewards at this current state,
|
|
||||||
func (di DelegationDistInfo) CurrentRewards(wc WithdrawContext, vi ValidatorDistInfo,
|
|
||||||
totalDelShares, delegatorShares sdk.Dec) DecCoins {
|
|
||||||
|
|
||||||
totalDelAccum := vi.GetTotalDelAccum(wc.Height, totalDelShares)
|
|
||||||
|
|
||||||
if vi.DelAccum.Accum.IsZero() {
|
|
||||||
return DecCoins{}
|
|
||||||
}
|
|
||||||
|
|
||||||
rewards := vi.CurrentPoolRewards(wc)
|
|
||||||
accum := di.GetDelAccum(wc.Height, delegatorShares)
|
|
||||||
tokens := rewards.MulDec(accum).QuoDec(totalDelAccum)
|
|
||||||
return tokens
|
|
||||||
}
|
|
|
@ -1,62 +0,0 @@
|
||||||
package types
|
|
||||||
|
|
||||||
import (
|
|
||||||
"testing"
|
|
||||||
|
|
||||||
"github.com/stretchr/testify/assert"
|
|
||||||
|
|
||||||
sdk "github.com/cosmos/cosmos-sdk/types"
|
|
||||||
stakingTypes "github.com/cosmos/cosmos-sdk/x/staking/types"
|
|
||||||
)
|
|
||||||
|
|
||||||
func TestWithdrawRewards(t *testing.T) {
|
|
||||||
|
|
||||||
// initialize
|
|
||||||
height := int64(0)
|
|
||||||
fp := InitialFeePool()
|
|
||||||
vi := NewValidatorDistInfo(valAddr1, height)
|
|
||||||
commissionRate := sdk.NewDecWithPrec(2, 2)
|
|
||||||
validatorTokens := sdk.NewDec(10)
|
|
||||||
validatorDelShares := sdk.NewDec(10)
|
|
||||||
totalBondedTokens := validatorTokens.Add(sdk.NewDec(90)) // validator-1 is 10% of total power
|
|
||||||
|
|
||||||
di1 := NewDelegationDistInfo(delAddr1, valAddr1, height)
|
|
||||||
di1Shares := sdk.NewDec(5) // this delegator has half the shares in the validator
|
|
||||||
|
|
||||||
di2 := NewDelegationDistInfo(delAddr2, valAddr1, height)
|
|
||||||
di2Shares := sdk.NewDec(5)
|
|
||||||
|
|
||||||
// simulate adding some stake for inflation
|
|
||||||
height = 10
|
|
||||||
fp.ValPool = DecCoins{NewDecCoin(stakingTypes.DefaultBondDenom, 1000)}
|
|
||||||
|
|
||||||
// withdraw rewards
|
|
||||||
wc := NewWithdrawContext(fp, height,
|
|
||||||
totalBondedTokens, validatorTokens, commissionRate)
|
|
||||||
di1, vi, fp, rewardRecv1 := di1.WithdrawRewards(wc, vi,
|
|
||||||
validatorDelShares, di1Shares)
|
|
||||||
|
|
||||||
assert.Equal(t, height, di1.DelPoolWithdrawalHeight)
|
|
||||||
assert.True(sdk.DecEq(t, sdk.NewDec(900), fp.TotalValAccum.Accum))
|
|
||||||
assert.True(sdk.DecEq(t, sdk.NewDec(900), fp.ValPool[0].Amount))
|
|
||||||
assert.True(sdk.DecEq(t, sdk.NewDec(49), vi.DelPool[0].Amount))
|
|
||||||
assert.True(sdk.DecEq(t, sdk.NewDec(2), vi.ValCommission[0].Amount))
|
|
||||||
assert.True(sdk.DecEq(t, sdk.NewDec(49), rewardRecv1[0].Amount))
|
|
||||||
|
|
||||||
// add more blocks and inflation
|
|
||||||
height = 20
|
|
||||||
fp.ValPool[0].Amount = fp.ValPool[0].Amount.Add(sdk.NewDec(1000))
|
|
||||||
|
|
||||||
// withdraw rewards
|
|
||||||
wc = NewWithdrawContext(fp, height,
|
|
||||||
totalBondedTokens, validatorTokens, commissionRate)
|
|
||||||
di2, vi, fp, rewardRecv2 := di2.WithdrawRewards(wc, vi,
|
|
||||||
validatorDelShares, di2Shares)
|
|
||||||
|
|
||||||
assert.Equal(t, height, di2.DelPoolWithdrawalHeight)
|
|
||||||
assert.True(sdk.DecEq(t, sdk.NewDec(1800), fp.TotalValAccum.Accum))
|
|
||||||
assert.True(sdk.DecEq(t, sdk.NewDec(1800), fp.ValPool[0].Amount))
|
|
||||||
assert.True(sdk.DecEq(t, sdk.NewDec(49), vi.DelPool[0].Amount))
|
|
||||||
assert.True(sdk.DecEq(t, sdk.NewDec(4), vi.ValCommission[0].Amount))
|
|
||||||
assert.True(sdk.DecEq(t, sdk.NewDec(98), rewardRecv2[0].Amount))
|
|
||||||
}
|
|
|
@ -0,0 +1,27 @@
|
||||||
|
package types
|
||||||
|
|
||||||
|
import (
|
||||||
|
sdk "github.com/cosmos/cosmos-sdk/types"
|
||||||
|
)
|
||||||
|
|
||||||
|
// starting info for a delegator reward period
|
||||||
|
// tracks the previous validator period, the delegation's amount
|
||||||
|
// of staking token, and the creation height (to check later on
|
||||||
|
// if any slashes have occurred)
|
||||||
|
// NOTE that even though validators are slashed to whole staking tokens, the
|
||||||
|
// delegators within the validator may be left with less than a full token,
|
||||||
|
// thus sdk.Dec is used
|
||||||
|
type DelegatorStartingInfo struct {
|
||||||
|
PreviousPeriod uint64 `json:"previous_period"` // period at which the delegation should withdraw starting from
|
||||||
|
Stake sdk.Dec `json:"stake"` // amount of staking token delegated
|
||||||
|
Height uint64 `json:"height"` // height at which delegation was created
|
||||||
|
}
|
||||||
|
|
||||||
|
// create a new DelegatorStartingInfo
|
||||||
|
func NewDelegatorStartingInfo(previousPeriod uint64, stake sdk.Dec, height uint64) DelegatorStartingInfo {
|
||||||
|
return DelegatorStartingInfo{
|
||||||
|
PreviousPeriod: previousPeriod,
|
||||||
|
Stake: stake,
|
||||||
|
Height: height,
|
||||||
|
}
|
||||||
|
}
|
|
@ -8,9 +8,10 @@ import (
|
||||||
type CodeType = sdk.CodeType
|
type CodeType = sdk.CodeType
|
||||||
|
|
||||||
const (
|
const (
|
||||||
DefaultCodespace sdk.CodespaceType = "DISTR"
|
DefaultCodespace sdk.CodespaceType = "DISTR"
|
||||||
CodeInvalidInput CodeType = 103
|
CodeInvalidInput CodeType = 103
|
||||||
CodeNoDistributionInfo CodeType = 104
|
CodeNoDistributionInfo CodeType = 104
|
||||||
|
CodeNoValidatorCommission CodeType = 105
|
||||||
)
|
)
|
||||||
|
|
||||||
func ErrNilDelegatorAddr(codespace sdk.CodespaceType) sdk.Error {
|
func ErrNilDelegatorAddr(codespace sdk.CodespaceType) sdk.Error {
|
||||||
|
@ -28,3 +29,6 @@ func ErrNoDelegationDistInfo(codespace sdk.CodespaceType) sdk.Error {
|
||||||
func ErrNoValidatorDistInfo(codespace sdk.CodespaceType) sdk.Error {
|
func ErrNoValidatorDistInfo(codespace sdk.CodespaceType) sdk.Error {
|
||||||
return sdk.NewError(codespace, CodeNoDistributionInfo, "no validator distribution info")
|
return sdk.NewError(codespace, CodeNoDistributionInfo, "no validator distribution info")
|
||||||
}
|
}
|
||||||
|
func ErrNoValidatorCommission(codespace sdk.CodespaceType) sdk.Error {
|
||||||
|
return sdk.NewError(codespace, CodeNoValidatorCommission, "no validator commission to withdraw")
|
||||||
|
}
|
||||||
|
|
|
@ -8,46 +8,18 @@ import (
|
||||||
|
|
||||||
// global fee pool for distribution
|
// global fee pool for distribution
|
||||||
type FeePool struct {
|
type FeePool struct {
|
||||||
TotalValAccum TotalAccum `json:"val_accum"` // total valdator accum held by validators
|
CommunityPool sdk.DecCoins `json:"community_pool"` // pool for community funds yet to be spent
|
||||||
ValPool DecCoins `json:"val_pool"` // funds for all validators which have yet to be withdrawn
|
|
||||||
CommunityPool DecCoins `json:"community_pool"` // pool for community funds yet to be spent
|
|
||||||
}
|
|
||||||
|
|
||||||
// update total validator accumulation factor
|
|
||||||
// NOTE: Do not call this except from ValidatorDistInfo.TakeFeePoolRewards().
|
|
||||||
func (f FeePool) UpdateTotalValAccum(height int64, totalBondedTokens sdk.Dec) FeePool {
|
|
||||||
f.TotalValAccum = f.TotalValAccum.UpdateForNewHeight(height, totalBondedTokens)
|
|
||||||
return f
|
|
||||||
}
|
|
||||||
|
|
||||||
// get the total validator accum for the fee pool without modifying the state
|
|
||||||
func (f FeePool) GetTotalValAccum(height int64, totalBondedTokens sdk.Dec) sdk.Dec {
|
|
||||||
return f.TotalValAccum.GetAccum(height, totalBondedTokens)
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// zero fee pool
|
// zero fee pool
|
||||||
func InitialFeePool() FeePool {
|
func InitialFeePool() FeePool {
|
||||||
return FeePool{
|
return FeePool{
|
||||||
TotalValAccum: NewTotalAccum(0),
|
CommunityPool: sdk.DecCoins{},
|
||||||
ValPool: DecCoins{},
|
|
||||||
CommunityPool: DecCoins{},
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// ValidateGenesis validates the fee pool for a genesis state
|
// ValidateGenesis validates the fee pool for a genesis state
|
||||||
func (f FeePool) ValidateGenesis() error {
|
func (f FeePool) ValidateGenesis() error {
|
||||||
if f.TotalValAccum.Accum.IsNegative() {
|
|
||||||
return fmt.Errorf("negative accum in distribution fee pool, is %v",
|
|
||||||
f.TotalValAccum.Accum.String())
|
|
||||||
}
|
|
||||||
if f.TotalValAccum.UpdateHeight < 0 {
|
|
||||||
return fmt.Errorf("negative update height in distribution fee pool, is %v",
|
|
||||||
f.TotalValAccum.UpdateHeight)
|
|
||||||
}
|
|
||||||
if f.ValPool.HasNegative() {
|
|
||||||
return fmt.Errorf("negative ValPool in distribution fee pool, is %v",
|
|
||||||
f.ValPool)
|
|
||||||
}
|
|
||||||
if f.CommunityPool.HasNegative() {
|
if f.CommunityPool.HasNegative() {
|
||||||
return fmt.Errorf("negative CommunityPool in distribution fee pool, is %v",
|
return fmt.Errorf("negative CommunityPool in distribution fee pool, is %v",
|
||||||
f.CommunityPool)
|
f.CommunityPool)
|
||||||
|
@ -55,3 +27,8 @@ func (f FeePool) ValidateGenesis() error {
|
||||||
|
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// outstanding (un-withdrawn) rewards for everyone
|
||||||
|
// excludes the community pool
|
||||||
|
// inexpensive to track, allows simple sanity checks
|
||||||
|
type OutstandingRewards = sdk.DecCoins
|
||||||
|
|
|
@ -8,13 +8,12 @@ import (
|
||||||
sdk "github.com/cosmos/cosmos-sdk/types"
|
sdk "github.com/cosmos/cosmos-sdk/types"
|
||||||
)
|
)
|
||||||
|
|
||||||
func TestUpdateTotalValAccum(t *testing.T) {
|
func TestValidateGenesis(t *testing.T) {
|
||||||
|
|
||||||
fp := InitialFeePool()
|
fp := InitialFeePool()
|
||||||
|
require.Nil(t, fp.ValidateGenesis())
|
||||||
|
|
||||||
fp = fp.UpdateTotalValAccum(5, sdk.NewDec(3))
|
fp2 := FeePool{CommunityPool: sdk.DecCoins{{"stake", sdk.NewDec(-1)}}}
|
||||||
require.True(sdk.DecEq(t, sdk.NewDec(15), fp.TotalValAccum.Accum))
|
require.NotNil(t, fp2.ValidateGenesis())
|
||||||
|
|
||||||
fp = fp.UpdateTotalValAccum(8, sdk.NewDec(2))
|
|
||||||
require.True(sdk.DecEq(t, sdk.NewDec(21), fp.TotalValAccum.Accum))
|
|
||||||
}
|
}
|
||||||
|
|
|
@ -13,62 +13,92 @@ type DelegatorWithdrawInfo struct {
|
||||||
WithdrawAddr sdk.AccAddress `json:"withdraw_addr"`
|
WithdrawAddr sdk.AccAddress `json:"withdraw_addr"`
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// used for import / export via genesis json
|
||||||
|
type ValidatorAccumulatedCommissionRecord struct {
|
||||||
|
ValidatorAddr sdk.ValAddress `json:"validator_addr"`
|
||||||
|
Accumulated ValidatorAccumulatedCommission `json:"accumulated"`
|
||||||
|
}
|
||||||
|
|
||||||
|
// used for import / export via genesis json
|
||||||
|
type ValidatorHistoricalRewardsRecord struct {
|
||||||
|
ValidatorAddr sdk.ValAddress `json:"validator_addr"`
|
||||||
|
Period uint64 `json:"period"`
|
||||||
|
Rewards ValidatorHistoricalRewards `json:"rewards"`
|
||||||
|
}
|
||||||
|
|
||||||
|
// used for import / export via genesis json
|
||||||
|
type ValidatorCurrentRewardsRecord struct {
|
||||||
|
ValidatorAddr sdk.ValAddress `json:"validator_addr"`
|
||||||
|
Rewards ValidatorCurrentRewards `json:"rewards"`
|
||||||
|
}
|
||||||
|
|
||||||
|
// used for import / export via genesis json
|
||||||
|
type DelegatorStartingInfoRecord struct {
|
||||||
|
DelegatorAddr sdk.AccAddress `json:"delegator_addr"`
|
||||||
|
ValidatorAddr sdk.ValAddress `json:"validator_addr"`
|
||||||
|
StartingInfo DelegatorStartingInfo `json:"starting_info"`
|
||||||
|
}
|
||||||
|
|
||||||
|
// used for import / export via genesis json
|
||||||
|
type ValidatorSlashEventRecord struct {
|
||||||
|
ValidatorAddr sdk.ValAddress `json:"validator_addr"`
|
||||||
|
Height uint64 `json:"height"`
|
||||||
|
Event ValidatorSlashEvent `json:"validator_slash_event"`
|
||||||
|
}
|
||||||
|
|
||||||
// GenesisState - all distribution state that must be provided at genesis
|
// GenesisState - all distribution state that must be provided at genesis
|
||||||
type GenesisState struct {
|
type GenesisState struct {
|
||||||
FeePool FeePool `json:"fee_pool"`
|
FeePool FeePool `json:"fee_pool"`
|
||||||
CommunityTax sdk.Dec `json:"community_tax"`
|
CommunityTax sdk.Dec `json:"community_tax"`
|
||||||
BaseProposerReward sdk.Dec `json:"base_proposer_reward"`
|
BaseProposerReward sdk.Dec `json:"base_proposer_reward"`
|
||||||
BonusProposerReward sdk.Dec `json:"bonus_proposer_reward"`
|
BonusProposerReward sdk.Dec `json:"bonus_proposer_reward"`
|
||||||
ValidatorDistInfos []ValidatorDistInfo `json:"validator_dist_infos"`
|
DelegatorWithdrawInfos []DelegatorWithdrawInfo `json:"delegator_withdraw_infos"`
|
||||||
DelegationDistInfos []DelegationDistInfo `json:"delegator_dist_infos"`
|
PreviousProposer sdk.ConsAddress `json:"previous_proposer"`
|
||||||
DelegatorWithdrawInfos []DelegatorWithdrawInfo `json:"delegator_withdraw_infos"`
|
OutstandingRewards sdk.DecCoins `json:"outstanding_rewards"`
|
||||||
PreviousProposer sdk.ConsAddress `json:"previous_proposer"`
|
ValidatorAccumulatedCommissions []ValidatorAccumulatedCommissionRecord `json:"validator_accumulated_commissions"`
|
||||||
|
ValidatorHistoricalRewards []ValidatorHistoricalRewardsRecord `json:"validator_historical_rewards"`
|
||||||
|
ValidatorCurrentRewards []ValidatorCurrentRewardsRecord `json:"validator_current_rewards"`
|
||||||
|
DelegatorStartingInfos []DelegatorStartingInfoRecord `json:"delegator_starting_infos"`
|
||||||
|
ValidatorSlashEvents []ValidatorSlashEventRecord `json:"validator_slash_events"`
|
||||||
}
|
}
|
||||||
|
|
||||||
func NewGenesisState(feePool FeePool, communityTax, baseProposerReward, bonusProposerReward sdk.Dec,
|
func NewGenesisState(feePool FeePool, communityTax, baseProposerReward, bonusProposerReward sdk.Dec,
|
||||||
vdis []ValidatorDistInfo, ddis []DelegationDistInfo, dwis []DelegatorWithdrawInfo, pp sdk.ConsAddress) GenesisState {
|
dwis []DelegatorWithdrawInfo, pp sdk.ConsAddress, r OutstandingRewards,
|
||||||
|
acc []ValidatorAccumulatedCommissionRecord, historical []ValidatorHistoricalRewardsRecord,
|
||||||
|
cur []ValidatorCurrentRewardsRecord, dels []DelegatorStartingInfoRecord,
|
||||||
|
slashes []ValidatorSlashEventRecord) GenesisState {
|
||||||
|
|
||||||
return GenesisState{
|
return GenesisState{
|
||||||
FeePool: feePool,
|
FeePool: feePool,
|
||||||
CommunityTax: communityTax,
|
CommunityTax: communityTax,
|
||||||
BaseProposerReward: baseProposerReward,
|
BaseProposerReward: baseProposerReward,
|
||||||
BonusProposerReward: bonusProposerReward,
|
BonusProposerReward: bonusProposerReward,
|
||||||
ValidatorDistInfos: vdis,
|
DelegatorWithdrawInfos: dwis,
|
||||||
DelegationDistInfos: ddis,
|
PreviousProposer: pp,
|
||||||
DelegatorWithdrawInfos: dwis,
|
OutstandingRewards: r,
|
||||||
PreviousProposer: pp,
|
ValidatorAccumulatedCommissions: acc,
|
||||||
|
ValidatorHistoricalRewards: historical,
|
||||||
|
ValidatorCurrentRewards: cur,
|
||||||
|
DelegatorStartingInfos: dels,
|
||||||
|
ValidatorSlashEvents: slashes,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// get raw genesis raw message for testing
|
// get raw genesis raw message for testing
|
||||||
func DefaultGenesisState() GenesisState {
|
func DefaultGenesisState() GenesisState {
|
||||||
return GenesisState{
|
return GenesisState{
|
||||||
FeePool: InitialFeePool(),
|
FeePool: InitialFeePool(),
|
||||||
CommunityTax: sdk.NewDecWithPrec(2, 2), // 2%
|
CommunityTax: sdk.NewDecWithPrec(2, 2), // 2%
|
||||||
BaseProposerReward: sdk.NewDecWithPrec(1, 2), // 1%
|
BaseProposerReward: sdk.NewDecWithPrec(1, 2), // 1%
|
||||||
BonusProposerReward: sdk.NewDecWithPrec(4, 2), // 4%
|
BonusProposerReward: sdk.NewDecWithPrec(4, 2), // 4%
|
||||||
}
|
DelegatorWithdrawInfos: []DelegatorWithdrawInfo{},
|
||||||
}
|
PreviousProposer: nil,
|
||||||
|
OutstandingRewards: sdk.DecCoins{},
|
||||||
// default genesis utility function, initialize for starting validator set
|
ValidatorAccumulatedCommissions: []ValidatorAccumulatedCommissionRecord{},
|
||||||
func DefaultGenesisWithValidators(valAddrs []sdk.ValAddress) GenesisState {
|
ValidatorHistoricalRewards: []ValidatorHistoricalRewardsRecord{},
|
||||||
|
ValidatorCurrentRewards: []ValidatorCurrentRewardsRecord{},
|
||||||
vdis := make([]ValidatorDistInfo, len(valAddrs))
|
DelegatorStartingInfos: []DelegatorStartingInfoRecord{},
|
||||||
ddis := make([]DelegationDistInfo, len(valAddrs))
|
ValidatorSlashEvents: []ValidatorSlashEventRecord{},
|
||||||
|
|
||||||
for i, valAddr := range valAddrs {
|
|
||||||
vdis[i] = NewValidatorDistInfo(valAddr, 0)
|
|
||||||
accAddr := sdk.AccAddress(valAddr)
|
|
||||||
ddis[i] = NewDelegationDistInfo(accAddr, valAddr, 0)
|
|
||||||
}
|
|
||||||
|
|
||||||
return GenesisState{
|
|
||||||
FeePool: InitialFeePool(),
|
|
||||||
CommunityTax: sdk.NewDecWithPrec(2, 2), // 2%
|
|
||||||
BaseProposerReward: sdk.NewDecWithPrec(1, 2), // 1%
|
|
||||||
BonusProposerReward: sdk.NewDecWithPrec(4, 2), // 4%
|
|
||||||
ValidatorDistInfos: vdis,
|
|
||||||
DelegationDistInfos: ddis,
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -19,7 +19,7 @@ type BankKeeper interface {
|
||||||
AddCoins(ctx sdk.Context, addr sdk.AccAddress, amt sdk.Coins) (sdk.Coins, sdk.Tags, sdk.Error)
|
AddCoins(ctx sdk.Context, addr sdk.AccAddress, amt sdk.Coins) (sdk.Coins, sdk.Tags, sdk.Error)
|
||||||
}
|
}
|
||||||
|
|
||||||
// from ante handler
|
// expected fee collection keeper
|
||||||
type FeeCollectionKeeper interface {
|
type FeeCollectionKeeper interface {
|
||||||
GetCollectedFees(ctx sdk.Context) sdk.Coins
|
GetCollectedFees(ctx sdk.Context) sdk.Coins
|
||||||
ClearCollectedFees(ctx sdk.Context)
|
ClearCollectedFees(ctx sdk.Context)
|
||||||
|
|
|
@ -5,11 +5,11 @@ import (
|
||||||
sdk "github.com/cosmos/cosmos-sdk/types"
|
sdk "github.com/cosmos/cosmos-sdk/types"
|
||||||
)
|
)
|
||||||
|
|
||||||
// Verify interface at compile time
|
// name to identify transaction types
|
||||||
var _, _ sdk.Msg = &MsgSetWithdrawAddress{}, &MsgWithdrawDelegatorRewardsAll{}
|
const MsgRoute = "distr"
|
||||||
var _, _ sdk.Msg = &MsgWithdrawDelegatorReward{}, &MsgWithdrawValidatorRewardsAll{}
|
|
||||||
|
|
||||||
//______________________________________________________________________
|
// Verify interface at compile time
|
||||||
|
var _, _, _ sdk.Msg = &MsgSetWithdrawAddress{}, &MsgWithdrawDelegatorReward{}, &MsgWithdrawValidatorCommission{}
|
||||||
|
|
||||||
// msg struct for changing the withdraw address for a delegator (or validator self-delegation)
|
// msg struct for changing the withdraw address for a delegator (or validator self-delegation)
|
||||||
type MsgSetWithdrawAddress struct {
|
type MsgSetWithdrawAddress struct {
|
||||||
|
@ -24,7 +24,7 @@ func NewMsgSetWithdrawAddress(delAddr, withdrawAddr sdk.AccAddress) MsgSetWithdr
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
func (msg MsgSetWithdrawAddress) Route() string { return RouterKey }
|
func (msg MsgSetWithdrawAddress) Route() string { return MsgRoute }
|
||||||
func (msg MsgSetWithdrawAddress) Type() string { return "set_withdraw_address" }
|
func (msg MsgSetWithdrawAddress) Type() string { return "set_withdraw_address" }
|
||||||
|
|
||||||
// Return address that must sign over msg.GetSignBytes()
|
// Return address that must sign over msg.GetSignBytes()
|
||||||
|
@ -52,46 +52,6 @@ func (msg MsgSetWithdrawAddress) ValidateBasic() sdk.Error {
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
//______________________________________________________________________
|
|
||||||
|
|
||||||
// msg struct for delegation withdraw for all of the delegator's delegations
|
|
||||||
type MsgWithdrawDelegatorRewardsAll struct {
|
|
||||||
DelegatorAddr sdk.AccAddress `json:"delegator_addr"`
|
|
||||||
}
|
|
||||||
|
|
||||||
func NewMsgWithdrawDelegatorRewardsAll(delAddr sdk.AccAddress) MsgWithdrawDelegatorRewardsAll {
|
|
||||||
return MsgWithdrawDelegatorRewardsAll{
|
|
||||||
DelegatorAddr: delAddr,
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
func (msg MsgWithdrawDelegatorRewardsAll) Route() string { return RouterKey }
|
|
||||||
func (msg MsgWithdrawDelegatorRewardsAll) Type() string { return "withdraw_delegation_rewards_all" }
|
|
||||||
|
|
||||||
// Return address that must sign over msg.GetSignBytes()
|
|
||||||
func (msg MsgWithdrawDelegatorRewardsAll) GetSigners() []sdk.AccAddress {
|
|
||||||
return []sdk.AccAddress{sdk.AccAddress(msg.DelegatorAddr)}
|
|
||||||
}
|
|
||||||
|
|
||||||
// get the bytes for the message signer to sign on
|
|
||||||
func (msg MsgWithdrawDelegatorRewardsAll) GetSignBytes() []byte {
|
|
||||||
b, err := MsgCdc.MarshalJSON(msg)
|
|
||||||
if err != nil {
|
|
||||||
panic(err)
|
|
||||||
}
|
|
||||||
return sdk.MustSortJSON(b)
|
|
||||||
}
|
|
||||||
|
|
||||||
// quick validity check
|
|
||||||
func (msg MsgWithdrawDelegatorRewardsAll) ValidateBasic() sdk.Error {
|
|
||||||
if msg.DelegatorAddr == nil {
|
|
||||||
return ErrNilDelegatorAddr(DefaultCodespace)
|
|
||||||
}
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
|
|
||||||
//______________________________________________________________________
|
|
||||||
|
|
||||||
// msg struct for delegation withdraw from a single validator
|
// msg struct for delegation withdraw from a single validator
|
||||||
type MsgWithdrawDelegatorReward struct {
|
type MsgWithdrawDelegatorReward struct {
|
||||||
DelegatorAddr sdk.AccAddress `json:"delegator_addr"`
|
DelegatorAddr sdk.AccAddress `json:"delegator_addr"`
|
||||||
|
@ -105,7 +65,7 @@ func NewMsgWithdrawDelegatorReward(delAddr sdk.AccAddress, valAddr sdk.ValAddres
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
func (msg MsgWithdrawDelegatorReward) Route() string { return RouterKey }
|
func (msg MsgWithdrawDelegatorReward) Route() string { return MsgRoute }
|
||||||
func (msg MsgWithdrawDelegatorReward) Type() string { return "withdraw_delegation_reward" }
|
func (msg MsgWithdrawDelegatorReward) Type() string { return "withdraw_delegation_reward" }
|
||||||
|
|
||||||
// Return address that must sign over msg.GetSignBytes()
|
// Return address that must sign over msg.GetSignBytes()
|
||||||
|
@ -133,29 +93,27 @@ func (msg MsgWithdrawDelegatorReward) ValidateBasic() sdk.Error {
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
//______________________________________________________________________
|
|
||||||
|
|
||||||
// msg struct for validator withdraw
|
// msg struct for validator withdraw
|
||||||
type MsgWithdrawValidatorRewardsAll struct {
|
type MsgWithdrawValidatorCommission struct {
|
||||||
ValidatorAddr sdk.ValAddress `json:"validator_addr"`
|
ValidatorAddr sdk.ValAddress `json:"validator_addr"`
|
||||||
}
|
}
|
||||||
|
|
||||||
func NewMsgWithdrawValidatorRewardsAll(valAddr sdk.ValAddress) MsgWithdrawValidatorRewardsAll {
|
func NewMsgWithdrawValidatorCommission(valAddr sdk.ValAddress) MsgWithdrawValidatorCommission {
|
||||||
return MsgWithdrawValidatorRewardsAll{
|
return MsgWithdrawValidatorCommission{
|
||||||
ValidatorAddr: valAddr,
|
ValidatorAddr: valAddr,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
func (msg MsgWithdrawValidatorRewardsAll) Route() string { return RouterKey }
|
func (msg MsgWithdrawValidatorCommission) Route() string { return MsgRoute }
|
||||||
func (msg MsgWithdrawValidatorRewardsAll) Type() string { return "withdraw_validator_rewards_all" }
|
func (msg MsgWithdrawValidatorCommission) Type() string { return "withdraw_validator_rewards_all" }
|
||||||
|
|
||||||
// Return address that must sign over msg.GetSignBytes()
|
// Return address that must sign over msg.GetSignBytes()
|
||||||
func (msg MsgWithdrawValidatorRewardsAll) GetSigners() []sdk.AccAddress {
|
func (msg MsgWithdrawValidatorCommission) GetSigners() []sdk.AccAddress {
|
||||||
return []sdk.AccAddress{sdk.AccAddress(msg.ValidatorAddr.Bytes())}
|
return []sdk.AccAddress{sdk.AccAddress(msg.ValidatorAddr.Bytes())}
|
||||||
}
|
}
|
||||||
|
|
||||||
// get the bytes for the message signer to sign on
|
// get the bytes for the message signer to sign on
|
||||||
func (msg MsgWithdrawValidatorRewardsAll) GetSignBytes() []byte {
|
func (msg MsgWithdrawValidatorCommission) GetSignBytes() []byte {
|
||||||
b, err := MsgCdc.MarshalJSON(msg)
|
b, err := MsgCdc.MarshalJSON(msg)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
panic(err)
|
panic(err)
|
||||||
|
@ -164,7 +122,7 @@ func (msg MsgWithdrawValidatorRewardsAll) GetSignBytes() []byte {
|
||||||
}
|
}
|
||||||
|
|
||||||
// quick validity check
|
// quick validity check
|
||||||
func (msg MsgWithdrawValidatorRewardsAll) ValidateBasic() sdk.Error {
|
func (msg MsgWithdrawValidatorCommission) ValidateBasic() sdk.Error {
|
||||||
if msg.ValidatorAddr == nil {
|
if msg.ValidatorAddr == nil {
|
||||||
return ErrNilValidatorAddr(DefaultCodespace)
|
return ErrNilValidatorAddr(DefaultCodespace)
|
||||||
}
|
}
|
||||||
|
|
|
@ -8,7 +8,7 @@ import (
|
||||||
sdk "github.com/cosmos/cosmos-sdk/types"
|
sdk "github.com/cosmos/cosmos-sdk/types"
|
||||||
)
|
)
|
||||||
|
|
||||||
// test ValidateBasic for MsgCreateValidator
|
// test ValidateBasic for MsgSetWithdrawAddress
|
||||||
func TestMsgSetWithdrawAddress(t *testing.T) {
|
func TestMsgSetWithdrawAddress(t *testing.T) {
|
||||||
tests := []struct {
|
tests := []struct {
|
||||||
delegatorAddr sdk.AccAddress
|
delegatorAddr sdk.AccAddress
|
||||||
|
@ -32,7 +32,7 @@ func TestMsgSetWithdrawAddress(t *testing.T) {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// test ValidateBasic for MsgEditValidator
|
// test ValidateBasic for MsgWithdrawDelegatorReward
|
||||||
func TestMsgWithdrawDelegatorReward(t *testing.T) {
|
func TestMsgWithdrawDelegatorReward(t *testing.T) {
|
||||||
tests := []struct {
|
tests := []struct {
|
||||||
delegatorAddr sdk.AccAddress
|
delegatorAddr sdk.AccAddress
|
||||||
|
@ -54,27 +54,8 @@ func TestMsgWithdrawDelegatorReward(t *testing.T) {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// test ValidateBasic and GetSigners for MsgCreateValidatorOnBehalfOf
|
// test ValidateBasic for MsgWithdrawValidatorCommission
|
||||||
func TestMsgWithdrawDelegatorRewardsAll(t *testing.T) {
|
func TestMsgWithdrawValidatorCommission(t *testing.T) {
|
||||||
tests := []struct {
|
|
||||||
delegatorAddr sdk.AccAddress
|
|
||||||
expectPass bool
|
|
||||||
}{
|
|
||||||
{delAddr1, true},
|
|
||||||
{emptyDelAddr, false},
|
|
||||||
}
|
|
||||||
for i, tc := range tests {
|
|
||||||
msg := NewMsgWithdrawDelegatorRewardsAll(tc.delegatorAddr)
|
|
||||||
if tc.expectPass {
|
|
||||||
require.Nil(t, msg.ValidateBasic(), "test index: %v", i)
|
|
||||||
} else {
|
|
||||||
require.NotNil(t, msg.ValidateBasic(), "test index: %v", i)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// test ValidateBasic for MsgDelegate
|
|
||||||
func TestMsgWithdrawValidatorRewardsAll(t *testing.T) {
|
|
||||||
tests := []struct {
|
tests := []struct {
|
||||||
validatorAddr sdk.ValAddress
|
validatorAddr sdk.ValAddress
|
||||||
expectPass bool
|
expectPass bool
|
||||||
|
@ -83,7 +64,7 @@ func TestMsgWithdrawValidatorRewardsAll(t *testing.T) {
|
||||||
{emptyValAddr, false},
|
{emptyValAddr, false},
|
||||||
}
|
}
|
||||||
for i, tc := range tests {
|
for i, tc := range tests {
|
||||||
msg := NewMsgWithdrawValidatorRewardsAll(tc.validatorAddr)
|
msg := NewMsgWithdrawValidatorCommission(tc.validatorAddr)
|
||||||
if tc.expectPass {
|
if tc.expectPass {
|
||||||
require.Nil(t, msg.ValidateBasic(), "test index: %v", i)
|
require.Nil(t, msg.ValidateBasic(), "test index: %v", i)
|
||||||
} else {
|
} else {
|
||||||
|
|
|
@ -1,40 +0,0 @@
|
||||||
package types
|
|
||||||
|
|
||||||
import (
|
|
||||||
sdk "github.com/cosmos/cosmos-sdk/types"
|
|
||||||
)
|
|
||||||
|
|
||||||
// total accumulation tracker
|
|
||||||
type TotalAccum struct {
|
|
||||||
UpdateHeight int64 `json:"update_height"`
|
|
||||||
Accum sdk.Dec `json:"accum"`
|
|
||||||
}
|
|
||||||
|
|
||||||
func NewTotalAccum(height int64) TotalAccum {
|
|
||||||
return TotalAccum{
|
|
||||||
UpdateHeight: height,
|
|
||||||
Accum: sdk.ZeroDec(),
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// update total accumulation factor for the new height
|
|
||||||
// CONTRACT: height should be greater than the old height
|
|
||||||
func (ta TotalAccum) UpdateForNewHeight(height int64, accumCreatedPerBlock sdk.Dec) TotalAccum {
|
|
||||||
blocks := height - ta.UpdateHeight
|
|
||||||
if blocks < 0 {
|
|
||||||
panic("reverse updated for new height")
|
|
||||||
}
|
|
||||||
ta.Accum = ta.Accum.Add(accumCreatedPerBlock.MulInt(sdk.NewInt(blocks)))
|
|
||||||
ta.UpdateHeight = height
|
|
||||||
return ta
|
|
||||||
}
|
|
||||||
|
|
||||||
// get total accumulation factor for the given height
|
|
||||||
// CONTRACT: height should be greater than the old height
|
|
||||||
func (ta TotalAccum) GetAccum(height int64, accumCreatedPerBlock sdk.Dec) sdk.Dec {
|
|
||||||
blocks := height - ta.UpdateHeight
|
|
||||||
if blocks < 0 {
|
|
||||||
panic("reverse updated for new height")
|
|
||||||
}
|
|
||||||
return ta.Accum.Add(accumCreatedPerBlock.MulInt(sdk.NewInt(blocks)))
|
|
||||||
}
|
|
|
@ -1,20 +0,0 @@
|
||||||
package types
|
|
||||||
|
|
||||||
import (
|
|
||||||
"testing"
|
|
||||||
|
|
||||||
"github.com/stretchr/testify/require"
|
|
||||||
|
|
||||||
sdk "github.com/cosmos/cosmos-sdk/types"
|
|
||||||
)
|
|
||||||
|
|
||||||
func TestTotalAccumUpdateForNewHeight(t *testing.T) {
|
|
||||||
|
|
||||||
ta := NewTotalAccum(0)
|
|
||||||
|
|
||||||
ta = ta.UpdateForNewHeight(5, sdk.NewDec(3))
|
|
||||||
require.True(sdk.DecEq(t, sdk.NewDec(15), ta.Accum))
|
|
||||||
|
|
||||||
ta = ta.UpdateForNewHeight(8, sdk.NewDec(2))
|
|
||||||
require.True(sdk.DecEq(t, sdk.NewDec(21), ta.Accum))
|
|
||||||
}
|
|
|
@ -0,0 +1,52 @@
|
||||||
|
package types
|
||||||
|
|
||||||
|
import (
|
||||||
|
sdk "github.com/cosmos/cosmos-sdk/types"
|
||||||
|
)
|
||||||
|
|
||||||
|
// historical rewards for a validator
|
||||||
|
// TODO add reference counter, ref https://github.com/cosmos/cosmos-sdk/pull/3099#discussion_r245747051
|
||||||
|
// height is implicit within the store key
|
||||||
|
type ValidatorHistoricalRewards = sdk.DecCoins
|
||||||
|
|
||||||
|
// current rewards and current period for a validator
|
||||||
|
// kept as a running counter and incremented each block
|
||||||
|
// as long as the validator's tokens remain constant
|
||||||
|
type ValidatorCurrentRewards struct {
|
||||||
|
Rewards sdk.DecCoins `json:"rewards"` // current rewards
|
||||||
|
Period uint64 `json:"period"` // current period
|
||||||
|
}
|
||||||
|
|
||||||
|
// create a new ValidatorCurrentRewards
|
||||||
|
func NewValidatorCurrentRewards(rewards sdk.DecCoins, period uint64) ValidatorCurrentRewards {
|
||||||
|
return ValidatorCurrentRewards{
|
||||||
|
Rewards: rewards,
|
||||||
|
Period: period,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// accumulated commission for a validator
|
||||||
|
// kept as a running counter, can be withdrawn at any time
|
||||||
|
type ValidatorAccumulatedCommission = sdk.DecCoins
|
||||||
|
|
||||||
|
// return the initial accumulated commission (zero)
|
||||||
|
func InitialValidatorAccumulatedCommission() ValidatorAccumulatedCommission {
|
||||||
|
return ValidatorAccumulatedCommission{}
|
||||||
|
}
|
||||||
|
|
||||||
|
// validator slash event
|
||||||
|
// height is implicit within the store key
|
||||||
|
// needed to calculate appropriate amounts of staking token
|
||||||
|
// for delegations which withdraw after a slash has occurred
|
||||||
|
type ValidatorSlashEvent struct {
|
||||||
|
ValidatorPeriod uint64 `json:"validator_period"` // period when the slash occurred
|
||||||
|
Fraction sdk.Dec `json:"fraction"` // slash fraction
|
||||||
|
}
|
||||||
|
|
||||||
|
// create a new ValidatorSlashEvent
|
||||||
|
func NewValidatorSlashEvent(validatorPeriod uint64, fraction sdk.Dec) ValidatorSlashEvent {
|
||||||
|
return ValidatorSlashEvent{
|
||||||
|
ValidatorPeriod: validatorPeriod,
|
||||||
|
Fraction: fraction,
|
||||||
|
}
|
||||||
|
}
|
|
@ -1,153 +0,0 @@
|
||||||
package types
|
|
||||||
|
|
||||||
import (
|
|
||||||
sdk "github.com/cosmos/cosmos-sdk/types"
|
|
||||||
)
|
|
||||||
|
|
||||||
// common parameters used in withdraws from validators
|
|
||||||
type WithdrawContext struct {
|
|
||||||
FeePool FeePool
|
|
||||||
Height int64 // block height
|
|
||||||
TotalPower sdk.Dec // total bonded tokens in the network
|
|
||||||
ValPower sdk.Dec // validator's bonded tokens
|
|
||||||
CommissionRate sdk.Dec // validator commission rate
|
|
||||||
}
|
|
||||||
|
|
||||||
func NewWithdrawContext(feePool FeePool, height int64, totalPower,
|
|
||||||
valPower, commissionRate sdk.Dec) WithdrawContext {
|
|
||||||
|
|
||||||
return WithdrawContext{
|
|
||||||
FeePool: feePool,
|
|
||||||
Height: height,
|
|
||||||
TotalPower: totalPower,
|
|
||||||
ValPower: valPower,
|
|
||||||
CommissionRate: commissionRate,
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
//_____________________________________________________________________________
|
|
||||||
|
|
||||||
// distribution info for a particular validator
|
|
||||||
type ValidatorDistInfo struct {
|
|
||||||
OperatorAddr sdk.ValAddress `json:"operator_addr"`
|
|
||||||
|
|
||||||
FeePoolWithdrawalHeight int64 `json:"fee_pool_withdrawal_height"` // last height this validator withdrew from the global pool
|
|
||||||
|
|
||||||
DelAccum TotalAccum `json:"del_accum"` // total accumulation factor held by delegators
|
|
||||||
DelPool DecCoins `json:"del_pool"` // rewards owed to delegators, commission has already been charged (includes proposer reward)
|
|
||||||
ValCommission DecCoins `json:"val_commission"` // commission collected by this validator (pending withdrawal)
|
|
||||||
}
|
|
||||||
|
|
||||||
func NewValidatorDistInfo(operatorAddr sdk.ValAddress, currentHeight int64) ValidatorDistInfo {
|
|
||||||
return ValidatorDistInfo{
|
|
||||||
OperatorAddr: operatorAddr,
|
|
||||||
FeePoolWithdrawalHeight: currentHeight,
|
|
||||||
DelPool: DecCoins{},
|
|
||||||
DelAccum: NewTotalAccum(currentHeight),
|
|
||||||
ValCommission: DecCoins{},
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// update total delegator accumululation
|
|
||||||
func (vi ValidatorDistInfo) UpdateTotalDelAccum(height int64, totalDelShares sdk.Dec) ValidatorDistInfo {
|
|
||||||
vi.DelAccum = vi.DelAccum.UpdateForNewHeight(height, totalDelShares)
|
|
||||||
return vi
|
|
||||||
}
|
|
||||||
|
|
||||||
// Get the total delegator accum within this validator at the provided height
|
|
||||||
func (vi ValidatorDistInfo) GetTotalDelAccum(height int64, totalDelShares sdk.Dec) sdk.Dec {
|
|
||||||
return vi.DelAccum.GetAccum(height, totalDelShares)
|
|
||||||
}
|
|
||||||
|
|
||||||
// Get the validator accum at the provided height
|
|
||||||
func (vi ValidatorDistInfo) GetValAccum(height int64, valTokens sdk.Dec) sdk.Dec {
|
|
||||||
blocks := height - vi.FeePoolWithdrawalHeight
|
|
||||||
return valTokens.MulInt(sdk.NewInt(blocks))
|
|
||||||
}
|
|
||||||
|
|
||||||
// Move any available accumulated fees in the FeePool to the validator's pool
|
|
||||||
// - updates validator info's FeePoolWithdrawalHeight, thus setting accum to 0
|
|
||||||
// - updates fee pool to latest height and total val accum w/ given totalBonded
|
|
||||||
// This is the only way to update the FeePool's validator TotalAccum.
|
|
||||||
// NOTE: This algorithm works as long as TakeFeePoolRewards is called after every power change.
|
|
||||||
// - called in ValidationDistInfo.WithdrawCommission
|
|
||||||
// - called in DelegationDistInfo.WithdrawRewards
|
|
||||||
// NOTE: When a delegator unbonds, say, onDelegationSharesModified ->
|
|
||||||
// WithdrawDelegationReward -> WithdrawRewards
|
|
||||||
func (vi ValidatorDistInfo) TakeFeePoolRewards(wc WithdrawContext) (
|
|
||||||
ValidatorDistInfo, FeePool) {
|
|
||||||
|
|
||||||
fp := wc.FeePool.UpdateTotalValAccum(wc.Height, wc.TotalPower)
|
|
||||||
|
|
||||||
if fp.TotalValAccum.Accum.IsZero() {
|
|
||||||
vi.FeePoolWithdrawalHeight = wc.Height
|
|
||||||
return vi, fp
|
|
||||||
}
|
|
||||||
|
|
||||||
// update the validators pool
|
|
||||||
accum := vi.GetValAccum(wc.Height, wc.ValPower)
|
|
||||||
vi.FeePoolWithdrawalHeight = wc.Height
|
|
||||||
|
|
||||||
if accum.GT(fp.TotalValAccum.Accum) {
|
|
||||||
panic("individual accum should never be greater than the total")
|
|
||||||
}
|
|
||||||
withdrawalTokens := fp.ValPool.MulDec(accum).QuoDec(fp.TotalValAccum.Accum) // XXX ensure this doesn't cause problems
|
|
||||||
remValPool := fp.ValPool.Minus(withdrawalTokens)
|
|
||||||
|
|
||||||
commission := withdrawalTokens.MulDec(wc.CommissionRate)
|
|
||||||
afterCommission := withdrawalTokens.Minus(commission)
|
|
||||||
|
|
||||||
fp.TotalValAccum.Accum = fp.TotalValAccum.Accum.Sub(accum)
|
|
||||||
fp.ValPool = remValPool
|
|
||||||
vi.ValCommission = vi.ValCommission.Plus(commission)
|
|
||||||
vi.DelPool = vi.DelPool.Plus(afterCommission)
|
|
||||||
|
|
||||||
return vi, fp
|
|
||||||
}
|
|
||||||
|
|
||||||
// withdraw commission rewards
|
|
||||||
func (vi ValidatorDistInfo) WithdrawCommission(wc WithdrawContext) (
|
|
||||||
vio ValidatorDistInfo, fpo FeePool, withdrawn DecCoins) {
|
|
||||||
|
|
||||||
vi, fp := vi.TakeFeePoolRewards(wc)
|
|
||||||
|
|
||||||
withdrawalTokens := vi.ValCommission
|
|
||||||
vi.ValCommission = DecCoins{} // zero
|
|
||||||
|
|
||||||
return vi, fp, withdrawalTokens
|
|
||||||
}
|
|
||||||
|
|
||||||
// get the validator's pool rewards at this current state,
|
|
||||||
func (vi ValidatorDistInfo) CurrentPoolRewards(
|
|
||||||
wc WithdrawContext) DecCoins {
|
|
||||||
|
|
||||||
fp := wc.FeePool
|
|
||||||
totalValAccum := fp.GetTotalValAccum(wc.Height, wc.TotalPower)
|
|
||||||
valAccum := vi.GetValAccum(wc.Height, wc.ValPower)
|
|
||||||
|
|
||||||
if valAccum.GT(totalValAccum) {
|
|
||||||
panic("individual accum should never be greater than the total")
|
|
||||||
}
|
|
||||||
withdrawalTokens := fp.ValPool.MulDec(valAccum).QuoDec(totalValAccum)
|
|
||||||
commission := withdrawalTokens.MulDec(wc.CommissionRate)
|
|
||||||
afterCommission := withdrawalTokens.Minus(commission)
|
|
||||||
pool := vi.DelPool.Plus(afterCommission)
|
|
||||||
return pool
|
|
||||||
}
|
|
||||||
|
|
||||||
// get the validator's commission pool rewards at this current state,
|
|
||||||
func (vi ValidatorDistInfo) CurrentCommissionRewards(
|
|
||||||
wc WithdrawContext) DecCoins {
|
|
||||||
|
|
||||||
fp := wc.FeePool
|
|
||||||
totalValAccum := fp.GetTotalValAccum(wc.Height, wc.TotalPower)
|
|
||||||
valAccum := vi.GetValAccum(wc.Height, wc.ValPower)
|
|
||||||
|
|
||||||
if valAccum.GT(totalValAccum) {
|
|
||||||
panic("individual accum should never be greater than the total")
|
|
||||||
}
|
|
||||||
withdrawalTokens := fp.ValPool.MulDec(valAccum).QuoDec(totalValAccum)
|
|
||||||
commission := withdrawalTokens.MulDec(wc.CommissionRate)
|
|
||||||
commissionPool := vi.ValCommission.Plus(commission)
|
|
||||||
return commissionPool
|
|
||||||
}
|
|
|
@ -1,92 +0,0 @@
|
||||||
package types
|
|
||||||
|
|
||||||
import (
|
|
||||||
"testing"
|
|
||||||
|
|
||||||
"github.com/stretchr/testify/assert"
|
|
||||||
"github.com/stretchr/testify/require"
|
|
||||||
|
|
||||||
sdk "github.com/cosmos/cosmos-sdk/types"
|
|
||||||
stakingTypes "github.com/cosmos/cosmos-sdk/x/staking/types"
|
|
||||||
)
|
|
||||||
|
|
||||||
func TestTakeFeePoolRewards(t *testing.T) {
|
|
||||||
|
|
||||||
// initialize
|
|
||||||
height := int64(0)
|
|
||||||
fp := InitialFeePool()
|
|
||||||
vi1 := NewValidatorDistInfo(valAddr1, height)
|
|
||||||
vi2 := NewValidatorDistInfo(valAddr2, height)
|
|
||||||
vi3 := NewValidatorDistInfo(valAddr3, height)
|
|
||||||
commissionRate1 := sdk.NewDecWithPrec(2, 2)
|
|
||||||
commissionRate2 := sdk.NewDecWithPrec(3, 2)
|
|
||||||
commissionRate3 := sdk.NewDecWithPrec(4, 2)
|
|
||||||
validatorTokens1 := sdk.NewDec(10)
|
|
||||||
validatorTokens2 := sdk.NewDec(40)
|
|
||||||
validatorTokens3 := sdk.NewDec(50)
|
|
||||||
totalBondedTokens := validatorTokens1.Add(validatorTokens2).Add(validatorTokens3)
|
|
||||||
|
|
||||||
// simulate adding some stake for inflation
|
|
||||||
height = 10
|
|
||||||
fp.ValPool = DecCoins{NewDecCoin(stakingTypes.DefaultBondDenom, 1000)}
|
|
||||||
|
|
||||||
vi1, fp = vi1.TakeFeePoolRewards(NewWithdrawContext(
|
|
||||||
fp, height, totalBondedTokens, validatorTokens1, commissionRate1))
|
|
||||||
require.True(sdk.DecEq(t, sdk.NewDec(900), fp.TotalValAccum.Accum))
|
|
||||||
assert.True(sdk.DecEq(t, sdk.NewDec(900), fp.ValPool[0].Amount))
|
|
||||||
assert.True(sdk.DecEq(t, sdk.NewDec(100-2), vi1.DelPool[0].Amount))
|
|
||||||
assert.True(sdk.DecEq(t, sdk.NewDec(2), vi1.ValCommission[0].Amount))
|
|
||||||
|
|
||||||
vi2, fp = vi2.TakeFeePoolRewards(NewWithdrawContext(
|
|
||||||
fp, height, totalBondedTokens, validatorTokens2, commissionRate2))
|
|
||||||
require.True(sdk.DecEq(t, sdk.NewDec(500), fp.TotalValAccum.Accum))
|
|
||||||
assert.True(sdk.DecEq(t, sdk.NewDec(500), fp.ValPool[0].Amount))
|
|
||||||
assert.True(sdk.DecEq(t, sdk.NewDec(400-12), vi2.DelPool[0].Amount))
|
|
||||||
assert.True(sdk.DecEq(t, vi2.ValCommission[0].Amount, sdk.NewDec(12)))
|
|
||||||
|
|
||||||
// add more blocks and inflation
|
|
||||||
height = 20
|
|
||||||
fp.ValPool[0].Amount = fp.ValPool[0].Amount.Add(sdk.NewDec(1000))
|
|
||||||
|
|
||||||
vi3, fp = vi3.TakeFeePoolRewards(NewWithdrawContext(
|
|
||||||
fp, height, totalBondedTokens, validatorTokens3, commissionRate3))
|
|
||||||
require.True(sdk.DecEq(t, sdk.NewDec(500), fp.TotalValAccum.Accum))
|
|
||||||
assert.True(sdk.DecEq(t, sdk.NewDec(500), fp.ValPool[0].Amount))
|
|
||||||
assert.True(sdk.DecEq(t, sdk.NewDec(1000-40), vi3.DelPool[0].Amount))
|
|
||||||
assert.True(sdk.DecEq(t, vi3.ValCommission[0].Amount, sdk.NewDec(40)))
|
|
||||||
}
|
|
||||||
|
|
||||||
func TestWithdrawCommission(t *testing.T) {
|
|
||||||
|
|
||||||
// initialize
|
|
||||||
height := int64(0)
|
|
||||||
fp := InitialFeePool()
|
|
||||||
vi := NewValidatorDistInfo(valAddr1, height)
|
|
||||||
commissionRate := sdk.NewDecWithPrec(2, 2)
|
|
||||||
validatorTokens := sdk.NewDec(10)
|
|
||||||
totalBondedTokens := validatorTokens.Add(sdk.NewDec(90)) // validator-1 is 10% of total power
|
|
||||||
|
|
||||||
// simulate adding some stake for inflation
|
|
||||||
height = 10
|
|
||||||
fp.ValPool = DecCoins{NewDecCoin(stakingTypes.DefaultBondDenom, 1000)}
|
|
||||||
|
|
||||||
// for a more fun staring condition, have an non-withdraw update
|
|
||||||
vi, fp = vi.TakeFeePoolRewards(NewWithdrawContext(
|
|
||||||
fp, height, totalBondedTokens, validatorTokens, commissionRate))
|
|
||||||
require.True(sdk.DecEq(t, sdk.NewDec(900), fp.TotalValAccum.Accum))
|
|
||||||
assert.True(sdk.DecEq(t, sdk.NewDec(900), fp.ValPool[0].Amount))
|
|
||||||
assert.True(sdk.DecEq(t, sdk.NewDec(100-2), vi.DelPool[0].Amount))
|
|
||||||
assert.True(sdk.DecEq(t, sdk.NewDec(2), vi.ValCommission[0].Amount))
|
|
||||||
|
|
||||||
// add more blocks and inflation
|
|
||||||
height = 20
|
|
||||||
fp.ValPool[0].Amount = fp.ValPool[0].Amount.Add(sdk.NewDec(1000))
|
|
||||||
|
|
||||||
vi, fp, commissionRecv := vi.WithdrawCommission(NewWithdrawContext(
|
|
||||||
fp, height, totalBondedTokens, validatorTokens, commissionRate))
|
|
||||||
require.True(sdk.DecEq(t, sdk.NewDec(1800), fp.TotalValAccum.Accum))
|
|
||||||
assert.True(sdk.DecEq(t, sdk.NewDec(1800), fp.ValPool[0].Amount))
|
|
||||||
assert.True(sdk.DecEq(t, sdk.NewDec(200-4), vi.DelPool[0].Amount))
|
|
||||||
assert.Zero(t, len(vi.ValCommission))
|
|
||||||
assert.True(sdk.DecEq(t, sdk.NewDec(4), commissionRecv[0].Amount))
|
|
||||||
}
|
|
|
@ -71,3 +71,5 @@ func (h Hooks) BeforeValidatorModified(_ sdk.Context, _ sdk.ValAddress)
|
||||||
func (h Hooks) BeforeDelegationCreated(_ sdk.Context, _ sdk.AccAddress, _ sdk.ValAddress) {}
|
func (h Hooks) BeforeDelegationCreated(_ sdk.Context, _ sdk.AccAddress, _ sdk.ValAddress) {}
|
||||||
func (h Hooks) BeforeDelegationSharesModified(_ sdk.Context, _ sdk.AccAddress, _ sdk.ValAddress) {}
|
func (h Hooks) BeforeDelegationSharesModified(_ sdk.Context, _ sdk.AccAddress, _ sdk.ValAddress) {}
|
||||||
func (h Hooks) BeforeDelegationRemoved(_ sdk.Context, _ sdk.AccAddress, _ sdk.ValAddress) {}
|
func (h Hooks) BeforeDelegationRemoved(_ sdk.Context, _ sdk.AccAddress, _ sdk.ValAddress) {}
|
||||||
|
func (h Hooks) AfterDelegationModified(_ sdk.Context, _ sdk.AccAddress, _ sdk.ValAddress) {}
|
||||||
|
func (h Hooks) BeforeValidatorSlashed(_ sdk.Context, _ sdk.ValAddress, _ sdk.Dec) {}
|
||||||
|
|
|
@ -45,6 +45,7 @@ func InitGenesis(ctx sdk.Context, keeper Keeper, data types.GenesisState) (res [
|
||||||
for _, delegation := range data.Bonds {
|
for _, delegation := range data.Bonds {
|
||||||
keeper.BeforeDelegationCreated(ctx, delegation.DelegatorAddr, delegation.ValidatorAddr)
|
keeper.BeforeDelegationCreated(ctx, delegation.DelegatorAddr, delegation.ValidatorAddr)
|
||||||
keeper.SetDelegation(ctx, delegation)
|
keeper.SetDelegation(ctx, delegation)
|
||||||
|
keeper.AfterDelegationModified(ctx, delegation.DelegatorAddr, delegation.ValidatorAddr)
|
||||||
}
|
}
|
||||||
|
|
||||||
for _, ubd := range data.UnbondingDelegations {
|
for _, ubd := range data.UnbondingDelegations {
|
||||||
|
|
|
@ -77,6 +77,7 @@ func (k Keeper) SetDelegation(ctx sdk.Context, delegation types.Delegation) {
|
||||||
store := ctx.KVStore(k.storeKey)
|
store := ctx.KVStore(k.storeKey)
|
||||||
b := types.MustMarshalDelegation(k.cdc, delegation)
|
b := types.MustMarshalDelegation(k.cdc, delegation)
|
||||||
store.Set(GetDelegationKey(delegation.DelegatorAddr, delegation.ValidatorAddr), b)
|
store.Set(GetDelegationKey(delegation.DelegatorAddr, delegation.ValidatorAddr), b)
|
||||||
|
k.AfterDelegationModified(ctx, delegation.DelegatorAddr, delegation.ValidatorAddr)
|
||||||
}
|
}
|
||||||
|
|
||||||
// remove a delegation from store
|
// remove a delegation from store
|
||||||
|
|
|
@ -59,3 +59,15 @@ func (k Keeper) BeforeDelegationRemoved(ctx sdk.Context, delAddr sdk.AccAddress,
|
||||||
k.hooks.BeforeDelegationRemoved(ctx, delAddr, valAddr)
|
k.hooks.BeforeDelegationRemoved(ctx, delAddr, valAddr)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func (k Keeper) AfterDelegationModified(ctx sdk.Context, delAddr sdk.AccAddress, valAddr sdk.ValAddress) {
|
||||||
|
if k.hooks != nil {
|
||||||
|
k.hooks.AfterDelegationModified(ctx, delAddr, valAddr)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (k Keeper) BeforeValidatorSlashed(ctx sdk.Context, valAddr sdk.ValAddress, fraction sdk.Dec) {
|
||||||
|
if k.hooks != nil {
|
||||||
|
k.hooks.BeforeValidatorSlashed(ctx, valAddr, fraction)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
|
@ -55,6 +55,11 @@ func (k Keeper) Slash(ctx sdk.Context, consAddr sdk.ConsAddress, infractionHeigh
|
||||||
operatorAddress := validator.GetOperator()
|
operatorAddress := validator.GetOperator()
|
||||||
k.BeforeValidatorModified(ctx, operatorAddress)
|
k.BeforeValidatorModified(ctx, operatorAddress)
|
||||||
|
|
||||||
|
// we need to calculate the *effective* slash fraction for distribution
|
||||||
|
if validator.Tokens.GT(sdk.ZeroInt()) {
|
||||||
|
k.BeforeValidatorSlashed(ctx, operatorAddress, slashAmountDec.Quo(sdk.NewDecFromInt(validator.Tokens)))
|
||||||
|
}
|
||||||
|
|
||||||
// Track remaining slash amount for the validator
|
// Track remaining slash amount for the validator
|
||||||
// This will decrease when we slash unbondings and
|
// This will decrease when we slash unbondings and
|
||||||
// redelegations, as that stake has since unbonded
|
// redelegations, as that stake has since unbonded
|
||||||
|
|
|
@ -82,17 +82,8 @@ func SupplyInvariants(ck bank.Keeper, k staking.Keeper,
|
||||||
// add community pool
|
// add community pool
|
||||||
loose = loose.Add(feePool.CommunityPool.AmountOf(stakingTypes.DefaultBondDenom))
|
loose = loose.Add(feePool.CommunityPool.AmountOf(stakingTypes.DefaultBondDenom))
|
||||||
|
|
||||||
// add validator distribution pool
|
// add yet-to-be-withdrawn
|
||||||
loose = loose.Add(feePool.ValPool.AmountOf(stakingTypes.DefaultBondDenom))
|
loose = loose.Add(d.GetOutstandingRewards(ctx).AmountOf(stakingTypes.DefaultBondDenom))
|
||||||
|
|
||||||
// add validator distribution commission and yet-to-be-withdrawn-by-delegators
|
|
||||||
d.IterateValidatorDistInfos(ctx,
|
|
||||||
func(_ int64, distInfo distribution.ValidatorDistInfo) (stop bool) {
|
|
||||||
loose = loose.Add(distInfo.DelPool.AmountOf(stakingTypes.DefaultBondDenom))
|
|
||||||
loose = loose.Add(distInfo.ValCommission.AmountOf(stakingTypes.DefaultBondDenom))
|
|
||||||
return false
|
|
||||||
},
|
|
||||||
)
|
|
||||||
|
|
||||||
// Loose tokens should equal coin supply plus unbonding delegations
|
// Loose tokens should equal coin supply plus unbonding delegations
|
||||||
// plus tokens on unbonded validators
|
// plus tokens on unbonded validators
|
||||||
|
|
|
@ -67,6 +67,7 @@ var (
|
||||||
KeyBondDenom = types.KeyBondDenom
|
KeyBondDenom = types.KeyBondDenom
|
||||||
|
|
||||||
DefaultParams = types.DefaultParams
|
DefaultParams = types.DefaultParams
|
||||||
|
DefaultBondDenom = types.DefaultBondDenom
|
||||||
InitialPool = types.InitialPool
|
InitialPool = types.InitialPool
|
||||||
NewValidator = types.NewValidator
|
NewValidator = types.NewValidator
|
||||||
NewDescription = types.NewDescription
|
NewDescription = types.NewDescription
|
||||||
|
|
|
@ -405,14 +405,15 @@ func (v Validator) BondedTokens() sdk.Int {
|
||||||
var _ sdk.Validator = Validator{}
|
var _ sdk.Validator = Validator{}
|
||||||
|
|
||||||
// nolint - for sdk.Validator
|
// nolint - for sdk.Validator
|
||||||
func (v Validator) GetJailed() bool { return v.Jailed }
|
func (v Validator) GetJailed() bool { return v.Jailed }
|
||||||
func (v Validator) GetMoniker() string { return v.Description.Moniker }
|
func (v Validator) GetMoniker() string { return v.Description.Moniker }
|
||||||
func (v Validator) GetStatus() sdk.BondStatus { return v.Status }
|
func (v Validator) GetStatus() sdk.BondStatus { return v.Status }
|
||||||
func (v Validator) GetOperator() sdk.ValAddress { return v.OperatorAddr }
|
func (v Validator) GetOperator() sdk.ValAddress { return v.OperatorAddr }
|
||||||
func (v Validator) GetConsPubKey() crypto.PubKey { return v.ConsPubKey }
|
func (v Validator) GetConsPubKey() crypto.PubKey { return v.ConsPubKey }
|
||||||
func (v Validator) GetConsAddr() sdk.ConsAddress { return sdk.ConsAddress(v.ConsPubKey.Address()) }
|
func (v Validator) GetConsAddr() sdk.ConsAddress { return sdk.ConsAddress(v.ConsPubKey.Address()) }
|
||||||
func (v Validator) GetPower() sdk.Int { return v.BondedTokens() }
|
func (v Validator) GetPower() sdk.Int { return v.BondedTokens() }
|
||||||
func (v Validator) GetTokens() sdk.Int { return v.Tokens }
|
func (v Validator) GetTokens() sdk.Int { return v.Tokens }
|
||||||
func (v Validator) GetCommission() sdk.Dec { return v.Commission.Rate }
|
func (v Validator) GetCommission() sdk.Dec { return v.Commission.Rate }
|
||||||
func (v Validator) GetDelegatorShares() sdk.Dec { return v.DelegatorShares }
|
func (v Validator) GetDelegatorShares() sdk.Dec { return v.DelegatorShares }
|
||||||
func (v Validator) GetBondHeight() int64 { return v.BondHeight }
|
func (v Validator) GetBondHeight() int64 { return v.BondHeight }
|
||||||
|
func (v Validator) GetDelegatorShareExRate() sdk.Dec { return v.DelegatorShareExRate() }
|
||||||
|
|
Loading…
Reference in New Issue