org.json4sbt.Extraction.scala Maven / Gradle / Ivy
The newest version!
/*
* Copyright 2009-2010 WorldWide Conferencing, LLC
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
package org.json4sbt
import java.lang.{Integer => JavaInteger, Long => JavaLong, Short => JavaShort, Byte => JavaByte, Boolean => JavaBoolean, Double => JavaDouble, Float => JavaFloat}
import java.math.{BigDecimal => JavaBigDecimal}
import java.util.Date
import java.sql.Timestamp
import reflect._
import scala.reflect.Manifest
import scala.collection.JavaConverters._
/** Function to extract values from JSON AST using case classes.
*
* See: ExtractionExamples.scala
*/
object Extraction {
/** Extract a case class from JSON.
* @see org.json4sbt.JsonAST.JValue#extract
* @throws MappingException is thrown if extraction fails
*/
def extract[A](json: JValue)(implicit formats: Formats, mf: Manifest[A]): A = {
try {
extract(json, Reflector.scalaTypeOf[A]).asInstanceOf[A]
} catch {
case e: MappingException => throw e
case e: Exception =>
throw new MappingException("unknown error", e)
}
}
/** Extract a case class from JSON.
* @see org.json4sbt.JsonAST.JValue#extract
*/
def extractOpt[A](json: JValue)(implicit formats: Formats, mf: Manifest[A]): Option[A] =
try { Option(extract(json)(formats, mf)) } catch { case _: MappingException => None }
def extract(json: JValue, target: TypeInfo)(implicit formats: Formats): Any = extract(json, ScalaType(target))
/** Decompose a case class into JSON.
*
* Example:
* case class Person(name: String, age: Int)
* implicit val formats = org.json4sbt.DefaultFormats
* Extraction.decompose(Person("joe", 25)) == JObject(JField("age",JInt(25)) :: JField("name",JString("joe")) :: Nil)
*
*/
def decomposeWithBuilder[T](a: Any, builder: JsonWriter[T])(implicit formats: Formats): T = {
internalDecomposeWithBuilder(a,builder)(formats)
builder.result
}
/** Load lazy val value
*
* This is a fix for failed lazy val serialization from FieldSerializer (see org.json4sbt.native.LazyValBugs test).
*
* We do this by finding the hidden lazy method which will have same name as the lazy val name
* but with suffix "$lzycompute" (for scala v2.10+), then invoke the method if found, and return the value.
*
* The "$lzycompute" method naming could be changed in future so this method must be adjusted if that happens.
*
* @param a Object to be serialized
* @param name Field name to be checked
* @param defaultValue Default value if lazy method is not found
* @return Value of invoked lazy method if found, else return the default value
*/
def loadLazyValValue(a: Any, name: String, defaultValue: Any) = {
try {
val method = a.getClass.getDeclaredMethod(name + "$lzycompute")
method.setAccessible(true)
method.invoke(a)
} catch {
case e: Exception => defaultValue
}
}
private[this] lazy val typesHaveNaN: Set[Class[_]] = Set(
classOf[Double],
classOf[Float],
classOf[java.lang.Double],
classOf[java.lang.Float]
)
/** Decompose a case class into JSON.
*
* This is broken out to avoid calling builder.result when we return from recursion
*/
def internalDecomposeWithBuilder[T](a: Any, builder: JsonWriter[T])(implicit formats: Formats):Unit = {
val current = builder
def prependTypeHint(clazz: Class[_], o: JObject) =
JObject(JField(formats.typeHintFieldName, JString(formats.typeHints.hintFor(clazz))) :: o.obj)
def addField(name: String, v: Any, obj: JsonWriter[T]) = v match {
case None => formats.emptyValueStrategy.noneValReplacement map (internalDecomposeWithBuilder(_, obj.startField(name)))
case oth => internalDecomposeWithBuilder(v, obj.startField(name))
}
val serializer = formats.typeHints.serialize
val any = a.asInstanceOf[AnyRef]
def decomposeObject(k: Class[_]) = {
val klass = Reflector.scalaTypeOf(k)
val descriptor = Reflector.describe(klass).asInstanceOf[reflect.ClassDescriptor]
val ctorParams = descriptor.mostComprehensive.map(_.name)
val iter = descriptor.properties.iterator
val obj = current.startObject()
if (formats.typeHints.containsHint(k)) {
val f = obj.startField(formats.typeHintFieldName)
f.string(formats.typeHints.hintFor(k))
}
val fs = formats.fieldSerializer(k)
while(iter.hasNext) {
val prop = iter.next()
val fieldVal = prop.get(any)
val n = prop.name
if (fs.isDefined) {
val fieldSerializer = fs.get
val ff = (fieldSerializer.serializer orElse Map((n, fieldVal) -> Some((n, fieldVal))))((n, fieldVal))
if (ff.isDefined) {
val Some((nn, vv)) = ff
val vvv = if (fieldSerializer.includeLazyVal) loadLazyValValue(a, nn, vv) else vv
addField(nn, vvv, obj)
}
} else if (ctorParams contains prop.name) addField(n, fieldVal, obj)
}
obj.endObject()
}
if (formats.customSerializer(formats).isDefinedAt(a)) {
current addJValue formats.customSerializer(formats)(a)
} else if (!serializer.isDefinedAt(a)) {
val k = if (any != null) any.getClass else null
// A series of if branches because of performance reasons
if (any == null) {
current.addJValue(JNull)
} else if (classOf[JValue].isAssignableFrom(k)) {
current.addJValue(any.asInstanceOf[JValue])
} else if (typesHaveNaN.contains(any.getClass) && any.toString == "NaN") {
current.addJValue(JNull)
} else if (Reflector.isPrimitive(any.getClass)) {
writePrimitive(any, current)(formats)
} else if (classOf[scala.collection.Map[_, _]].isAssignableFrom(k)) {
val obj = current.startObject()
val iter = any.asInstanceOf[scala.collection.Map[_, _]].iterator
while(iter.hasNext) {
iter.next() match {
case (k: String, v) => addField(k, v, obj)
case (k: Symbol, v) => addField(k.name, v, obj)
case (k: Int, v) => addField(k.toString, v, obj)
case (k: Long, v) => addField(k.toString, v, obj)
case (k: Date, v) => addField(formats.dateFormat.format(k), v, obj)
case (k: JavaInteger, v) => addField(k.toString, v, obj)
case (k: BigInt, v) => addField(k.toString, v, obj)
case (k: JavaLong, v) => addField(k.toString, v, obj)
case (k: Short, v) => addField(k.toString, v, obj)
case (k: JavaShort, v) => addField(k.toString, v, obj)
case (k, v) => {
val customKeySerializer = formats.customKeySerializer(formats)
if(customKeySerializer.isDefinedAt(k)) {
addField(customKeySerializer(k), v, obj)
} else {
fail("Do not know how to serialize key of type " + k.getClass + ". " +
"Consider implementing a CustomKeySerializer.")
}
}
}
}
obj.endObject()
} else if (classOf[Iterable[_]].isAssignableFrom(k)) {
val arr = current.startArray()
val iter = any.asInstanceOf[Iterable[_]].iterator
while(iter.hasNext) { internalDecomposeWithBuilder(iter.next(), arr) }
arr.endArray()
} else if (classOf[java.util.Collection[_]].isAssignableFrom(k)) {
val arr = current.startArray()
val iter = any.asInstanceOf[java.util.Collection[_]].iterator
while(iter.hasNext) { internalDecomposeWithBuilder(iter.next(), arr) }
arr.endArray()
} else if (k.isArray) {
val arr = current.startArray()
val iter = any.asInstanceOf[Array[_]].iterator
while(iter.hasNext) { internalDecomposeWithBuilder(iter.next(), arr) }
arr.endArray()
} else if (classOf[Option[_]].isAssignableFrom(k)) {
val v = any.asInstanceOf[Option[_]]
if (v.isDefined) {
internalDecomposeWithBuilder(v.get, current)
}
} else if (classOf[Either[_, _]].isAssignableFrom(k)) {
val v = any.asInstanceOf[Either[_, _]]
if (v.isLeft) {
internalDecomposeWithBuilder(v.left.get, current)
} else {
internalDecomposeWithBuilder(v.right.get, current)
}
} else if (classOf[(_, _)].isAssignableFrom(k)) {
any.asInstanceOf[(_, _)] match {
case (k: String, v) =>
val obj = current.startObject()
addField(k, v, obj)
obj.endObject()
case (k: Symbol, v) =>
val obj = current.startObject()
addField(k.name, v, obj)
obj.endObject()
case _: (_, _) =>
decomposeObject(k)
}
} else {
decomposeObject(k)
}
} else current addJValue prependTypeHint(any.getClass, serializer(any))
}
/** Decompose a case class into JSON.
*
* Example:
* case class Person(name: String, age: Int)
* implicit val formats = org.json4sbt.DefaultFormats
* Extraction.decompose(Person("joe", 25)) == JObject(JField("age",JInt(25)) :: JField("name",JString("joe")) :: Nil)
*
*/
def decompose(a: Any)(implicit formats: Formats): JValue =
decomposeWithBuilder(a, if (formats.wantsBigDecimal) JsonWriter.bigDecimalAst else JsonWriter.ast)
private[this] def writePrimitive(a: Any, builder: JsonWriter[_])(implicit formats: Formats) = a match {
case x: String => builder.string(x)
case x: Int => builder.int(x)
case x: Long => builder.long(x)
case x: Double => builder.double(x)
case x: Float => builder.float(x)
case x: Byte => builder.byte(x)
case x: BigInt => builder.bigInt(x)
case x: BigDecimal => builder.bigDecimal(x)
case x: Boolean => builder.boolean(x)
case x: Short => builder.short(x)
case x: java.lang.Integer => builder.int(x.intValue())
case x: java.lang.Long => builder.long(x.longValue())
case x: java.lang.Double => builder.double(x.doubleValue())
case x: java.lang.Float => builder.float(x.floatValue())
case x: java.lang.Byte => builder.byte(x.byteValue())
case x: java.lang.Boolean => builder.boolean(x.booleanValue())
case x: java.lang.Short => builder.short(x.shortValue())
case x: java.math.BigDecimal => builder.bigDecimal(x)
case x: Date => builder.string(formats.dateFormat.format(x))
case x: Symbol => builder.string(x.name)
case _ => sys.error("not a primitive " + a.asInstanceOf[AnyRef].getClass)
}
/** Flattens the JSON to a key/value map.
*/
def flatten(json: JValue)(implicit formats: Formats = DefaultFormats): Map[String, String] = {
def escapePath(str: String) = str
def flatten0(path: String, json: JValue): Map[String, String] = {
json match {
case JNothing | JNull => Map()
case JString(s) => Map(path -> ("\"" + ParserUtil.quote(s) + "\""))
case JDouble(num) => Map(path -> num.toString)
case JDecimal(num) => Map(path -> num.toString)
case JLong(num) => Map(path -> num.toString)
case JInt(num) => Map(path -> num.toString)
case JBool(value) => Map(path -> value.toString)
case JObject(obj) => obj.foldLeft(Map[String, String]()) { case (map, (name, value)) =>
map ++ flatten0(path + "." + escapePath(name), value)
}
case JArray(arr) => arr.length match {
case 0 => Map(path -> "[]")
case _ => arr.foldLeft((Map[String, String](), 0)) {
(tuple, value) => (tuple._1 ++ flatten0(path + "[" + tuple._2 + "]", value), tuple._2 + 1)
}._1
}
}
}
flatten0("", json)
}
/** Unflattens a key/value map to a JSON object.
*/
def unflatten(map: Map[String, String], useBigDecimalForDouble: Boolean = false, useBigIntForLong: Boolean = true): JValue = {
import scala.util.matching.Regex
def extractValue(value: String): JValue = value.toLowerCase match {
case "" => JNothing
case "null" => JNull
case "true" => JBool.True
case "false" => JBool.False
case "[]" => JArray(Nil)
case x @ _ =>
if (value.charAt(0).isDigit) {
if (value.indexOf('.') == -1) {
if (useBigIntForLong) JInt(BigInt(value))
else JLong(value.toLong)
} else {
if (!useBigDecimalForDouble) JDouble(ParserUtil.parseDouble(value))
else JDecimal(BigDecimal(value))
}
}
else JString(ParserUtil.unquote(value.substring(1)))
}
def submap(prefix: String): Map[String, String] =
map.withFilter(t => t._1.startsWith(prefix)).map(
t => (t._1.substring(prefix.length), t._2)
)
val ArrayProp = new Regex("""^(\.([^\.\[]+))\[(\d+)\].*$""")
val ArrayElem = new Regex("""^(\[(\d+)\]).*$""")
val OtherProp = new Regex("""^(\.([^\.\[]+)).*$""")
val uniquePaths = map.keys.foldLeft[Set[String]](Set()) {
(set, key) =>
key match {
case ArrayProp(p, f, i) => set + p
case OtherProp(p, f) => set + p
case ArrayElem(p, i) => set + p
case x @ _ => set + x
}
}.toList.sortWith(_ < _) // Sort is necessary to get array order right
uniquePaths.foldLeft[JValue](JNothing) { (jvalue, key) =>
jvalue.merge(key match {
case ArrayProp(p, f, i) => JObject(List(JField(f, unflatten(submap(key)))))
case ArrayElem(p, i) => JArray(List(unflatten(submap(key))))
case OtherProp(p, f) => JObject(List(JField(f, unflatten(submap(key)))))
case "" => extractValue(map(key))
})
}
}
def extract(json: JValue, scalaType: ScalaType)(implicit formats: Formats): Any = {
if (scalaType.isEither) {
import scala.util.control.Exception.allCatch
(allCatch opt {
Left(extract(json, scalaType.typeArgs(0)))
} orElse (allCatch opt {
Right(extract(json, scalaType.typeArgs(1)))
})).getOrElse(fail("Expected value but got " + json))
} else if (scalaType.isOption) {
customOrElse(scalaType, json)(_.toOption flatMap (j => Option(extract(j, scalaType.typeArgs.head))))
} else if (scalaType.isMap) {
json match {
case JObject(xs) => {
val kta = scalaType.typeArgs(0)
val ta = scalaType.typeArgs(1)
Map(xs.map(x => (convert(x._1, kta, formats), extract(x._2, ta))): _*)
}
case x => fail("Expected object but got " + x)
}
} else if (scalaType.isCollection) {
customOrElse(scalaType, json)(new CollectionBuilder(_, scalaType).result)
} else if (classOf[(_, _)].isAssignableFrom(scalaType.erasure) && (classOf[String].isAssignableFrom(scalaType.typeArgs.head.erasure) || classOf[Symbol].isAssignableFrom(scalaType.typeArgs.head.erasure) )) {
val ta = scalaType.typeArgs(1)
json match {
case JObject(xs :: Nil) =>
if (classOf[Symbol].isAssignableFrom(scalaType.typeArgs.head.erasure)) (Symbol(xs._1), extract(xs._2, ta))
else (xs._1, extract(xs._2, ta))
case x => fail("Expected object with 1 element but got " + x)
}
} else {
Reflector.describe(scalaType) match {
case PrimitiveDescriptor(tpe, default) => convert(json, tpe, formats, default) //customOrElse(tpe, json)(convert(_, tpe, formats, default))
case o : ClassDescriptor if o.erasure.isSingleton =>
if (json==JObject(List.empty))
o.erasure.singletonInstance.getOrElse(sys.error(s"Not a case object: ${o.erasure}"))
else
sys.error(s"Expected empty parameter list for singleton instance, got ${json} instead")
case c: ClassDescriptor => new ClassInstanceBuilder(json, c).result
}
}
}
private class CollectionBuilder(json: JValue, tpe: ScalaType)(implicit formats: Formats) {
private[this] val typeArg = tpe.typeArgs.head
private[this] def mkCollection(constructor: Array[_] => Any) = {
val array: Array[_] = json match {
case JArray(arr) => arr.map(extract(_, typeArg)).toArray
case JNothing | JNull => Array[AnyRef]()
case x => fail("Expected collection but got " + x + " for root " + json + " and mapping " + tpe)
}
constructor(array)
}
private[this] def mkTypedArray(a: Array[_]) = {
import java.lang.reflect.Array.{newInstance => newArray}
a.foldLeft((newArray(typeArg.erasure, a.length), 0)) { (tuple, e) => {
java.lang.reflect.Array.set(tuple._1, tuple._2, e)
(tuple._1, tuple._2 + 1)
}}._1
}
def result: Any = {
val custom = formats.customDeserializer(formats)
if (custom.isDefinedAt(tpe.typeInfo, json)) custom(tpe.typeInfo, json)
else if (tpe.erasure == classOf[List[_]]) mkCollection(a => List(a: _*))
else if (tpe.erasure == classOf[Set[_]]) mkCollection(a => Set(a: _*))
else if (tpe.erasure == classOf[scala.collection.mutable.Set[_]]) mkCollection(a => scala.collection.mutable.Set(a: _*))
else if (tpe.erasure == classOf[scala.collection.mutable.Seq[_]]) mkCollection(a => scala.collection.mutable.Seq(a: _*))
else if (tpe.erasure == classOf[java.util.ArrayList[_]]) mkCollection(a => new java.util.ArrayList[Any](a.toList.asJavaCollection))
else if (tpe.erasure.isArray) mkCollection(mkTypedArray)
else if (classOf[scala.collection.generic.GenericTraversableTemplate[_, Any]].isAssignableFrom(tpe.erasure)) {
reflect.ScalaSigReader.companions(tpe.erasure.getName) match {
case Some(tuple) => // can't `case Some((_, Some(c)))` due to Scala 2.10 bug
tuple match {
case (_, Some(c)) =>
import language.reflectiveCalls
val companion = c.asInstanceOf[{def apply(elems: Seq[_]): Any}]
mkCollection(a => companion(a.toSeq))
case _ =>
fail("Expected collection but got " + tpe)
}
case _ =>
fail("Expected collection but got " + tpe)
}
} else fail("Expected collection but got " + tpe)
}
}
private class ClassInstanceBuilder(json: JValue, descr: ClassDescriptor)(implicit formats: Formats) {
private object TypeHint {
def unapply(fs: List[JField]): Option[(String, List[JField])] =
if (formats.typeHints == NoTypeHints) None
else {
fs.partition(_._1 == formats.typeHintFieldName) match {
case (Nil, _) => None
case (t, f) => Some((t.head._2.values.toString, f))
}
}
}
private[this] var _constructor: ConstructorDescriptor = null
private[this] def constructor = {
if (_constructor == null) {
_constructor =
if (descr.constructors.size == 1) descr.constructors.head
else {
val argNames = json match {
case JObject(fs) => fs.map(_._1)
case _ => Nil
}
val r = descr.bestMatching(argNames)
r.getOrElse(fail("No constructor for type " + descr.erasure + ", " + json))
}
}
_constructor
}
private[this] def setFields(a: AnyRef) = json match {
case JObject(fields) =>
formats.fieldSerializer(a.getClass) map { serializer =>
val ctorArgs = constructor.params.map(_.name)
val fieldsToSet = descr.properties.filterNot(f => ctorArgs.contains(f.name))
val idPf: PartialFunction[JField, JField] = { case f => f }
val jsonSerializers = (fields map { f =>
val JField(n, v) = (serializer.deserializer orElse idPf)(f)
(n, (n, v))
}).toMap
fieldsToSet foreach { prop =>
jsonSerializers get prop.name foreach { case (_, v) =>
val vv = extract(v, prop.returnType)
// If includeLazyVal is set, try to find and initialize lazy val.
// This is to prevent the extracted value to be overwritten by the lazy val initialization.
if (serializer.includeLazyVal) loadLazyValValue(a, prop.name, vv) else ()
prop.set(a, vv)
}
}
}
a
case _ => a
}
private[this] def buildCtorArg(json: JValue, descr: ConstructorParamDescriptor) = {
val default = descr.defaultValue
def defv(v: Any) = if (default.isDefined) default.get() else v
if (descr.isOptional && json == JNothing) defv(None)
else {
try {
val x = if (json == JNothing && default.isDefined) default.get() else extract(json, descr.argType)
if (descr.isOptional) { if (x == null) defv(None) else x }
else if (x == null) {
if(!default.isDefined && descr.argType <:< ScalaType(manifest[AnyVal])) {
throw new MappingException("Null invalid value for a sub-type of AnyVal")
} else {
defv(x)
}
}
else x
} catch {
case e @ MappingException(msg, _) =>
if (descr.isOptional && !formats.strictOptionParsing) defv(None) else fail("No usable value for " + descr.name + "\n" + msg, e)
}
}
}
private[this] def instantiate = {
val jconstructor = constructor.constructor
val deserializedJson = json match {
case JObject(fields) =>
formats.fieldSerializer(descr.erasure.erasure) map { serializer =>
val idPf: PartialFunction[JField, JField] = { case f => f }
JObject(fields map { f =>
(serializer.deserializer orElse idPf)(f)
})
} getOrElse json
case other: JValue => other
}
val args = constructor.params.map(a => buildCtorArg(deserializedJson \ a.name, a))
try {
if (jconstructor.getDeclaringClass == classOf[java.lang.Object]) {
deserializedJson match {
case JObject(TypeHint(t, fs)) => mkWithTypeHint(t: String, fs: List[JField], descr.erasure)
case v: JValue => v.values
}
} else {
val instance = jconstructor.invoke(descr.companion, args)
setFields(instance.asInstanceOf[AnyRef])
}
} catch {
case e @ (_:IllegalArgumentException | _:InstantiationException) =>
val argsTypeComparisonResult = {
val constructorParamTypes = jconstructor.getParameterTypes().map(paramType => Some(paramType.asInstanceOf[Class[Any]]))
val argTypes = args.map(arg => Some(if (arg != null) arg.getClass.asInstanceOf[Class[Any]] else null))
constructorParamTypes.zipAll(argTypes, None, None).map {
case (None, Some(argType)) =>
s"REDUNDANT(${argType.getName})"
case (Some(constructorParamType), None) =>
s"MISSING(${constructorParamType.getName})"
case (Some(constructorParamType), Some(argType)) if argType == null || constructorParamType.isAssignableFrom(argType) =>
"MATCH"
case (Some(constructorParamType), Some(argType)) =>
s"${argType.getName}(${argType.getClassLoader}) !<: ${constructorParamType.getName}(${constructorParamType.getClassLoader})"
}
}
fail("Parsed JSON values do not match with class constructor\nargs=" +
args.mkString(",") + "\narg types=" + args.map(a => if (a != null)
a.asInstanceOf[AnyRef].getClass.getName else "null").mkString(",") +
"\nexecutable=" + jconstructor +
"\ncause=" + e.getMessage +
"\ntypes comparison result=" + argsTypeComparisonResult.mkString(","))
}
}
private[this] def mkWithTypeHint(typeHint: String, fields: List[JField], typeInfo: ScalaType) = {
val obj = JObject(fields filterNot (_._1 == formats.typeHintFieldName))
val deserializer = formats.typeHints.deserialize
if (!deserializer.isDefinedAt(typeHint, obj)) {
val concreteClass = formats.typeHints.classFor(typeHint) getOrElse fail("Do not know how to deserialize '" + typeHint + "'")
extract(obj, typeInfo.copy(erasure = concreteClass))
} else deserializer(typeHint, obj)
}
def result: Any =
customOrElse(descr.erasure, json){
case JNull if formats.allowNull => null
case JNull if !formats.allowNull =>
fail("Did not find value which can be converted into " + descr.fullName)
case JObject(TypeHint(t, fs)) => mkWithTypeHint(t, fs, descr.erasure)
case _ => instantiate
}
}
private[this] def customOrElse(target: ScalaType, json: JValue)(thunk: JValue => Any)(implicit formats: Formats): Any = {
val custom = formats.customDeserializer(formats)
val targetType = target.typeInfo
if (custom.isDefinedAt(targetType, json)) {
custom(targetType, json)
} else thunk(json)
}
private[this] def convert(key: String, target: ScalaType, formats: Formats): Any = {
val targetType = target.erasure
targetType match {
case tt if tt == classOf[String] => key
case tt if tt == classOf[Symbol] => Symbol(key)
case tt if tt == classOf[Int] => key.toInt
case tt if tt == classOf[JavaInteger] => new JavaInteger(key.toInt)
case tt if tt == classOf[BigInt] => key.toInt
case tt if tt == classOf[Long] => key.toLong
case tt if tt == classOf[JavaLong] => new JavaLong(key.toLong)
case tt if tt == classOf[Short] => key.toShort
case tt if tt == classOf[JavaShort] => new JavaShort(key.toShort)
case tt if tt == classOf[Date] => formatDate(key, formats)
case tt if tt == classOf[Timestamp] => formatTimestamp(key, formats)
case _ =>
val deserializer = formats.customKeyDeserializer(formats)
val typeInfo = TypeInfo(targetType, None)
if(deserializer.isDefinedAt((typeInfo, key))) {
deserializer((typeInfo, key))
} else {
fail("Do not know how to deserialize key of type " + targetType + ". Consider implementing a CustomKeyDeserializer.")
}
}
}
private[this] def convert(json: JValue, target: ScalaType, formats: Formats, default: Option[() => Any]): Any = {
val targetType = target.erasure
json match {
case JInt(x) if (targetType == classOf[Int]) => x.intValue
case JInt(x) if (targetType == classOf[JavaInteger]) => new JavaInteger(x.intValue)
case JInt(x) if (targetType == classOf[BigInt]) => x
case JInt(x) if (targetType == classOf[Long]) => x.longValue
case JInt(x) if (targetType == classOf[JavaLong]) => new JavaLong(x.longValue)
case JInt(x) if (targetType == classOf[Double]) => x.doubleValue
case JInt(x) if (targetType == classOf[JavaDouble]) => new JavaDouble(x.doubleValue)
case JInt(x) if (targetType == classOf[Float]) => x.floatValue
case JInt(x) if (targetType == classOf[JavaFloat]) => new JavaFloat(x.floatValue)
case JInt(x) if (targetType == classOf[Short]) => x.shortValue
case JInt(x) if (targetType == classOf[JavaShort]) => new JavaShort(x.shortValue)
case JInt(x) if (targetType == classOf[Byte]) => x.byteValue
case JInt(x) if (targetType == classOf[JavaByte]) => new JavaByte(x.byteValue)
case JInt(x) if (targetType == classOf[String]) => x.toString
case JInt(x) if (targetType == classOf[Number]) => x.longValue
case JInt(x) if (targetType == classOf[BigDecimal]) => BigDecimal(x)
case JInt(x) if (targetType == classOf[JavaBigDecimal]) => BigDecimal(x).bigDecimal
case JLong(x) if (targetType == classOf[Int]) => x.intValue
case JLong(x) if (targetType == classOf[JavaInteger]) => new JavaInteger(x.intValue)
case JLong(x) if (targetType == classOf[BigInt]) => x
case JLong(x) if (targetType == classOf[Long]) => x.longValue
case JLong(x) if (targetType == classOf[JavaLong]) => new JavaLong(x.longValue)
case JLong(x) if (targetType == classOf[Double]) => x.doubleValue
case JLong(x) if (targetType == classOf[JavaDouble]) => new JavaDouble(x.doubleValue)
case JLong(x) if (targetType == classOf[Float]) => x.floatValue
case JLong(x) if (targetType == classOf[JavaFloat]) => new JavaFloat(x.floatValue)
case JLong(x) if (targetType == classOf[Short]) => x.shortValue
case JLong(x) if (targetType == classOf[JavaShort]) => new JavaShort(x.shortValue)
case JLong(x) if (targetType == classOf[Byte]) => x.byteValue
case JLong(x) if (targetType == classOf[JavaByte]) => new JavaByte(x.byteValue)
case JLong(x) if (targetType == classOf[String]) => x.toString
case JLong(x) if (targetType == classOf[Number]) => x.longValue
case JLong(x) if (targetType == classOf[BigDecimal]) => BigDecimal(x)
case JLong(x) if (targetType == classOf[JavaBigDecimal]) => BigDecimal(x).bigDecimal
case JDouble(x) if (targetType == classOf[Double]) => x
case JDouble(x) if (targetType == classOf[JavaDouble]) => new JavaDouble(x)
case JDouble(x) if (targetType == classOf[Float]) => x.floatValue
case JDouble(x) if (targetType == classOf[JavaFloat]) => new JavaFloat(x.floatValue)
case JDouble(x) if (targetType == classOf[String]) => x.toString
case JDouble(x) if (targetType == classOf[Int]) => x.intValue
case JDouble(x) if (targetType == classOf[Long]) => x.longValue
case JDouble(x) if (targetType == classOf[Number]) => x
case JDouble(x) if (targetType == classOf[BigDecimal]) => BigDecimal(x)
case JDouble(x) if (targetType == classOf[JavaBigDecimal]) => BigDecimal(x).bigDecimal
case JDecimal(x) if (targetType == classOf[Double]) => x.doubleValue()
case JDecimal(x) if (targetType == classOf[JavaDouble]) => new JavaDouble(x.doubleValue())
case JDecimal(x) if (targetType == classOf[BigDecimal]) => x
case JDecimal(x) if (targetType == classOf[JavaBigDecimal]) => x.bigDecimal
case JDecimal(x) if (targetType == classOf[Float]) => x.floatValue
case JDecimal(x) if (targetType == classOf[JavaFloat]) => new JavaFloat(x.floatValue)
case JDecimal(x) if (targetType == classOf[String]) => x.toString
case JDecimal(x) if (targetType == classOf[Int]) => x.intValue
case JDecimal(x) if (targetType == classOf[Long]) => x.longValue
case JDecimal(x) if (targetType == classOf[Number]) => x
case JString(s) if (targetType == classOf[String]) => s
case JString(s) if (targetType == classOf[Symbol]) => Symbol(s)
case JString(s) if (targetType == classOf[Date]) => formatDate(s, formats)
case JString(s) if (targetType == classOf[Timestamp]) => formatTimestamp(s, formats)
case JBool(x) if (targetType == classOf[Boolean]) => x
case JBool(x) if (targetType == classOf[JavaBoolean]) => new JavaBoolean(x)
case j: JValue if (targetType == classOf[JValue]) => j
case j: JObject if (targetType == classOf[JObject]) => j
case j: JArray if (targetType == classOf[JArray]) => j
case JNull if formats.allowNull => null
case JNull if !formats.allowNull =>
fail("Did not find value which can be converted into " + targetType.getName)
case JNothing =>
default map (_.apply()) getOrElse fail("Did not find value which can be converted into " + targetType.getName)
case _ =>
val custom = formats.customDeserializer(formats)
val typeInfo = target.typeInfo
if (custom.isDefinedAt(typeInfo, json)) custom(typeInfo, json)
else fail("Do not know how to convert " + json + " into " + targetType)
}
}
private[this] def formatTimestamp(s: String, formats: Formats): Timestamp = {
new Timestamp(formats.dateFormat.parse(s).getOrElse(fail("Invalid date '" + s + "'")).getTime)
}
private[this] def formatDate(s: String, formats: Formats): Date = {
formats.dateFormat.parse(s).getOrElse(fail("Invalid date '" + s + "'"))
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy