cosmos-sdk/server/util.go

228 lines
5.8 KiB
Go
Raw Normal View History

package server
import (
2018-04-21 19:26:46 -07:00
"encoding/json"
2018-04-23 12:15:50 -07:00
"net"
"os"
"path/filepath"
2018-04-23 12:15:50 -07:00
"github.com/pkg/errors"
"github.com/spf13/cobra"
"github.com/spf13/viper"
"github.com/cosmos/cosmos-sdk/client"
"github.com/cosmos/cosmos-sdk/codec"
"github.com/cosmos/cosmos-sdk/server/config"
"github.com/cosmos/cosmos-sdk/version"
tcmd "github.com/tendermint/tendermint/cmd/tendermint/commands"
cfg "github.com/tendermint/tendermint/config"
"github.com/tendermint/tendermint/libs/cli"
tmflags "github.com/tendermint/tendermint/libs/cli/flags"
"github.com/tendermint/tendermint/libs/log"
)
2018-04-18 21:49:24 -07:00
// server context
2018-04-05 03:31:33 -07:00
type Context struct {
Config *cfg.Config
Logger log.Logger
}
2018-04-05 04:16:20 -07:00
func NewDefaultContext() *Context {
return NewContext(
cfg.DefaultConfig(),
log.NewTMLogger(log.NewSyncWriter(os.Stdout)),
)
}
2018-04-05 03:31:33 -07:00
func NewContext(config *cfg.Config, logger log.Logger) *Context {
return &Context{config, logger}
}
2018-04-21 19:26:46 -07:00
//___________________________________________________________________________________
2018-04-05 03:31:33 -07:00
// PersistentPreRunEFn returns a PersistentPreRunE function for cobra
// that initailizes the passed in context with a properly configured
// logger and config objecy
func PersistentPreRunEFn(context *Context) func(*cobra.Command, []string) error {
return func(cmd *cobra.Command, args []string) error {
if cmd.Name() == version.VersionCmd.Name() {
return nil
}
config, err := interceptLoadConfig()
if err != nil {
return err
}
err = validateConfig(config)
if err != nil {
return err
}
logger := log.NewTMLogger(log.NewSyncWriter(os.Stdout))
logger, err = tmflags.ParseLogLevel(config.LogLevel, logger, cfg.DefaultLogLevel())
if err != nil {
return err
}
if viper.GetBool(cli.TraceFlag) {
logger = log.NewTracingLogger(logger)
}
logger = logger.With("module", "main")
context.Config = config
context.Logger = logger
return nil
}
}
// If a new config is created, change some of the default tendermint settings
func interceptLoadConfig() (conf *cfg.Config, err error) {
tmpConf := cfg.DefaultConfig()
err = viper.Unmarshal(tmpConf)
if err != nil {
// TODO: Handle with #870
panic(err)
}
rootDir := tmpConf.RootDir
configFilePath := filepath.Join(rootDir, "config/config.toml")
// Intercept only if the file doesn't already exist
if _, err := os.Stat(configFilePath); os.IsNotExist(err) {
// the following parse config is needed to create directories
conf, _ = tcmd.ParseConfig()
conf.ProfListenAddress = "localhost:6060"
conf.P2P.RecvRate = 5120000
conf.P2P.SendRate = 5120000
conf.TxIndex.IndexAllTags = true
conf.Consensus.TimeoutCommit = 5000
cfg.WriteConfigFile(configFilePath, conf)
// Fall through, just so that its parsed into memory.
}
if conf == nil {
conf, err = tcmd.ParseConfig()
}
cosmosConfigFilePath := filepath.Join(rootDir, "config/gaiad.toml")
viper.SetConfigName("cosmos")
_ = viper.MergeInConfig()
var cosmosConf *config.Config
if _, err := os.Stat(cosmosConfigFilePath); os.IsNotExist(err) {
cosmosConf, _ := config.ParseConfig()
config.WriteConfigFile(cosmosConfigFilePath, cosmosConf)
}
if cosmosConf == nil {
_, err = config.ParseConfig()
}
return
}
2018-04-05 03:24:53 -07:00
// validate the config with the sdk's requirements.
func validateConfig(conf *cfg.Config) error {
if conf.Consensus.CreateEmptyBlocks == false {
return errors.New("config option CreateEmptyBlocks = false is currently unsupported")
}
return nil
}
2018-04-18 21:49:24 -07:00
// add server commands
2018-04-05 03:24:53 -07:00
func AddCommands(
ctx *Context, cdc *codec.Codec,
2018-04-23 17:05:58 -07:00
rootCmd *cobra.Command, appInit AppInit,
2018-04-27 17:36:11 -07:00
appCreator AppCreator, appExport AppExporter) {
2018-04-05 03:24:53 -07:00
2018-04-21 19:26:46 -07:00
rootCmd.PersistentFlags().String("log_level", ctx.Config.LogLevel, "Log level")
2018-04-05 04:16:20 -07:00
2018-05-31 18:38:15 -07:00
tendermintCmd := &cobra.Command{
Use: "tendermint",
Short: "Tendermint subcommands",
}
2018-05-31 18:38:15 -07:00
tendermintCmd.AddCommand(
2018-04-21 19:26:46 -07:00
ShowNodeIDCmd(ctx),
ShowValidatorCmd(ctx),
ShowAddressCmd(ctx),
)
rootCmd.AddCommand(
InitCmd(ctx, cdc, appInit),
TestnetFilesCmd(ctx, cdc, appInit),
StartCmd(ctx, appCreator),
2018-06-01 01:35:39 -07:00
UnsafeResetAllCmd(ctx),
client.LineBreak,
2018-05-31 18:38:15 -07:00
tendermintCmd,
2018-04-27 17:36:11 -07:00
ExportCmd(ctx, cdc, appExport),
client.LineBreak,
2018-04-05 03:24:53 -07:00
version.VersionCmd,
)
}
2018-04-21 19:26:46 -07:00
//___________________________________________________________________________________
// InsertKeyJSON inserts a new JSON field/key with a given value to an existing
// JSON message. An error is returned if any serialization operation fails.
//
// NOTE: The ordering of the keys returned as the resulting JSON message is
// non-deterministic, so the client should not rely on key ordering.
func InsertKeyJSON(cdc *codec.Codec, baseJSON []byte, key string, value json.RawMessage) ([]byte, error) {
2018-04-21 19:26:46 -07:00
var jsonMap map[string]json.RawMessage
if err := cdc.UnmarshalJSON(baseJSON, &jsonMap); err != nil {
2018-04-21 19:26:46 -07:00
return nil, err
}
2018-04-21 19:26:46 -07:00
jsonMap[key] = value
bz, err := codec.MarshalJSONIndent(cdc, jsonMap)
2018-04-21 19:26:46 -07:00
return json.RawMessage(bz), err
}
2018-04-23 12:15:50 -07:00
// https://stackoverflow.com/questions/23558425/how-do-i-get-the-local-ip-address-in-go
// TODO there must be a better way to get external IP
func externalIP() (string, error) {
ifaces, err := net.Interfaces()
if err != nil {
return "", err
}
for _, iface := range ifaces {
if skipInterface(iface) {
continue
2018-04-23 12:15:50 -07:00
}
addrs, err := iface.Addrs()
if err != nil {
return "", err
}
for _, addr := range addrs {
ip := addrToIP(addr)
2018-04-23 12:15:50 -07:00
if ip == nil || ip.IsLoopback() {
continue
}
ip = ip.To4()
if ip == nil {
continue // not an ipv4 address
}
return ip.String(), nil
}
}
return "", errors.New("are you connected to the network?")
}
func skipInterface(iface net.Interface) bool {
if iface.Flags&net.FlagUp == 0 {
return true // interface down
}
if iface.Flags&net.FlagLoopback != 0 {
return true // loopback interface
}
return false
}
func addrToIP(addr net.Addr) net.IP {
var ip net.IP
switch v := addr.(type) {
case *net.IPNet:
ip = v.IP
case *net.IPAddr:
ip = v.IP
}
return ip
}