500 lines
13 KiB
Go
500 lines
13 KiB
Go
package server
|
|
|
|
import (
|
|
"encoding/json"
|
|
"fmt"
|
|
"io/ioutil"
|
|
"os"
|
|
"path"
|
|
"path/filepath"
|
|
"sort"
|
|
|
|
"github.com/pkg/errors"
|
|
"github.com/spf13/cobra"
|
|
"github.com/spf13/pflag"
|
|
"github.com/spf13/viper"
|
|
|
|
"github.com/cosmos/cosmos-sdk/crypto/keys"
|
|
"github.com/tendermint/tendermint/crypto"
|
|
|
|
cfg "github.com/tendermint/tendermint/config"
|
|
tmcli "github.com/tendermint/tendermint/libs/cli"
|
|
cmn "github.com/tendermint/tendermint/libs/common"
|
|
dbm "github.com/tendermint/tendermint/libs/db"
|
|
"github.com/tendermint/tendermint/p2p"
|
|
pvm "github.com/tendermint/tendermint/privval"
|
|
tmtypes "github.com/tendermint/tendermint/types"
|
|
|
|
clkeys "github.com/cosmos/cosmos-sdk/client/keys"
|
|
"github.com/cosmos/cosmos-sdk/codec"
|
|
serverconfig "github.com/cosmos/cosmos-sdk/server/config"
|
|
sdk "github.com/cosmos/cosmos-sdk/types"
|
|
)
|
|
|
|
//Parameter names, for init gen-tx command
|
|
var (
|
|
FlagName = "name"
|
|
FlagClientHome = "home-client"
|
|
FlagOWK = "owk"
|
|
)
|
|
|
|
//parameter names, init command
|
|
var (
|
|
FlagOverwrite = "overwrite"
|
|
FlagWithTxs = "with-txs"
|
|
FlagIP = "ip"
|
|
FlagChainID = "chain-id"
|
|
)
|
|
|
|
// genesis piece structure for creating combined genesis
|
|
type GenesisTx struct {
|
|
NodeID string `json:"node_id"`
|
|
IP string `json:"ip"`
|
|
Validator tmtypes.GenesisValidator `json:"validator"`
|
|
AppGenTx json.RawMessage `json:"app_gen_tx"`
|
|
}
|
|
|
|
// Storage for init command input parameters
|
|
type InitConfig struct {
|
|
ChainID string
|
|
GenTxs bool
|
|
GenTxsDir string
|
|
Overwrite bool
|
|
}
|
|
|
|
// get cmd to initialize all files for tendermint and application
|
|
func GenTxCmd(ctx *Context, cdc *codec.Codec, appInit AppInit) *cobra.Command {
|
|
cmd := &cobra.Command{
|
|
Use: "gen-tx",
|
|
Short: "Create genesis transaction file (under [--home]/config/gentx/gentx-[nodeID].json)",
|
|
Args: cobra.NoArgs,
|
|
RunE: func(_ *cobra.Command, args []string) error {
|
|
|
|
config := ctx.Config
|
|
config.SetRoot(viper.GetString(tmcli.HomeFlag))
|
|
|
|
ip := viper.GetString(FlagIP)
|
|
if len(ip) == 0 {
|
|
eip, err := externalIP()
|
|
if err != nil {
|
|
return err
|
|
}
|
|
ip = eip
|
|
}
|
|
|
|
genTxConfig := serverconfig.GenTx{
|
|
viper.GetString(FlagName),
|
|
viper.GetString(FlagClientHome),
|
|
viper.GetBool(FlagOWK),
|
|
ip,
|
|
}
|
|
cliPrint, genTxFile, err := gentxWithConfig(cdc, appInit, config, genTxConfig)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
toPrint := struct {
|
|
AppMessage json.RawMessage `json:"app_message"`
|
|
GenTxFile json.RawMessage `json:"gen_tx_file"`
|
|
}{
|
|
cliPrint,
|
|
genTxFile,
|
|
}
|
|
out, err := codec.MarshalJSONIndent(cdc, toPrint)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
fmt.Println(string(out))
|
|
return nil
|
|
},
|
|
}
|
|
cmd.Flags().String(FlagIP, "", "external facing IP to use if left blank IP will be retrieved from this machine")
|
|
cmd.Flags().AddFlagSet(appInit.FlagsAppGenTx)
|
|
return cmd
|
|
}
|
|
|
|
func gentxWithConfig(cdc *codec.Codec, appInit AppInit, config *cfg.Config, genTxConfig serverconfig.GenTx) (
|
|
cliPrint json.RawMessage, genTxFile json.RawMessage, err error) {
|
|
nodeKey, err := p2p.LoadOrGenNodeKey(config.NodeKeyFile())
|
|
if err != nil {
|
|
return
|
|
}
|
|
nodeID := string(nodeKey.ID())
|
|
pubKey := readOrCreatePrivValidator(config)
|
|
|
|
appGenTx, cliPrint, validator, err := appInit.AppGenTx(cdc, pubKey, genTxConfig)
|
|
if err != nil {
|
|
return
|
|
}
|
|
|
|
tx := GenesisTx{
|
|
NodeID: nodeID,
|
|
IP: genTxConfig.IP,
|
|
Validator: validator,
|
|
AppGenTx: appGenTx,
|
|
}
|
|
bz, err := codec.MarshalJSONIndent(cdc, tx)
|
|
if err != nil {
|
|
return
|
|
}
|
|
genTxFile = json.RawMessage(bz)
|
|
name := fmt.Sprintf("gentx-%v.json", nodeID)
|
|
writePath := filepath.Join(config.RootDir, "config", "gentx")
|
|
file := filepath.Join(writePath, name)
|
|
err = cmn.EnsureDir(writePath, 0700)
|
|
if err != nil {
|
|
return
|
|
}
|
|
err = cmn.WriteFile(file, bz, 0644)
|
|
if err != nil {
|
|
return
|
|
}
|
|
|
|
// Write updated config with moniker
|
|
config.Moniker = genTxConfig.Name
|
|
configFilePath := filepath.Join(config.RootDir, "config", "config.toml")
|
|
cfg.WriteConfigFile(configFilePath, config)
|
|
|
|
return
|
|
}
|
|
|
|
// get cmd to initialize all files for tendermint and application
|
|
func InitCmd(ctx *Context, cdc *codec.Codec, appInit AppInit) *cobra.Command {
|
|
cmd := &cobra.Command{
|
|
Use: "init",
|
|
Short: "Initialize genesis config, priv-validator file, and p2p-node file",
|
|
Args: cobra.NoArgs,
|
|
RunE: func(_ *cobra.Command, _ []string) error {
|
|
|
|
config := ctx.Config
|
|
config.SetRoot(viper.GetString(tmcli.HomeFlag))
|
|
initConfig := InitConfig{
|
|
viper.GetString(FlagChainID),
|
|
viper.GetBool(FlagWithTxs),
|
|
filepath.Join(config.RootDir, "config", "gentx"),
|
|
viper.GetBool(FlagOverwrite),
|
|
}
|
|
|
|
chainID, nodeID, appMessage, err := initWithConfig(cdc, appInit, config, initConfig)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
// print out some key information
|
|
toPrint := struct {
|
|
ChainID string `json:"chain_id"`
|
|
NodeID string `json:"node_id"`
|
|
AppMessage json.RawMessage `json:"app_message"`
|
|
}{
|
|
chainID,
|
|
nodeID,
|
|
appMessage,
|
|
}
|
|
out, err := codec.MarshalJSONIndent(cdc, toPrint)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
fmt.Println(string(out))
|
|
return nil
|
|
},
|
|
}
|
|
cmd.Flags().BoolP(FlagOverwrite, "o", false, "overwrite the genesis.json file")
|
|
cmd.Flags().String(FlagChainID, "", "genesis file chain-id, if left blank will be randomly created")
|
|
cmd.Flags().Bool(FlagWithTxs, false, "apply existing genesis transactions from [--home]/config/gentx/")
|
|
cmd.Flags().AddFlagSet(appInit.FlagsAppGenState)
|
|
cmd.Flags().AddFlagSet(appInit.FlagsAppGenTx) // need to add this flagset for when no GenTx's provided
|
|
cmd.AddCommand(GenTxCmd(ctx, cdc, appInit))
|
|
return cmd
|
|
}
|
|
|
|
func initWithConfig(cdc *codec.Codec, appInit AppInit, config *cfg.Config, initConfig InitConfig) (
|
|
chainID string, nodeID string, appMessage json.RawMessage, err error) {
|
|
nodeKey, err := p2p.LoadOrGenNodeKey(config.NodeKeyFile())
|
|
if err != nil {
|
|
return
|
|
}
|
|
nodeID = string(nodeKey.ID())
|
|
pubKey := readOrCreatePrivValidator(config)
|
|
|
|
if initConfig.ChainID == "" {
|
|
initConfig.ChainID = fmt.Sprintf("test-chain-%v", cmn.RandStr(6))
|
|
}
|
|
chainID = initConfig.ChainID
|
|
|
|
genFile := config.GenesisFile()
|
|
if !initConfig.Overwrite && cmn.FileExists(genFile) {
|
|
err = fmt.Errorf("genesis.json file already exists: %v", genFile)
|
|
return
|
|
}
|
|
|
|
// process genesis transactions, or otherwise create one for defaults
|
|
var appGenTxs []json.RawMessage
|
|
var validators []tmtypes.GenesisValidator
|
|
var persistentPeers string
|
|
|
|
if initConfig.GenTxs {
|
|
validators, appGenTxs, persistentPeers, err = processGenTxs(initConfig.GenTxsDir, cdc)
|
|
if err != nil {
|
|
return
|
|
}
|
|
config.P2P.PersistentPeers = persistentPeers
|
|
configFilePath := filepath.Join(config.RootDir, "config", "config.toml")
|
|
cfg.WriteConfigFile(configFilePath, config)
|
|
} else {
|
|
genTxConfig := serverconfig.GenTx{
|
|
viper.GetString(FlagName),
|
|
viper.GetString(FlagClientHome),
|
|
viper.GetBool(FlagOWK),
|
|
"127.0.0.1",
|
|
}
|
|
|
|
// Write updated config with moniker
|
|
config.Moniker = genTxConfig.Name
|
|
configFilePath := filepath.Join(config.RootDir, "config", "config.toml")
|
|
cfg.WriteConfigFile(configFilePath, config)
|
|
appGenTx, am, validator, err := appInit.AppGenTx(cdc, pubKey, genTxConfig)
|
|
appMessage = am
|
|
if err != nil {
|
|
return "", "", nil, err
|
|
}
|
|
validators = []tmtypes.GenesisValidator{validator}
|
|
appGenTxs = []json.RawMessage{appGenTx}
|
|
}
|
|
|
|
appState, err := appInit.AppGenState(cdc, appGenTxs)
|
|
if err != nil {
|
|
return
|
|
}
|
|
|
|
err = writeGenesisFile(cdc, genFile, initConfig.ChainID, validators, appState)
|
|
if err != nil {
|
|
return
|
|
}
|
|
|
|
return
|
|
}
|
|
|
|
// append a genesis-piece
|
|
func processGenTxs(genTxsDir string, cdc *codec.Codec) (
|
|
validators []tmtypes.GenesisValidator, appGenTxs []json.RawMessage, persistentPeers string, err error) {
|
|
|
|
var fos []os.FileInfo
|
|
fos, err = ioutil.ReadDir(genTxsDir)
|
|
if err != nil {
|
|
return
|
|
}
|
|
|
|
genTxs := make(map[string]GenesisTx)
|
|
var nodeIDs []string
|
|
for _, fo := range fos {
|
|
filename := path.Join(genTxsDir, fo.Name())
|
|
if !fo.IsDir() && (path.Ext(filename) != ".json") {
|
|
continue
|
|
}
|
|
|
|
// get the genTx
|
|
var bz []byte
|
|
bz, err = ioutil.ReadFile(filename)
|
|
if err != nil {
|
|
return
|
|
}
|
|
var genTx GenesisTx
|
|
err = cdc.UnmarshalJSON(bz, &genTx)
|
|
if err != nil {
|
|
return
|
|
}
|
|
|
|
genTxs[genTx.NodeID] = genTx
|
|
nodeIDs = append(nodeIDs, genTx.NodeID)
|
|
}
|
|
|
|
sort.Strings(nodeIDs)
|
|
|
|
for _, nodeID := range nodeIDs {
|
|
genTx := genTxs[nodeID]
|
|
|
|
// combine some stuff
|
|
validators = append(validators, genTx.Validator)
|
|
appGenTxs = append(appGenTxs, genTx.AppGenTx)
|
|
|
|
// Add a persistent peer
|
|
comma := ","
|
|
if len(persistentPeers) == 0 {
|
|
comma = ""
|
|
}
|
|
persistentPeers += fmt.Sprintf("%s%s@%s:26656", comma, genTx.NodeID, genTx.IP)
|
|
}
|
|
|
|
return
|
|
}
|
|
|
|
//________________________________________________________________________________________
|
|
|
|
// read of create the private key file for this config
|
|
func readOrCreatePrivValidator(tmConfig *cfg.Config) crypto.PubKey {
|
|
// private validator
|
|
privValFile := tmConfig.PrivValidatorFile()
|
|
var privValidator *pvm.FilePV
|
|
if cmn.FileExists(privValFile) {
|
|
privValidator = pvm.LoadFilePV(privValFile)
|
|
} else {
|
|
privValidator = pvm.GenFilePV(privValFile)
|
|
privValidator.Save()
|
|
}
|
|
return privValidator.GetPubKey()
|
|
}
|
|
|
|
// writeGenesisFile creates and writes the genesis configuration to disk. An
|
|
// error is returned if building or writing the configuration to file fails.
|
|
// nolint: unparam
|
|
func writeGenesisFile(cdc *codec.Codec, genesisFile, chainID string, validators []tmtypes.GenesisValidator, appState json.RawMessage) error {
|
|
genDoc := tmtypes.GenesisDoc{
|
|
ChainID: chainID,
|
|
Validators: validators,
|
|
AppState: appState,
|
|
}
|
|
|
|
if err := genDoc.ValidateAndComplete(); err != nil {
|
|
return err
|
|
}
|
|
|
|
return genDoc.SaveAs(genesisFile)
|
|
}
|
|
|
|
//_____________________________________________________________________
|
|
|
|
// Core functionality passed from the application to the server init command
|
|
type AppInit struct {
|
|
|
|
// flags required for application init functions
|
|
FlagsAppGenState *pflag.FlagSet
|
|
FlagsAppGenTx *pflag.FlagSet
|
|
|
|
// create the application genesis tx
|
|
AppGenTx func(cdc *codec.Codec, pk crypto.PubKey, genTxConfig serverconfig.GenTx) (
|
|
appGenTx, cliPrint json.RawMessage, validator tmtypes.GenesisValidator, err error)
|
|
|
|
// AppGenState creates the core parameters initialization. It takes in a
|
|
// pubkey meant to represent the pubkey of the validator of this machine.
|
|
AppGenState func(cdc *codec.Codec, appGenTxs []json.RawMessage) (appState json.RawMessage, err error)
|
|
}
|
|
|
|
//_____________________________________________________________________
|
|
|
|
// simple default application init
|
|
var DefaultAppInit = AppInit{
|
|
AppGenTx: SimpleAppGenTx,
|
|
AppGenState: SimpleAppGenState,
|
|
}
|
|
|
|
// simple genesis tx
|
|
type SimpleGenTx struct {
|
|
Addr sdk.AccAddress `json:"addr"`
|
|
}
|
|
|
|
// Generate a genesis transaction
|
|
func SimpleAppGenTx(cdc *codec.Codec, pk crypto.PubKey, genTxConfig serverconfig.GenTx) (
|
|
appGenTx, cliPrint json.RawMessage, validator tmtypes.GenesisValidator, err error) {
|
|
|
|
var addr sdk.AccAddress
|
|
var secret string
|
|
addr, secret, err = GenerateCoinKey()
|
|
if err != nil {
|
|
return
|
|
}
|
|
|
|
var bz []byte
|
|
simpleGenTx := SimpleGenTx{addr}
|
|
bz, err = cdc.MarshalJSON(simpleGenTx)
|
|
if err != nil {
|
|
return
|
|
}
|
|
appGenTx = json.RawMessage(bz)
|
|
|
|
mm := map[string]string{"secret": secret}
|
|
bz, err = cdc.MarshalJSON(mm)
|
|
if err != nil {
|
|
return
|
|
}
|
|
cliPrint = json.RawMessage(bz)
|
|
|
|
validator = tmtypes.GenesisValidator{
|
|
PubKey: pk,
|
|
Power: 10,
|
|
}
|
|
return
|
|
}
|
|
|
|
// create the genesis app state
|
|
func SimpleAppGenState(cdc *codec.Codec, appGenTxs []json.RawMessage) (appState json.RawMessage, err error) {
|
|
|
|
if len(appGenTxs) != 1 {
|
|
err = errors.New("must provide a single genesis transaction")
|
|
return
|
|
}
|
|
|
|
var genTx SimpleGenTx
|
|
err = cdc.UnmarshalJSON(appGenTxs[0], &genTx)
|
|
if err != nil {
|
|
return
|
|
}
|
|
|
|
appState = json.RawMessage(fmt.Sprintf(`{
|
|
"accounts": [{
|
|
"address": "%s",
|
|
"coins": [
|
|
{
|
|
"denom": "mycoin",
|
|
"amount": "9007199254740992"
|
|
}
|
|
]
|
|
}]
|
|
}`, genTx.Addr))
|
|
return
|
|
}
|
|
|
|
//___________________________________________________________________________________________
|
|
|
|
// GenerateCoinKey returns the address of a public key, along with the secret
|
|
// phrase to recover the private key.
|
|
func GenerateCoinKey() (sdk.AccAddress, string, error) {
|
|
|
|
// construct an in-memory key store
|
|
keybase := keys.New(
|
|
dbm.NewMemDB(),
|
|
)
|
|
|
|
// generate a private key, with recovery phrase
|
|
info, secret, err := keybase.CreateMnemonic("name", keys.English, "pass", keys.Secp256k1)
|
|
if err != nil {
|
|
return sdk.AccAddress([]byte{}), "", err
|
|
}
|
|
addr := info.GetPubKey().Address()
|
|
return sdk.AccAddress(addr), secret, nil
|
|
}
|
|
|
|
// GenerateSaveCoinKey returns the address of a public key, along with the secret
|
|
// phrase to recover the private key.
|
|
func GenerateSaveCoinKey(clientRoot, keyName, keyPass string, overwrite bool) (sdk.AccAddress, string, error) {
|
|
|
|
// get the keystore from the client
|
|
keybase, err := clkeys.GetKeyBaseFromDir(clientRoot)
|
|
if err != nil {
|
|
return sdk.AccAddress([]byte{}), "", err
|
|
}
|
|
|
|
// ensure no overwrite
|
|
if !overwrite {
|
|
_, err := keybase.Get(keyName)
|
|
if err == nil {
|
|
return sdk.AccAddress([]byte{}), "", errors.New("key already exists, overwrite is disabled")
|
|
}
|
|
}
|
|
|
|
// generate a private key, with recovery phrase
|
|
info, secret, err := keybase.CreateMnemonic(keyName, keys.English, keyPass, keys.Secp256k1)
|
|
if err != nil {
|
|
return sdk.AccAddress([]byte{}), "", err
|
|
}
|
|
addr := info.GetPubKey().Address()
|
|
return sdk.AccAddress(addr), secret, nil
|
|
}
|