/ / Konwertowanie (bardzo) skomplikowanych obiektów JSON na obiekty Scala z klasami przypadków w Json4s - json, scala, klasa case, json4s

Konwertowanie (bardzo) skomplikowanych obiektów JSON na obiekty Scala z klasami przypadków w Json4s - json, scala, case-class, json4s

Mam bardzo skomplikowany plik JSON, który wygląda tak:

       {
"Animals": [
[
100,
"Mammals",
[
1,
"Cat",
50,
45,
57,
-1
],
[
2,
"Dog",
31,
44,
18,
-1
]
],
[
159,
"Reptiles",
[
1,
"Lizard",
11,
12,
9,
-1
]
]
]
}

Próbuję przeanalizować tę strukturę i jakoś uzyskać z niej obiekty scala.

Oto moja próba:

case class Facts(number: Int, subTypeOfAnimal: String, data: List[Int])

case class Animaltype(value: Int, typeOfAnimal: String, characteristics: List[Facts])

case class Animal(rows: List[Animaltype])

To oczywiście nie konwertuje danych. Zwraca JNothing. Zastanawiam się, jak właściwie mogę wyrazić złożone JArrays w JArrays tego rodzaju.

Każda pomoc byłaby przydatna

Dzięki!

Odpowiedzi:

2 dla odpowiedzi № 1

Możesz zdefiniować CustomSerializers dla Facts i AnimalType.

import scala.util.Try

import org.json4s._
import org.json4s.JsonDSL._
import org.json4s.native.JsonMethods._
import org.json4s.native.Serialization

case class Facts(number: Int, subTypeOfAnimal: String, data: List[Int])
case class AnimalType(value: Int, typeOfAnimal: String, characteristics: List[Facts])
case class Animal(Animals: List[AnimalType])

implicit val formats = Serialization.formats(NoTypeHints) +
new AnimalTypeSerializer + new FactsSerializer

class FactsSerializer extends CustomSerializer[Facts](format => ( {
case JArray(JInt(nr) :: JString(subType) :: data) =>
Facts(nr.toInt, subType, data.collect{ case JInt(i) => i.toInt})
}, { case _ => throw new RuntimeException("No serializing")}))

class AnimalTypeSerializer extends CustomSerializer[AnimalType](format => ( {
case JArray(JInt(value) :: JString(typeAnimal) :: factsArrays) =>
val facts = factsArrays.collect { case facts: JArray =>
Try(facts.extract[Facts]).toOption
}.flatten
AnimalType(value.toInt, typeAnimal, facts)
}, { case _ => throw new RuntimeException("No serializing")}))

Jeśli przyjmiesz wartość json jako wartość json możesz go deserializować za pomocą:

parse(json).extract[Animal]
// Animal = Animal(List(
//   AnimalType(100,Mammals,List(
//     Facts(1,Cat,List(50, 45, 57, -1)), Facts(2,Dog,List(31, 44, 18, -1))
//   )),
//   AnimalType(159,Reptiles,List(
//     Facts(1,Lizard,List(11, 12, 9, -1))
//   ))
// ))