小编典典

在Revel中解析JSON日期时间

go

我正在尝试解析json请求中的Datetime。该请求看起来像这样:

{
  "startedAt": "2017-06-01 10:39",
  ...
}

它被解码成的结构看起来像这样:

type MyStruct struct {
  StartedAt time.Time `json:"startedAt" bson:"startedAt"`
  ...
}

解码行如下所示:

json.NewDecoder(c.Request.Body).Decode(&MyStruct)

Revel返回此错误:

interface conversion: error is *time.ParseError, not *errors.Error

根据此处的revel文档,https://revel.github.io/manual/parameters.html

The SQL standard date time formats of 2006-01-02, 2006-01-02 15:04 are built in.

在同一文档中,他们还说您可以附加如下格式:

func init() {
    revel.TimeFormats = append(revel.TimeFormats, "01/02/2006")
}

为了验证格式是否在数组中,我尝试了以下方法:

func init() {
    revel.TimeFormats = append(revel.TimeFormats, "2016-06-01 12:12")
}

在绝望的最后一幕中,我尝试以revel将返回json时间的格式提交:

{
  "startedAt": "2017-06-22T12:22:16.265618819-05:00",
  ...
}

在这一点上,我不确定该去哪里。有谁能够陶醉解析日期时间?


更新:我在下面尝试了RickyA的解决方案,但是现在出现了解析错误。

parsing time ""Mon, 02 Jan 2006 15:04:05 -0700"" as "Mon, 02 Jan 2006 15:04:05 -0700": cannot parse ""Mon, 02 Jan 2006 15:04:05 -0700"" as "Mon"

甚至更奇怪的是,我实施了一些破解措施以使其在内部运行。我将请求时间字段更改为字符串,并为其提供了ToX函数,该函数将其转换为字符串。该函数有效,但是将其移入UnmarshalJSON函数后会失败。

另外,我无法确定这是否是错误:

func (t *AnyTime) UnmarshalJSON(b []byte) error {
  fmt.Println(time.RFC1123Z)
  fmt.Println(string(b))
  fmt.Println(reflect.TypeOf(time.RFC1123Z))
  fmt.Println(reflect.TypeOf(string(b)))
  ...

这输出:

Mon, 02 Jan 2006 15:04:05 -0700
"Mon, 02 Jan 2006 15:04:05 -0700"
string
string

请注意,字符串中只有1个带有双引号。这使我相信,传递给UnmarshalJSON的字节数组在其字符串中如何具有双引号。


最终更新:所以我很确定双引号是有意的,并且在某种意义上是可行的。UnmarshalJSON将从’:’到’,’传递所有内容,其中包括双引号。我猜测这样做是为了使其也可以支持整数和布尔值。我不喜欢该解决方案,但这是我解决的方法:

func (t *AnyTime) UnmarshalJSON(b []byte) error {
  var err error
  sTime := string(b)
  sTime = strings.TrimSuffix(sTime, "\"")
  sTime = strings.TrimPrefix(sTime, "\"")
  t.Time, err = time.Parse(time.RFC1123Z, sTime)
  ...

阅读 281

收藏
2020-07-02

共1个答案

小编典典

我遇到了同样的问题,最终为创建了一个自定义类型 time.Time

package genericfields

import (
    "encoding/json"
    "time"
    "strings"

    "gopkg.in/mgo.v2/bson"
)

// ANYTIME //

// AnyTime accepts any time format for its unmarshaling //
type AnyTime struct{ time.Time }

func (t AnyTime) MarshalJSON() ([]byte, error) {
    return json.Marshal(t.Time)
}

func (t *AnyTime) UnmarshalJSON(b []byte) error {
    err := json.Unmarshal(b, &t.Time)
    if err != nil { //assume non tz time input
        bstr := strings.Trim(string(b), `"`)
        t.Time, err = time.Parse("2006-01-02T15:04:05", bstr)
        if err != nil {
            return err //TODO add more formats to try
        }
    }
    return nil
}

func (t AnyTime) GetBSON() (interface{}, error) {
    return t.Time, nil
}

func (t *AnyTime) SetBSON(raw bson.Raw) error {
    var tm time.Time
    err := raw.Unmarshal(&tm)
    if err != nil {
        return err
    }
    t.Time = tm.UTC()
    return nil
}

func (t AnyTime) ToTime() time.Time {
    return t.Time
}

用法:

type MyStruct struct {
  StartedAt genericfields.AnyTime `json:"startedAt" bson:"startedAt"`
  ...
}

您可能需要对输入进行time.Parse一些调整。

2020-07-02