Невозможно преобразовать сгенерированные Scrooge классы в карты, используя Shapeless
Я использую Scrooge для генерации классов. Они выглядят примерно так, вот пример:
object Flags extends ThriftStructCodec3[Flags] {
private val NoPassthroughFields = immutable$Map.empty[Short, TFieldBlob]
val Struct = new TStruct("Flags")
val IsDangerousField = new TField("isDangerous", TType.BOOL, 1)
val IsDangerousFieldManifest = implicitly[Manifest[Boolean]]
val IsWildField = new TField("isWild", TType.BOOL, 2)
val IsWildFieldManifest = implicitly[Manifest[Boolean]]
def apply(
isDangerous: Option[Boolean] = None,
isWild: Option[Boolean] = None
): Flags =
new Immutable(
isDangerous,
isWild
)
def unapply(_item: Flags): Option[scala.Product2[Option[Boolean], Option[Boolean]]] = Some(_item)
object Immutable extends ThriftStructCodec3[Flags] {
override def encode(_item: Flags, _oproto: TProtocol) { _item.write(_oproto) }
override def decode(_iprot: TProtocol): Flags = Flags.decode(_iprot)
}
/**
* The default read-only implementation of Flags. You typically should not need to
* directly reference this class; instead, use the Flags.apply method to construct
* new instances.
*/
class Immutable(
val isDangerous: Option[Boolean],
val isWild: Option[Boolean],
override val _passthroughFields: immutable$Map[Short, TFieldBlob]
) extends Flags {
def this(
isDangerous: Option[Boolean] = None,
isWild: Option[Boolean] = None
) = this(
isDangerous,
isWild,
Map.empty
)
}
}
trait Flags
extends ThriftStruct
with scala.Product2[Option[Boolean], Option[Boolean]]
with java.io.Serializable
{
import Flags._
def isDangerous: Option[Boolean]
def isWild: Option[Boolean]
def _passthroughFields: immutable$Map[Short, TFieldBlob] = immutable$Map.empty
def _1 = isDangerous
def _2 = isWild
override def productArity: Int = 2
override def productElement(n: Int): Any = n match {
case 0 => this.isDangerous
case 1 => this.isWild
case _ => throw new IndexOutOfBoundsException(n.toString)
}
override def productPrefix: String = "Flags"
}
Цель состоит в том, чтобы с помощью Shapeless сгенерировать из этих классов вложенную карту, выполнить с ней некоторую операцию, а затем преобразовать карту обратно в сгенерированные классы.
Использование преобразования вложенных классов наблюдений во вложенные Карты с использованием Shapeless и преобразование Map[String,Any] в класс наблюдений с использованием Shapeless Мне удалось собрать проект, который отлично работает с базовыми классами наблюдений. Помимо примеров, представленных в ответах переполнения стека, я также реализовал импликации для Option[A].
Теперь, поскольку Scrooge не генерирует надлежащие классы падежей, я вместо этого пытался использовать класс Immutable (например, Flags.Immutable
). Выполнение этого означает, что я должен определить последствия для всех экономных классов, чтобы преобразовать их в NameOfClass.Immutable
, Все идет нормально.
Но у меня есть 2 вещи, которые я, кажется, не понимаю (полную реализацию можно найти в ClassToMap.scala
файл):
(1) При создании карты из экономичных классов неявный Option, похоже, игнорируется.
implicit def hconsToMapRecOption[K <: Symbol, V, R <: HList, T <: HList]
(implicit
wit: Witness.Aux[K],
gen: LabelledGeneric.Aux[V, R],
tmrT: Lazy[ToMapRec[T]],
tmrH: Lazy[ToMapRec[R]]
): ToMapRec[FieldType[K, Option[V]] :: T] = new ToMapRec[FieldType[K, Option[V]] :: T] {
override def apply(l: FieldType[K, Option[V]] :: T): Map[String, Any] = {
tmrT.value(l.tail) + (wit.value.name -> l.head.map(value => tmrH.value(gen.to(value))))
}
}
(2) При работе с экономными союзами мое неявное не используется.
implicit def hconsToMapRecGoatData[K <: Symbol, R <: HList, T <: HList]
(implicit
wit: Witness.Aux[K],
gen: LabelledGeneric.Aux[MyGoat, R],
tmrT: Lazy[ToMapRec[T]],
tmrH: Lazy[ToMapRec[R]]
): ToMapRec[FieldType[K, MyGoatData] :: T] = new ToMapRec[FieldType[K, MyGoatData] :: T] {
override def apply(l: FieldType[K, MyGoatData] :: T): Map[String, Any] = {
tmrT.value(l.tail) + (wit.value.name -> tmrH.value(gen.to(l.head.goat)))
}
}
Я не тратил так много времени, рассматривая карту для реализации класса, но я предполагаю, что она должна отражать класс для отображения.