I'm trying to create inline def to generate Json codec for any enums in Scala 3. For this I need to have access to valueOf method of the parent of the enum. Something like this:
inline def gen[T](using JsonCodec[String], T <:< reflect.Enum): JsonCodec[T] = ???
How this can be achieved?
After reading comments now I changed my code to:
val decoder: JsonDecoder[T] = JsonDecoder[String].mapOrFail(v => Try(${getEnum[T]}(v)).fold(e => Left(e.getMessage), v => Right(v)))
val encoder: JsonEncoder[T] = JsonEncoder[String].contramap(_.toString)
JsonCodec.apply(encoder, decoder)
def getEnum[T: Type](using Quotes): Expr[String => T] =
import quotes.reflect.*
val companion = Ref(TypeTree.of[T].symbol.companionModule)
Select.unique(companion, "valueOf").asExprOf[String => T]
And compiler complains with:
Malformed macro.
Expected the splice ${...} to be at the top of the RHS:
inline def foo(inline x: X, ..., y: Y): Int = ${ impl('x, ... 'y) }
* The contents of the splice must call a static method
* All arguments must be quoted
As a json library you seem to use https://zio.github.io/zio-json
I guess the error is understandable. You're not supposed to use macro implementations directly like that.
Either you have a macro (inline method)
getEnum
and its implementationgetEnumImpl
(returning anExpr
) and you usegetEnum
(notgetEnumImpl
)or
getEnum
is a macro implementation itself and you use it in another macro implementationNow the error is
You can wrap
Select.unique
withApply
(and replace functionsString => T
with methods). So eitheror
Also in order to derive
JsonCodec
for enums you can useMirror
Testing:
There can be also cases depending on a parameter like in
We can also use https://github.com/typelevel/shapeless-3 for derivation or derive in Shapeless 2 style