472 lines
13 KiB
Go
472 lines
13 KiB
Go
package stake
|
|
|
|
import (
|
|
"bytes"
|
|
|
|
sdk "github.com/cosmos/cosmos-sdk/types"
|
|
"github.com/cosmos/cosmos-sdk/wire"
|
|
"github.com/cosmos/cosmos-sdk/x/bank"
|
|
abci "github.com/tendermint/abci/types"
|
|
)
|
|
|
|
// keeper of the staking store
|
|
type Keeper struct {
|
|
storeKey sdk.StoreKey
|
|
cdc *wire.Codec
|
|
coinKeeper bank.Keeper
|
|
|
|
// caches
|
|
gs Pool
|
|
params Params
|
|
|
|
// codespace
|
|
codespace sdk.CodespaceType
|
|
}
|
|
|
|
func NewKeeper(cdc *wire.Codec, key sdk.StoreKey, ck bank.Keeper, codespace sdk.CodespaceType) Keeper {
|
|
keeper := Keeper{
|
|
storeKey: key,
|
|
cdc: cdc,
|
|
coinKeeper: ck,
|
|
codespace: codespace,
|
|
}
|
|
return keeper
|
|
}
|
|
|
|
// get the current in-block validator operation counter
|
|
func (k Keeper) getCounter(ctx sdk.Context) int16 {
|
|
store := ctx.KVStore(k.storeKey)
|
|
b := store.Get(CounterKey)
|
|
if b == nil {
|
|
return 0
|
|
}
|
|
var counter int16
|
|
err := k.cdc.UnmarshalBinary(b, &counter)
|
|
if err != nil {
|
|
panic(err)
|
|
}
|
|
return counter
|
|
}
|
|
|
|
// set the current in-block validator operation counter
|
|
func (k Keeper) setCounter(ctx sdk.Context, counter int16) {
|
|
store := ctx.KVStore(k.storeKey)
|
|
bz, err := k.cdc.MarshalBinary(counter)
|
|
if err != nil {
|
|
panic(err)
|
|
}
|
|
store.Set(CounterKey, bz)
|
|
}
|
|
|
|
//_________________________________________________________________________
|
|
|
|
// get a single candidate
|
|
func (k Keeper) GetCandidate(ctx sdk.Context, addr sdk.Address) (candidate Candidate, found bool) {
|
|
store := ctx.KVStore(k.storeKey)
|
|
b := store.Get(GetCandidateKey(addr))
|
|
if b == nil {
|
|
return candidate, false
|
|
}
|
|
err := k.cdc.UnmarshalBinary(b, &candidate)
|
|
if err != nil {
|
|
panic(err)
|
|
}
|
|
return candidate, true
|
|
}
|
|
|
|
// Get the set of all candidates, retrieve a maxRetrieve number of records
|
|
func (k Keeper) GetCandidates(ctx sdk.Context, maxRetrieve int16) (candidates Candidates) {
|
|
store := ctx.KVStore(k.storeKey)
|
|
iterator := store.Iterator(subspace(CandidatesKey))
|
|
|
|
candidates = make([]Candidate, maxRetrieve)
|
|
i := 0
|
|
for ; ; i++ {
|
|
if !iterator.Valid() || i > int(maxRetrieve-1) {
|
|
iterator.Close()
|
|
break
|
|
}
|
|
bz := iterator.Value()
|
|
var candidate Candidate
|
|
err := k.cdc.UnmarshalBinary(bz, &candidate)
|
|
if err != nil {
|
|
panic(err)
|
|
}
|
|
candidates[i] = candidate
|
|
iterator.Next()
|
|
}
|
|
return candidates[:i] // trim
|
|
}
|
|
|
|
func (k Keeper) setCandidate(ctx sdk.Context, candidate Candidate) {
|
|
store := ctx.KVStore(k.storeKey)
|
|
address := candidate.Address
|
|
|
|
// retreive the old candidate record
|
|
oldCandidate, oldFound := k.GetCandidate(ctx, address)
|
|
|
|
// if found, copy the old block height and counter
|
|
if oldFound {
|
|
candidate.ValidatorBondHeight = oldCandidate.ValidatorBondHeight
|
|
candidate.ValidatorBondCounter = oldCandidate.ValidatorBondCounter
|
|
}
|
|
|
|
// marshal the candidate record and add to the state
|
|
bz, err := k.cdc.MarshalBinary(candidate)
|
|
if err != nil {
|
|
panic(err)
|
|
}
|
|
store.Set(GetCandidateKey(candidate.Address), bz)
|
|
|
|
// if the voting power is the same no need to update any of the other indexes
|
|
if oldFound && oldCandidate.Assets.Equal(candidate.Assets) {
|
|
return
|
|
}
|
|
|
|
updateHeight := false
|
|
|
|
// update the list ordered by voting power
|
|
if oldFound {
|
|
if !k.isNewValidator(ctx, store, candidate.Address) {
|
|
updateHeight = true
|
|
}
|
|
// else already in the validator set - retain the old validator height and counter
|
|
store.Delete(GetValidatorKey(address, oldCandidate.Assets, oldCandidate.ValidatorBondHeight, oldCandidate.ValidatorBondCounter, k.cdc))
|
|
} else {
|
|
updateHeight = true
|
|
}
|
|
|
|
if updateHeight {
|
|
// wasn't a candidate or wasn't in the validator set, update the validator block height and counter
|
|
candidate.ValidatorBondHeight = ctx.BlockHeight()
|
|
counter := k.getCounter(ctx)
|
|
candidate.ValidatorBondCounter = counter
|
|
k.setCounter(ctx, counter+1)
|
|
}
|
|
|
|
// update the candidate record
|
|
bz, err = k.cdc.MarshalBinary(candidate)
|
|
if err != nil {
|
|
panic(err)
|
|
}
|
|
store.Set(GetCandidateKey(candidate.Address), bz)
|
|
|
|
// marshal the new validator record
|
|
validator := candidate.validator()
|
|
bz, err = k.cdc.MarshalBinary(validator)
|
|
if err != nil {
|
|
panic(err)
|
|
}
|
|
|
|
store.Set(GetValidatorKey(address, validator.Power, validator.Height, validator.Counter, k.cdc), bz)
|
|
|
|
// add to the validators to update list if is already a validator
|
|
// or is a new validator
|
|
setAcc := false
|
|
if store.Get(GetRecentValidatorKey(address)) != nil {
|
|
setAcc = true
|
|
|
|
// want to check in the else statement because inefficient
|
|
} else if k.isNewValidator(ctx, store, address) {
|
|
setAcc = true
|
|
}
|
|
if setAcc {
|
|
bz, err = k.cdc.MarshalBinary(validator.abciValidator(k.cdc))
|
|
if err != nil {
|
|
panic(err)
|
|
}
|
|
store.Set(GetAccUpdateValidatorKey(validator.Address), bz)
|
|
|
|
}
|
|
|
|
return
|
|
}
|
|
|
|
func (k Keeper) removeCandidate(ctx sdk.Context, address sdk.Address) {
|
|
|
|
// first retreive the old candidate record
|
|
candidate, found := k.GetCandidate(ctx, address)
|
|
if !found {
|
|
return
|
|
}
|
|
|
|
// delete the old candidate record
|
|
store := ctx.KVStore(k.storeKey)
|
|
store.Delete(GetCandidateKey(address))
|
|
store.Delete(GetValidatorKey(address, candidate.Assets, candidate.ValidatorBondHeight, candidate.ValidatorBondCounter, k.cdc))
|
|
|
|
// delete from recent and power weighted validator groups if the validator
|
|
// exists and add validator with zero power to the validator updates
|
|
if store.Get(GetRecentValidatorKey(address)) == nil {
|
|
return
|
|
}
|
|
bz, err := k.cdc.MarshalBinary(candidate.validator().abciValidatorZero(k.cdc))
|
|
if err != nil {
|
|
panic(err)
|
|
}
|
|
store.Set(GetAccUpdateValidatorKey(address), bz)
|
|
store.Delete(GetRecentValidatorKey(address))
|
|
}
|
|
|
|
//___________________________________________________________________________
|
|
|
|
// Get the validator set from the candidates. The correct subset is retrieved
|
|
// by iterating through an index of the candidates sorted by power, stored
|
|
// using the ValidatorsKey. Simultaniously the most recent the validator
|
|
// records are updated in store with the RecentValidatorsKey. This store is
|
|
// used to determine if a candidate is a validator without needing to iterate
|
|
// over the subspace as we do in GetValidators
|
|
func (k Keeper) GetValidators(ctx sdk.Context) (validators []Validator) {
|
|
store := ctx.KVStore(k.storeKey)
|
|
|
|
// clear the recent validators store, add to the ToKickOut Temp store
|
|
iterator := store.Iterator(subspace(RecentValidatorsKey))
|
|
for ; iterator.Valid(); iterator.Next() {
|
|
addr := AddrFromKey(iterator.Key())
|
|
|
|
// iterator.Value is the validator object
|
|
store.Set(GetToKickOutValidatorKey(addr), iterator.Value())
|
|
store.Delete(iterator.Key())
|
|
}
|
|
iterator.Close()
|
|
|
|
// add the actual validator power sorted store
|
|
maxValidators := k.GetParams(ctx).MaxValidators
|
|
iterator = store.ReverseIterator(subspace(ValidatorsKey)) // largest to smallest
|
|
validators = make([]Validator, maxValidators)
|
|
i := 0
|
|
for ; ; i++ {
|
|
if !iterator.Valid() || i > int(maxValidators-1) {
|
|
iterator.Close()
|
|
break
|
|
}
|
|
bz := iterator.Value()
|
|
var validator Validator
|
|
err := k.cdc.UnmarshalBinary(bz, &validator)
|
|
if err != nil {
|
|
panic(err)
|
|
}
|
|
validators[i] = validator
|
|
|
|
// remove from ToKickOut group
|
|
store.Delete(GetToKickOutValidatorKey(validator.Address))
|
|
|
|
// also add to the recent validators group
|
|
store.Set(GetRecentValidatorKey(validator.Address), bz)
|
|
|
|
iterator.Next()
|
|
}
|
|
|
|
// add any kicked out validators to the acc change
|
|
iterator = store.Iterator(subspace(ToKickOutValidatorsKey))
|
|
for ; iterator.Valid(); iterator.Next() {
|
|
key := iterator.Key()
|
|
addr := AddrFromKey(key)
|
|
|
|
// get the zero abci validator from the ToKickOut iterator value
|
|
bz := iterator.Value()
|
|
var validator Validator
|
|
err := k.cdc.UnmarshalBinary(bz, &validator)
|
|
if err != nil {
|
|
panic(err)
|
|
}
|
|
bz, err = k.cdc.MarshalBinary(validator.abciValidatorZero(k.cdc))
|
|
if err != nil {
|
|
panic(err)
|
|
}
|
|
|
|
store.Set(GetAccUpdateValidatorKey(addr), bz)
|
|
store.Delete(key)
|
|
}
|
|
iterator.Close()
|
|
|
|
return validators[:i] // trim
|
|
}
|
|
|
|
// TODO this is madly inefficient because need to call every time we set a candidate
|
|
// Should use something better than an iterator maybe?
|
|
// Used to determine if something has just been added to the actual validator set
|
|
func (k Keeper) isNewValidator(ctx sdk.Context, store sdk.KVStore, address sdk.Address) bool {
|
|
// add the actual validator power sorted store
|
|
maxVal := k.GetParams(ctx).MaxValidators
|
|
iterator := store.ReverseIterator(subspace(ValidatorsKey)) // largest to smallest
|
|
for i := 0; ; i++ {
|
|
if !iterator.Valid() || i > int(maxVal-1) {
|
|
iterator.Close()
|
|
break
|
|
}
|
|
bz := iterator.Value()
|
|
var val Validator
|
|
err := k.cdc.UnmarshalBinary(bz, &val)
|
|
if err != nil {
|
|
panic(err)
|
|
}
|
|
if bytes.Equal(val.Address, address) {
|
|
return true
|
|
}
|
|
iterator.Next()
|
|
}
|
|
|
|
return false
|
|
}
|
|
|
|
// Is the address provided a part of the most recently saved validator group?
|
|
func (k Keeper) IsRecentValidator(ctx sdk.Context, address sdk.Address) bool {
|
|
store := ctx.KVStore(k.storeKey)
|
|
if store.Get(GetRecentValidatorKey(address)) == nil {
|
|
return false
|
|
}
|
|
return true
|
|
}
|
|
|
|
//_________________________________________________________________________
|
|
// Accumulated updates to the validator set
|
|
|
|
// get the most recently updated validators
|
|
func (k Keeper) getAccUpdateValidators(ctx sdk.Context) (updates []abci.Validator) {
|
|
store := ctx.KVStore(k.storeKey)
|
|
|
|
iterator := store.Iterator(subspace(AccUpdateValidatorsKey)) //smallest to largest
|
|
for ; iterator.Valid(); iterator.Next() {
|
|
valBytes := iterator.Value()
|
|
var val abci.Validator
|
|
err := k.cdc.UnmarshalBinary(valBytes, &val)
|
|
if err != nil {
|
|
panic(err)
|
|
}
|
|
updates = append(updates, val)
|
|
}
|
|
iterator.Close()
|
|
return
|
|
}
|
|
|
|
// remove all validator update entries
|
|
func (k Keeper) clearAccUpdateValidators(ctx sdk.Context) {
|
|
store := ctx.KVStore(k.storeKey)
|
|
|
|
// delete subspace
|
|
iterator := store.Iterator(subspace(AccUpdateValidatorsKey))
|
|
for ; iterator.Valid(); iterator.Next() {
|
|
store.Delete(iterator.Key())
|
|
}
|
|
iterator.Close()
|
|
}
|
|
|
|
//_____________________________________________________________________
|
|
|
|
// load a delegator bond
|
|
func (k Keeper) GetDelegatorBond(ctx sdk.Context,
|
|
delegatorAddr, candidateAddr sdk.Address) (bond DelegatorBond, found bool) {
|
|
|
|
store := ctx.KVStore(k.storeKey)
|
|
delegatorBytes := store.Get(GetDelegatorBondKey(delegatorAddr, candidateAddr, k.cdc))
|
|
if delegatorBytes == nil {
|
|
return bond, false
|
|
}
|
|
|
|
err := k.cdc.UnmarshalBinary(delegatorBytes, &bond)
|
|
if err != nil {
|
|
panic(err)
|
|
}
|
|
return bond, true
|
|
}
|
|
|
|
// load all bonds of a delegator
|
|
func (k Keeper) GetDelegatorBonds(ctx sdk.Context, delegator sdk.Address, maxRetrieve int16) (bonds []DelegatorBond) {
|
|
store := ctx.KVStore(k.storeKey)
|
|
delegatorPrefixKey := GetDelegatorBondsKey(delegator, k.cdc)
|
|
iterator := store.Iterator(subspace(delegatorPrefixKey)) //smallest to largest
|
|
|
|
bonds = make([]DelegatorBond, maxRetrieve)
|
|
i := 0
|
|
for ; ; i++ {
|
|
if !iterator.Valid() || i > int(maxRetrieve-1) {
|
|
iterator.Close()
|
|
break
|
|
}
|
|
bondBytes := iterator.Value()
|
|
var bond DelegatorBond
|
|
err := k.cdc.UnmarshalBinary(bondBytes, &bond)
|
|
if err != nil {
|
|
panic(err)
|
|
}
|
|
bonds[i] = bond
|
|
iterator.Next()
|
|
}
|
|
return bonds[:i] // trim
|
|
}
|
|
|
|
func (k Keeper) setDelegatorBond(ctx sdk.Context, bond DelegatorBond) {
|
|
store := ctx.KVStore(k.storeKey)
|
|
b, err := k.cdc.MarshalBinary(bond)
|
|
if err != nil {
|
|
panic(err)
|
|
}
|
|
store.Set(GetDelegatorBondKey(bond.DelegatorAddr, bond.CandidateAddr, k.cdc), b)
|
|
}
|
|
|
|
func (k Keeper) removeDelegatorBond(ctx sdk.Context, bond DelegatorBond) {
|
|
store := ctx.KVStore(k.storeKey)
|
|
store.Delete(GetDelegatorBondKey(bond.DelegatorAddr, bond.CandidateAddr, k.cdc))
|
|
}
|
|
|
|
//_______________________________________________________________________
|
|
|
|
// load/save the global staking params
|
|
func (k Keeper) GetParams(ctx sdk.Context) (params Params) {
|
|
// check if cached before anything
|
|
if k.params != (Params{}) {
|
|
return k.params
|
|
}
|
|
store := ctx.KVStore(k.storeKey)
|
|
b := store.Get(ParamKey)
|
|
if b == nil {
|
|
panic("Stored params should not have been nil")
|
|
}
|
|
|
|
err := k.cdc.UnmarshalBinary(b, ¶ms)
|
|
if err != nil {
|
|
panic(err)
|
|
}
|
|
return
|
|
}
|
|
func (k Keeper) setParams(ctx sdk.Context, params Params) {
|
|
store := ctx.KVStore(k.storeKey)
|
|
b, err := k.cdc.MarshalBinary(params)
|
|
if err != nil {
|
|
panic(err)
|
|
}
|
|
store.Set(ParamKey, b)
|
|
k.params = Params{} // clear the cache
|
|
}
|
|
|
|
//_______________________________________________________________________
|
|
|
|
// load/save the pool
|
|
func (k Keeper) GetPool(ctx sdk.Context) (gs Pool) {
|
|
// check if cached before anything
|
|
if k.gs != (Pool{}) {
|
|
return k.gs
|
|
}
|
|
store := ctx.KVStore(k.storeKey)
|
|
b := store.Get(PoolKey)
|
|
if b == nil {
|
|
panic("Stored pool should not have been nil")
|
|
}
|
|
err := k.cdc.UnmarshalBinary(b, &gs)
|
|
if err != nil {
|
|
panic(err) // This error should never occur big problem if does
|
|
}
|
|
return
|
|
}
|
|
|
|
func (k Keeper) setPool(ctx sdk.Context, p Pool) {
|
|
store := ctx.KVStore(k.storeKey)
|
|
b, err := k.cdc.MarshalBinary(p)
|
|
if err != nil {
|
|
panic(err)
|
|
}
|
|
store.Set(PoolKey, b)
|
|
k.gs = Pool{} // clear the cache
|
|
}
|