Logo Questions Linux Laravel Mysql Ubuntu Git Menu
 

Playframework JSON parsing - Null pointer exception - when array is present

I am using a playframework for JSON parsing and I am facing following NullPointerException:

My data model is following:

case class SearchLikeThisResult(total: Int, max_score: Double, hits: Seq[Hits])
case class Hits( index: String, typ: String, id: String, score: Double)

My Readers are as follows:

object SearchLikeThisHits {

  import play.api.libs.functional.syntax._

  implicit val searchLikeThisResult: Reads[SearchLikeThisResult] = (
    (JsPath \ "total").read[Int] and
    (JsPath \ "max_score").read[Double] and
    (JsPath \ "hits").read[Seq[Hits]]
    )(SearchLikeThisResult.apply _)


  implicit val hitsReads: Reads[Hits] = (
    (JsPath \ "_index").read[String] and
      (JsPath \ "_type").read[String] and
      (JsPath \ "_id").read[String] and
      (JsPath \ "_score").read[Double]
    )(Hits.apply _)

}

Then I have following test code:

import play.api.libs.json.{JsValue, Json}

object Test extends App{

  val str = """{"total": 53, "max_score": 3.2948244, "hits": [
                                 {
                                     "_index": "hovno",
                                     "_type": "BYT",
                                     "_id": "3413569628",
                                     "_score": 3.2948244
                                 },
                                 {
                                      "_index": "hovno22",
                                      "_type": "BYT",
                                      "_id": "3413569628",
                                      "_score": 3.2948244
                                 }
                             ]
              }"""

  import SearchLikeThisHits.searchLikeThisResult

  val json = Json.parse(str)
  val r = json.as[SearchLikeThisResult]
}

This results in following NullPointerException:

Exception in thread "main" java.lang.NullPointerException
    at play.api.libs.json.Json$.fromJson(Json.scala:115)
    at play.api.libs.json.DefaultReads$$anon$2$$anonfun$reads$6.apply(Reads.scala:448)
    at play.api.libs.json.DefaultReads$$anon$2$$anonfun$reads$6.apply(Reads.scala:447)
    at scala.collection.LinearSeqOptimized$class.foldLeft(LinearSeqOptimized.scala:111)
    at scala.collection.immutable.List.foldLeft(List.scala:84)
    at scala.collection.generic.TraversableForwarder$class.foldLeft(TraversableForwarder.scala:41)
    at scala.collection.mutable.ListBuffer.foldLeft(ListBuffer.scala:45)
    at play.api.libs.json.DefaultReads$$anon$2.reads(Reads.scala:447)
    at play.api.libs.json.PathReads$$anonfun$at$1$$anonfun$apply$2.apply(JsConstraints.scala:36)
    at play.api.libs.json.PathReads$$anonfun$at$1$$anonfun$apply$2.apply(JsConstraints.scala:36)
    at play.api.libs.json.JsResult$class.flatMap(JsResult.scala:103)
    at play.api.libs.json.JsSuccess.flatMap(JsResult.scala:9)
    at play.api.libs.json.PathReads$$anonfun$at$1.apply(JsConstraints.scala:36)
    at play.api.libs.json.PathReads$$anonfun$at$1.apply(JsConstraints.scala:36)
    at play.api.libs.json.Reads$$anon$8.reads(Reads.scala:101)
    at play.api.libs.json.Reads$$anon$3$$anon$4.reads(Reads.scala:81)
    at play.api.libs.json.Reads$$anonfun$map$1.apply(Reads.scala:28)
    at play.api.libs.json.Reads$$anonfun$map$1.apply(Reads.scala:28)
    at play.api.libs.json.Reads$$anon$8.reads(Reads.scala:101)
    at play.api.libs.json.JsValue$class.as(JsValue.scala:65)
    at play.api.libs.json.JsObject.as(JsValue.scala:166)
    at models.data.Test$delayedInit$body.apply(Test.scala:33)
    at scala.Function0$class.apply$mcV$sp(Function0.scala:40)
    at scala.runtime.AbstractFunction0.apply$mcV$sp(AbstractFunction0.scala:12)
    at scala.App$$anonfun$main$1.apply(App.scala:71)
    at scala.App$$anonfun$main$1.apply(App.scala:71)
    at scala.collection.immutable.List.foreach(List.scala:318)
    at scala.collection.generic.TraversableForwarder$class.foreach(TraversableForwarder.scala:32)
    at scala.App$class.main(App.scala:71)
    at models.data.Test$.main(Test.scala:9)
    at models.data.Test.main(Test.scala)
    at sun.reflect.NativeMethodAccessorImpl.invoke0(Native Method)
    at sun.reflect.NativeMethodAccessorImpl.invoke(NativeMethodAccessorImpl.java:57)
    at sun.reflect.DelegatingMethodAccessorImpl.invoke(DelegatingMethodAccessorImpl.java:43)
    at java.lang.reflect.Method.invoke(Method.java:606)
    at com.intellij.rt.execution.application.AppMain.main(AppMain.java:134)

If I remove an json array that all works. I tried similar example from playframework json support web pages which worked correctly. I can't spot the problem here. My guess based on debugging is that there might be an issue with implicits but dont know how to help myself more. I tried "-Xprint:typer" compiler option but haven't found anything suspicious there.

Any hint would be appreciated

like image 547
jaksky Avatar asked Mar 26 '15 13:03

jaksky


1 Answers

It's an initialization order problem. You need to define Reads[Hits] before Reads[SearchLikeThisResult]. It compiles because the symbol exists in the code, when when Reads[SearchLikeThisResult] is initialized, Reads[Hits] is not. It goes unnoticed until it attempts to parse the array of Hits, and hits the NPE.

So just swap the order. This is related to this answer.

implicit val hitsReads: Reads[Hits] = (
    (JsPath \ "_index").read[String] and
    (JsPath \ "_type").read[String] and
    (JsPath \ "_id").read[String] and
    (JsPath \ "_score").read[Double]
)(Hits.apply _)

implicit val searchLikeThisResult: Reads[SearchLikeThisResult] = (
    (JsPath \ "total").read[Int] and
    (JsPath \ "max_score").read[Double] and
    (JsPath \ "hits").read[Seq[Hits]]
)(SearchLikeThisResult.apply _)
like image 168
Michael Zajac Avatar answered Nov 04 '22 12:11

Michael Zajac