2015-08-10 20:38:45 -07:00
|
|
|
package types
|
2014-09-03 19:21:19 -07:00
|
|
|
|
|
|
|
import (
|
2014-12-09 18:49:04 -08:00
|
|
|
"bytes"
|
2014-10-18 01:42:33 -07:00
|
|
|
"fmt"
|
2014-09-03 19:21:19 -07:00
|
|
|
"io"
|
|
|
|
|
2015-11-01 11:34:08 -08:00
|
|
|
"github.com/tendermint/go-crypto"
|
2015-10-22 17:39:06 -07:00
|
|
|
"github.com/tendermint/go-wire"
|
2017-04-27 19:01:18 -04:00
|
|
|
"github.com/tendermint/go-wire/data"
|
|
|
|
cmn "github.com/tendermint/tmlibs/common"
|
2014-09-03 19:21:19 -07:00
|
|
|
)
|
|
|
|
|
2014-12-09 18:49:04 -08:00
|
|
|
// Volatile state for each Validator
|
2017-05-16 21:31:19 +02:00
|
|
|
// NOTE: The Accum is not included in Validator.Hash();
|
|
|
|
// make sure to update that method if changes are made here
|
2014-09-03 19:21:19 -07:00
|
|
|
type Validator struct {
|
2017-04-27 19:01:18 -04:00
|
|
|
Address data.Bytes `json:"address"`
|
2016-11-16 20:58:53 -05:00
|
|
|
PubKey crypto.PubKey `json:"pub_key"`
|
|
|
|
VotingPower int64 `json:"voting_power"`
|
2017-05-16 21:31:19 +02:00
|
|
|
|
|
|
|
Accum int64 `json:"accum"`
|
2014-09-03 19:21:19 -07:00
|
|
|
}
|
|
|
|
|
2016-11-19 19:32:35 -05:00
|
|
|
func NewValidator(pubKey crypto.PubKey, votingPower int64) *Validator {
|
|
|
|
return &Validator{
|
|
|
|
Address: pubKey.Address(),
|
|
|
|
PubKey: pubKey,
|
|
|
|
VotingPower: votingPower,
|
|
|
|
Accum: 0,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-09-03 19:21:19 -07:00
|
|
|
// Creates a new copy of the validator so we can mutate accum.
|
2015-07-10 09:55:23 -07:00
|
|
|
// Panics if the validator is nil.
|
2014-09-03 19:21:19 -07:00
|
|
|
func (v *Validator) Copy() *Validator {
|
2014-11-05 03:08:29 -08:00
|
|
|
vCopy := *v
|
|
|
|
return &vCopy
|
2014-09-03 19:21:19 -07:00
|
|
|
}
|
|
|
|
|
2014-10-11 21:27:58 -07:00
|
|
|
// Returns the one with higher Accum.
|
|
|
|
func (v *Validator) CompareAccum(other *Validator) *Validator {
|
|
|
|
if v == nil {
|
|
|
|
return other
|
2014-09-14 15:37:32 -07:00
|
|
|
}
|
2014-10-11 21:27:58 -07:00
|
|
|
if v.Accum > other.Accum {
|
|
|
|
return v
|
|
|
|
} else if v.Accum < other.Accum {
|
|
|
|
return other
|
|
|
|
} else {
|
2014-12-09 18:49:04 -08:00
|
|
|
if bytes.Compare(v.Address, other.Address) < 0 {
|
2014-10-11 21:27:58 -07:00
|
|
|
return v
|
2014-12-09 18:49:04 -08:00
|
|
|
} else if bytes.Compare(v.Address, other.Address) > 0 {
|
2014-10-11 21:27:58 -07:00
|
|
|
return other
|
|
|
|
} else {
|
2017-04-27 19:01:18 -04:00
|
|
|
cmn.PanicSanity("Cannot compare identical validators")
|
2015-07-19 23:42:52 +00:00
|
|
|
return nil
|
2014-09-03 19:21:19 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-10-18 01:42:33 -07:00
|
|
|
func (v *Validator) String() string {
|
2015-07-10 09:55:23 -07:00
|
|
|
if v == nil {
|
|
|
|
return "nil-Validator"
|
|
|
|
}
|
2017-05-17 01:08:41 +02:00
|
|
|
return fmt.Sprintf("Validator{%v %v VP:%v A:%v}",
|
2014-12-09 18:49:04 -08:00
|
|
|
v.Address,
|
|
|
|
v.PubKey,
|
2014-11-05 03:08:29 -08:00
|
|
|
v.VotingPower,
|
|
|
|
v.Accum)
|
|
|
|
}
|
|
|
|
|
2017-05-16 21:31:19 +02:00
|
|
|
// Hash computes the unique ID of a validator with a given voting power.
|
|
|
|
// It exludes the Accum value, which changes with every round.
|
2014-11-05 03:08:29 -08:00
|
|
|
func (v *Validator) Hash() []byte {
|
2017-05-16 21:31:19 +02:00
|
|
|
return wire.BinaryRipemd160(struct {
|
|
|
|
Address data.Bytes
|
|
|
|
PubKey crypto.PubKey
|
|
|
|
VotingPower int64
|
|
|
|
}{
|
|
|
|
v.Address,
|
|
|
|
v.PubKey,
|
|
|
|
v.VotingPower,
|
|
|
|
})
|
2014-10-18 01:42:33 -07:00
|
|
|
}
|
|
|
|
|
2014-10-11 21:27:58 -07:00
|
|
|
//-------------------------------------
|
2014-09-14 15:37:32 -07:00
|
|
|
|
2014-10-11 21:27:58 -07:00
|
|
|
var ValidatorCodec = validatorCodec{}
|
2014-09-14 15:37:32 -07:00
|
|
|
|
2014-10-11 21:27:58 -07:00
|
|
|
type validatorCodec struct{}
|
2014-09-03 19:21:19 -07:00
|
|
|
|
2015-11-10 13:10:43 -08:00
|
|
|
func (vc validatorCodec) Encode(o interface{}, w io.Writer, n *int, err *error) {
|
2015-07-25 15:45:45 -07:00
|
|
|
wire.WriteBinary(o.(*Validator), w, n, err)
|
2014-09-03 19:21:19 -07:00
|
|
|
}
|
|
|
|
|
2015-11-10 13:10:43 -08:00
|
|
|
func (vc validatorCodec) Decode(r io.Reader, n *int, err *error) interface{} {
|
|
|
|
return wire.ReadBinary(&Validator{}, r, 0, n, err)
|
2014-09-03 19:21:19 -07:00
|
|
|
}
|
|
|
|
|
2014-10-11 21:27:58 -07:00
|
|
|
func (vc validatorCodec) Compare(o1 interface{}, o2 interface{}) int {
|
2017-04-27 19:01:18 -04:00
|
|
|
cmn.PanicSanity("ValidatorCodec.Compare not implemented")
|
2015-07-19 23:42:52 +00:00
|
|
|
return 0
|
2014-10-07 00:43:34 -07:00
|
|
|
}
|
2015-12-31 15:02:38 -08:00
|
|
|
|
|
|
|
//--------------------------------------------------------------------------------
|
|
|
|
// For testing...
|
|
|
|
|
2017-09-22 12:00:37 -04:00
|
|
|
// RandValidator returns a randomized validator, useful for testing.
|
|
|
|
// UNSTABLE
|
2017-09-18 22:05:33 -04:00
|
|
|
func RandValidator(randPower bool, minPower int64) (*Validator, *PrivValidatorFS) {
|
2017-04-27 19:01:18 -04:00
|
|
|
_, tempFilePath := cmn.Tempfile("priv_validator_")
|
2017-09-18 22:05:33 -04:00
|
|
|
privVal := GenPrivValidatorFS(tempFilePath)
|
2015-12-31 15:02:38 -08:00
|
|
|
votePower := minPower
|
|
|
|
if randPower {
|
2017-04-27 19:01:18 -04:00
|
|
|
votePower += int64(cmn.RandUint32())
|
2015-12-31 15:02:38 -08:00
|
|
|
}
|
2017-09-21 16:32:02 -04:00
|
|
|
val := NewValidator(privVal.GetPubKey(), votePower)
|
2015-12-31 15:02:38 -08:00
|
|
|
return val, privVal
|
|
|
|
}
|