15

使用InsertOne创建新文档后,当我返回结果时,我得到一个数字数组而不是ObjectID. 在数据库中,id 生成良好。

type User struct {
  ID       string
  Email    string
  Username string
  Password string
}

var db = ...

// UserStore creates user
func UserStore(c echo.Context) (err error) {

  coll := db.Collection("users")

  u := new(User)
  if err = c.Bind(u); err != nil {
    return c.JSON(http.StatusInternalServerError, err)
  }

  result, err := coll.InsertOne(
    context.Background(),
    bson.NewDocument(
        bson.EC.String("email", u.Email),
        bson.EC.String("username", u.Username),
        bson.EC.String("password", u.Password),
    ),
  )
  if err != nil {
    return c.JSON(http.StatusInternalServerError, err)
  }

  return c.JSON(http.StatusCreated, result)
}

这会返回类似InsertedID: [90, 217, 85, 109, 184, 249, 162, 204, 249, 103, 214, 121]而不是正常的ObjectID. 如何ObjectID从新插入的文档中返回实际值?

4

1 回答 1

41

成功Collection.InsertOne()将返回 type 的结果mongo.InsertOneResult,这是一个包装新插入文档的 ID 的结构:

type InsertOneResult struct {
    // The identifier that was inserted.
    InsertedID interface{}
}

官方的 MongoDB Go 驱动程序使用该primitive.ObjectID类型来表示 MongoDB ObjectIds。这种类型是一个简单的字节数组:

type ObjectID [12]byte

要访问这种类型,您需要导入原始包

import "go.mongodb.org/mongo-driver/bson/primitive"

InsertOneResult.InsertedID将持有动态类型的primitive.ObjectID. 该primitive.ObjectID类型未定义自定义 JSON 封送方法(未实现json.Marshaler),这意味着当结果转换为 JSON 时,将使用默认的封送规则,对于字节数组(不是切片),您看到的是:十进制ObjectID 的字节表示。

您不应该将类型值转换InsertOneResult为 JSON,因为它(或者更确切地说primitive.ObjectID它本身)不是“JSON 友好的”(至少在当前版本中不是)。

而是创建/包装您自己的类型,在其中定义您希望结果在 JSON 中的样子,例如:

if oid, ok := result.InsertedID.(primitive.ObjectID); ok {
    return c.JSON(http.StatusCreated, map[string]interface{}{
        "id": oid.String(),
    })
} else {
    // Not objectid.ObjectID, do what you want
}

上面的代码会产生这样的 JSON 响应:

{"id":"ObjectID(\"5ad9a913478c26d220afb681\")"}

或者,如果您只想要十六进制表示:

if oid, ok := result.InsertedID.(primitive.ObjectID); ok {
    return c.JSON(http.StatusCreated, map[string]interface{}{
        "id": oid.Hex(),
    })
}

这将是:

{"id":"5ad9a913478c26d220afb681"}
于 2018-04-20T08:48:55.693 回答