在 go 中使用预构造的字符串作为 bson.M 进行 mgo 查询

Use a pre-constructed string as bson.M for mgo query in go

这里更大的目标是在 go 中设置更新查询,仅设置请求中发送给它的那些字段。

例如,我有一个文档要更新,我允许用户通过简单地在请求中指定它们来更新可变数量的字段,就像这样 -

{
"field1": valueOfField1,
"field2": valueOfField2,
"field3": valueOfField3,
...
}

这里的问题是,当我使用 json.Decode 将此输入解码为大多数字段可选的自定义结构类型时,输入中不存在的值将保留 nil

我的结构看起来像这样 -

type Fields struct {
    Field1       string      `bson:"field1" json:"field1,omitempty"`
    Field2       string      `bson:"field2" json:"field2"`
    Field3       time.Time   `bson:"field3,omitempty" json:"field3,omitempty"`
    Field4       bool        `bson:"field4,omitempty" json:"field4,omitempty"`
...
}

现在在我的更新查询中,我说,

bson.M{"$set": bson.M{"field1": body.Field1, "field2": body.Field2, "field3": body.Field3, "field4": body.Field4, ...}}

问题是,如果输入中不存在这些字段之一,它仍然会覆盖数据库中的现有值并使其为空。

为了避免这种情况,我希望这个 {"field1": body.Field1, "field2": body.Field2, "field3": body.Field3, "field4": body.Field4, ...} 部分是根据传入的字段动态构建的。

为此,我对输入做了 json.Marshal,就像这样 -

finalbody, err := json.Marshal(body)

然后我尝试在 $set 字段中使用它作为 -

bson.M{"$set": string(finalbody)}

当然,这给了我一个错误说 - "Modifiers operate on fields but we found a string instead"。 该字符串与 bson.M 完全一样,只是它不是 bson.M 即 {"field1": valueOfField1, "field2": valueOfField2, "field3": valueOfField1, ...}

我哪里错了? [...代表'and so on']

我通过将 finalbody 解组到映射[string]接口{}然后将其用作更新映射找到了解决方案。

所以,

finalbody, err := json.Marshal(body)
var finalbodymap map[string]interface{}
json.Unmarshal(finalbody, &finalbodymap)

当然,你需要添加一些错误处理所以最终的代码是这样的-

finalbody, err := json.Marshal(body)
    if err != nil {
        log.Println(err)
        return
    }
var finalbodymap map[string]interface{}
    if err = json.Unmarshal(finalbody, &finalbodymap); err != nil{
        log.Println(err)
    }

然后在更新查询中,我可以简单地写 -

bson.M{"$set": finalbodymap}

这里的一个问题是,在编组时,它会将任何 time.Time 类型值设置为 nil,即“0001-01-01T00:00:00Z”。我怀疑某些其他类型也可以观察到这种行为。