|
|
|
@ -34,37 +34,7 @@ func voteToStep(vote *Vote) int8 {
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Signer is an interface that defines how to sign votes.
|
|
|
|
|
// It is the caller's duty to verify the msg before calling Sign,
|
|
|
|
|
// eg. to avoid double signing.
|
|
|
|
|
// Currently, the only callers are SignVote and SignProposal.
|
|
|
|
|
type Signer interface {
|
|
|
|
|
PubKey() crypto.PubKey
|
|
|
|
|
Sign(msg []byte) (crypto.Signature, error)
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// DefaultSigner implements Signer.
|
|
|
|
|
// It uses a standard 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
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// PubKey implements Signer. It should return the public key that corresponds
|
|
|
|
|
// to the private key used for signing.
|
|
|
|
|
func (ds *DefaultSigner) PubKey() crypto.PubKey {
|
|
|
|
|
return ds.PrivKey.PubKey()
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// PrivValidator defines the functionality of a local Tendermint validator.
|
|
|
|
|
type PrivValidator interface {
|
|
|
|
|
Address() data.Bytes // redundant since .PubKey().Address()
|
|
|
|
|
PubKey() crypto.PubKey
|
|
|
|
@ -72,60 +42,45 @@ type PrivValidator interface {
|
|
|
|
|
SignVote(chainID string, vote *Vote) error
|
|
|
|
|
SignProposal(chainID string, proposal *Proposal) error
|
|
|
|
|
SignHeartbeat(chainID string, heartbeat *Heartbeat) error
|
|
|
|
|
|
|
|
|
|
Reset()
|
|
|
|
|
|
|
|
|
|
SetFile(file string)
|
|
|
|
|
Save()
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// DefaultPrivValidator implements the functionality for signing blocks.
|
|
|
|
|
type DefaultPrivValidator struct {
|
|
|
|
|
Info PrivValidatorInfo `json:"info"`
|
|
|
|
|
Signer *DefaultSigner `json:"signer"`
|
|
|
|
|
// 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 {
|
|
|
|
|
ID ValidatorID `json:"id"`
|
|
|
|
|
Signer Signer `json:"signer"`
|
|
|
|
|
|
|
|
|
|
// mutable state to be persisted to disk
|
|
|
|
|
// after each signature to prevent double signing
|
|
|
|
|
mtx sync.Mutex
|
|
|
|
|
Info LastSignedInfo `json:"info"`
|
|
|
|
|
|
|
|
|
|
// For persistence.
|
|
|
|
|
// Overloaded for testing.
|
|
|
|
|
filePath string
|
|
|
|
|
mtx sync.Mutex
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
func (pv *DefaultPrivValidator) Address() data.Bytes {
|
|
|
|
|
return pv.Info.Address
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
func (pv *DefaultPrivValidator) PubKey() crypto.PubKey {
|
|
|
|
|
return pv.Info.PubKey
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
type PrivValidatorInfo struct {
|
|
|
|
|
Address data.Bytes `json:"address"`
|
|
|
|
|
PubKey crypto.PubKey `json:"pub_key"`
|
|
|
|
|
LastHeight int `json:"last_height"`
|
|
|
|
|
LastRound int `json:"last_round"`
|
|
|
|
|
LastStep int8 `json:"last_step"`
|
|
|
|
|
LastSignature crypto.Signature `json:"last_signature,omitempty"` // so we dont lose signatures
|
|
|
|
|
LastSignBytes data.Bytes `json:"last_signbytes,omitempty"` // so we dont lose signatures
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
func LoadOrGenPrivValidator(filePath string) *DefaultPrivValidator {
|
|
|
|
|
var privValidator *DefaultPrivValidator
|
|
|
|
|
// 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 {
|
|
|
|
|
var PrivValidatorFS *PrivValidatorFS
|
|
|
|
|
if _, err := os.Stat(filePath); err == nil {
|
|
|
|
|
privValidator = LoadPrivValidator(filePath)
|
|
|
|
|
PrivValidatorFS = LoadPrivValidatorFS(filePath)
|
|
|
|
|
} else {
|
|
|
|
|
privValidator = GenPrivValidator()
|
|
|
|
|
privValidator.SetFile(filePath)
|
|
|
|
|
privValidator.Save()
|
|
|
|
|
PrivValidatorFS = GenPrivValidatorFS(filePath)
|
|
|
|
|
PrivValidatorFS.Save()
|
|
|
|
|
}
|
|
|
|
|
return privValidator
|
|
|
|
|
return PrivValidatorFS
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
func LoadPrivValidator(filePath string) *DefaultPrivValidator {
|
|
|
|
|
// LoadPrivValidatorFS loads a PrivValidatorFS from the filePath.
|
|
|
|
|
func LoadPrivValidatorFS(filePath string) *PrivValidatorFS {
|
|
|
|
|
privValJSONBytes, err := ioutil.ReadFile(filePath)
|
|
|
|
|
if err != nil {
|
|
|
|
|
Exit(err.Error())
|
|
|
|
|
}
|
|
|
|
|
privVal := DefaultPrivValidator{}
|
|
|
|
|
privVal := PrivValidatorFS{}
|
|
|
|
|
err = json.Unmarshal(privValJSONBytes, &privVal)
|
|
|
|
|
if err != nil {
|
|
|
|
|
Exit(Fmt("Error reading PrivValidator from %v: %v\n", filePath, err))
|
|
|
|
@ -135,56 +90,58 @@ func LoadPrivValidator(filePath string) *DefaultPrivValidator {
|
|
|
|
|
return &privVal
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Generates a new validator with private key.
|
|
|
|
|
func GenPrivValidator() *DefaultPrivValidator {
|
|
|
|
|
// 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()
|
|
|
|
|
pubKey := privKey.PubKey()
|
|
|
|
|
return &DefaultPrivValidator{
|
|
|
|
|
Info: PrivValidatorInfo{
|
|
|
|
|
Address: pubKey.Address(),
|
|
|
|
|
PubKey: pubKey,
|
|
|
|
|
return &PrivValidatorFS{
|
|
|
|
|
ID: ValidatorID{privKey.PubKey().Address(), privKey.PubKey()},
|
|
|
|
|
Info: LastSignedInfo{
|
|
|
|
|
LastStep: stepNone,
|
|
|
|
|
},
|
|
|
|
|
Signer: NewDefaultSigner(privKey),
|
|
|
|
|
filePath: "",
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// LoadPrivValidatorWithSigner instantiates a private validator with a custom
|
|
|
|
|
// signer object. Tendermint tracks state in the PrivValidator that might be
|
|
|
|
|
// saved to disk. Please supply a filepath where Tendermint can save the
|
|
|
|
|
// private validator.
|
|
|
|
|
func LoadPrivValidatorWithSigner(signer *DefaultSigner, filePath string) *DefaultPrivValidator {
|
|
|
|
|
return &DefaultPrivValidator{
|
|
|
|
|
Info: PrivValidatorInfo{
|
|
|
|
|
Address: signer.PubKey().Address(),
|
|
|
|
|
PubKey: signer.PubKey(),
|
|
|
|
|
LastStep: stepNone,
|
|
|
|
|
},
|
|
|
|
|
Signer: signer,
|
|
|
|
|
filePath: filePath,
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Overwrite address and pubkey for convenience
|
|
|
|
|
/*func (privVal *DefaultPrivValidator) setPubKeyAndAddress() {
|
|
|
|
|
privVal.PubKey = privVal.Signer.PubKey()
|
|
|
|
|
privVal.Address = privVal.PubKey.Address()
|
|
|
|
|
}*/
|
|
|
|
|
// 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.
|
|
|
|
|
func LoadPrivValidatorFSWithSigner(filePath string, signerFunc func(ValidatorID) Signer) *PrivValidatorFS {
|
|
|
|
|
privValJSONBytes, err := ioutil.ReadFile(filePath)
|
|
|
|
|
if err != nil {
|
|
|
|
|
Exit(err.Error())
|
|
|
|
|
}
|
|
|
|
|
privVal := PrivValidatorFS{}
|
|
|
|
|
err = json.Unmarshal(privValJSONBytes, &privVal)
|
|
|
|
|
if err != nil {
|
|
|
|
|
Exit(Fmt("Error reading PrivValidator from %v: %v\n", filePath, err))
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
func (privVal *DefaultPrivValidator) SetFile(filePath string) {
|
|
|
|
|
privVal.mtx.Lock()
|
|
|
|
|
defer privVal.mtx.Unlock()
|
|
|
|
|
privVal.filePath = filePath
|
|
|
|
|
privVal.Signer = signerFunc(privVal.ID)
|
|
|
|
|
return &privVal
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
func (privVal *DefaultPrivValidator) Save() {
|
|
|
|
|
// Address returns the address of the validator.
|
|
|
|
|
func (pv *PrivValidatorFS) Address() data.Bytes {
|
|
|
|
|
return pv.ID.Address
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// PubKey returns the public key of the validator.
|
|
|
|
|
func (pv *PrivValidatorFS) PubKey() crypto.PubKey {
|
|
|
|
|
return pv.ID.PubKey
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Save persists the PrivValidatorFS to disk.
|
|
|
|
|
func (privVal *PrivValidatorFS) Save() {
|
|
|
|
|
privVal.mtx.Lock()
|
|
|
|
|
defer privVal.mtx.Unlock()
|
|
|
|
|
privVal.save()
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
func (privVal *DefaultPrivValidator) save() {
|
|
|
|
|
func (privVal *PrivValidatorFS) save() {
|
|
|
|
|
if privVal.filePath == "" {
|
|
|
|
|
PanicSanity("Cannot save PrivValidator: filePath not set")
|
|
|
|
|
}
|
|
|
|
@ -200,8 +157,9 @@ func (privVal *DefaultPrivValidator) save() {
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Reset resets all fields in the PrivValidatorFS.Info.
|
|
|
|
|
// NOTE: Unsafe!
|
|
|
|
|
func (privVal *DefaultPrivValidator) Reset() {
|
|
|
|
|
func (privVal *PrivValidatorFS) Reset() {
|
|
|
|
|
privVal.Info.LastHeight = 0
|
|
|
|
|
privVal.Info.LastRound = 0
|
|
|
|
|
privVal.Info.LastStep = 0
|
|
|
|
@ -210,11 +168,8 @@ func (privVal *DefaultPrivValidator) Reset() {
|
|
|
|
|
privVal.Save()
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
func (privVal *DefaultPrivValidator) GetAddress() []byte {
|
|
|
|
|
return privVal.Address()
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
func (privVal *DefaultPrivValidator) SignVote(chainID string, vote *Vote) error {
|
|
|
|
|
// SignVote signs a canonical representation of the vote, along with the chainID.
|
|
|
|
|
func (privVal *PrivValidatorFS) SignVote(chainID string, vote *Vote) error {
|
|
|
|
|
privVal.mtx.Lock()
|
|
|
|
|
defer privVal.mtx.Unlock()
|
|
|
|
|
signature, err := privVal.signBytesHRS(vote.Height, vote.Round, voteToStep(vote), SignBytes(chainID, vote))
|
|
|
|
@ -225,7 +180,8 @@ func (privVal *DefaultPrivValidator) SignVote(chainID string, vote *Vote) error
|
|
|
|
|
return nil
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
func (privVal *DefaultPrivValidator) SignProposal(chainID string, proposal *Proposal) error {
|
|
|
|
|
// SignProposal signs a canonical representation of the proposal, along with the chainID.
|
|
|
|
|
func (privVal *PrivValidatorFS) SignProposal(chainID string, proposal *Proposal) error {
|
|
|
|
|
privVal.mtx.Lock()
|
|
|
|
|
defer privVal.mtx.Unlock()
|
|
|
|
|
signature, err := privVal.signBytesHRS(proposal.Height, proposal.Round, stepPropose, SignBytes(chainID, proposal))
|
|
|
|
@ -236,8 +192,17 @@ func (privVal *DefaultPrivValidator) SignProposal(chainID string, proposal *Prop
|
|
|
|
|
return nil
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// SignHeartbeat signs a canonical representation of the heartbeat, along with the chainID.
|
|
|
|
|
func (privVal *PrivValidatorFS) SignHeartbeat(chainID string, heartbeat *Heartbeat) error {
|
|
|
|
|
privVal.mtx.Lock()
|
|
|
|
|
defer privVal.mtx.Unlock()
|
|
|
|
|
var err error
|
|
|
|
|
heartbeat.Signature, err = privVal.Signer.Sign(SignBytes(chainID, heartbeat))
|
|
|
|
|
return err
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// check if there's a regression. Else sign and write the hrs+signature to disk
|
|
|
|
|
func (privVal *DefaultPrivValidator) signBytesHRS(height, round int, step int8, signBytes []byte) (crypto.Signature, error) {
|
|
|
|
|
func (privVal *PrivValidatorFS) signBytesHRS(height, round int, step int8, signBytes []byte) (crypto.Signature, error) {
|
|
|
|
|
sig := crypto.Signature{}
|
|
|
|
|
info := privVal.Info
|
|
|
|
|
// If height regression, err
|
|
|
|
@ -287,32 +252,68 @@ func (privVal *DefaultPrivValidator) signBytesHRS(height, round int, step int8,
|
|
|
|
|
privVal.save()
|
|
|
|
|
|
|
|
|
|
return sig, nil
|
|
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
func (privVal *DefaultPrivValidator) SignHeartbeat(chainID string, heartbeat *Heartbeat) error {
|
|
|
|
|
privVal.mtx.Lock()
|
|
|
|
|
defer privVal.mtx.Unlock()
|
|
|
|
|
var err error
|
|
|
|
|
heartbeat.Signature, err = privVal.Signer.Sign(SignBytes(chainID, heartbeat))
|
|
|
|
|
return err
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
func (privVal *DefaultPrivValidator) String() string {
|
|
|
|
|
// String returns a string representation of the PrivValidatorFS.
|
|
|
|
|
func (privVal *PrivValidatorFS) String() string {
|
|
|
|
|
info := privVal.Info
|
|
|
|
|
return fmt.Sprintf("PrivValidator{%v LH:%v, LR:%v, LS:%v}", info.Address, info.LastHeight, info.LastRound, info.LastStep)
|
|
|
|
|
return fmt.Sprintf("PrivValidator{%v LH:%v, LR:%v, LS:%v}", privVal.Address(), info.LastHeight, info.LastRound, info.LastStep)
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
//-------------------------------------
|
|
|
|
|
|
|
|
|
|
type PrivValidatorsByAddress []*DefaultPrivValidator
|
|
|
|
|
// ValidatorID contains the identity of the validator.
|
|
|
|
|
type ValidatorID struct {
|
|
|
|
|
Address data.Bytes `json:"address"`
|
|
|
|
|
PubKey crypto.PubKey `json:"pub_key"`
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// LastSignedInfo contains information about the latest
|
|
|
|
|
// data signed by a validator to help prevent double signing.
|
|
|
|
|
type LastSignedInfo struct {
|
|
|
|
|
LastHeight int `json:"last_height"`
|
|
|
|
|
LastRound int `json:"last_round"`
|
|
|
|
|
LastStep int8 `json:"last_step"`
|
|
|
|
|
LastSignature crypto.Signature `json:"last_signature,omitempty"` // so we dont lose signatures
|
|
|
|
|
LastSignBytes data.Bytes `json:"last_signbytes,omitempty"` // so we dont lose signatures
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// 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
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
//-------------------------------------
|
|
|
|
|
|
|
|
|
|
type PrivValidatorsByAddress []*PrivValidatorFS
|
|
|
|
|
|
|
|
|
|
func (pvs PrivValidatorsByAddress) Len() int {
|
|
|
|
|
return len(pvs)
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
func (pvs PrivValidatorsByAddress) Less(i, j int) bool {
|
|
|
|
|
return bytes.Compare(pvs[i].Info.Address, pvs[j].Info.Address) == -1
|
|
|
|
|
return bytes.Compare(pvs[i].Address(), pvs[j].Address()) == -1
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
func (pvs PrivValidatorsByAddress) Swap(i, j int) {
|
|
|
|
|