我个人建议您不要使用此选项。提供格式化数据通常是通过读取 STDIN 或从文件中完成的。这种解决方案通常更灵活,允许您添加标志来指定文件的格式(JSON、XML 等)。
在参数中提供文件名而不是原始 JSON 字符串可以增加与其他软件的更好互操作性,以及其他好处,例如使用计算机的磁盘而不是计算机的内存/RAM 来缓冲数据。
我个人的建议是:
- 对选项和配置使用标志,类似于 HTTP 的查询参数。
- 对数据使用标准输入/文件句柄,类似于 HTTP 的请求正文。
但是,如果您坚持使用标志:
Cobra 没有对 JSON 结构的内置支持。但是,pflagpflag.(*FlagSet).Var()
包(Cobra 使用的标志库)允许您通过该方法定义要用作标志的自定义值类型。您必须创建一个实现pflag.Value
接口的新类型:
type Value interface {
String() string
Set(string) error
Type() string
}
要制作自定义 JSON 解析类型,您可以编写以下代码以使用内置encoding/json
包:
import (
"encoding/json"
)
type JSONFlag struct {
Target interface{}
}
// String is used both by fmt.Print and by Cobra in help text
func (f *JSONFlag) String() string {
b, err := json.Marshal(f.Target)
if err != nil {
return "failed to marshal object"
}
return string(b)
}
// Set must have pointer receiver so it doesn't change the value of a copy
func (f *JSONFlag) Set(v string) error {
return json.Unmarshal([]byte(v), f.Target)
}
// Type is only used in help text
func (f *JSONFlag) Type() string {
return "json"
}
然后要使用这个新的pflag.Value
兼容类型,你可以这样写:
import (
"fmt"
"github.com/spf13/cobra"
)
type MyStruct struct {
StringArray []string
}
func init() {
var flagMyStringArray []string
var myCmd = &cobra.Command{
Use: "mycmd",
Short: "A brief description of your command",
Run: func(cmd *cobra.Command, args []string) {
myStruct := MyStruct{StringArray: flagMyStringArray}
fmt.Printf("myStruct.StringArray contains %d elements:\n", len(myStruct.StringArray))
for i, s := range myStruct.StringArray {
fmt.Printf("idx=%d: %q", i, s)
}
},
}
rootCmd.AddCommand(myCmd)
myCmd.Flags().Var(&JSONFlag{&flagMyStringArray}, "paramname", `this is the description`)
}
示例用法:
$ go run . mycmd --paramname 'hello'
Error: invalid argument "hello" for "--paramname" flag: invalid character 'h' looking for beginning of value
Usage:
test mycmd [flags]
Flags:
-h, --help help for mycmd
--paramname json this is the description
exit status 1
$ go run . mycmd --paramname '["(stringA, stringB)", "stringC"]'
myStruct.StringArray contains 2 elements:
idx=0: "(stringA, stringB)"
idx=1: "stringC"