Golang 和 MongoDB:带过滤器的 DeleteMany
Golang and MongoDB: DeleteMany with filter
我尝试使用 Go 的官方 mongodb 驱动程序从 Go 应用程序读取、写入和删除数据 (go.mongodb.org/mongo-driver).
这是我要使用的结构:
Contact struct {
ID xid.ID `json:"contact_id" bson:"contact_id"`
SurName string `json:"surname" bson:"surname"`
PreName string `json:"prename" bson:"prename"`
}
// xid is https://github.com/rs/xid
我省略了添加到集合中的代码,因为它正在查找。
我可以使用以下代码(缩写)获取具有特定 contact_id 的联系人列表:
filter := bson.D{}
cursor, err := contactCollection.Find(nil, filter)
for cur.Next(context.TODO()) {
...
}
这有效,returns 文档。我考虑过对删除或匹配的 get 做同样的事情:
// delete - abbreviated
filter := bson.M{"contact_id": id}
result, _ := contactCollection.DeleteMany(nil, filter)
// result.DeletedCount is always 0, err is nil
if err != nil {
sendError(c, err) // helper function
return
}
c.JSON(200, gin.H{
"ok": true,
"message": fmt.Sprintf("deleted %d patients", result.DeletedCount),
}) // will be called, it is part of a webservice done with gin
// get complete
func Get(c *gin.Context) {
defer c.Done()
id := c.Param("id")
filter := bson.M{"contact_id": id}
cur, err := contactCollection.Find(nil, filter)
if err != nil {
sendError(c, err) // helper function
return
} // no error
contacts := make([]types.Contact, 0)
for cur.Next(context.TODO()) { // nothing returned
// create a value into which the single document can be decoded
var elem types.Contact
err := cur.Decode(&elem)
if err != nil {
sendError(c, err) // helper function
return
}
contacts = append(contacts, elem)
}
c.JSON(200, contacts)
}
为什么相同的过滤器对删除不起作用?
编辑: 插入代码如下:
_, _ = contactCollection.InsertOne(context.TODO(), Contact{
ID: "abcdefg",
SurName: "Demo",
PreName: "on Whosebug",
})
Contact.ID
是xid.ID
类型,是字节数组:
type ID [rawLen]byte
因此,您在使用 string
文字指定 ID
字段的值时提供的插入代码将是 compile-time 错误:
_, _ = contactCollection.InsertOne(context.TODO(), Contact{
ID: "abcdefg",
SurName: "Demo",
PreName: "on Whosebug",
})
稍后在您的评论中您澄清了上面的插入代码只是一个示例,而不是您实际执行的方式。在您的真实代码中,您从请求中解组联系人(或其 ID 字段)。
xid.ID
有自己的解组逻辑,这可能会以不同方式解释输入数据,并可能导致 ID 表示与您的输入不同的 string
值。 ID.UnmarshalJSON()
定义如何将 string
ID 转换为 xid.ID
:
func (id *ID) UnmarshalJSON(b []byte) error {
s := string(b)
if s == "null" {
*id = nilID
return nil
}
return id.UnmarshalText(b[1 : len(b)-1])
}
如您所见,第一个字节被截断,ID.UnmarshalText()
在其上做了更多 "magic"(如果您感兴趣,请查看源代码)。
All-in-all,为了避免这种"transformations"在您不知情的情况下在后台发生,请使用简单的string
类型作为您的ID,并在需要存储的地方自行进行必要的转换/ 发送您的 ID。
对于ID字段,你应该使用bson包提供的primitive.ObjectID
。
"go.mongodb.org/mongo-driver/bson/primitive"
ID primitive.ObjectID `json:"_id" bson:"_id"`
我尝试使用 Go 的官方 mongodb 驱动程序从 Go 应用程序读取、写入和删除数据 (go.mongodb.org/mongo-driver).
这是我要使用的结构:
Contact struct {
ID xid.ID `json:"contact_id" bson:"contact_id"`
SurName string `json:"surname" bson:"surname"`
PreName string `json:"prename" bson:"prename"`
}
// xid is https://github.com/rs/xid
我省略了添加到集合中的代码,因为它正在查找。
我可以使用以下代码(缩写)获取具有特定 contact_id 的联系人列表:
filter := bson.D{}
cursor, err := contactCollection.Find(nil, filter)
for cur.Next(context.TODO()) {
...
}
这有效,returns 文档。我考虑过对删除或匹配的 get 做同样的事情:
// delete - abbreviated
filter := bson.M{"contact_id": id}
result, _ := contactCollection.DeleteMany(nil, filter)
// result.DeletedCount is always 0, err is nil
if err != nil {
sendError(c, err) // helper function
return
}
c.JSON(200, gin.H{
"ok": true,
"message": fmt.Sprintf("deleted %d patients", result.DeletedCount),
}) // will be called, it is part of a webservice done with gin
// get complete
func Get(c *gin.Context) {
defer c.Done()
id := c.Param("id")
filter := bson.M{"contact_id": id}
cur, err := contactCollection.Find(nil, filter)
if err != nil {
sendError(c, err) // helper function
return
} // no error
contacts := make([]types.Contact, 0)
for cur.Next(context.TODO()) { // nothing returned
// create a value into which the single document can be decoded
var elem types.Contact
err := cur.Decode(&elem)
if err != nil {
sendError(c, err) // helper function
return
}
contacts = append(contacts, elem)
}
c.JSON(200, contacts)
}
为什么相同的过滤器对删除不起作用?
编辑: 插入代码如下:
_, _ = contactCollection.InsertOne(context.TODO(), Contact{
ID: "abcdefg",
SurName: "Demo",
PreName: "on Whosebug",
})
Contact.ID
是xid.ID
类型,是字节数组:
type ID [rawLen]byte
因此,您在使用 string
文字指定 ID
字段的值时提供的插入代码将是 compile-time 错误:
_, _ = contactCollection.InsertOne(context.TODO(), Contact{
ID: "abcdefg",
SurName: "Demo",
PreName: "on Whosebug",
})
稍后在您的评论中您澄清了上面的插入代码只是一个示例,而不是您实际执行的方式。在您的真实代码中,您从请求中解组联系人(或其 ID 字段)。
xid.ID
有自己的解组逻辑,这可能会以不同方式解释输入数据,并可能导致 ID 表示与您的输入不同的 string
值。 ID.UnmarshalJSON()
定义如何将 string
ID 转换为 xid.ID
:
func (id *ID) UnmarshalJSON(b []byte) error {
s := string(b)
if s == "null" {
*id = nilID
return nil
}
return id.UnmarshalText(b[1 : len(b)-1])
}
如您所见,第一个字节被截断,ID.UnmarshalText()
在其上做了更多 "magic"(如果您感兴趣,请查看源代码)。
All-in-all,为了避免这种"transformations"在您不知情的情况下在后台发生,请使用简单的string
类型作为您的ID,并在需要存储的地方自行进行必要的转换/ 发送您的 ID。
对于ID字段,你应该使用bson包提供的primitive.ObjectID
。
"go.mongodb.org/mongo-driver/bson/primitive"
ID primitive.ObjectID `json:"_id" bson:"_id"`