I want to write an anorm.ToStatement instance for generic enumeratum.values.ValueEnumEntry[ValueType]. But seems like ValueType is not inferred correctly.
My code:
import java.sql.PreparedStatement
import anorm._
import enumeratum.values._
sealed trait Test extends IntEnumEntry
object Test extends IntEnum[Test] {
case object One extends Test {val value = 1}
val values = findValues
}
implicit def valueEnumEntryToStatement[ValueType: ToStatement, A <: ValueEnumEntry[ValueType]]: ToStatement[A] =
(s: PreparedStatement, index: Int, v: A) => implicitly[ToStatement[ValueType]].set(s, index, v.value)
val toStatement: ToStatement[Test] = implicitly[ToStatement[Test]] //NOT OK: diverging implicit expansion for type anorm.ToStatement[ValueType]
It fails on diverging implicit expansion for type anorm.ToStatement[ValueType].
But if I set Int type explicitly, it works.
//either
implicit def valueEnumEntryToStatement[A <: ValueEnumEntry[Int]]: ToStatement[A] =
(s: PreparedStatement, index: Int, v: A) => implicitly[ToStatement[Int]].set(s, index, v.value)
val toStatement: ToStatement[Test] = implicitly[ToStatement[Test]] //OK
//or
def valueEnumEntryToStatement[ValueType: ToStatement, A <: ValueEnumEntry[ValueType]]: ToStatement[A] =
(s: PreparedStatement, index: Int, v: A) => implicitly[ToStatement[ValueType]].set(s, index, v.value)
implicit def intEnumEntryToStatement[A <: IntEnumEntry]: ToStatement[A] = valueEnumEntryToStatement[Int, A]
val toStatement: ToStatement[Test] = implicitly[ToStatement[Test]]
Is there a way to make a general solution without explicitly setting a type?
The trouble is that
ValueTypeis a type parameter in traitValueEnumEntryand there is no type memberValueTypethere (so we can't use typeTest#ValueType). Try defining a type class