/
genesis.go
225 lines (197 loc) · 8.57 KB
/
genesis.go
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
package staking
import (
"fmt"
supplyexported "github.com/cosmos/cosmos-sdk/x/supply/exported"
abci "github.com/tendermint/tendermint/abci/types"
tmtypes "github.com/tendermint/tendermint/types"
sdk "github.com/cosmos/cosmos-sdk/types"
"github.com/zenchainprotocol/zenchain-node/x/staking/exported"
"github.com/zenchainprotocol/zenchain-node/x/staking/types"
)
// 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
// Returns final validator set after applying all declaration and delegations
func InitGenesis(ctx sdk.Context, keeper Keeper, accountKeeper types.AccountKeeper,
supplyKeeper types.SupplyKeeper, data types.GenesisState) (res []abci.ValidatorUpdate) {
bondedTokens, notBondedTokens := sdk.ZeroDec(), sdk.ZeroDec()
// 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.
ctx = ctx.WithBlockHeight(1 - sdk.ValidatorUpdateDelay)
keeper.SetParams(ctx, data.Params)
keeper.SetLastTotalPower(ctx, data.LastTotalPower)
for _, validator := range data.Validators {
initValidator(ctx, validator, keeper, &bondedTokens, data.Exported)
}
for _, delegator := range data.Delegators {
initDelegator(ctx, delegator, keeper, &bondedTokens)
}
for _, ubd := range data.UnbondingDelegations {
initUnbondingDelegation(ctx, ubd, keeper, ¬BondedTokens)
}
for _, sharesExported := range data.AllShares {
keeper.SetShares(ctx, sharesExported.DelAddress, sharesExported.ValidatorAddress, sharesExported.Shares)
}
for _, proxyDelegatorKeyExported := range data.ProxyDelegatorKeys {
keeper.SetProxyBinding(ctx, proxyDelegatorKeyExported.ProxyAddr, proxyDelegatorKeyExported.DelAddr, false)
}
checkPools(ctx, keeper, sdk.NewDecCoinFromDec(data.Params.BondDenom, bondedTokens),
sdk.NewDecCoinFromDec(data.Params.BondDenom, notBondedTokens), data.Exported)
// don't need to run Tendermint updates if we exported
if data.Exported {
for _, lv := range data.LastValidatorPowers {
keeper.SetLastValidatorPower(ctx, lv.Address, lv.Power)
validator, found := keeper.GetValidator(ctx, lv.Address)
if !found {
panic(fmt.Sprintf("validator %s not found", lv.Address))
}
update := validator.ABCIValidatorUpdate()
update.Power = lv.Power // keep the next-val-set offset, use the last power for the first block
res = append(res, update)
}
} else {
res = keeper.ApplyAndReturnValidatorSetUpdates(ctx)
}
return res
}
// assume that there is only okt in pool, if not panics
func checkTokenSum(tokenSum sdk.SysCoin, pool supplyexported.ModuleAccountI) {
poolCoins := pool.GetCoins()
if !poolCoins.IsZero() {
if len(poolCoins) != 1 {
panic(fmt.Sprintf("only okt in %s, but there are %d kinds of coins", pool.GetName(), len(poolCoins)))
}
if !tokenSum.ToCoins().IsEqual(poolCoins) {
panic(fmt.Sprintf("coins in %s don't match the token sum, tokenSum: %s, poolCoins: %s",
pool.GetName(), tokenSum.String(), poolCoins.String()))
}
}
}
func checkPools(ctx sdk.Context, keeper Keeper, bondedDecCoin, notBondedDecCoin sdk.SysCoin, isExported bool) {
bondedPool := keeper.GetBondedPool(ctx)
if bondedPool == nil {
panic(fmt.Sprintf("%s module account has not been set", types.BondedPoolName))
}
notBondedPool := keeper.GetNotBondedPool(ctx)
if notBondedPool == nil {
panic(fmt.Sprintf("%s module account has not been set", types.NotBondedPoolName))
}
if isExported {
checkTokenSum(bondedDecCoin, bondedPool)
checkTokenSum(notBondedDecCoin, notBondedPool)
}
}
func initUnbondingDelegation(ctx sdk.Context, ubd UndelegationInfo, keeper Keeper, notBondedTokens *sdk.Dec) {
keeper.SetUndelegating(ctx, ubd)
keeper.SetAddrByTimeKeyWithNilValue(ctx, ubd.CompletionTime, ubd.DelegatorAddress)
*notBondedTokens = notBondedTokens.Add(ubd.Quantity)
}
func initDelegator(ctx sdk.Context, delegator Delegator, keeper Keeper, pBondedTokens *sdk.Dec) {
keeper.SetDelegator(ctx, delegator)
*pBondedTokens = pBondedTokens.Add(delegator.Tokens)
}
func initValidator(ctx sdk.Context, valExported ValidatorExport, keeper Keeper, pBondedTokens *sdk.Dec, exported bool) {
validator := valExported.Import()
keeper.SetValidator(ctx, validator)
// manually set indices for the first time
keeper.SetValidatorByConsAddr(ctx, validator)
keeper.SetValidatorByPowerIndex(ctx, validator)
// call the creation hook if not exported
if !exported {
keeper.AfterValidatorCreated(ctx, validator.OperatorAddress)
}
// update timeslice if necessary
if validator.IsUnbonding() {
keeper.InsertValidatorQueue(ctx, validator)
}
// all the msd on validator should be added into bonded pool
*pBondedTokens = pBondedTokens.Add(validator.MinSelfDelegation)
}
// ExportGenesis returns a GenesisState for a given context and keeper
// The GenesisState will contain the pool, params, validators, and bonds found in the keeper
func ExportGenesis(ctx sdk.Context, keeper Keeper) types.GenesisState {
params := keeper.GetParams(ctx)
lastTotalPower := keeper.GetLastTotalPower(ctx)
validators := keeper.GetAllValidators(ctx)
var delegators []types.Delegator
keeper.IterateDelegator(ctx, func(_ int64, delegator types.Delegator) (stop bool) {
delegators = append(delegators, delegator)
return false
})
var undelegationInfos []types.UndelegationInfo
keeper.IterateUndelegationInfo(ctx, func(_ int64, ubd types.UndelegationInfo) (stop bool) {
undelegationInfos = append(undelegationInfos, ubd)
return false
})
var lastValidatorPowers []types.LastValidatorPower
keeper.IterateLastValidatorPowers(ctx, func(addr sdk.ValAddress, power int64) (stop bool) {
lastValidatorPowers = append(lastValidatorPowers, types.NewLastValidatorPower(addr, power))
return false
})
var sharesExportedSlice []types.SharesExported
keeper.IterateShares(ctx,
func(_ int64, delAddr sdk.AccAddress, valAddr sdk.ValAddress, shares types.Shares) (stop bool) {
sharesExportedSlice = append(sharesExportedSlice, types.NewSharesExported(delAddr, valAddr, shares))
return false
})
var proxyDelegatorKeys []types.ProxyDelegatorKeyExported
keeper.IterateProxy(ctx, []byte{}, false, func(_ int64, delAddr, proxyAddr sdk.AccAddress) (stop bool) {
proxyDelegatorKeys = append(proxyDelegatorKeys, types.NewProxyDelegatorKeyExported(delAddr, proxyAddr))
return false
})
return types.GenesisState{
Params: params,
LastTotalPower: lastTotalPower,
LastValidatorPowers: lastValidatorPowers,
Validators: validators.Export(),
Delegators: delegators,
UnbondingDelegations: undelegationInfos,
AllShares: sharesExportedSlice,
ProxyDelegatorKeys: proxyDelegatorKeys,
Exported: true,
}
}
// GetLatestGenesisValidator returns a slice of bonded genesis validators
func GetLatestGenesisValidator(ctx sdk.Context, keeper Keeper) (vals []tmtypes.GenesisValidator) {
keeper.IterateLastValidators(ctx, func(_ int64, validator exported.ValidatorI) (stop bool) {
vals = append(vals, tmtypes.GenesisValidator{
PubKey: validator.GetConsPubKey(),
Power: validator.GetConsensusPower(),
Name: validator.GetMoniker(),
})
return false
})
return
}
// 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 {
err := validateGenesisStateValidators(data.Validators)
if err != nil {
return err
}
return data.Params.Validate()
}
func validateGenesisStateValidators(valsExported []types.ValidatorExported) (err error) {
valsLen := len(valsExported)
addrMap := make(map[string]bool, valsLen)
for i := 0; i < valsLen; i++ {
valExported := valsExported[i]
strKey := valExported.ConsPubKey
if _, ok := addrMap[strKey]; ok {
return fmt.Errorf("duplicate validator in genesis state: moniker %v, address %v",
valExported.Description.Moniker, valExported.ConsAddress())
}
if valExported.Jailed && valExported.IsBonded() {
return fmt.Errorf("validator is bonded and jailed in genesis state: moniker %v, address %v",
valExported.Description.Moniker, valExported.ConsAddress())
}
if valExported.DelegatorShares.IsZero() {
return fmt.Errorf("it's impossible for a validator with zero delegator shares, validator: %v", valExported)
}
addrMap[strKey] = true
}
return
}