使用反应式mongo将案例类映射到mongodb文档

时间:2019-01-03 14:41:58

标签: mongodb scala playframework play-reactivemongo

下面是我代表链接的简单文档。我正在scala中使用reactivemongo。

在编译过程中出现此错误:

  

app / components / Link.scala:60:11:未找到用于类型的Json解串器   组件。尝试为此实现隐式读取或格式化   类型。 [错误] .one [链接])[错误] ^ [错误]一个错误   找到

我在Link随播对象中创建了隐式对象,并将其也导入了我的LinkRepo类。

我可以正确处理mongo文档_id吗?
我应该使用String来映射文档ID,对最佳实践感到困惑吗?我是否必须在某个时候将String转换为BSONObjectID?

package components

import javax.inject.Inject
import reactivemongo.bson._

import reactivemongo.api.ReadPreference
import reactivemongo.api.collections.bson.BSONCollection
import reactivemongo.bson.{ BSONDocument, BSONObjectID }
import reactivemongo.api.commands.{ UpdateWriteResult, WriteResult, Upserted }
import reactivemongo.api.commands.bson.BSONUpdateCommand._
import reactivemongo.api.commands.bson.BSONUpdateCommandImplicits._

case class Link(id: Link.ID,
                name: String,
                url: String)


object Link {

  type ID = String

  implicit val linkReader: BSONDocumentReader[Link] =
    BSONDocumentReader[Link] { doc: BSONDocument =>
      Link(
        doc.getAs[String]("id").getOrElse(""),
        doc.getAs[String]("name").getOrElse(""),
        doc.getAs[String]("url").getOrElse(""))
    }

  implicit val linkWriter: BSONDocumentWriter[Link] =
    BSONDocumentWriter[Link] { link: Link =>
      BSONDocument(
        "id" -> link.id,
        "name" -> link.name,
        "url" -> link.url)
    }

}

import scala.concurrent.{ ExecutionContext, Future }
import reactivemongo.bson.{ BSONDocument, BSONObjectID }

import reactivemongo.api.{ Cursor, ReadPreference }
import reactivemongo.api.commands.WriteResult

import reactivemongo.play.json._
import reactivemongo.play.json.collection.JSONCollection

import play.modules.reactivemongo.ReactiveMongoApi

class LinkRepo @Inject()(implicit ec: ExecutionContext, reactiveMongoApi: ReactiveMongoApi) {
  import Link._

  def linksCol: Future[JSONCollection] = reactiveMongoApi.database.map(_.collection("links"))

  def byId(id: Link.ID): Future[Option[Link]] = {
    linksCol.flatMap(_.find(
      selector = BSONDocument("_id" -> id),
      projection = Option.empty[BSONDocument])
      .one[Link])
  }



}

我的sbt具有以下lib版本:

scalaVersion := "2.12.7"

libraryDependencies += guice
libraryDependencies ++= Seq(
  guice,
  "joda-time" % "joda-time" % "2.9.9",
  "net.ruippeixotog" %% "scala-scraper" % "2.1.0",
  "org.reactivemongo"      %% "play2-reactivemongo" % "0.16.0-play26",
  "org.scalatestplus.play" %% "scalatestplus-play" % "3.1.2" % Test
)

插件:

addSbtPlugin("com.typesafe.play" % "sbt-plugin" % "2.6.20")

1 个答案:

答案 0 :(得分:1)

据我所知,您不需要使用BSONObjectID,但建议使用。 但是您应该在mongo中使用“ _id”字段,否则默认行为是在创建新文件时在文档上创建_id ...,这样您的文档将具有_id和id。

因此,即使您的案例类是ID,也应从“ _id”获取ID并将其写入“ _id”。

根据处理程序,对于基本案例类,您可以使用其提供的宏:

implicit val linkHandler: BSONDocumentHandler[Link] = Macros.handler

这将为您的链接生成读写器。

使用宏时,您有一个注释,可用于突出显示您的ID在mongo上为“ _id”:

import reactivemongo.bson.Macros.Annotations.Key

case class Link(
  @Key("_id")
  id: Link.ID,
  name: String,
  url: String
)