在 Go 中解组 Yaml 时,在列表中的键上设置默认值的方法是什么?

Vad*_*rod 5 yaml default list go

我有一个像下面这样的 YAML。

connections:
  - name: demo
    hosts:
      - demo.example.com:9200
    username: admin
    password: password
    ssl: true
    ssl_verify: true
    version: 7
  - name: test
    hosts:
      - "test.example.com:9200"
    username: admin
    password: password
Run Code Online (Sandbox Code Playgroud)

如您所见sslssl_verify并且未在列表的第二项中设置。我希望它们默认为true,但是,它没有发生。我尝试了不同的解决方案。

  1. Viper 默认值 - 不起作用。
viper.SetDefault("connections[].ssl", "true")
Run Code Online (Sandbox Code Playgroud)
  1. https://github.com/creasty/defaults - 不起作用。
type Config struct {
    Connections []struct {
        Name      string
        Hosts     []string
        Username  string
        Password  string
        Ssl       bool `default:"true"`
        SslVerify bool `default:"true"`
        Version   int
    }
}

...

err := defaults.Set(config)

Run Code Online (Sandbox Code Playgroud)
  1. 手动循环遍历结构列表。虽然这些方法适用于字符串,但它不适用于布尔值,因为它们false在解组后已经初始化,我们不确定是否false由用户输入。

  2. 使用带有布尔值的指针。这是因为未初始化的值等于nil并且它们很容易被捕获。但是,在使用 config struct 时需要取消引用指针,这不是很方便。或者,可以基于来自解组的结构生成新的结构。

type Config struct {
    Connections []struct {
        Name      string
        Hosts     []string
        Username  string
        Password  string
        Ssl       *bool
        SslVerify *bool
        Version   int
    }
}
Run Code Online (Sandbox Code Playgroud)
  1. 使用 hashmap 而不是 struct。这是有效的,因为空值没有被初始化,但是,它需要在访问它们之前对映射元素运行检查或将映射转换为结构。

解决方案 4 和 5 可能会奏效,但我想知道是否有比这更好的方法。

有任何想法吗?

Vad*_*rod 10

我想到了。github.com/creasty/defaults我在回调中使用UnmarshalYAML


type Config struct {
    Connections []Connection
}

type Connection struct {
    Name      string
    Hosts     []string
    Username  string
    Password  string
    Ssl       bool `default:"true"`
    SslVerify bool `default:"true" yaml:"ssl_verify"`
    Version   int  `version:"7"`
}

func (s *Connection) UnmarshalYAML(unmarshal func(interface{}) error) error {
    defaults.Set(s)

    type plain Connection
    if err := unmarshal((*plain)(s)); err != nil {
        return err
    }

    return nil
}
Run Code Online (Sandbox Code Playgroud)

此外,此解决方案与https://github.com/dealancer/validate配合良好,可用于验证目的。


Mar*_*erg 3

您可以使用一个函数来完成这项工作:

type Connection struct {
    Name      string
    Hosts     []string
    Username  string
    Password  string
    Ssl       *bool
    SslVerify bool
    Version   int
}

// If Ssl is nil, returns true
// otherwise the value
func (c Connection) IsSSL() bool {
    return c.Ssl == nil || *c.Ssl
}

type Config struct {
    Connections []Connection
}
Run Code Online (Sandbox Code Playgroud)

编辑

或者,更好的是,简单地反转布尔值的逻辑:

type Connection struct {
    Name          string
    Hosts         []string
    Username      string
    Password      string
    SkipSsl       bool
    SkipSslVerify bool
    Version       int
}
Run Code Online (Sandbox Code Playgroud)

这样,除非在配置中明确告知不同,否则将使用 SSL - 当有人读取配置时,这会很明显。