Подтвердить что ты не робот

Json unmarshal time, которое отсутствует в формате RFC 3339

Каков подходящий способ обработки десериализации различных временных форматов в Go? Пакет encoding/json кажется полностью жестким только в RFC 3339. Я могу десериализоваться в строку, преобразовать ее в RFC 3339, а затем отключить ее, но я действительно не хочу этого делать. Любые лучшие решения?

4b9b3361

Ответ 1

Вам нужно будет выполнить json.Marshaler/json.Unmarshaler на пользовательский тип и вместо этого используйте пример:

type CustomTime struct {
    time.Time
}

const ctLayout = "2006/01/02|15:04:05"

func (ct *CustomTime) UnmarshalJSON(b []byte) (err error) {
    s := strings.Trim(string(b), "\"")
    if s == "null" {
       ct.Time = time.Time{}
       return
    }
    ct.Time, err = time.Parse(ctLayout, s)
    return
}

func (ct *CustomTime) MarshalJSON() ([]byte, error) {
  if ct.Time.UnixNano() == nilTime {
    return []byte("null"), nil
  }
  return []byte(fmt.Sprintf("\"%s\"", ct.Time.Format(ctLayout))), nil
}

var nilTime = (time.Time{}).UnixNano()
func (ct *CustomTime) IsSet() bool {
    return ct.UnixNano() != nilTime
}

type Args struct {
    Time CustomTime
}

var data = `
    {"Time": "2014/08/01|11:27:18"}
`

func main() {
    a := Args{}
    fmt.Println(json.Unmarshal([]byte(data), &a))
    fmt.Println(a.Time.String())
}

изменить: добавлен CustomTime.IsSet(), чтобы проверить, действительно ли он установлен или нет, для справки в будущем.

Ответ 2

Кодирование/декодирование выполняется time.Time в методах MarshalJSON и UnamrshalJSON. Вы можете создать свой собственный тип time.Time и переопределить эти функции для работы с json, но вы хотите.

type Time struct {
    time.Time
}

// returns time.Now() no matter what!
func (t *Time) UnmarshalJSON(b []byte) error {
    // you can now parse b as thoroughly as you want

    *t = Time{time.Now()}
    return nil
}

type Config struct {
    T Time
}

func main() {
    c := Config{}

    json.Unmarshal([]byte(`{"T": "bad-time"}`), &c)

    fmt.Printf("%+v\n", c)
}