tendermint/types/genesis.go

114 lines
3.2 KiB
Go
Raw Normal View History

2015-11-01 11:34:08 -08:00
package types
import (
"encoding/json"
"io/ioutil"
2015-11-01 11:34:08 -08:00
"time"
"github.com/tendermint/go-crypto"
2017-04-27 16:01:18 -07:00
cmn "github.com/tendermint/tmlibs/common"
2015-11-01 11:34:08 -08:00
)
//------------------------------------------------------------
// core types for a genesis definition
2017-06-26 14:20:27 -07:00
// GenesisValidator is an initial validator.
2015-11-01 11:34:08 -08:00
type GenesisValidator struct {
PubKey crypto.PubKey `json:"pub_key"`
2017-09-21 11:37:34 -07:00
Power int64 `json:"power"`
Name string `json:"name"`
2015-11-01 11:34:08 -08:00
}
2017-06-26 14:20:27 -07:00
// GenesisDoc defines the initial conditions for a tendermint blockchain, in particular its validator set.
2015-11-01 11:34:08 -08:00
type GenesisDoc struct {
GenesisTime time.Time `json:"genesis_time"`
ChainID string `json:"chain_id"`
ConsensusParams *ConsensusParams `json:"consensus_params,omitempty"`
Validators []GenesisValidator `json:"validators"`
2018-02-03 00:42:59 -08:00
AppHash cmn.HexBytes `json:"app_hash"`
2018-02-08 12:30:43 -08:00
AppState json.RawMessage `json:"app_state,omitempty"`
2015-11-01 11:34:08 -08:00
}
2017-06-26 14:20:27 -07:00
// SaveAs is a utility method for saving GenensisDoc as a JSON file.
2015-12-03 09:51:10 -08:00
func (genDoc *GenesisDoc) SaveAs(file string) error {
genDocBytes, err := cdc.MarshalJSON(genDoc)
if err != nil {
return err
}
2017-04-27 16:01:18 -07:00
return cmn.WriteFile(file, genDocBytes, 0644)
2015-12-03 09:51:10 -08:00
}
2017-06-26 14:20:27 -07:00
// ValidatorHash returns the hash of the validator set contained in the GenesisDoc
func (genDoc *GenesisDoc) ValidatorHash() []byte {
vals := make([]*Validator, len(genDoc.Validators))
for i, v := range genDoc.Validators {
2017-09-21 11:37:34 -07:00
vals[i] = NewValidator(v.PubKey, v.Power)
}
vset := NewValidatorSet(vals)
return vset.Hash()
}
2017-09-11 13:48:41 -07:00
// ValidateAndComplete checks that all necessary fields are present
// and fills in defaults for optional fields left empty
func (genDoc *GenesisDoc) ValidateAndComplete() error {
if genDoc.ChainID == "" {
return cmn.NewError("Genesis doc must include non-empty chain_id")
}
2017-09-11 13:48:41 -07:00
if genDoc.ConsensusParams == nil {
genDoc.ConsensusParams = DefaultConsensusParams()
} else {
if err := genDoc.ConsensusParams.Validate(); err != nil {
2017-09-11 13:48:41 -07:00
return err
}
}
2017-09-11 13:48:41 -07:00
if len(genDoc.Validators) == 0 {
return cmn.NewError("The genesis file must have at least one validator")
2017-09-11 13:48:41 -07:00
}
2017-09-21 11:37:34 -07:00
for _, v := range genDoc.Validators {
if v.Power == 0 {
return cmn.NewError("The genesis file cannot contain validators with no voting power: %v", v)
2017-09-21 11:37:34 -07:00
}
}
2017-09-11 13:48:41 -07:00
if genDoc.GenesisTime.IsZero() {
genDoc.GenesisTime = time.Now()
}
return nil
}
//------------------------------------------------------------
// Make genesis state from file
// GenesisDocFromJSON unmarshalls JSON data into a GenesisDoc.
func GenesisDocFromJSON(jsonBlob []byte) (*GenesisDoc, error) {
genDoc := GenesisDoc{}
err := cdc.UnmarshalJSON(jsonBlob, &genDoc)
2017-09-11 13:48:41 -07:00
if err != nil {
return nil, err
}
if err := genDoc.ValidateAndComplete(); err != nil {
return nil, err
}
return &genDoc, err
2015-11-01 11:34:08 -08:00
}
// GenesisDocFromFile reads JSON data from a file and unmarshalls it into a GenesisDoc.
func GenesisDocFromFile(genDocFile string) (*GenesisDoc, error) {
jsonBlob, err := ioutil.ReadFile(genDocFile)
if err != nil {
return nil, cmn.ErrorWrap(err, "Couldn't read GenesisDoc file")
}
genDoc, err := GenesisDocFromJSON(jsonBlob)
if err != nil {
return nil, cmn.ErrorWrap(err, cmn.Fmt("Error reading GenesisDoc at %v", genDocFile))
}
return genDoc, nil
}