7

Cobra 和 Viper 中的文档让我感到困惑。我做了cobra init fooproject,然后在我做的项目目录中cobra add bar。我有一个PersistentFlag命名的foo,这是命令中的 init 函数root

func Execute() {
    if err := RootCmd.Execute(); err != nil {
        fmt.Println(err)
        os.Exit(-1)
    }

    fmt.Println(cfgFile)
    fmt.Println("fooString is: ", fooString)
}


func init() {
    cobra.OnInitialize(initConfig)

    // Here you will define your flags and configuration settings.
    // Cobra supports Persistent Flags, which, if defined here,
    // will be global for your application.

    RootCmd.PersistentFlags().StringVar(&cfgFile, "config", "", "config file (default is $HOME/.fooproject.yaml)")
    RootCmd.PersistentFlags().StringVar(&fooString, "foo", "", "loaded from config")

    viper.BindPFlag("foo", RootCmd.PersistentFlags().Lookup("foo"))
    // Cobra also supports local flags, which will only run
    // when this action is called directly.
    RootCmd.Flags().BoolP("toggle", "t", false, "Help message for toggle")
}

我的配置文件看起来像这样......

---
foo: aFooString

当我打电话时,go run main.go我看到了这个......

A longer description that spans multiple lines and likely contains
examples and usage of using your application. For example:

Cobra is a CLI library for Go that empowers applications.
This application is a tool to generate the needed files
to quickly create a Cobra application.

Usage:
  fooproject [command]

Available Commands:
  bar         A brief description of your command
  help        Help about any command

Flags:
      --config string   config file (default is $HOME/.fooproject.yaml)
      --foo string      loaded from config
  -h, --help            help for fooproject
  -t, --toggle          Help message for toggle

Use "fooproject [command] --help" for more information about a command.

fooString is:

当我打电话时,go run main.go bar我看到这个...

Using config file: my/gopath/github.com/user/fooproject/.fooproject.yaml
bar called

fooString is:

所以它正在使用配置文件,但他们似乎都没有阅读它。也许我误解了 Cobra 和 Viper 的工作方式。有任何想法吗?

4

4 回答 4

11

要结合spf13/cobraand spf13/viper,首先用 Cobra 定义标志:

RootCmd.PersistentFlags().StringP("foo", "", "loaded from config")

用毒蛇绑定它:

viper.BindPFlag("foo", RootCmd.PersistentFlags().Lookup("foo"))

并通过蝰蛇方法获取变量:

fmt.Println("fooString is: ", viper.GetString("foo"))
于 2017-05-08T14:48:03.940 回答
4

由于 Viper 值在某种程度上不如 pflags(例如,不支持自定义数据类型),我对“使用 Viper 检索值”的回答不满意,并最终编写了小的帮助类型来将值放回 pflags。

type viperPFlagBinding struct {
        configName string
        flagValue  pflag.Value
}

type viperPFlagHelper struct {
        bindings []viperPFlagBinding
}

func (vch *viperPFlagHelper) BindPFlag(configName string, flag *pflag.Flag) (err error) {
        err = viper.BindPFlag(configName, flag)
        if err == nil {
                vch.bindings = append(vch.bindings, viperPFlagBinding{configName, flag.Value})
        }
        return
}

func (vch *viperPFlagHelper) setPFlagsFromViper() {
        for _, v := range vch.bindings {
                v.flagValue.Set(viper.GetString(v.configName))
        }
}


func main() {
        var rootCmd = &cobra.Command{}
        var viperPFlagHelper viperPFlagHelper

        rootCmd.PersistentFlags().StringVar(&config.Password, "password", "", "API server password (remote HTTPS mode only)")
        viperPFlagHelper.BindPFlag("password", rootCmd.Flag("password"))

        rootCmd.PersistentPreRunE = func(cmd *cobra.Command, args []string) error {
                err := viper.ReadInConfig()
                if err != nil {
                        if _, ok := err.(viper.ConfigFileNotFoundError); !ok {
                                return err
                        }
                }

                viperPFlagHelper.setPFlagsFromViper()

                return nil
        }
}
于 2018-07-06T22:49:38.103 回答
4

对于那些面临同样问题的人,问题出在这个电话上:

cobra.OnInitialize(initConfig)

函数 initConfig 不直接执行,而是附加到初始化器数组https://github.com/spf13/cobra/blob/master/cobra.go#L80

因此,存储在配置文件中的值将在Run执行该字段时加载:

  rootCmd = &cobra.Command{
    Use:   "example",
    Short: "example cmd",
    Run: func(cmd *cobra.Command, args []string) { // OnInitialize is called first
      fmt.Println(viper.AllKeys())
    },  
  }
于 2020-03-24T21:45:59.767 回答
1

@WGH 的答案是正确的,您可以使用 rootCmd.PersistenPreRun 中的持久标志做一些事情,这将在所有其他子命令中可用。

var rootCmd = &cobra.Command{
    PersistentPreRun: func(_ *cobra.Command, _ []string) {
        if persistentflag {
            // do something with persistentflag
        }
    },
于 2021-07-27T03:25:11.583 回答