2019-01-11 12:08:01 -08:00
|
|
|
package staking
|
2018-05-12 15:22:59 -07:00
|
|
|
|
2018-06-06 09:38:13 -07:00
|
|
|
import (
|
2018-10-07 21:12:37 -07:00
|
|
|
"fmt"
|
|
|
|
|
2018-07-18 23:39:40 -07:00
|
|
|
abci "github.com/tendermint/tendermint/abci/types"
|
|
|
|
tmtypes "github.com/tendermint/tendermint/types"
|
|
|
|
|
2018-06-06 09:38:13 -07:00
|
|
|
sdk "github.com/cosmos/cosmos-sdk/types"
|
2019-06-05 10:42:25 -07:00
|
|
|
"github.com/cosmos/cosmos-sdk/x/staking/exported"
|
2020-06-12 07:54:37 -07:00
|
|
|
"github.com/cosmos/cosmos-sdk/x/staking/keeper"
|
2019-01-11 12:08:01 -08:00
|
|
|
"github.com/cosmos/cosmos-sdk/x/staking/types"
|
2018-06-06 09:38:13 -07:00
|
|
|
)
|
2018-05-12 15:22:59 -07:00
|
|
|
|
2018-12-10 02:49:37 -08:00
|
|
|
// InitGenesis sets the pool and parameters for the provided keeper. For each
|
|
|
|
// validator in data, it sets that validator in the keeper along with manually
|
|
|
|
// setting the indexes. In addition, it also sets any delegations found in
|
|
|
|
// data. Finally, it updates the bonded validators.
|
2018-07-18 23:39:40 -07:00
|
|
|
// Returns final validator set after applying all declaration and delegations
|
2020-01-30 13:31:16 -08:00
|
|
|
func InitGenesis(
|
2020-06-12 07:54:37 -07:00
|
|
|
ctx sdk.Context, keeper keeper.Keeper, accountKeeper types.AccountKeeper,
|
2020-04-20 08:22:12 -07:00
|
|
|
bankKeeper types.BankKeeper, data types.GenesisState,
|
2020-01-30 13:31:16 -08:00
|
|
|
) (res []abci.ValidatorUpdate) {
|
2019-06-28 13:11:27 -07:00
|
|
|
bondedTokens := sdk.ZeroInt()
|
|
|
|
notBondedTokens := sdk.ZeroInt()
|
2018-10-05 05:11:36 -07:00
|
|
|
|
2019-01-16 02:35:18 -08:00
|
|
|
// We need to pretend to be "n blocks before genesis", where "n" is the
|
|
|
|
// validator update delay, so that e.g. slashing periods are correctly
|
|
|
|
// initialized for the validator set e.g. with a one-block offset - the
|
|
|
|
// first TM block is at height 1, so state updates applied from
|
|
|
|
// genesis.json are in block 0.
|
2019-02-13 15:01:50 -08:00
|
|
|
ctx = ctx.WithBlockHeight(1 - sdk.ValidatorUpdateDelay)
|
2018-10-05 05:11:36 -07:00
|
|
|
|
2018-10-03 09:37:06 -07:00
|
|
|
keeper.SetParams(ctx, data.Params)
|
2018-11-08 16:28:28 -08:00
|
|
|
keeper.SetLastTotalPower(ctx, data.LastTotalPower)
|
|
|
|
|
2018-12-10 02:49:37 -08:00
|
|
|
for _, validator := range data.Validators {
|
2018-11-08 16:28:28 -08:00
|
|
|
keeper.SetValidator(ctx, validator)
|
2018-07-09 19:51:13 -07:00
|
|
|
|
2018-10-03 09:37:06 -07:00
|
|
|
// Manually set indices for the first time
|
2018-09-19 16:33:12 -07:00
|
|
|
keeper.SetValidatorByConsAddr(ctx, validator)
|
2018-12-07 16:04:52 -08:00
|
|
|
keeper.SetValidatorByPowerIndex(ctx, validator)
|
2019-02-07 17:41:23 -08:00
|
|
|
|
|
|
|
// Call the creation hook if not exported
|
2019-01-24 13:01:32 -08:00
|
|
|
if !data.Exported {
|
2019-02-25 07:16:52 -08:00
|
|
|
keeper.AfterValidatorCreated(ctx, validator.OperatorAddress)
|
2019-01-24 13:01:32 -08:00
|
|
|
}
|
2018-11-08 16:28:28 -08:00
|
|
|
|
2019-06-28 13:11:27 -07:00
|
|
|
// update timeslice if necessary
|
2019-06-04 15:06:58 -07:00
|
|
|
if validator.IsUnbonding() {
|
2018-11-08 16:28:28 -08:00
|
|
|
keeper.InsertValidatorQueue(ctx, validator)
|
|
|
|
}
|
2019-06-28 13:11:27 -07:00
|
|
|
|
|
|
|
switch validator.GetStatus() {
|
|
|
|
case sdk.Bonded:
|
|
|
|
bondedTokens = bondedTokens.Add(validator.GetTokens())
|
|
|
|
case sdk.Unbonding, sdk.Unbonded:
|
|
|
|
notBondedTokens = notBondedTokens.Add(validator.GetTokens())
|
|
|
|
default:
|
|
|
|
panic("invalid validator status")
|
|
|
|
}
|
2018-05-12 15:22:59 -07:00
|
|
|
}
|
2018-07-03 21:21:36 -07:00
|
|
|
|
2019-02-13 15:01:50 -08:00
|
|
|
for _, delegation := range data.Delegations {
|
2019-02-07 17:41:23 -08:00
|
|
|
// Call the before-creation hook if not exported
|
2019-01-24 13:01:32 -08:00
|
|
|
if !data.Exported {
|
2019-02-25 07:16:52 -08:00
|
|
|
keeper.BeforeDelegationCreated(ctx, delegation.DelegatorAddress, delegation.ValidatorAddress)
|
2019-01-24 13:01:32 -08:00
|
|
|
}
|
2019-06-28 13:11:27 -07:00
|
|
|
|
2020-05-02 12:26:59 -07:00
|
|
|
keeper.SetDelegation(ctx, delegation)
|
2019-02-07 17:41:23 -08:00
|
|
|
// Call the after-modification hook if not exported
|
2019-01-24 13:01:32 -08:00
|
|
|
if !data.Exported {
|
2019-02-25 07:16:52 -08:00
|
|
|
keeper.AfterDelegationModified(ctx, delegation.DelegatorAddress, delegation.ValidatorAddress)
|
2019-01-24 13:01:32 -08:00
|
|
|
}
|
2018-05-12 15:22:59 -07:00
|
|
|
}
|
2018-07-03 21:21:36 -07:00
|
|
|
|
2018-11-08 16:28:28 -08:00
|
|
|
for _, ubd := range data.UnbondingDelegations {
|
|
|
|
keeper.SetUnbondingDelegation(ctx, ubd)
|
2020-05-02 12:26:59 -07:00
|
|
|
|
2019-01-16 02:35:18 -08:00
|
|
|
for _, entry := range ubd.Entries {
|
|
|
|
keeper.InsertUBDQueue(ctx, ubd, entry.CompletionTime)
|
2019-06-28 13:11:27 -07:00
|
|
|
notBondedTokens = notBondedTokens.Add(entry.Balance)
|
2019-01-16 02:35:18 -08:00
|
|
|
}
|
2018-11-08 16:28:28 -08:00
|
|
|
}
|
|
|
|
|
|
|
|
for _, red := range data.Redelegations {
|
|
|
|
keeper.SetRedelegation(ctx, red)
|
2020-05-02 12:26:59 -07:00
|
|
|
|
2019-01-16 02:35:18 -08:00
|
|
|
for _, entry := range red.Entries {
|
|
|
|
keeper.InsertRedelegationQueue(ctx, red, entry.CompletionTime)
|
|
|
|
}
|
2018-11-08 16:28:28 -08:00
|
|
|
}
|
|
|
|
|
2019-06-28 13:11:27 -07:00
|
|
|
bondedCoins := sdk.NewCoins(sdk.NewCoin(data.Params.BondDenom, bondedTokens))
|
|
|
|
notBondedCoins := sdk.NewCoins(sdk.NewCoin(data.Params.BondDenom, notBondedTokens))
|
|
|
|
|
|
|
|
// check if the unbonded and bonded pools accounts exists
|
|
|
|
bondedPool := keeper.GetBondedPool(ctx)
|
|
|
|
if bondedPool == nil {
|
|
|
|
panic(fmt.Sprintf("%s module account has not been set", types.BondedPoolName))
|
|
|
|
}
|
|
|
|
|
|
|
|
// TODO remove with genesis 2-phases refactor https://github.com/cosmos/cosmos-sdk/issues/2862
|
|
|
|
// add coins if not provided on genesis
|
2020-01-30 13:31:16 -08:00
|
|
|
if bankKeeper.GetAllBalances(ctx, bondedPool.GetAddress()).IsZero() {
|
|
|
|
if err := bankKeeper.SetBalances(ctx, bondedPool.GetAddress(), bondedCoins); err != nil {
|
2019-06-28 13:11:27 -07:00
|
|
|
panic(err)
|
|
|
|
}
|
2020-01-30 13:31:16 -08:00
|
|
|
|
2020-04-20 12:32:10 -07:00
|
|
|
accountKeeper.SetModuleAccount(ctx, bondedPool)
|
2019-06-28 13:11:27 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
notBondedPool := keeper.GetNotBondedPool(ctx)
|
|
|
|
if notBondedPool == nil {
|
|
|
|
panic(fmt.Sprintf("%s module account has not been set", types.NotBondedPoolName))
|
|
|
|
}
|
|
|
|
|
2020-01-30 13:31:16 -08:00
|
|
|
if bankKeeper.GetAllBalances(ctx, notBondedPool.GetAddress()).IsZero() {
|
|
|
|
if err := bankKeeper.SetBalances(ctx, notBondedPool.GetAddress(), notBondedCoins); err != nil {
|
2019-06-28 13:11:27 -07:00
|
|
|
panic(err)
|
|
|
|
}
|
2020-01-30 13:31:16 -08:00
|
|
|
|
2020-04-20 12:32:10 -07:00
|
|
|
accountKeeper.SetModuleAccount(ctx, notBondedPool)
|
2019-06-28 13:11:27 -07:00
|
|
|
}
|
|
|
|
|
2018-11-13 06:46:14 -08:00
|
|
|
// don't need to run Tendermint updates if we exported
|
2018-11-13 06:08:14 -08:00
|
|
|
if data.Exported {
|
|
|
|
for _, lv := range data.LastValidatorPowers {
|
|
|
|
keeper.SetLastValidatorPower(ctx, lv.Address, lv.Power)
|
2018-11-26 04:21:23 -08:00
|
|
|
validator, found := keeper.GetValidator(ctx, lv.Address)
|
2020-05-02 12:26:59 -07:00
|
|
|
|
2018-11-26 04:21:23 -08:00
|
|
|
if !found {
|
2019-06-28 13:11:27 -07:00
|
|
|
panic(fmt.Sprintf("validator %s not found", lv.Address))
|
2018-11-26 04:21:23 -08:00
|
|
|
}
|
2020-05-02 12:26:59 -07:00
|
|
|
|
2018-11-26 04:21:23 -08:00
|
|
|
update := validator.ABCIValidatorUpdate()
|
2019-02-05 21:30:48 -08:00
|
|
|
update.Power = lv.Power // keep the next-val-set offset, use the last power for the first block
|
2018-11-26 04:21:23 -08:00
|
|
|
res = append(res, update)
|
2018-11-13 06:08:14 -08:00
|
|
|
}
|
2018-11-13 06:38:29 -08:00
|
|
|
} else {
|
|
|
|
res = keeper.ApplyAndReturnValidatorSetUpdates(ctx)
|
2018-11-13 06:08:14 -08:00
|
|
|
}
|
|
|
|
|
2019-05-16 08:25:32 -07:00
|
|
|
return res
|
2018-05-12 15:22:59 -07:00
|
|
|
}
|
|
|
|
|
2018-11-08 16:28:28 -08:00
|
|
|
// ExportGenesis returns a GenesisState for a given context and keeper. The
|
2018-07-03 21:21:36 -07:00
|
|
|
// GenesisState will contain the pool, params, validators, and bonds found in
|
|
|
|
// the keeper.
|
2020-06-12 07:54:37 -07:00
|
|
|
func ExportGenesis(ctx sdk.Context, keeper keeper.Keeper) types.GenesisState {
|
2018-11-08 16:28:28 -08:00
|
|
|
var unbondingDelegations []types.UnbondingDelegation
|
2020-05-02 12:26:59 -07:00
|
|
|
|
2018-11-08 16:28:28 -08:00
|
|
|
keeper.IterateUnbondingDelegations(ctx, func(_ int64, ubd types.UnbondingDelegation) (stop bool) {
|
|
|
|
unbondingDelegations = append(unbondingDelegations, ubd)
|
|
|
|
return false
|
|
|
|
})
|
2020-05-02 12:26:59 -07:00
|
|
|
|
2018-11-08 16:28:28 -08:00
|
|
|
var redelegations []types.Redelegation
|
2020-05-02 12:26:59 -07:00
|
|
|
|
2018-11-08 16:28:28 -08:00
|
|
|
keeper.IterateRedelegations(ctx, func(_ int64, red types.Redelegation) (stop bool) {
|
|
|
|
redelegations = append(redelegations, red)
|
|
|
|
return false
|
|
|
|
})
|
2020-05-02 12:26:59 -07:00
|
|
|
|
2018-11-13 06:08:14 -08:00
|
|
|
var lastValidatorPowers []types.LastValidatorPower
|
2020-05-02 12:26:59 -07:00
|
|
|
|
2019-02-05 21:30:48 -08:00
|
|
|
keeper.IterateLastValidatorPowers(ctx, func(addr sdk.ValAddress, power int64) (stop bool) {
|
2019-08-19 09:06:27 -07:00
|
|
|
lastValidatorPowers = append(lastValidatorPowers, types.LastValidatorPower{Address: addr, Power: power})
|
2018-11-13 06:08:14 -08:00
|
|
|
return false
|
|
|
|
})
|
2018-07-03 21:21:36 -07:00
|
|
|
|
2018-06-26 19:00:12 -07:00
|
|
|
return types.GenesisState{
|
2020-04-07 15:33:30 -07:00
|
|
|
Params: keeper.GetParams(ctx),
|
|
|
|
LastTotalPower: keeper.GetLastTotalPower(ctx),
|
2018-11-13 06:08:14 -08:00
|
|
|
LastValidatorPowers: lastValidatorPowers,
|
2020-04-07 15:33:30 -07:00
|
|
|
Validators: keeper.GetAllValidators(ctx),
|
|
|
|
Delegations: keeper.GetAllDelegations(ctx),
|
2018-11-08 16:28:28 -08:00
|
|
|
UnbondingDelegations: unbondingDelegations,
|
|
|
|
Redelegations: redelegations,
|
2018-11-13 06:08:14 -08:00
|
|
|
Exported: true,
|
2018-05-12 15:22:59 -07:00
|
|
|
}
|
|
|
|
}
|
2018-06-06 09:38:13 -07:00
|
|
|
|
2018-07-03 21:21:36 -07:00
|
|
|
// WriteValidators returns a slice of bonded genesis validators.
|
2020-06-12 07:54:37 -07:00
|
|
|
func WriteValidators(ctx sdk.Context, keeper keeper.Keeper) (vals []tmtypes.GenesisValidator) {
|
2019-06-05 10:42:25 -07:00
|
|
|
keeper.IterateLastValidators(ctx, func(_ int64, validator exported.ValidatorI) (stop bool) {
|
2018-06-06 09:38:13 -07:00
|
|
|
vals = append(vals, tmtypes.GenesisValidator{
|
2020-06-29 11:56:08 -07:00
|
|
|
Address: validator.GetConsAddr().Bytes(),
|
|
|
|
PubKey: validator.GetConsPubKey(),
|
|
|
|
Power: validator.GetConsensusPower(),
|
|
|
|
Name: validator.GetMoniker(),
|
2018-06-06 09:38:13 -07:00
|
|
|
})
|
2018-07-03 21:21:36 -07:00
|
|
|
|
2018-06-06 09:38:13 -07:00
|
|
|
return false
|
|
|
|
})
|
2018-07-03 21:21:36 -07:00
|
|
|
|
2018-06-06 09:38:13 -07:00
|
|
|
return
|
|
|
|
}
|
2018-10-07 21:12:37 -07:00
|
|
|
|
|
|
|
// ValidateGenesis validates the provided staking genesis state to ensure the
|
|
|
|
// expected invariants holds. (i.e. params in correct bounds, no duplicate validators)
|
|
|
|
func ValidateGenesis(data types.GenesisState) error {
|
2020-05-02 12:26:59 -07:00
|
|
|
if err := validateGenesisStateValidators(data.Validators); err != nil {
|
2018-10-07 21:12:37 -07:00
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
2020-05-02 12:26:59 -07:00
|
|
|
return data.Params.Validate()
|
2018-10-07 21:12:37 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
func validateGenesisStateValidators(validators []types.Validator) (err error) {
|
|
|
|
addrMap := make(map[string]bool, len(validators))
|
2020-05-02 12:26:59 -07:00
|
|
|
|
2018-10-07 21:12:37 -07:00
|
|
|
for i := 0; i < len(validators); i++ {
|
|
|
|
val := validators[i]
|
2020-02-06 11:21:02 -08:00
|
|
|
strKey := string(val.GetConsPubKey().Bytes())
|
|
|
|
|
2018-10-07 21:12:37 -07:00
|
|
|
if _, ok := addrMap[strKey]; ok {
|
2020-02-06 11:21:02 -08:00
|
|
|
return fmt.Errorf("duplicate validator in genesis state: moniker %v, address %v", val.Description.Moniker, val.GetConsAddr())
|
2018-10-07 21:12:37 -07:00
|
|
|
}
|
2020-05-02 12:26:59 -07:00
|
|
|
|
2019-06-04 15:06:58 -07:00
|
|
|
if val.Jailed && val.IsBonded() {
|
2020-02-06 11:21:02 -08:00
|
|
|
return fmt.Errorf("validator is bonded and jailed in genesis state: moniker %v, address %v", val.Description.Moniker, val.GetConsAddr())
|
2018-10-07 21:12:37 -07:00
|
|
|
}
|
2020-05-02 12:26:59 -07:00
|
|
|
|
2019-06-04 15:06:58 -07:00
|
|
|
if val.DelegatorShares.IsZero() && !val.IsUnbonding() {
|
2018-11-08 16:28:28 -08:00
|
|
|
return fmt.Errorf("bonded/unbonded genesis validator cannot have zero delegator shares, validator: %v", val)
|
2018-10-07 21:12:37 -07:00
|
|
|
}
|
2020-02-06 11:21:02 -08:00
|
|
|
|
2018-10-07 21:12:37 -07:00
|
|
|
addrMap[strKey] = true
|
|
|
|
}
|
2020-02-06 11:21:02 -08:00
|
|
|
|
2018-10-07 21:12:37 -07:00
|
|
|
return
|
|
|
|
}
|