mirror of
https://github.com/ipfs/kubo.git
synced 2025-07-15 07:58:15 +08:00
426 lines
9.7 KiB
Go
426 lines
9.7 KiB
Go
package commands
|
|
|
|
import (
|
|
"bytes"
|
|
"encoding/json"
|
|
"errors"
|
|
"fmt"
|
|
"io"
|
|
"io/ioutil"
|
|
"os"
|
|
"os/exec"
|
|
"strings"
|
|
|
|
cmds "github.com/ipfs/go-ipfs/commands"
|
|
e "github.com/ipfs/go-ipfs/core/commands/e"
|
|
repo "github.com/ipfs/go-ipfs/repo"
|
|
config "github.com/ipfs/go-ipfs/repo/config"
|
|
fsrepo "github.com/ipfs/go-ipfs/repo/fsrepo"
|
|
|
|
"gx/ipfs/QmceUdzxkimdYsgtX733uNgzf1DLHyBKN6ehGSp85ayppM/go-ipfs-cmdkit"
|
|
)
|
|
|
|
type ConfigField struct {
|
|
Key string
|
|
Value interface{}
|
|
}
|
|
|
|
var ConfigCmd = &cmds.Command{
|
|
Helptext: cmdkit.HelpText{
|
|
Tagline: "Get and set ipfs config values.",
|
|
ShortDescription: `
|
|
'ipfs config' controls configuration variables. It works like 'git config'.
|
|
The configuration values are stored in a config file inside your ipfs
|
|
repository.`,
|
|
LongDescription: `
|
|
'ipfs config' controls configuration variables. It works
|
|
much like 'git config'. The configuration values are stored in a config
|
|
file inside your IPFS repository.
|
|
|
|
Examples:
|
|
|
|
Get the value of the 'Datastore.Path' key:
|
|
|
|
$ ipfs config Datastore.Path
|
|
|
|
Set the value of the 'Datastore.Path' key:
|
|
|
|
$ ipfs config Datastore.Path ~/.ipfs/datastore
|
|
`,
|
|
},
|
|
|
|
Arguments: []cmdkit.Argument{
|
|
cmdkit.StringArg("key", true, false, "The key of the config entry (e.g. \"Addresses.API\")."),
|
|
cmdkit.StringArg("value", false, false, "The value to set the config entry to."),
|
|
},
|
|
Options: []cmdkit.Option{
|
|
cmdkit.BoolOption("bool", "Set a boolean value."),
|
|
cmdkit.BoolOption("json", "Parse stringified JSON."),
|
|
},
|
|
Run: func(req cmds.Request, res cmds.Response) {
|
|
args := req.Arguments()
|
|
key := args[0]
|
|
|
|
var output *ConfigField
|
|
defer func() {
|
|
if output != nil {
|
|
res.SetOutput(output)
|
|
} else {
|
|
res.SetOutput(nil)
|
|
}
|
|
}()
|
|
|
|
// This is a temporary fix until we move the private key out of the config file
|
|
switch strings.ToLower(key) {
|
|
case "identity", "identity.privkey":
|
|
res.SetError(fmt.Errorf("cannot show or change private key through API"), cmdkit.ErrNormal)
|
|
return
|
|
default:
|
|
}
|
|
|
|
r, err := fsrepo.Open(req.InvocContext().ConfigRoot)
|
|
if err != nil {
|
|
res.SetError(err, cmdkit.ErrNormal)
|
|
return
|
|
}
|
|
defer r.Close()
|
|
if len(args) == 2 {
|
|
value := args[1]
|
|
|
|
if parseJson, _, _ := req.Option("json").Bool(); parseJson {
|
|
var jsonVal interface{}
|
|
if err := json.Unmarshal([]byte(value), &jsonVal); err != nil {
|
|
err = fmt.Errorf("failed to unmarshal json. %s", err)
|
|
res.SetError(err, cmdkit.ErrNormal)
|
|
return
|
|
}
|
|
|
|
output, err = setConfig(r, key, jsonVal)
|
|
} else if isbool, _, _ := req.Option("bool").Bool(); isbool {
|
|
output, err = setConfig(r, key, value == "true")
|
|
} else {
|
|
output, err = setConfig(r, key, value)
|
|
}
|
|
} else {
|
|
output, err = getConfig(r, key)
|
|
}
|
|
if err != nil {
|
|
res.SetError(err, cmdkit.ErrNormal)
|
|
return
|
|
}
|
|
},
|
|
Marshalers: cmds.MarshalerMap{
|
|
cmds.Text: func(res cmds.Response) (io.Reader, error) {
|
|
if len(res.Request().Arguments()) == 2 {
|
|
return nil, nil // dont output anything
|
|
}
|
|
|
|
if res.Error() != nil {
|
|
return nil, res.Error()
|
|
}
|
|
|
|
v, err := unwrapOutput(res.Output())
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
vf, ok := v.(*ConfigField)
|
|
if !ok {
|
|
return nil, e.TypeErr(vf, v)
|
|
}
|
|
|
|
buf, err := config.HumanOutput(vf.Value)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
buf = append(buf, byte('\n'))
|
|
return bytes.NewReader(buf), nil
|
|
},
|
|
},
|
|
Type: ConfigField{},
|
|
Subcommands: map[string]*cmds.Command{
|
|
"show": configShowCmd,
|
|
"edit": configEditCmd,
|
|
"replace": configReplaceCmd,
|
|
"profile": configProfileCmd,
|
|
},
|
|
}
|
|
|
|
var configShowCmd = &cmds.Command{
|
|
Helptext: cmdkit.HelpText{
|
|
Tagline: "Output config file contents.",
|
|
ShortDescription: `
|
|
NOTE: For security reasons, this command will omit your private key. If you would like to make a full backup of your config (private key included), you must copy the config file from your repo.
|
|
`,
|
|
},
|
|
|
|
Run: func(req cmds.Request, res cmds.Response) {
|
|
cfgPath := req.InvocContext().ConfigRoot
|
|
fname, err := config.Filename(cfgPath)
|
|
if err != nil {
|
|
res.SetError(err, cmdkit.ErrNormal)
|
|
return
|
|
}
|
|
|
|
data, err := ioutil.ReadFile(fname)
|
|
if err != nil {
|
|
res.SetError(err, cmdkit.ErrNormal)
|
|
return
|
|
}
|
|
|
|
var cfg map[string]interface{}
|
|
err = json.Unmarshal(data, &cfg)
|
|
if err != nil {
|
|
res.SetError(err, cmdkit.ErrNormal)
|
|
return
|
|
}
|
|
|
|
err = scrubValue(cfg, []string{config.IdentityTag, config.PrivKeyTag})
|
|
if err != nil {
|
|
res.SetError(err, cmdkit.ErrNormal)
|
|
return
|
|
}
|
|
|
|
output, err := config.HumanOutput(cfg)
|
|
if err != nil {
|
|
res.SetError(err, cmdkit.ErrNormal)
|
|
return
|
|
}
|
|
|
|
res.SetOutput(bytes.NewReader(output))
|
|
},
|
|
}
|
|
|
|
func scrubValue(m map[string]interface{}, key []string) error {
|
|
find := func(m map[string]interface{}, k string) (string, interface{}, bool) {
|
|
lckey := strings.ToLower(k)
|
|
for mkey, val := range m {
|
|
lcmkey := strings.ToLower(mkey)
|
|
if lckey == lcmkey {
|
|
return mkey, val, true
|
|
}
|
|
}
|
|
return "", nil, false
|
|
}
|
|
|
|
cur := m
|
|
for _, k := range key[:len(key)-1] {
|
|
foundk, val, ok := find(cur, k)
|
|
if !ok {
|
|
return fmt.Errorf("failed to find specified key")
|
|
}
|
|
|
|
if foundk != k {
|
|
// case mismatch, calling this an error
|
|
return fmt.Errorf("case mismatch in config, expected %q but got %q", k, foundk)
|
|
}
|
|
|
|
mval, mok := val.(map[string]interface{})
|
|
if !mok {
|
|
return fmt.Errorf("%s was not a map", foundk)
|
|
}
|
|
|
|
cur = mval
|
|
}
|
|
|
|
todel, _, ok := find(cur, key[len(key)-1])
|
|
if !ok {
|
|
return fmt.Errorf("%s, not found", strings.Join(key, "."))
|
|
}
|
|
|
|
delete(cur, todel)
|
|
return nil
|
|
}
|
|
|
|
var configEditCmd = &cmds.Command{
|
|
Helptext: cmdkit.HelpText{
|
|
Tagline: "Open the config file for editing in $EDITOR.",
|
|
ShortDescription: `
|
|
To use 'ipfs config edit', you must have the $EDITOR environment
|
|
variable set to your preferred text editor.
|
|
`,
|
|
},
|
|
|
|
Run: func(req cmds.Request, res cmds.Response) {
|
|
filename, err := config.Filename(req.InvocContext().ConfigRoot)
|
|
if err != nil {
|
|
res.SetError(err, cmdkit.ErrNormal)
|
|
return
|
|
}
|
|
|
|
err = editConfig(filename)
|
|
if err != nil {
|
|
res.SetError(err, cmdkit.ErrNormal)
|
|
}
|
|
},
|
|
}
|
|
|
|
var configReplaceCmd = &cmds.Command{
|
|
Helptext: cmdkit.HelpText{
|
|
Tagline: "Replace the config with <file>.",
|
|
ShortDescription: `
|
|
Make sure to back up the config file first if necessary, as this operation
|
|
can't be undone.
|
|
`,
|
|
},
|
|
|
|
Arguments: []cmdkit.Argument{
|
|
cmdkit.FileArg("file", true, false, "The file to use as the new config."),
|
|
},
|
|
Run: func(req cmds.Request, res cmds.Response) {
|
|
// has to be called
|
|
res.SetOutput(nil)
|
|
|
|
r, err := fsrepo.Open(req.InvocContext().ConfigRoot)
|
|
if err != nil {
|
|
res.SetError(err, cmdkit.ErrNormal)
|
|
return
|
|
}
|
|
defer r.Close()
|
|
|
|
file, err := req.Files().NextFile()
|
|
if err != nil {
|
|
res.SetError(err, cmdkit.ErrNormal)
|
|
return
|
|
}
|
|
defer file.Close()
|
|
|
|
err = replaceConfig(r, file)
|
|
if err != nil {
|
|
res.SetError(err, cmdkit.ErrNormal)
|
|
return
|
|
}
|
|
},
|
|
}
|
|
|
|
var configProfileCmd = &cmds.Command{
|
|
Helptext: cmdkit.HelpText{
|
|
Tagline: "Apply profiles to config.",
|
|
ShortDescription: fmt.Sprintf(`
|
|
Available profiles:
|
|
%s
|
|
`, buildProfileHelp()),
|
|
},
|
|
|
|
Subcommands: map[string]*cmds.Command{
|
|
"apply": configProfileApplyCmd,
|
|
},
|
|
}
|
|
|
|
var configProfileApplyCmd = &cmds.Command{
|
|
Helptext: cmdkit.HelpText{
|
|
Tagline: "Apply profile to config.",
|
|
},
|
|
Arguments: []cmdkit.Argument{
|
|
cmdkit.StringArg("profile", true, false, "The profile to apply to the config."),
|
|
},
|
|
Run: func(req cmds.Request, res cmds.Response) {
|
|
profile, ok := config.Profiles[req.Arguments()[0]]
|
|
if !ok {
|
|
res.SetError(fmt.Errorf("%s is not a profile", req.Arguments()[0]), cmdkit.ErrNormal)
|
|
return
|
|
}
|
|
|
|
err := transformConfig(req.InvocContext().ConfigRoot, req.Arguments()[0], profile.Transform)
|
|
if err != nil {
|
|
res.SetError(err, cmdkit.ErrNormal)
|
|
return
|
|
}
|
|
res.SetOutput(nil)
|
|
},
|
|
}
|
|
|
|
func buildProfileHelp() string {
|
|
var out string
|
|
|
|
for name, profile := range config.Profiles {
|
|
dlines := strings.Split(profile.Description, "\n")
|
|
for i := range dlines {
|
|
dlines[i] = " " + dlines[i]
|
|
}
|
|
|
|
out = out + fmt.Sprintf(" '%s':\n%s\n", name, strings.Join(dlines, "\n"))
|
|
}
|
|
|
|
return out
|
|
}
|
|
|
|
func transformConfig(configRoot string, configName string, transformer config.Transformer) error {
|
|
r, err := fsrepo.Open(configRoot)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
defer r.Close()
|
|
|
|
cfg, err := r.Config()
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
err = transformer(cfg)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
_, err = r.BackupConfig("pre-" + configName + "-")
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
return r.SetConfig(cfg)
|
|
}
|
|
|
|
func getConfig(r repo.Repo, key string) (*ConfigField, error) {
|
|
value, err := r.GetConfigKey(key)
|
|
if err != nil {
|
|
return nil, fmt.Errorf("failed to get config value: %q", err)
|
|
}
|
|
return &ConfigField{
|
|
Key: key,
|
|
Value: value,
|
|
}, nil
|
|
}
|
|
|
|
func setConfig(r repo.Repo, key string, value interface{}) (*ConfigField, error) {
|
|
err := r.SetConfigKey(key, value)
|
|
if err != nil {
|
|
return nil, fmt.Errorf("failed to set config value: %s (maybe use --json?)", err)
|
|
}
|
|
return getConfig(r, key)
|
|
}
|
|
|
|
func editConfig(filename string) error {
|
|
editor := os.Getenv("EDITOR")
|
|
if editor == "" {
|
|
return errors.New("ENV variable $EDITOR not set")
|
|
}
|
|
|
|
cmd := exec.Command("sh", "-c", editor+" "+filename)
|
|
cmd.Stdin, cmd.Stdout, cmd.Stderr = os.Stdin, os.Stdout, os.Stderr
|
|
return cmd.Run()
|
|
}
|
|
|
|
func replaceConfig(r repo.Repo, file io.Reader) error {
|
|
var cfg config.Config
|
|
if err := json.NewDecoder(file).Decode(&cfg); err != nil {
|
|
return errors.New("failed to decode file as config")
|
|
}
|
|
if len(cfg.Identity.PrivKey) != 0 {
|
|
return errors.New("setting private key with API is not supported")
|
|
}
|
|
|
|
keyF, err := getConfig(r, config.PrivKeySelector)
|
|
if err != nil {
|
|
return fmt.Errorf("failed to get PrivKey")
|
|
}
|
|
|
|
pkstr, ok := keyF.Value.(string)
|
|
if !ok {
|
|
return fmt.Errorf("private key in config was not a string")
|
|
}
|
|
|
|
cfg.Identity.PrivKey = pkstr
|
|
|
|
return r.SetConfig(&cfg)
|
|
}
|