嵌套期货响应与Play和ReactiveMongo的编译器错误

时间:2013-08-13 10:23:47

标签: mongodb scala playframework-2.1 future reactivemongo

我有一个mongo条目,其中一个字段是一个列表。如果输入不存在,我想添加新的。如果它存在,我想在列表中附加一个新元素。

最后我想向客户端返回Ok,但只有在操作成功完成后才能返回。不是严格要求,但在我看来,这是对用户最有意义的。

这就是我目前所拥有的 - 它可以工作,但在更新时它会覆盖旧列表,而不是附加新元素。

def myMethod(value:String, value2:String) = Action {

  Async {
    val myElement = Json.obj(
      "key" -> value2
    )

    val myDBEntry = Json.obj(
      "key" -> value,
      "list" -> List(myElement)
    )

    collection.update(Json.obj("key" -> value), myDBEntry, upsert = true).map(lastError =>
      Ok("Mongo LastError: %s".format(lastError)))
  }
}

为了检查列表是否存在并附加元素/创建一个新列表,我开始使用类似的东西(这取代了之前代码中的collection.update块):

val futureResult:Future[Option[JsObject]] = collection.find(Json.obj("key" -> value)).one[JsObject]

futureResult.map { result =>

  if (result.isEmpty) {
      collection.insert(myDBEntry).map(lastError =>
        Ok("Mongo LastError: %s".format(lastError)))


  } else {
      //this not the correct command yet - but compiler already fails because method is not returning correct future 
      collection.update(Json.obj("key" -> value), myDBEntry, upsert = true).map(lastError =>
        Ok("Mongo LastError: %s".format(lastError)))
  }


}

但编译器似乎不喜欢这种嵌套:“类型不匹配,预期:未来[结果],实际:未来:[未来[SimpleResult [Nothing]]]”

无论如何我觉得这种方式有点尴尬,必须有更优雅的方法来解决这个问题,但我是Futures和ReactiveMongo的新手并且不知道。我该如何解决?

编辑: 我还发现this post但我认为在DB操作完成之前返回响应并且我不希望这样。

1 个答案:

答案 0 :(得分:4)

尝试更改此行:

futureResult.map { result =>

对此:

futureResult.flatMap { result =>

当您在未来和地图块内部执行地图返回另一个未来时,您需要使用flatMap,因为它会使嵌套变平。