yaml:解组错误:无法将字符串解组到时间中

时间:2018-08-09 19:33:02

标签: go yaml unmarshalling

我的结构如下:

type Connect struct {
     ClientID string `yaml:"clientid"`
     Password string `yaml:"password"`
     Timeout  time.Duration `yaml:"timeout"`
}

c1 := `
    id: 'client1'
    password: 'hhhhhhha'
    timeout: 10
    `

c2 := `
    id: 'client2'
    password: 'llllllla'
    timeout: '10'
    `

c3 := `
    id: 'client3'
    password: 'hhhhhhha'
    timeout: 10s
    `

c4 := `
    id: 'client4'
    password: 'llllllla'
    timeout: '10s'
    `

如上所示,超时类型为time.Duration,默认单位为纳秒,但我想得到结果:c1 && c2有错误,c3 && c4有效( Timeout的配置必须具有单位)。我应该如何为Yaml重写UnmarshalYAML()方法?非常感谢。

3 个答案:

答案 0 :(得分:1)

如果您无法通过Timeout的{​​{1}}方法执行此操作,则一种方法是为实现UnmarshalYAML的自定义类型实现Unmarshaler接口:

Connect

答案 1 :(得分:1)

我将在UnmarshalYAML函数中创建一个别名类型,以便可以将所有值解编为某些原始类型。然后,我将重写那些匹配的值并转换那些不匹配的值:

package main

import (
    "fmt"
    "time"

    "gopkg.in/yaml.v2"
)

type Connect struct {
    ClientID string        `yaml:"clientid"`
    Password string        `yaml:"password"`
    Timeout  time.Duration `yaml:"timeout"`
}

func (ut *Connect) UnmarshalYAML(unmarshal func(interface{}) error) error {
    type alias struct {
        ClientID string `yaml:"clientid"`
        Password string `yaml:"password"`
        Timeout  string `yaml:"timeout"`
    }

    var tmp alias
    if err := unmarshal(&tmp); err != nil {
        return err
    }

    t, err := time.ParseDuration(tmp.Timeout)
    if err != nil {
        return fmt.Errorf("failed to parse '%s' to time.Duration: %v", tmp.Timeout, err)
    }

    ut.ClientID = tmp.ClientID
    ut.Password = tmp.Password
    ut.Timeout = t

    return nil
}

func main() {
    c1 := `
id: 'client1'
password: 'hhhhhhha'
timeout: 10
`

    c2 := `
id: 'client2'
password: 'llllllla'
timeout: '10'
`

    c3 := `
id: 'client3'
password: 'hhhhhhha'
timeout: 10s
`

    c4 := `
id: 'client4'
password: 'llllllla'
timeout: '10s'
`

    cc := []string{c1, c2, c3, c4}
    for i, cstr := range cc {
        var c Connect
        err := yaml.Unmarshal([]byte(cstr), &c)
        if err != nil {
            fmt.Printf("Error for c%d: %v\n", (i + 1), err)
            continue
        }
        fmt.Printf("c%d: %+v\n", (i + 1), c)
    }
}

输出如下:

$ go run main.go
Error for c1: failed to parse '10' to time.Duration: time: missing unit in duration10
Error for c2: failed to parse '10' to time.Duration: time: missing unit in duration10
c3: {ClientID: Password:hhhhhhha Timeout:10s}
c4: {ClientID: Password:llllllla Timeout:10s}

答案 2 :(得分:0)

type system struct {
    ...
    TokenLifeTime yamlTimeDur  `yaml:"TokenLifeTime"`
}

type yamlTimeDur time.Duration

func (t *yamlTimeDur) UnmarshalYAML(unmarshal func(interface{}) error) error {
    var tm string
    if err := unmarshal(&tm); err != nil {
        return err
    }

    td, err := time.ParseDuration(tm)
    if err != nil {
        return fmt.Errorf("failed to parse '%s' to time.Duration: %v", tm, err)
    }

    *t = yamlTimeDur(td)
    return nil
}

func (t *yamlTimeDur) Duration() time.Duration {
    return time.Duration(*t)
}