¿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 № 1El 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.