-
Notifications
You must be signed in to change notification settings - Fork 3.7k
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
Add client config subcommand to CLI (#8953)
* add client config * addressed reviewers comments * refactored,ready for review * fixed linter issues and addressed reviewers comments * Bump golangci-lint * fix linter warnings * fix some tests Co-authored-by: Alessio Treglia <[email protected]> Co-authored-by: Amaury <[email protected]>
- Loading branch information
1 parent
413938c
commit 410d8ed
Showing
15 changed files
with
332 additions
and
21 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,96 @@ | ||
package config | ||
|
||
import ( | ||
"encoding/json" | ||
"fmt" | ||
"path/filepath" | ||
|
||
tmcli "github.com/tendermint/tendermint/libs/cli" | ||
|
||
"github.com/spf13/cobra" | ||
|
||
"github.com/cosmos/cosmos-sdk/client" | ||
"github.com/cosmos/cosmos-sdk/client/flags" | ||
) | ||
|
||
// Cmd returns a CLI command to interactively create an application CLI | ||
// config file. | ||
func Cmd() *cobra.Command { | ||
cmd := &cobra.Command{ | ||
Use: "config <key> [value]", | ||
Short: "Create or query an application CLI configuration file", | ||
RunE: runConfigCmd, | ||
Args: cobra.RangeArgs(0, 2), | ||
} | ||
return cmd | ||
} | ||
|
||
func runConfigCmd(cmd *cobra.Command, args []string) error { | ||
clientCtx := client.GetClientContextFromCmd(cmd) | ||
configPath := filepath.Join(clientCtx.HomeDir, "config") | ||
|
||
conf, err := getClientConfig(configPath, clientCtx.Viper) | ||
if err != nil { | ||
return fmt.Errorf("couldn't get client config: %v", err) | ||
} | ||
|
||
switch len(args) { | ||
case 0: | ||
// print all client config fields to sdt out | ||
s, _ := json.MarshalIndent(conf, "", "\t") | ||
cmd.Println(string(s)) | ||
|
||
case 1: | ||
// it's a get | ||
key := args[0] | ||
|
||
switch key { | ||
case flags.FlagChainID: | ||
cmd.Println(conf.ChainID) | ||
case flags.FlagKeyringBackend: | ||
cmd.Println(conf.KeyringBackend) | ||
case tmcli.OutputFlag: | ||
cmd.Println(conf.Output) | ||
case flags.FlagNode: | ||
cmd.Println(conf.Node) | ||
case flags.FlagBroadcastMode: | ||
cmd.Println(conf.BroadcastMode) | ||
default: | ||
err := errUnknownConfigKey(key) | ||
return fmt.Errorf("couldn't get the value for the key: %v, error: %v", key, err) | ||
} | ||
|
||
case 2: | ||
// it's set | ||
key, value := args[0], args[1] | ||
|
||
switch key { | ||
case flags.FlagChainID: | ||
conf.SetChainID(value) | ||
case flags.FlagKeyringBackend: | ||
conf.SetKeyringBackend(value) | ||
case tmcli.OutputFlag: | ||
conf.SetOutput(value) | ||
case flags.FlagNode: | ||
conf.SetNode(value) | ||
case flags.FlagBroadcastMode: | ||
conf.SetBroadcastMode(value) | ||
default: | ||
return errUnknownConfigKey(key) | ||
} | ||
|
||
confFile := filepath.Join(configPath, "client.toml") | ||
if err := writeConfigToFile(confFile, conf); err != nil { | ||
return fmt.Errorf("could not write client config to the file: %v", err) | ||
} | ||
|
||
default: | ||
panic("cound not execute config command") | ||
} | ||
|
||
return nil | ||
} | ||
|
||
func errUnknownConfigKey(key string) error { | ||
return fmt.Errorf("unknown configuration key: %q", key) | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,96 @@ | ||
package config | ||
|
||
import ( | ||
"fmt" | ||
"os" | ||
"path/filepath" | ||
|
||
"github.com/cosmos/cosmos-sdk/client" | ||
) | ||
|
||
// Default constants | ||
const ( | ||
chainID = "" | ||
keyringBackend = "os" | ||
output = "text" | ||
node = "tcp://localhost:26657" | ||
broadcastMode = "sync" | ||
) | ||
|
||
type ClientConfig struct { | ||
ChainID string `mapstructure:"chain-id" json:"chain-id"` | ||
KeyringBackend string `mapstructure:"keyring-backend" json:"keyring-backend"` | ||
Output string `mapstructure:"output" json:"output"` | ||
Node string `mapstructure:"node" json:"node"` | ||
BroadcastMode string `mapstructure:"broadcast-mode" json:"broadcast-mode"` | ||
} | ||
|
||
// defaultClientConfig returns the reference to ClientConfig with default values. | ||
func defaultClientConfig() *ClientConfig { | ||
return &ClientConfig{chainID, keyringBackend, output, node, broadcastMode} | ||
} | ||
|
||
func (c *ClientConfig) SetChainID(chainID string) { | ||
c.ChainID = chainID | ||
} | ||
|
||
func (c *ClientConfig) SetKeyringBackend(keyringBackend string) { | ||
c.KeyringBackend = keyringBackend | ||
} | ||
|
||
func (c *ClientConfig) SetOutput(output string) { | ||
c.Output = output | ||
} | ||
|
||
func (c *ClientConfig) SetNode(node string) { | ||
c.Node = node | ||
} | ||
|
||
func (c *ClientConfig) SetBroadcastMode(broadcastMode string) { | ||
c.BroadcastMode = broadcastMode | ||
} | ||
|
||
// ReadFromClientConfig reads values from client.toml file and updates them in client Context | ||
func ReadFromClientConfig(ctx client.Context) (client.Context, error) { | ||
configPath := filepath.Join(ctx.HomeDir, "config") | ||
configFilePath := filepath.Join(configPath, "client.toml") | ||
conf := defaultClientConfig() | ||
|
||
// if config.toml file does not exist we create it and write default ClientConfig values into it. | ||
if _, err := os.Stat(configFilePath); os.IsNotExist(err) { | ||
if err := ensureConfigPath(configPath); err != nil { | ||
return ctx, fmt.Errorf("couldn't make client config: %v", err) | ||
} | ||
|
||
if err := writeConfigToFile(configFilePath, conf); err != nil { | ||
return ctx, fmt.Errorf("could not write client config to the file: %v", err) | ||
} | ||
} | ||
|
||
conf, err := getClientConfig(configPath, ctx.Viper) | ||
if err != nil { | ||
return ctx, fmt.Errorf("couldn't get client config: %v", err) | ||
} | ||
// we need to update KeyringDir field on Client Context first cause it is used in NewKeyringFromBackend | ||
ctx = ctx.WithOutputFormat(conf.Output). | ||
WithChainID(conf.ChainID) | ||
|
||
keyring, err := client.NewKeyringFromBackend(ctx, conf.KeyringBackend) | ||
if err != nil { | ||
return ctx, fmt.Errorf("couldn't get key ring: %v", err) | ||
} | ||
|
||
ctx = ctx.WithKeyring(keyring) | ||
|
||
// https://github.com/cosmos/cosmos-sdk/issues/8986 | ||
client, err := client.NewClientFromNode(conf.Node) | ||
if err != nil { | ||
return ctx, fmt.Errorf("couldn't get client from nodeURI: %v", err) | ||
} | ||
|
||
ctx = ctx.WithNodeURI(conf.Node). | ||
WithClient(client). | ||
WithBroadcastMode(conf.BroadcastMode) | ||
|
||
return ctx, nil | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,70 @@ | ||
package config | ||
|
||
import ( | ||
"bytes" | ||
"io/ioutil" | ||
"os" | ||
"text/template" | ||
|
||
"github.com/spf13/viper" | ||
) | ||
|
||
const defaultConfigTemplate = `# This is a TOML config file. | ||
# For more information, see https://github.com/toml-lang/toml | ||
############################################################################### | ||
### Client Configuration ### | ||
############################################################################### | ||
# The network chain ID | ||
chain-id = "{{ .ChainID }}" | ||
# The keyring's backend, where the keys are stored (os|file|kwallet|pass|test|memory) | ||
keyring-backend = "{{ .KeyringBackend }}" | ||
# CLI output format (text|json) | ||
output = "{{ .Output }}" | ||
# <host>:<port> to Tendermint RPC interface for this chain | ||
node = "{{ .Node }}" | ||
# Transaction broadcasting mode (sync|async|block) | ||
broadcast-mode = "{{ .BroadcastMode }}" | ||
` | ||
|
||
// writeConfigToFile parses defaultConfigTemplate, renders config using the template and writes it to | ||
// configFilePath. | ||
func writeConfigToFile(configFilePath string, config *ClientConfig) error { | ||
var buffer bytes.Buffer | ||
|
||
tmpl := template.New("clientConfigFileTemplate") | ||
configTemplate, err := tmpl.Parse(defaultConfigTemplate) | ||
if err != nil { | ||
return err | ||
} | ||
|
||
if err := configTemplate.Execute(&buffer, config); err != nil { | ||
return err | ||
} | ||
|
||
return ioutil.WriteFile(configFilePath, buffer.Bytes(), 0600) | ||
} | ||
|
||
// ensureConfigPath creates a directory configPath if it does not exist | ||
func ensureConfigPath(configPath string) error { | ||
return os.MkdirAll(configPath, os.ModePerm) | ||
} | ||
|
||
// getClientConfig reads values from client.toml file and unmarshalls them into ClientConfig | ||
func getClientConfig(configPath string, v *viper.Viper) (*ClientConfig, error) { | ||
v.AddConfigPath(configPath) | ||
v.SetConfigName("client") | ||
v.SetConfigType("toml") | ||
|
||
if err := v.ReadInConfig(); err != nil { | ||
return nil, err | ||
} | ||
|
||
conf := new(ClientConfig) | ||
if err := v.Unmarshal(conf); err != nil { | ||
return nil, err | ||
} | ||
|
||
return conf, nil | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Oops, something went wrong.