распылить атрибут json со специальным символом

У меня есть json, где имя атрибута имеет специальный символ. Пытаюсь разобрать его с помощью spray json. Ниже приведен код. Как я могу получить имя атрибута в json @xml: lang для анализа в классе case.

import spray.json._
import DefaultJsonProtocol._

object SpecialCharInName extends App {

  case class Person(name: String, `@xml:lang`: String)

  val json = """ {"name":"MyName", "@xml:lang":"US"} """

  object PersonProtocol extends DefaultJsonProtocol {
    implicit val personFormat = jsonFormat2(Person)
  }

  import PersonProtocol._
  val person = json.parseJson
  val personClass = person.convertTo[Person]

  println(personClass)

}

Приведенный выше код выдает исключение, как показано ниже.

[error] (run-main-0) spray.json.DeserializationException: Object is missing required member '@xml$colonlang'
spray.json.DeserializationException: Object is missing required member '@xml$colonlang'
at spray.json.package$.deserializationError(package.scala:23)
at spray.json.ProductFormats$class.fromField(ProductFormats.scala:60)
at c.c.s.f.v.d.SpecialCharInName$PersonProtocol$.fromField(SpecialCharInName.scala:12)
at spray.json.ProductFormatsInstances$$anon$2.read(ProductFormatsInstances.scala:56)
at spray.json.ProductFormatsInstances$$anon$2.read(ProductFormatsInstances.scala:46)
at spray.json.JsValue.convertTo(JsValue.scala:31)
at c.c.s.f.v.d.SpecialCharInName$.delayedEndpoint$com$comcast$sv$fabric$vimond$domain$SpecialCharInName$1(SpecialCharInName.scala:20)
at c.c.s.f.v.d.SpecialCharInName$delayedInit$body.apply(SpecialCharInName.scala:6)
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:76)
at scala.App$$anonfun$main$1.apply(App.scala:76)
at scala.collection.immutable.List.foreach(List.scala:381)
at scala.collection.generic.TraversableForwarder$class.foreach(TraversableForwarder.scala:35)
at scala.App$class.main(App.scala:76)
at c.c.s.f.v.d.SpecialCharInName$.main(SpecialCharInName.scala:6)
at c.c.s.f.v.d.SpecialCharInName.main(SpecialCharInName.scala)
at sun.reflect.NativeMethodAccessorImpl.invoke0(Native Method)
at sun.reflect.NativeMethodAccessorImpl.invoke(NativeMethodAccessorImpl.java:62)
at sun.reflect.DelegatingMethodAccessorImpl.invoke(DelegatingMethodAccessorImpl.java:43)
at java.lang.reflect.Method.invoke(Method.java:497)
Caused by: java.util.NoSuchElementException: key not found: @xml$colonlang
at scala.collection.MapLike$class.default(MapLike.scala:228)
at scala.collection.AbstractMap.default(Map.scala:59)
at scala.collection.MapLike$class.apply(MapLike.scala:141)
at scala.collection.AbstractMap.apply(Map.scala:59)
at spray.json.ProductFormats$class.fromField(ProductFormats.scala:57)
at c.c.s.f.v.d.SpecialCharInName$PersonProtocol$.fromField(SpecialCharInName.scala:12)
at spray.json.ProductFormatsInstances$$anon$2.read(ProductFormatsInstances.scala:56)
at spray.json.ProductFormatsInstances$$anon$2.read(ProductFormatsInstances.scala:46)
at spray.json.JsValue.convertTo(JsValue.scala:31)
at c.c.s.f.v.d.SpecialCharInName$.delayedEndpoint$com$comcast$sv$fabric$vimond$domain$SpecialCharInName$1(SpecialCharInName.scala:20)
at c.c.s.f.v.d.SpecialCharInName$delayedInit$body.apply(SpecialCharInName.scala:6)
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:76)
at scala.App$$anonfun$main$1.apply(App.scala:76)
at scala.collection.immutable.List.foreach(List.scala:381)
at scala.collection.generic.TraversableForwarder$class.foreach(TraversableForwarder.scala:35)
at scala.App$class.main(App.scala:76)
at c.c.s.f.v.d.SpecialCharInName$.main(SpecialCharInName.scala:6)
at c.c.s.f.v.d.SpecialCharInName.main(SpecialCharInName.scala)
at sun.reflect.NativeMethodAccessorImpl.invoke0(Native Method)
at sun.reflect.NativeMethodAccessorImpl.invoke(NativeMethodAccessorImpl.java:62)
at sun.reflect.DelegatingMethodAccessorImpl.invoke(DelegatingMethodAccessorImpl.java:43)
at java.lang.reflect.Method.invoke(Method.java:497)

Заранее благодарны за Вашу помощь.

Саад


person saad    schedule 31.10.2015    source источник


Ответы (1)


import spray.json._
import DefaultJsonProtocol._

object SpecialCharInName extends App {

  case class Person(name: String, `@xml:lang`: String)

  val json = """ {"name":"MyName", "@xml:lang":"US"} """

  object PersonProtocol extends DefaultJsonProtocol {
    implicit val personFormat = jsonFormat(Person.apply, "name", "@xml:lang")
  }

  import PersonProtocol._

  val person = json.parseJson
  val personClass = person.convertTo[Person]
  println(personClass)
}

Наслаждаться :)

Проще говоря, используйте специальный метод jsonFormat, который принимает метод компоновщика (например, apply) и список именованных свойств, которые должны присутствовать в анализируемом документе JSON. Стандартный в вашем случае не очень подходит.

person jdevelop    schedule 31.10.2015
comment
Мне было бы интересно узнать, github.com/fommil/spray-json-shapeless поддерживает это из коробки. Если нет, поднимите тикет со своим примером. - person fommil; 31.10.2015