Play Framework:类型不匹配JsValue / JsValueWrapper

时间:2014-03-03 23:11:35

标签: json scala playframework

我需要转换此JSON

{ "matchItem": { "collection": { "fieldName": [ "value1", "value2" ] } } }

这个MongoDB投影:

{ "collection": { "$elemMatch": { "fieldName": "value1", "fieldName": "value2" } }

以下是我的代码:

def toProjection(json: JsValue) = {
  json.getOpt(__ \ "matchItem").map { value =>
    val obj = value.as[JsObject]
    for (key <- obj.keys) { obj.getOpt(__ \ key).map { matchObj =>
      for (matchKey <- matchObj.as[JsObject].keys) { matchObj.getOpt(__ \ matchKey).map { items =>
        val fields = items.as[Seq[JsValue]].map(item => (matchKey -> item))
        seq += key -> Json.obj("$elemMatch" -> Json.obj(fields))
      }}
    }}
  }
  if (seq.length > 0) JsObject(seq) else json
}

val json = """{ "matchItem": { "collection": { "fieldName": [ "value1", "value2" ] } } }"""
val proj = toProjection(json)

此代码无法编译,我总是收到以下错误消息:

[error] /home/j3d/Projects/test/app/services/Test.scala:82: type mismatch;
[error]  found   : Seq[(String, play.api.libs.json.JsValue)]
[error]  required: (String, play.api.libs.json.Json.JsValueWrapper)
[error]                 seq += fieldMaps.fromPublic(key) -> Json.obj("$elemMatch" -> Json.obj(fields))
[error]                                                                                       ^

我有点失落。我知道Json.obj是构建JsObject实例的辅助方法:

JsObject(Seq(
  "key1" -> JsString("value1"),
  "key2" -> JsString("value2")
  ...
))

......相当于:

Json.obj("key1" -> "value1", "key2" -> "value2")

在上面的代码中,fields值是Seq[(String, play.api.libs.json.JsValue)] ...所以我不明白为什么它不起作用。任何帮助都会非常感激。

2 个答案:

答案 0 :(得分:5)

以下是解决方案:

def toProjection(json: JsValue) = {
  json.getOpt(__ \ "matchItem").map { value =>
    val obj = value.as[JsObject]
    for (key <- obj.keys) { obj.getOpt(__ \ key).map { matchObj =>
      for (matchKey <- matchObj.as[JsObject].keys) { matchObj.getOpt(__ \ matchKey).map { items =>
        val fields = items.as[Seq[JsValue]].map(item => (matchKey -> item))
     // seq += key -> Json.obj("$elemMatch" -> Json.obj(fields))
        seq += key -> Json.obj("$elemMatch" -> JsObject(fields))
      }}
    }}
  }
  if (seq.length > 0) JsObject(seq) else json
}

Json.obj(fields)替换JsObject(fields)可以解决问题。

我希望它有所帮助。

答案 1 :(得分:0)

在实施我自己的解决方案并认为必须有一种更简便的方法后,到这里结束。看起来好像没有。这是另一个可行的解决方案:

def flatten(update: JsObject): JsObject = {
    def f(key: String, value: JsValue): Seq[(String, JsValue)] = {
      value match {
        case o: JsObject => o.fields.flatMap(kv => f(s"$key.${kv._1}", kv._2))
        case v: JsValue => Seq(key -> v)
      }
    }
    JsObject(update.fields.flatMap(kv => f(kv._1, kv._2)))
  }