/ / Spray-json deserializando objeto anidado - json, scala, spray-json

Spray-json deserializando objeto anidado - json, scala, spray-json

¿Cómo deserializar correctamente los objetos anidados en spray-json?

    import spray.json._

case class Person(name: String)

case class Color(n: String, r: Int, g: Int, b: Int, p: Person)

object MyJsonProtocol extends DefaultJsonProtocol {

implicit object ColorJsonFormat extends RootJsonFormat[Color] {
def write(c: Color) = JsObject(
"color-name" -> JsString(c.n),
"Green" -> JsNumber(c.g),
"Red" -> JsNumber(c.r),
"Blue" -> JsNumber(c.b),
"person-field" -> JsObject("p-name" -> JsString(c.p.name))
)

def read(value: JsValue) = {
value.asJsObject.getFields("color-name", "Red", "Green", "Blue", "person-field") match {
case Seq(JsString(name), JsNumber(red), JsNumber(green), JsNumber(blue), JsObject(person)) =>
Color(name, red.toInt, green.toInt, blue.toInt, null) //gotta replace null with correct deserializer
case _ => throw new DeserializationException("Color expected")
}
}
}

}

import MyJsonProtocol._

val jsValue = Color("CadetBlue", 95, 158, 160, Person("guest")).toJson

jsValue.prettyPrint

val color = jsValue.convertTo[Color] //person is missing of course

En una nota lateral, ¿cómo spray-json ayuda a serializar un mapa de campos (con un mapa anidado para objetos anidados)?

Respuestas

16 para la respuesta № 1

El siguiente ejemplo muestra JSON -> ResumenÁrbol de sintaxis -> Scala Case Classes and back con nombres de campos personalizados y soporte para miembros opcionales de la clase de casos. El ejemplo se deriva de la documentación de spray-json en https://github.com/spray/spray-json para la versión 1.2.5.

package rando

import spray.json._

case class Color(name: String, red: Int, green: Int, blue: Int)

case class Team(name: String, color: Option[Color])

object MyJsonProtocol extends DefaultJsonProtocol {
implicit val colorFormat = jsonFormat(Color, "name", "r", "g", "b")
implicit val teamFormat = jsonFormat(Team, "name", "jersey")
}
import MyJsonProtocol._

object GoSox extends App {
val obj = Team("Red Sox", Some(Color("Red", 255, 0, 0)))
val ast = obj.toJson
println(obj)
println(ast.prettyPrint)
println(ast.convertTo[Team])
println("""{ "name": "Red Sox", "jersey": null }""".asJson.convertTo[Team])
println("""{ "name": "Red Sox" }""".asJson.convertTo[Team])
}

El ejemplo produce lo siguiente cuando se ejecuta:

Team(Red Sox,Some(Color(Red,255,0,0)))
{
"name": "Red Sox",
"jersey": {
"name": "Red",
"r": 255,
"g": 0,
"b": 0
}
}
Team(Red Sox,Some(Color(Red,255,0,0)))
Team(Red Sox,None)
Team(Red Sox,None)

1 para la respuesta № 2

A su pregunta restante: cómo reutilizar las conversiones JSON dentro de un tipo de envoltura:

"person-field" -> JsObject("p-name" -> JsString(c.p.name))

Yo cambiaría esto a

"person-field" -> p.toJson

De esta manera, estás dejando que el Person clase de caso JSONify en sí, en lugar de introducir otra forma en el objeto de ajuste. SECO y mas sencillo.

Y:

case Seq(JsString(name), JsNumber(red), JsNumber(green), JsNumber(blue), JsObject(person)) =>
Color(name, red.toInt, green.toInt, blue.toInt, null)

Utilizar el .convertTo[Person] aquí:

case Seq(JsString(name), JsNumber(red), JsNumber(green), JsNumber(blue), jsv) =>
Color(name, red.toInt, green.toInt, blue.toInt, jsv.convertTo[Person])

Si hay problemas, por favor pida más ayuda. Tengo una estructura similar en mi propio proyecto, pero no intenté ejecutarlos en este contexto.