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/QmQp2a2Hhb7F6eK2A5hN8f9aJy4mtkEikL9Zj4cgB7d1dD/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: ` WARNING: Your private key is stored in the config file, and it will be included in the output of this command. `, }, 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 .", 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: cmds.HelpText{ Tagline: "Apply profiles to config.", }, Subcommands: map[string]*cmds.Command{ "apply": configProfileApplyCmd, "revert": configProfileRevertCmd, }, } var configProfileApplyCmd = &cmds.Command{ Helptext: cmds.HelpText{ Tagline: "Apply profile to config.", }, Arguments: []cmds.Argument{ cmds.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]), cmds.ErrNormal) return } err := transformConfig(req.InvocContext().ConfigRoot, profile.Apply) if err != nil { res.SetError(err, cmds.ErrNormal) return } }, } var configProfileRevertCmd = &cmds.Command{ Helptext: cmds.HelpText{ Tagline: "Revert profile changes.", ShortDescription: `Reverts profile-related changes to the config. Reverting some profiles may damage the configuration or not be possible. Backing up the config before running this command is advised.`, }, Arguments: []cmds.Argument{ cmds.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]), cmds.ErrNormal) return } err := transformConfig(req.InvocContext().ConfigRoot, profile.Unapply) if err != nil { res.SetError(err, cmds.ErrNormal) return } }, } func transformConfig(configRoot 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 } 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) }