Отображение класса дела в документ mongodb с использованием реактивного монго - PullRequest
0 голосов
/ 03 января 2019

Ниже приведен мой простой документ, представляющий ссылку. Для этого я использую реактивмонго в скале.

Я получаю эту ошибку во время компиляции:

app / components / Link.scala: 60: 11: не найден десериализатор Json для типа components.Link. Попробуйте реализовать неявное чтение или формат для этого тип. [ошибка] .one [ссылка]) [ошибка] ^ [ошибка] одна ошибка найдено

Я создал имплициты в моем объекте-компаньоне Link, который также импортировал в свой класс LinkRepo.

Правильно ли я обрабатываю документ на монго _id?
Должен ли я использовать строку для сопоставления с идентификатором документа, не понимая, что является лучшей практикой? Нужно ли преобразовывать строку в 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 голосов
/ 03 января 2019

Насколько я знаю, вам НЕ НУЖНО использовать BSONObjectID, но это было бы рекомендовано. Но, тем не менее, вам следует использовать поле «_id» в mongo, иначе поведение по умолчанию будет создавать _id для вашего документа при создании нового ... так что ваш документ будет иметь _id и id.

Таким образом, вы должны получить идентификатор из "_id" и записать его в "_id", даже если ваш класс дел - id.

В соответствии с обработчиками, для базовых классов case вы можете использовать предоставленные им макросы:

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

это создаст Reader and Writer для вашей ссылки.

При использовании макросов у вас есть аннотации, которые вы можете использовать для выделения своего идентификатора: "_id" на mongo:

import reactivemongo.bson.Macros.Annotations.Key

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