com.rojoma.json.v3.-impl.util.AutomaticJsonCodecBuilderImpl.scala Maven / Gradle / Ivy
package com.rojoma.json.v3
package `-impl`.util
import scala.collection.mutable
import scala.reflect.macros.ParseException
import codec._
import util.{JsonKey, AlternativeJsonKey, JsonKeyStrategy, Strategy, LazyCodec, NullForNone, ForbidUnknownFields, AllowMissing}
import MacroCompat._
abstract class AutomaticJsonCodecBuilderImpl[T] extends MacroCompat with MacroCommon {
val c: Context
import c.universe._
implicit val Ttag: c.WeakTypeTag[T]
private val T = weakTypeOf[T]
private val Tname = TypeTree(T)
private def identityStrat(x: String) = x
private def underscoreStrat(x: String) = CamelSplit(x).map(_.toLowerCase).mkString("_")
private def freshTermName(): TermName = toTermName(c.freshName())
private def nameStrategy(thing: Symbol, default: String => String): String => String = {
checkAnn(thing, typeOf[JsonKeyStrategy])
thing.annotations.reverse.find { ann => isType(ann.tree.tpe, typeOf[JsonKeyStrategy]) } match {
case Some(ann) =>
findValue(ann) match {
case Some(strat : Symbol) =>
try {
Strategy.valueOf(strat.name.decodedName.toString) match {
case Strategy.Identity =>
identityStrat _
case Strategy.Underscore =>
underscoreStrat _
}
} catch {
case _: IllegalArgumentException =>
default
}
case _ =>
default
}
case None =>
default
}
}
private val nfnDefault = {
checkAnn(T.typeSymbol, typeOf[NullForNone])
T.typeSymbol.annotations.exists { ann => isType(ann.tree.tpe, typeOf[NullForNone]) }
}
private val defaultNameStrategy = nameStrategy(T.typeSymbol, identityStrat)
private val forbidUnknownFields = T.typeSymbol.annotations.exists { ann => isType(ann.tree.tpe, typeOf[ForbidUnknownFields]) }
// since v2 and v3 share the same names for their annotations, warn if we find one that isn't
// the same type but is the same name and don't find one that IS the right type.
def checkAnn(param: Symbol, t: Type) {
param.annotations.find { ann => ann.tree.tpe.erasure.typeSymbol.name == t.typeSymbol.name && !isType(ann.tree.tpe, t) }.foreach { ann =>
if(!param.annotations.exists { ann => isType(ann.tree.tpe, t) }) {
c.warning(posOf(param, ann), "Found non-v3 `" + t.typeSymbol.name.decodedName + "' annotation; did you accidentally import v2's?")
}
}
}
private def hasLazyAnnotation(param: TermSymbol) = {
checkAnn(param, typeOf[LazyCodec])
param.annotations.exists { ann => isType(ann.tree.tpe, typeOf[LazyCodec]) }
}
private def computeJsonNames(param: TermSymbol): Seq[String] = {
var names = List(nameStrategy(param, defaultNameStrategy)(param.name.decodedName.toString))
checkAnn(param, typeOf[JsonKey])
checkAnn(param, typeOf[AlternativeJsonKey])
val keyAnnotations = param.annotations.filter { ann => isType(ann.tree.tpe, typeOf[JsonKey]) }
if(keyAnnotations.size > 1) {
c.abort(posOf(param, keyAnnotations(1)), "Found multiple JsonKey annotations")
}
val keyAnnotation = keyAnnotations.headOption
for(ann <- keyAnnotation)
findValue(ann) match {
case Some(s: String) =>
names = List(s)
case _ =>
c.abort(posOf(param, ann), "Unable to find value for JsonKey annotation")
}
val alternativeKeyAnnotations = param.annotations.filter { ann => isType(ann.tree.tpe, typeOf[AlternativeJsonKey]) }
for(ann <- alternativeKeyAnnotations)
findValue(ann) match {
case Some(s : String) =>
names = s :: names
case _ =>
c.abort(posOf(param, ann), "Unable to find value for AlternativeJsonKey annotation")
}
names.reverse
}
private def findAccessor(param: TermSymbol) =
param.name.asInstanceOf[TermName]
private def findType(param: TermSymbol) =
param.typeSignature.asSeenFrom(T, T.typeSymbol)
private def findCodecType(param: TermSymbol) = {
val tpe = findType(param)
if(isType(tpe.erasure, typeOf[Option[_]].erasure)) {
val TypeRef(_, _, c) = tpe
c.head
} else {
tpe
}
}
private def isOption(param: TermSymbol) = {
val tpe = param.typeSignature.asSeenFrom(T, T.typeSymbol)
isType(tpe.erasure, typeOf[Option[_]].erasure)
}
private def hasNullForNoneAnnotation(param: TermSymbol) = {
checkAnn(param, typeOf[NullForNone])
nfnDefault || param.annotations.exists(ann => isType(ann.tree.tpe, typeOf[NullForNone]))
}
private def defaultIfMissing(param: TermSymbol): Option[Tree] = {
checkAnn(param, typeOf[AllowMissing])
val keyAnnotations = param.annotations.filter { ann => isType(ann.tree.tpe, typeOf[AllowMissing]) }
if(keyAnnotations.size > 1) {
c.abort(posOf(param, keyAnnotations(1)), "Found multiple AllowMissing annotations")
}
keyAnnotations.headOption.map { ann =>
findValue(ann) match {
case Some(s: String) =>
try {
c.parse(s)
} catch {
case e: ParseException =>
c.abort(posOf(param, ann), "AllowMissing parse error: " + e.getMessage)
}
case _ =>
c.abort(posOf(param, ann), "Unable to find value for AllowMissing annotation")
}
}
}
private case class FieldInfo(encName: TermName, decName: TermName, isLazy: Boolean, jsonNames: Seq[String], accessorName: TermName, missingMethodName: TermName, errorAugmenterMethodName: TermName, typ: Type, codecType: Type, isOption: Boolean, isNullForNone: Boolean, defaultIfMissing: Option[(Tree, TermName)], position: Position)
private val fieldss = locally {
val seenNames = new mutable.HashSet[String]
val buffer = new mutable.ListBuffer[List[FieldInfo]]
for {
member <- T.members
if member.isMethod && member.asMethod.isPrimaryConstructor
} {
val mem = member.asMethod
if(mem.owner == T.typeSymbol) {
for {
params <- mem.paramLists
} {
if(params.isEmpty || !params.head.asTerm.isImplicit) {
val fieldList =
for { rawParam <- params }
yield {
val param = rawParam.asTerm
val names = computeJsonNames(param)
names.find(seenNames) match {
case Some(name) =>
c.abort(param.pos, s"The name `$name' is already used by the codec for $Tname")
case None =>
seenNames ++= names
}
FieldInfo(
freshTermName(),
freshTermName(),
hasLazyAnnotation(param),
names,
findAccessor(param),
freshTermName(),
freshTermName(),
findType(param),
findCodecType(param),
isOption(param),
hasNullForNoneAnnotation(param),
defaultIfMissing(param).map((_, freshTermName())),
mem.pos
)
}
buffer += fieldList
}
}
}
}
buffer.toList
}
private val fields = fieldss.flatten
// three names for temporary variables used during the encoding/decoding process
private val tmp = freshTermName()
private val tmp2 = freshTermName()
private val tmp3 = freshTermName()
private val tmp4 = freshTermName()
private val lValue = freshTermName()
private val rValue = freshTermName()
private val expectedFields = freshTermName()
// the name of the thing being encoded or decoded
private val param = freshTermName()
private def encodes = fields.map { fi =>
val enc = q"_root_.com.rojoma.json.v3.codec.JsonEncode[${TypeTree(fi.codecType)}]"
if(fi.isLazy) {
q"private[this] lazy val ${fi.encName} = $enc"
} else {
q"private[this] val ${fi.encName} = $enc"
}
}
private def encoder = locally {
val encoderMap = freshTermName()
def encoderMapUpdates = for(fi <- fields) yield {
if(fi.isOption) {
if(fi.isNullForNone || fi.defaultIfMissing.nonEmpty) {
if(fi.isNullForNone && fi.defaultIfMissing.nonEmpty) {
c.warning(fi.position, "NullForNone is implied by AllowMissing on Options")
}
q"""$encoderMap(${fi.jsonNames.head}) = {
val $tmp = $param.${fi.accessorName}
if($tmp.isInstanceOf[_root_.scala.Some[_]]) ${fi.encName}.encode($tmp.get)
else _root_.com.rojoma.json.v3.ast.JNull
}"""
} else {
q"""val $tmp = $param.${fi.accessorName}
if($tmp.isInstanceOf[_root_.scala.Some[_]]) $encoderMap(${fi.jsonNames.head}) = ${fi.encName}.encode($tmp.get)"""
}
} else {
q"$encoderMap(${fi.jsonNames.head}) = ${fi.encName}.encode($param.${fi.accessorName})"
}
}
q"""def encode($param: $Tname) = {
val $encoderMap = new _root_.scala.collection.mutable.LinkedHashMap[_root_.scala.Predef.String, _root_.com.rojoma.json.v3.ast.JValue]
..$encoderMapUpdates
_root_.com.rojoma.json.v3.ast.JObject($encoderMap)
}"""
}
private def decodes = fields.map { fi =>
val dec = q"_root_.com.rojoma.json.v3.codec.JsonDecode[${TypeTree(fi.codecType)}]"
if(fi.isLazy) {
q"private[this] lazy val ${fi.decName} = $dec"
} else {
q"private[this] val ${fi.decName} = $dec"
}
}
private def defaults = fields.flatMap { case fi =>
fi.defaultIfMissing.map { case (dflt, dfltName) =>
q"private[this] def ${dfltName}: ${fi.typ} = $dflt"
}
}
private def missingMethods: List[DefDef] = fields.filter { fi => !fi.isOption && fi.defaultIfMissing.isEmpty }.map { fi =>
q"""private[this] def ${fi.missingMethodName}: _root_.scala.Left[_root_.com.rojoma.json.v3.codec.DecodeError, _root_.scala.Nothing] =
_root_.scala.Left(_root_.com.rojoma.json.v3.codec.DecodeError.MissingField(${fi.jsonNames.head},
_root_.com.rojoma.json.v3.codec.Path.empty))"""
}
private def errorAugmenterMethods: List[DefDef] = fields.map { fi =>
q"""private[this] def ${fi.errorAugmenterMethodName}($tmp3 : _root_.scala.Either[_root_.com.rojoma.json.v3.codec.DecodeError, _root_.scala.Any], $tmp4 : _root_.scala.Predef.String): _root_.scala.Left[_root_.com.rojoma.json.v3.codec.DecodeError, _root_.scala.Nothing] =
_root_.scala.Left($lValue($tmp3.asInstanceOf[_root_.scala.Left[_root_.com.rojoma.json.v3.codec.DecodeError, _root_.scala.Any]]).augment(_root_.com.rojoma.json.v3.codec.Path.Field($tmp4)))"""
}
private def fieldSet =
if(forbidUnknownFields) {
List(q"""private[this] val $expectedFields = _root_.scala.collection.immutable.Set(..${fields.flatMap(_.jsonNames)})""")
} else {
Nil
}
def decoder = locally {
val obj = freshTermName()
val tmps = fieldss.map { _.map { _ => freshTermName() } }
val decoderMapExtractions: List[ValDef] = for((fi,tmp) <- fields.zip(tmps.flatten)) yield {
val expr =
if(fi.isOption) {
fi.jsonNames.foldRight[Tree](fi.defaultIfMissing.fold[Tree](q"""_root_.scala.None""") { case (_, name) => q"""$name""" }) { (jsonName, otherwise) =>
q"""val $tmp2 = $obj.get($jsonName)
if($tmp2.isInstanceOf[_root_.scala.Some[_]]) {
val $tmp3 = ${fi.decName}.decode($tmp2.get)
if($tmp3.isInstanceOf[_root_.scala.Right[_,_]]) Some($rValue($tmp3.asInstanceOf[_root_.scala.Right[_root_.scala.Any, ${TypeTree(fi.codecType)}]]))
else if(_root_.com.rojoma.json.v3.ast.JNull == $tmp2.get) _root_.scala.None
else return ${fi.errorAugmenterMethodName}($tmp3, $jsonName)
} else $otherwise"""
}
} else {
fi.jsonNames.foldRight[Tree](fi.defaultIfMissing.fold[Tree](q"""return ${fi.missingMethodName}""") { case (_, name) => q"""$name""" }) { (jsonName, otherwise) =>
q"""val $tmp2 = ${obj}.get($jsonName)
if($tmp2.isInstanceOf[_root_.scala.Some[_]]) {
val $tmp3 = ${fi.decName}.decode($tmp2.get)
if($tmp3.isInstanceOf[_root_.scala.Right[_,_]]) $rValue($tmp3.asInstanceOf[_root_.scala.Right[_root_.scala.Any, ${TypeTree(fi.codecType)}]])
else return ${fi.errorAugmenterMethodName}($tmp3, $jsonName)
} else $otherwise"""
}
}
q"val $tmp = $expr"
}
val create = // not sure how to do this with quasiquote...
tmps.foldLeft(Select(New(TypeTree(T)), toTermName("")) : Tree) { (seed, arglist) =>
Apply(seed, arglist.map(Ident(_)))
}
def checkUnknownFields =
if(forbidUnknownFields) {
List(q"""val $tmp = $obj.keysIterator.filterNot($expectedFields)
if($tmp.hasNext) {
return _root_.scala.Left(
_root_.com.rojoma.json.v3.codec.DecodeError.InvalidField($tmp.next())
)
}""")
} else {
Nil
}
q"""def decode($param: _root_.com.rojoma.json.v3.ast.JValue): _root_.scala.Either[_root_.com.rojoma.json.v3.codec.DecodeError, $Tname] =
if($param.isInstanceOf[_root_.com.rojoma.json.v3.ast.JObject]) {
val $obj = $param.asInstanceOf[_root_.com.rojoma.json.v3.ast.JObject].fields
..$decoderMapExtractions
..$checkUnknownFields
_root_.scala.Right($create)
} else {
_root_.scala.Left(_root_.com.rojoma.json.v3.codec.DecodeError.InvalidType(
_root_.com.rojoma.json.v3.ast.JObject,
$param.jsonType,
_root_.com.rojoma.json.v3.codec.Path.empty))
}"""
}
private def lValueDef =
if(EitherCompat.hasValue)
q"""@_root_.scala.inline private[this] def $lValue[K,V](a: Left[K,V]) = a.value"""
else
q"""@_root_.scala.inline private[this] def $lValue[K,V](a: Left[K,V]) = a.a"""
private def rValueDef =
if(EitherCompat.hasValue)
q"""@_root_.scala.inline private[this] def $rValue[K,V](b: Right[K,V]) = b.value"""
else
q"""@_root_.scala.inline private[this] def $rValue[K,V](b: Right[K,V]) = b.b"""
private def eitherValues = List(lValueDef, rValueDef)
private def encode: c.Expr[JsonEncode[T] with JObjectEncode[T]] = {
val tree =
q"""(new _root_.com.rojoma.json.v3.codec.JsonEncode[$Tname] with _root_.com.rojoma.json.v3.`-impl`.util.JObjectEncode[$Tname] {
..$encodes
$encoder
override def toString = ${"#"}
}) : _root_.com.rojoma.json.v3.codec.JsonEncode[$Tname] with _root_.com.rojoma.json.v3.`-impl`.util.JObjectEncode[$Tname]"""
// println(tree)
c.Expr[JsonEncode[T] with JObjectEncode[T]](tree)
}
private def decode: c.Expr[JsonDecode[T]] = {
val tree =
q"""(new _root_.com.rojoma.json.v3.codec.JsonDecode[$Tname] {
..$eitherValues
..$decodes
..$defaults
..$missingMethods
..$errorAugmenterMethods
..$fieldSet
$decoder
override def toString = ${"#"}
}) : _root_.com.rojoma.json.v3.codec.JsonDecode[$Tname]"""
// println(tree)
c.Expr[JsonDecode[T]](tree)
}
private def codec: c.Expr[JsonEncode[T] with JsonDecode[T] with JObjectEncode[T]] = {
val tree = q"""(new _root_.com.rojoma.json.v3.codec.JsonEncode[$Tname] with _root_.com.rojoma.json.v3.codec.JsonDecode[$Tname] with _root_.com.rojoma.json.v3.`-impl`.util.JObjectEncode[$Tname] {
..$eitherValues
..$encodes
..$decodes
..$defaults
..$missingMethods
..$errorAugmenterMethods
..$fieldSet
$encoder
$decoder
override def toString = ${"#"}
}) : _root_.com.rojoma.json.v3.codec.JsonEncode[$Tname] with _root_.com.rojoma.json.v3.codec.JsonDecode[$Tname] with _root_.com.rojoma.json.v3.`-impl`.util.JObjectEncode[$Tname]"""
// println(tree)
c.Expr[JsonEncode[T] with JsonDecode[T] with JObjectEncode[T]](tree)
}
}
object AutomaticJsonCodecBuilderImpl {
def encode[T : ctx.WeakTypeTag](ctx: Context): ctx.Expr[JsonEncode[T] with JObjectEncode[T]] = {
val b = new {
val c: ctx.type = ctx
val Ttag = implicitly[c.WeakTypeTag[T]]
} with AutomaticJsonCodecBuilderImpl[T]
b.encode
}
def decode[T : ctx.WeakTypeTag](ctx: Context): ctx.Expr[JsonDecode[T]] = {
val b = new {
val c: ctx.type = ctx
val Ttag = implicitly[c.WeakTypeTag[T]]
} with AutomaticJsonCodecBuilderImpl[T]
b.decode
}
def codec[T : ctx.WeakTypeTag](ctx: Context): ctx.Expr[JsonEncode[T] with JsonDecode[T] with JObjectEncode[T]] = {
val b = new {
val c: ctx.type = ctx
val Ttag = implicitly[c.WeakTypeTag[T]]
} with AutomaticJsonCodecBuilderImpl[T]
b.codec
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy