Вот как вы можете это сделать:
import cats.syntax.functor._
import io.circe.Decoder, io.circe.generic.auto._
sealed trait PrimitiveWrapper
case class IntWrapper(v: Int) extends PrimitiveWrapper
case class StringWrapper(v: String) extends PrimitiveWrapper
case class FloatWrapper(v: Float) extends PrimitiveWrapper
case class Person(name: String, age: PrimitiveWrapper)
object GenericDerivation {
implicit val decodePrimitiveWrapper: Decoder[PrimitiveWrapper] =
List[Decoder[PrimitiveWrapper]](
Decoder.decodeInt.map(IntWrapper).widen,
Decoder.decodeString.map(StringWrapper).widen,
Decoder.decodeFloat.map(FloatWrapper).widen
).reduceLeft(_ or _)
def main(args: Array[String]): Unit = {
import io.circe.parser.decode
println(decode[Person]("""{"name" : "john", "age" : 31 }"""))
println(decode[Person]("""{"name" : "john", "age" : "thirty one" }"""))
println(decode[Person]("""{"name" : "john", "age" : 31.3 }"""))
// Prints
// Right(Person(john,IntWrapper(31)))
// Right(Person(john,StringWrapper(thirty one)))
// Right(Person(john,FloatWrapper(31.3)))
}
}
Примечание: Следующее получение разбирается с использованием IntWrapper
println(decode[Person]("""{"name" : "john", "age" : 31.0 }"""))
Обновление: Как отметил @Travis, decodePrimitiveWrapper
можно записать так:
implicit val decodePrimitiveWrapper: Decoder[PrimitiveWrapper] =
Decoder.decodeInt.map(IntWrapper).widen[PrimitiveWrapper] or
Decoder.decodeString.map(StringWrapper).widen[PrimitiveWrapper] or
Decoder.decodeFloat.map(FloatWrapper).widen[PrimitiveWrapper]