2015-08-10 20:38:45 -07:00
|
|
|
package types
|
2014-09-04 03:32:38 -07:00
|
|
|
|
|
|
|
import (
|
2015-08-10 20:38:45 -07:00
|
|
|
"bytes"
|
2017-03-21 14:46:15 -07:00
|
|
|
"encoding/json"
|
2014-12-31 16:14:26 -08:00
|
|
|
"errors"
|
2014-12-09 18:49:04 -08:00
|
|
|
"fmt"
|
|
|
|
"io/ioutil"
|
2015-10-18 07:20:05 -07:00
|
|
|
"os"
|
2014-12-22 18:49:37 -08:00
|
|
|
"sync"
|
2017-12-21 12:37:27 -08:00
|
|
|
"time"
|
2014-12-09 18:49:04 -08:00
|
|
|
|
2017-05-02 00:53:32 -07:00
|
|
|
crypto "github.com/tendermint/go-crypto"
|
2017-09-20 15:40:41 -07:00
|
|
|
cmn "github.com/tendermint/tmlibs/common"
|
2014-12-09 18:49:04 -08:00
|
|
|
)
|
|
|
|
|
2017-09-11 12:45:12 -07:00
|
|
|
// TODO: type ?
|
2014-12-09 18:49:04 -08:00
|
|
|
const (
|
2018-01-04 10:52:41 -08:00
|
|
|
stepNone int8 = 0 // Used to distinguish the initial state
|
|
|
|
stepPropose int8 = 1
|
|
|
|
stepPrevote int8 = 2
|
|
|
|
stepPrecommit int8 = 3
|
2014-09-04 03:32:38 -07:00
|
|
|
)
|
|
|
|
|
2015-08-10 20:38:45 -07:00
|
|
|
func voteToStep(vote *Vote) int8 {
|
2014-12-09 18:49:04 -08:00
|
|
|
switch vote.Type {
|
2015-08-10 20:38:45 -07:00
|
|
|
case VoteTypePrevote:
|
2014-12-09 18:49:04 -08:00
|
|
|
return stepPrevote
|
2015-08-10 20:38:45 -07:00
|
|
|
case VoteTypePrecommit:
|
2014-12-09 18:49:04 -08:00
|
|
|
return stepPrecommit
|
|
|
|
default:
|
2017-09-20 15:40:41 -07:00
|
|
|
cmn.PanicSanity("Unknown vote type")
|
2015-07-19 16:42:52 -07:00
|
|
|
return 0
|
2014-12-09 18:49:04 -08:00
|
|
|
}
|
|
|
|
}
|
2014-09-04 03:32:38 -07:00
|
|
|
|
2017-09-21 09:19:21 -07:00
|
|
|
// PrivValidator defines the functionality of a local Tendermint validator
|
|
|
|
// that signs votes, proposals, and heartbeats, and never double signs.
|
2017-09-18 15:12:31 -07:00
|
|
|
type PrivValidator interface {
|
2018-01-05 22:26:51 -08:00
|
|
|
GetAddress() Address // redundant since .PubKey().Address()
|
2017-09-21 13:32:02 -07:00
|
|
|
GetPubKey() crypto.PubKey
|
2017-09-18 15:12:31 -07:00
|
|
|
|
|
|
|
SignVote(chainID string, vote *Vote) error
|
|
|
|
SignProposal(chainID string, proposal *Proposal) error
|
|
|
|
SignHeartbeat(chainID string, heartbeat *Heartbeat) error
|
|
|
|
}
|
|
|
|
|
2017-09-18 19:05:33 -07:00
|
|
|
// PrivValidatorFS implements PrivValidator using data persisted to disk
|
|
|
|
// to prevent double signing. The Signer itself can be mutated to use
|
|
|
|
// something besides the default, for instance a hardware signer.
|
|
|
|
type PrivValidatorFS struct {
|
2018-02-03 00:42:59 -08:00
|
|
|
Address Address `json:"address"`
|
2017-09-21 13:32:02 -07:00
|
|
|
PubKey crypto.PubKey `json:"pub_key"`
|
2017-12-01 17:04:53 -08:00
|
|
|
LastHeight int64 `json:"last_height"`
|
2017-09-21 13:32:02 -07:00
|
|
|
LastRound int `json:"last_round"`
|
|
|
|
LastStep int8 `json:"last_step"`
|
|
|
|
LastSignature crypto.Signature `json:"last_signature,omitempty"` // so we dont lose signatures
|
2018-02-03 00:42:59 -08:00
|
|
|
LastSignBytes cmn.HexBytes `json:"last_signbytes,omitempty"` // so we dont lose signatures
|
2017-09-18 19:05:33 -07:00
|
|
|
|
2017-09-21 13:32:02 -07:00
|
|
|
// PrivKey should be empty if a Signer other than the default is being used.
|
|
|
|
PrivKey crypto.PrivKey `json:"priv_key"`
|
|
|
|
Signer `json:"-"`
|
2017-09-18 15:12:31 -07:00
|
|
|
|
|
|
|
// For persistence.
|
|
|
|
// Overloaded for testing.
|
|
|
|
filePath string
|
2017-09-21 13:32:02 -07:00
|
|
|
mtx sync.Mutex
|
2017-05-25 10:39:36 -07:00
|
|
|
}
|
|
|
|
|
2017-09-21 13:58:18 -07:00
|
|
|
// Signer is an interface that defines how to sign messages.
|
|
|
|
// It is the caller's duty to verify the msg before calling Sign,
|
|
|
|
// eg. to avoid double signing.
|
|
|
|
// Currently, the only callers are SignVote, SignProposal, and SignHeartbeat.
|
|
|
|
type Signer interface {
|
|
|
|
Sign(msg []byte) (crypto.Signature, error)
|
|
|
|
}
|
|
|
|
|
|
|
|
// DefaultSigner implements Signer.
|
|
|
|
// It uses a standard, unencrypted crypto.PrivKey.
|
|
|
|
type DefaultSigner struct {
|
|
|
|
PrivKey crypto.PrivKey `json:"priv_key"`
|
|
|
|
}
|
|
|
|
|
|
|
|
// NewDefaultSigner returns an instance of DefaultSigner.
|
|
|
|
func NewDefaultSigner(priv crypto.PrivKey) *DefaultSigner {
|
|
|
|
return &DefaultSigner{
|
|
|
|
PrivKey: priv,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Sign implements Signer. It signs the byte slice with a private key.
|
|
|
|
func (ds *DefaultSigner) Sign(msg []byte) (crypto.Signature, error) {
|
|
|
|
return ds.PrivKey.Sign(msg), nil
|
|
|
|
}
|
|
|
|
|
2017-09-21 13:32:02 -07:00
|
|
|
// GetAddress returns the address of the validator.
|
2017-09-21 09:19:21 -07:00
|
|
|
// Implements PrivValidator.
|
2018-01-05 22:26:51 -08:00
|
|
|
func (pv *PrivValidatorFS) GetAddress() Address {
|
2017-09-21 13:32:02 -07:00
|
|
|
return pv.Address
|
2014-10-20 19:02:10 -07:00
|
|
|
}
|
|
|
|
|
2017-09-21 13:32:02 -07:00
|
|
|
// GetPubKey returns the public key of the validator.
|
2017-09-21 09:19:21 -07:00
|
|
|
// Implements PrivValidator.
|
2017-09-21 13:32:02 -07:00
|
|
|
func (pv *PrivValidatorFS) GetPubKey() crypto.PubKey {
|
|
|
|
return pv.PubKey
|
2014-12-09 18:49:04 -08:00
|
|
|
}
|
|
|
|
|
2017-09-21 13:58:18 -07:00
|
|
|
// GenPrivValidatorFS generates a new validator with randomly generated private key
|
|
|
|
// and sets the filePath, but does not call Save().
|
|
|
|
func GenPrivValidatorFS(filePath string) *PrivValidatorFS {
|
|
|
|
privKey := crypto.GenPrivKeyEd25519().Wrap()
|
|
|
|
return &PrivValidatorFS{
|
|
|
|
Address: privKey.PubKey().Address(),
|
|
|
|
PubKey: privKey.PubKey(),
|
|
|
|
PrivKey: privKey,
|
|
|
|
LastStep: stepNone,
|
|
|
|
Signer: NewDefaultSigner(privKey),
|
|
|
|
filePath: filePath,
|
2015-10-18 07:20:05 -07:00
|
|
|
}
|
2017-09-12 15:18:07 -07:00
|
|
|
}
|
|
|
|
|
2017-09-21 13:58:18 -07:00
|
|
|
// LoadPrivValidatorFS loads a PrivValidatorFS from the filePath.
|
|
|
|
func LoadPrivValidatorFS(filePath string) *PrivValidatorFS {
|
2017-09-21 14:08:17 -07:00
|
|
|
return LoadPrivValidatorFSWithSigner(filePath, func(privVal PrivValidator) Signer {
|
|
|
|
return NewDefaultSigner(privVal.(*PrivValidatorFS).PrivKey)
|
|
|
|
})
|
2017-09-18 19:05:33 -07:00
|
|
|
}
|
2015-10-18 07:20:05 -07:00
|
|
|
|
2017-09-21 13:58:18 -07:00
|
|
|
// LoadOrGenPrivValidatorFS loads a PrivValidatorFS from the given filePath
|
|
|
|
// or else generates a new one and saves it to the filePath.
|
|
|
|
func LoadOrGenPrivValidatorFS(filePath string) *PrivValidatorFS {
|
2017-09-21 21:05:39 -07:00
|
|
|
var privVal *PrivValidatorFS
|
2017-09-21 13:58:18 -07:00
|
|
|
if _, err := os.Stat(filePath); err == nil {
|
2017-09-21 21:05:39 -07:00
|
|
|
privVal = LoadPrivValidatorFS(filePath)
|
2017-09-21 13:58:18 -07:00
|
|
|
} else {
|
2017-09-21 21:05:39 -07:00
|
|
|
privVal = GenPrivValidatorFS(filePath)
|
|
|
|
privVal.Save()
|
2017-09-21 13:58:18 -07:00
|
|
|
}
|
2017-09-21 21:05:39 -07:00
|
|
|
return privVal
|
2017-09-21 13:58:18 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
// LoadPrivValidatorWithSigner loads a PrivValidatorFS with a custom
|
|
|
|
// signer object. The PrivValidatorFS handles double signing prevention by persisting
|
|
|
|
// data to the filePath, while the Signer handles the signing.
|
|
|
|
// If the filePath does not exist, the PrivValidatorFS must be created manually and saved.
|
2017-09-21 14:08:17 -07:00
|
|
|
func LoadPrivValidatorFSWithSigner(filePath string, signerFunc func(PrivValidator) Signer) *PrivValidatorFS {
|
2017-09-21 13:58:18 -07:00
|
|
|
privValJSONBytes, err := ioutil.ReadFile(filePath)
|
|
|
|
if err != nil {
|
|
|
|
cmn.Exit(err.Error())
|
|
|
|
}
|
2017-09-21 14:08:17 -07:00
|
|
|
privVal := &PrivValidatorFS{}
|
2017-09-21 13:58:18 -07:00
|
|
|
err = json.Unmarshal(privValJSONBytes, &privVal)
|
|
|
|
if err != nil {
|
|
|
|
cmn.Exit(cmn.Fmt("Error reading PrivValidator from %v: %v\n", filePath, err))
|
|
|
|
}
|
|
|
|
|
|
|
|
privVal.filePath = filePath
|
2017-09-21 14:08:17 -07:00
|
|
|
privVal.Signer = signerFunc(privVal)
|
|
|
|
return privVal
|
2015-04-03 16:15:52 -07:00
|
|
|
}
|
|
|
|
|
2017-09-18 19:05:33 -07:00
|
|
|
// Save persists the PrivValidatorFS to disk.
|
|
|
|
func (privVal *PrivValidatorFS) Save() {
|
2014-12-22 18:49:37 -08:00
|
|
|
privVal.mtx.Lock()
|
|
|
|
defer privVal.mtx.Unlock()
|
|
|
|
privVal.save()
|
|
|
|
}
|
|
|
|
|
2017-09-18 19:05:33 -07:00
|
|
|
func (privVal *PrivValidatorFS) save() {
|
2015-05-15 18:05:09 -07:00
|
|
|
if privVal.filePath == "" {
|
2017-09-20 15:40:41 -07:00
|
|
|
cmn.PanicSanity("Cannot save PrivValidator: filePath not set")
|
2015-05-15 18:05:09 -07:00
|
|
|
}
|
2017-03-21 14:46:15 -07:00
|
|
|
jsonBytes, err := json.Marshal(privVal)
|
|
|
|
if err != nil {
|
|
|
|
// `@; BOOM!!!
|
2017-09-20 15:40:41 -07:00
|
|
|
cmn.PanicCrisis(err)
|
2017-03-21 14:46:15 -07:00
|
|
|
}
|
2017-09-20 15:40:41 -07:00
|
|
|
err = cmn.WriteFileAtomic(privVal.filePath, jsonBytes, 0600)
|
2014-12-28 00:44:56 -08:00
|
|
|
if err != nil {
|
2015-04-18 12:48:14 -07:00
|
|
|
// `@; BOOM!!!
|
2017-09-20 15:40:41 -07:00
|
|
|
cmn.PanicCrisis(err)
|
2014-12-28 00:44:56 -08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-09-21 13:58:18 -07:00
|
|
|
// Reset resets all fields in the PrivValidatorFS.
|
|
|
|
// NOTE: Unsafe!
|
|
|
|
func (privVal *PrivValidatorFS) Reset() {
|
|
|
|
privVal.LastHeight = 0
|
|
|
|
privVal.LastRound = 0
|
|
|
|
privVal.LastStep = 0
|
|
|
|
privVal.LastSignature = crypto.Signature{}
|
|
|
|
privVal.LastSignBytes = nil
|
|
|
|
privVal.Save()
|
|
|
|
}
|
|
|
|
|
2017-12-20 12:18:15 -08:00
|
|
|
// SignVote signs a canonical representation of the vote, along with the
|
|
|
|
// chainID. Implements PrivValidator.
|
2017-09-21 13:58:18 -07:00
|
|
|
func (privVal *PrivValidatorFS) SignVote(chainID string, vote *Vote) error {
|
|
|
|
privVal.mtx.Lock()
|
|
|
|
defer privVal.mtx.Unlock()
|
2018-01-04 10:52:41 -08:00
|
|
|
if err := privVal.signVote(chainID, vote); err != nil {
|
2017-09-21 13:58:18 -07:00
|
|
|
return errors.New(cmn.Fmt("Error signing vote: %v", err))
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2017-12-20 12:18:15 -08:00
|
|
|
// SignProposal signs a canonical representation of the proposal, along with
|
|
|
|
// the chainID. Implements PrivValidator.
|
2017-09-21 13:58:18 -07:00
|
|
|
func (privVal *PrivValidatorFS) SignProposal(chainID string, proposal *Proposal) error {
|
|
|
|
privVal.mtx.Lock()
|
|
|
|
defer privVal.mtx.Unlock()
|
2018-01-04 10:52:41 -08:00
|
|
|
if err := privVal.signProposal(chainID, proposal); err != nil {
|
2017-09-21 13:58:18 -07:00
|
|
|
return fmt.Errorf("Error signing proposal: %v", err)
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2017-12-21 13:28:05 -08:00
|
|
|
// returns error if HRS regression or no LastSignBytes. returns true if HRS is unchanged
|
2017-12-21 12:37:27 -08:00
|
|
|
func (privVal *PrivValidatorFS) checkHRS(height int64, round int, step int8) (bool, error) {
|
2017-09-21 13:32:02 -07:00
|
|
|
if privVal.LastHeight > height {
|
2017-12-21 12:37:27 -08:00
|
|
|
return false, errors.New("Height regression")
|
2017-09-21 13:32:02 -07:00
|
|
|
}
|
2017-12-20 12:18:15 -08:00
|
|
|
|
2017-09-21 13:32:02 -07:00
|
|
|
if privVal.LastHeight == height {
|
|
|
|
if privVal.LastRound > round {
|
2017-12-21 12:37:27 -08:00
|
|
|
return false, errors.New("Round regression")
|
2017-09-21 13:32:02 -07:00
|
|
|
}
|
2017-12-20 12:18:15 -08:00
|
|
|
|
2017-09-21 13:32:02 -07:00
|
|
|
if privVal.LastRound == round {
|
|
|
|
if privVal.LastStep > step {
|
2017-12-21 12:37:27 -08:00
|
|
|
return false, errors.New("Step regression")
|
2017-09-21 13:32:02 -07:00
|
|
|
} else if privVal.LastStep == step {
|
|
|
|
if privVal.LastSignBytes != nil {
|
|
|
|
if privVal.LastSignature.Empty() {
|
2017-12-20 12:18:15 -08:00
|
|
|
panic("privVal: LastSignature is nil but LastSignBytes is not!")
|
2017-09-21 13:32:02 -07:00
|
|
|
}
|
2017-12-21 12:37:27 -08:00
|
|
|
return true, nil
|
2017-09-21 13:32:02 -07:00
|
|
|
}
|
2017-12-21 12:37:27 -08:00
|
|
|
return false, errors.New("No LastSignature found")
|
2017-09-21 13:32:02 -07:00
|
|
|
}
|
|
|
|
}
|
2017-09-18 20:16:14 -07:00
|
|
|
}
|
2017-12-21 12:37:27 -08:00
|
|
|
return false, nil
|
|
|
|
}
|
2017-09-18 20:16:14 -07:00
|
|
|
|
2018-01-04 10:52:41 -08:00
|
|
|
// signVote checks if the vote is good to sign and sets the vote signature.
|
|
|
|
// It may need to set the timestamp as well if the vote is otherwise the same as
|
|
|
|
// a previously signed vote (ie. we crashed after signing but before the vote hit the WAL).
|
|
|
|
func (privVal *PrivValidatorFS) signVote(chainID string, vote *Vote) error {
|
|
|
|
height, round, step := vote.Height, vote.Round, voteToStep(vote)
|
|
|
|
signBytes := SignBytes(chainID, vote)
|
2017-12-21 12:37:27 -08:00
|
|
|
|
|
|
|
sameHRS, err := privVal.checkHRS(height, round, step)
|
2017-09-21 13:32:02 -07:00
|
|
|
if err != nil {
|
2018-01-04 10:52:41 -08:00
|
|
|
return err
|
2017-09-18 20:16:14 -07:00
|
|
|
}
|
|
|
|
|
2017-12-21 12:37:27 -08:00
|
|
|
// We might crash before writing to the wal,
|
2018-01-04 10:52:41 -08:00
|
|
|
// causing us to try to re-sign for the same HRS.
|
|
|
|
// If signbytes are the same, use the last signature.
|
|
|
|
// If they only differ by timestamp, use last timestamp and signature
|
|
|
|
// Otherwise, return error
|
2017-12-21 12:37:27 -08:00
|
|
|
if sameHRS {
|
2018-01-04 10:52:41 -08:00
|
|
|
if bytes.Equal(signBytes, privVal.LastSignBytes) {
|
|
|
|
vote.Signature = privVal.LastSignature
|
|
|
|
} else if timestamp, ok := checkVotesOnlyDifferByTimestamp(privVal.LastSignBytes, signBytes); ok {
|
|
|
|
vote.Timestamp = timestamp
|
|
|
|
vote.Signature = privVal.LastSignature
|
|
|
|
} else {
|
|
|
|
err = fmt.Errorf("Conflicting data")
|
2017-12-21 12:37:27 -08:00
|
|
|
}
|
2018-01-04 10:52:41 -08:00
|
|
|
return err
|
2017-12-21 12:37:27 -08:00
|
|
|
}
|
|
|
|
|
2018-01-04 10:52:41 -08:00
|
|
|
// It passed the checks. Sign the vote
|
|
|
|
sig, err := privVal.Sign(signBytes)
|
2017-12-21 12:37:27 -08:00
|
|
|
if err != nil {
|
2018-01-04 10:52:41 -08:00
|
|
|
return err
|
2017-12-21 12:37:27 -08:00
|
|
|
}
|
|
|
|
privVal.saveSigned(height, round, step, signBytes, sig)
|
2018-01-04 10:52:41 -08:00
|
|
|
vote.Signature = sig
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// signProposal checks if the proposal is good to sign and sets the proposal signature.
|
|
|
|
// It may need to set the timestamp as well if the proposal is otherwise the same as
|
|
|
|
// a previously signed proposal ie. we crashed after signing but before the proposal hit the WAL).
|
|
|
|
func (privVal *PrivValidatorFS) signProposal(chainID string, proposal *Proposal) error {
|
|
|
|
height, round, step := proposal.Height, proposal.Round, stepPropose
|
|
|
|
signBytes := SignBytes(chainID, proposal)
|
|
|
|
|
|
|
|
sameHRS, err := privVal.checkHRS(height, round, step)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
// We might crash before writing to the wal,
|
|
|
|
// causing us to try to re-sign for the same HRS.
|
|
|
|
// If signbytes are the same, use the last signature.
|
|
|
|
// If they only differ by timestamp, use last timestamp and signature
|
|
|
|
// Otherwise, return error
|
|
|
|
if sameHRS {
|
|
|
|
if bytes.Equal(signBytes, privVal.LastSignBytes) {
|
|
|
|
proposal.Signature = privVal.LastSignature
|
|
|
|
} else if timestamp, ok := checkProposalsOnlyDifferByTimestamp(privVal.LastSignBytes, signBytes); ok {
|
|
|
|
proposal.Timestamp = timestamp
|
|
|
|
proposal.Signature = privVal.LastSignature
|
|
|
|
} else {
|
|
|
|
err = fmt.Errorf("Conflicting data")
|
|
|
|
}
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
// It passed the checks. Sign the proposal
|
|
|
|
sig, err := privVal.Sign(signBytes)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
privVal.saveSigned(height, round, step, signBytes, sig)
|
|
|
|
proposal.Signature = sig
|
|
|
|
return nil
|
2017-12-21 12:37:27 -08:00
|
|
|
}
|
|
|
|
|
|
|
|
// Persist height/round/step and signature
|
|
|
|
func (privVal *PrivValidatorFS) saveSigned(height int64, round int, step int8,
|
|
|
|
signBytes []byte, sig crypto.Signature) {
|
|
|
|
|
2017-09-21 13:32:02 -07:00
|
|
|
privVal.LastHeight = height
|
|
|
|
privVal.LastRound = round
|
|
|
|
privVal.LastStep = step
|
|
|
|
privVal.LastSignature = sig
|
|
|
|
privVal.LastSignBytes = signBytes
|
|
|
|
privVal.save()
|
2017-09-18 20:16:14 -07:00
|
|
|
}
|
|
|
|
|
2017-09-21 13:58:18 -07:00
|
|
|
// SignHeartbeat signs a canonical representation of the heartbeat, along with the chainID.
|
|
|
|
// Implements PrivValidator.
|
|
|
|
func (privVal *PrivValidatorFS) SignHeartbeat(chainID string, heartbeat *Heartbeat) error {
|
|
|
|
privVal.mtx.Lock()
|
|
|
|
defer privVal.mtx.Unlock()
|
|
|
|
var err error
|
2017-09-21 18:44:36 -07:00
|
|
|
heartbeat.Signature, err = privVal.Sign(SignBytes(chainID, heartbeat))
|
2017-09-21 13:58:18 -07:00
|
|
|
return err
|
2016-08-14 09:31:24 -07:00
|
|
|
}
|
|
|
|
|
2017-09-21 09:19:21 -07:00
|
|
|
// String returns a string representation of the PrivValidatorFS.
|
|
|
|
func (privVal *PrivValidatorFS) String() string {
|
2017-09-21 13:32:02 -07:00
|
|
|
return fmt.Sprintf("PrivValidator{%v LH:%v, LR:%v, LS:%v}", privVal.GetAddress(), privVal.LastHeight, privVal.LastRound, privVal.LastStep)
|
2017-09-21 09:19:21 -07:00
|
|
|
}
|
|
|
|
|
2015-08-10 20:38:45 -07:00
|
|
|
//-------------------------------------
|
|
|
|
|
2017-09-18 19:05:33 -07:00
|
|
|
type PrivValidatorsByAddress []*PrivValidatorFS
|
2015-08-10 20:38:45 -07:00
|
|
|
|
|
|
|
func (pvs PrivValidatorsByAddress) Len() int {
|
|
|
|
return len(pvs)
|
|
|
|
}
|
|
|
|
|
|
|
|
func (pvs PrivValidatorsByAddress) Less(i, j int) bool {
|
2017-09-21 13:32:02 -07:00
|
|
|
return bytes.Compare(pvs[i].GetAddress(), pvs[j].GetAddress()) == -1
|
2015-08-10 20:38:45 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
func (pvs PrivValidatorsByAddress) Swap(i, j int) {
|
|
|
|
it := pvs[i]
|
|
|
|
pvs[i] = pvs[j]
|
|
|
|
pvs[j] = it
|
|
|
|
}
|
2017-12-21 12:37:27 -08:00
|
|
|
|
|
|
|
//-------------------------------------
|
|
|
|
|
2018-01-04 10:52:41 -08:00
|
|
|
// returns the timestamp from the lastSignBytes.
|
|
|
|
// returns true if the only difference in the votes is their timestamp.
|
|
|
|
func checkVotesOnlyDifferByTimestamp(lastSignBytes, newSignBytes []byte) (time.Time, bool) {
|
2017-12-21 12:37:27 -08:00
|
|
|
var lastVote, newVote CanonicalJSONOnceVote
|
|
|
|
if err := json.Unmarshal(lastSignBytes, &lastVote); err != nil {
|
|
|
|
panic(fmt.Sprintf("LastSignBytes cannot be unmarshalled into vote: %v", err))
|
|
|
|
}
|
|
|
|
if err := json.Unmarshal(newSignBytes, &newVote); err != nil {
|
|
|
|
panic(fmt.Sprintf("signBytes cannot be unmarshalled into vote: %v", err))
|
|
|
|
}
|
|
|
|
|
2018-01-04 10:52:41 -08:00
|
|
|
lastTime, err := time.Parse(timeFormat, lastVote.Vote.Timestamp)
|
|
|
|
if err != nil {
|
|
|
|
panic(err)
|
|
|
|
}
|
|
|
|
|
2017-12-21 12:37:27 -08:00
|
|
|
// set the times to the same value and check equality
|
|
|
|
now := CanonicalTime(time.Now())
|
|
|
|
lastVote.Vote.Timestamp = now
|
|
|
|
newVote.Vote.Timestamp = now
|
|
|
|
lastVoteBytes, _ := json.Marshal(lastVote)
|
|
|
|
newVoteBytes, _ := json.Marshal(newVote)
|
|
|
|
|
2018-01-04 10:52:41 -08:00
|
|
|
return lastTime, bytes.Equal(newVoteBytes, lastVoteBytes)
|
2017-12-21 12:37:27 -08:00
|
|
|
}
|
|
|
|
|
2018-01-04 10:52:41 -08:00
|
|
|
// returns the timestamp from the lastSignBytes.
|
2017-12-21 12:37:27 -08:00
|
|
|
// returns true if the only difference in the proposals is their timestamp
|
2018-01-04 10:52:41 -08:00
|
|
|
func checkProposalsOnlyDifferByTimestamp(lastSignBytes, newSignBytes []byte) (time.Time, bool) {
|
2017-12-21 12:37:27 -08:00
|
|
|
var lastProposal, newProposal CanonicalJSONOnceProposal
|
|
|
|
if err := json.Unmarshal(lastSignBytes, &lastProposal); err != nil {
|
|
|
|
panic(fmt.Sprintf("LastSignBytes cannot be unmarshalled into proposal: %v", err))
|
|
|
|
}
|
|
|
|
if err := json.Unmarshal(newSignBytes, &newProposal); err != nil {
|
|
|
|
panic(fmt.Sprintf("signBytes cannot be unmarshalled into proposal: %v", err))
|
|
|
|
}
|
|
|
|
|
2018-01-04 10:52:41 -08:00
|
|
|
lastTime, err := time.Parse(timeFormat, lastProposal.Proposal.Timestamp)
|
|
|
|
if err != nil {
|
|
|
|
panic(err)
|
|
|
|
}
|
|
|
|
|
2017-12-21 12:37:27 -08:00
|
|
|
// set the times to the same value and check equality
|
|
|
|
now := CanonicalTime(time.Now())
|
|
|
|
lastProposal.Proposal.Timestamp = now
|
|
|
|
newProposal.Proposal.Timestamp = now
|
|
|
|
lastProposalBytes, _ := json.Marshal(lastProposal)
|
|
|
|
newProposalBytes, _ := json.Marshal(newProposal)
|
|
|
|
|
2018-01-04 10:52:41 -08:00
|
|
|
return lastTime, bytes.Equal(newProposalBytes, lastProposalBytes)
|
2017-12-21 12:37:27 -08:00
|
|
|
}
|