All Downloads are FREE. Search and download functionalities are using the official Maven repository.

harness.deriving.K0.scala Maven / Gradle / Ivy

package harness.deriving

import cats.syntax.option.*
import cats.syntax.traverse.*
import scala.compiletime.*
import scala.deriving.Mirror
import scala.reflect.ClassTag

abstract class K0T[UB] {

  type Kind[C, O <: UB] = C {
    type MirroredType = O
    type MirroredMonoType = O
    type MirroredElemTypes <: Tuple
  }

  type Generic[O <: UB] = Kind[Mirror, O]
  type ProductGeneric[O <: UB] = Kind[Mirror.Product, O]
  type SumGeneric[O <: UB] = Kind[Mirror.Sum, O]

  type FlatFieldInstances[T <: Tuple, F[_ <: UB]] <: Tuple =
    T match {
      case EmptyTuple => EmptyTuple
      case x *: xs    => F[x] *: FlatFieldInstances[xs, F]
    }

  type FieldInstances[T <: Tuple, W[_], F[_ <: UB]] <: Tuple =
    T match {
      case EmptyTuple => EmptyTuple
      case x *: xs    => W[F[x]] *: FieldInstances[xs, W, F]
    }

  inline def summonFlatFieldInstances[T <: Tuple, G[_ <: UB]]: List[G[UB]] =
    summonAll[FlatFieldInstances[T, G]].toIArray.toList.asInstanceOf[List[G[UB]]]

  inline def summonFieldInstances[T <: Tuple, F[_], G[_ <: UB]]: List[F[G[UB]]] =
    summonAll[FieldInstances[T, F, G]].toIArray.toList.asInstanceOf[List[F[G[UB]]]]

  final class ProductInstances[F <: UB, T[_ <: UB]](val m: ProductGeneric[F])(
      val ev: m.MirroredMonoType <:< Product,
      val rawInstances: List[LazyDerived[T[UB]]],
  ) {

    lazy val instances: List[T[UB]] = rawInstances.map(_.derived)

    object instantiate {

      def tuple(tuple: m.MirroredElemTypes): F =
        m.asInstanceOf[Mirror.ProductOf[m.MirroredMonoType]].fromTuple(tuple.asInstanceOf)

      def tupleUnsafe(tuple: Tuple): F =
        m.asInstanceOf[Mirror.ProductOf[m.MirroredMonoType]].fromTuple(tuple.asInstanceOf)

      def seqUnsafe(seq: Seq[Any]): F =
        tupleUnsafe(Tuple.fromArray(seq.toArray[Any]))

    }

    final case class withInstance(a: F) {

      private def productElements: List[UB] = ev(a).productIterator.toList.asInstanceOf[List[UB]]

      object map {

        def apply[B](f: [t <: UB] => (T[t], t) => B): List[B] =
          productElements.zip(instances).map { case (b, i) => f(i, b) }

        def withLabels[B](labels: Labelling[F])(f: [t <: UB] => (String, T[t], t) => B): List[B] =
          labels.elemLabels.zip(productElements).zip(instances).map { case ((l, b), i) => f(l, i, b) }

      }

      object foldLeft {

        def apply[R](z: R)(f: [t <: UB] => (R, T[t], t) => R): R =
          productElements.zip(instances).foldLeft(z) { case (acc, (b, i)) => f(acc, i, b) }

        def withLabels[R](labels: Labelling[F], z: R)(f: [t <: UB] => (R, String, T[t], t) => R): R =
          labels.elemLabels.zip(productElements).zip(instances).foldLeft(z) { case (acc, ((l, b), i)) => f(acc, l, i, b) }

      }

    }

    object withoutInstance {

      object foldLeft {

        def apply[R](z: R)(f: [t <: UB] => (R, T[t]) => R): R =
          instances.foldLeft(z) { case (acc, i) => f(acc, i) }

        def withLabels[R](labels: Labelling[F], z: R)(f: [t <: UB] => (R, String, T[t]) => R): R =
          labels.elemLabels.zip(instances).foldLeft(z) { case (acc, (l, i)) => f(acc, l, i) }

      }

      object mapInstantiate {

        def apply(f: [t <: UB] => T[t] => t): F =
          instantiate.seqUnsafe(instances.map { i => f(i) })

        def withLabels(labels: Labelling[F])(f: [t <: UB] => (String, T[t]) => t): F =
          instantiate.seqUnsafe(labels.elemLabels.zip(instances).map { case (l, i) => f(l, i) })

      }

      object mapInstantiateEither {

        def apply[L](f: [t <: UB] => T[t] => Either[L, t]): Either[L, F] =
          instances.traverse { i => f(i) }.map(instantiate.seqUnsafe)

        def withLabels[L](labels: Labelling[F])(f: [t <: UB] => (String, T[t]) => Either[L, t]): Either[L, F] =
          labels.elemLabels.zip(instances).traverse { case (l, i) => f(l, i) }.map(instantiate.seqUnsafe)

      }

    }

  }
  object ProductInstances {
    inline given of[F <: UB, T[_ <: UB]](using m: ProductGeneric[F]): ProductInstances[F, T] =
      new ProductInstances[F, T](m)(
        summonInline[m.MirroredMonoType <:< Product],
        summonFieldInstances[m.MirroredElemTypes, LazyDerived, T],
      )
  }

  final class SumInstances[F <: UB, T[_ <: UB]](val m: SumGeneric[F])(
      val children: List[T[UB]],
  ) {

    def narrow[G[B <: UB] <: T[B]](implicit fCt: ClassTag[T[m.MirroredType]], gCt: ClassTag[G[m.MirroredType]]): SumInstances[F, G] =
      new SumInstances[F, G](m)(
        children.asInstanceOf[List[Matchable]].map {
          case gCt(c) => c.asInstanceOf[G[UB]]
          case other  => throw new RuntimeException(s"Unable to narrow ${fCt.runtimeClass.getName} to ${gCt.runtimeClass.getName} ($other)")
        },
      )

    final case class withInstance(a: F) {
      val ord: Int = m.ordinal(a)

      object use {

        def apply[B](f: [t <: F] => (i: T[t], t: t) => B): B =
          f[F](children(ord).asInstanceOf[T[F]], a)

        def withLabels[B](labels: Labelling[F])(f: [t <: F] => (l: String, i: T[t], t: t) => B): B =
          f(labels.elemLabels(ord), children(ord).asInstanceOf, a.asInstanceOf)

      }

    }

    def instanceFromLabels(labels: Labelling[F], label: String): Option[T[F]] =
      labels.elemLabels.indexOf(label) match {
        case -1 => None
        case i  => children(i).asInstanceOf[T[F]].some
      }

  }
  object SumInstances {
    inline given of[A <: UB, F[_ <: UB]](using m: SumGeneric[A]): SumInstances[A, F] =
      new SumInstances[A, F](m)(
        summonFieldInstances[m.MirroredElemTypes, Derived, F].map(_.derived),
      )
  }

  trait Derivable[T[_ <: UB]] {

    inline implicit def genProduct[F <: UB](implicit m: ProductGeneric[F]): Derived[T[F]]

    inline implicit def genSum[F <: UB](implicit m: SumGeneric[F]): Derived[T[F]]

    inline final def summonOrDerive[F <: UB](using m: Generic[F]): T[F] =
      compiletime.summonFrom {
        case t: T[F] => t
        case _       => derived[F]
      }

    inline final def derived[F <: UB](using m: Generic[F]): T[F] =
      inline m match {
        case m: ProductGeneric[F] => genProduct[F](using m).derived
        case m: SumGeneric[F]     => genSum[F](using m).derived
      }

  }

}

object K0 extends K0T[Any] {

  type UnionGeneric[A] = UnionMirror[A]
  type IntersectionGeneric[A] = IntersectionMirror[A]

  trait DerivableUnion[T[_]] {

    protected inline def foldUnion[A, B](ta: T[A], tb: T[B]): T[A | B]

    final inline def deriveUnion[A](implicit ev: UnionGeneric[A]): T[A] =
      summonFlatFieldInstances[ev.ElementTypes, T]
        .reduceLeft { (a, b) => foldUnion(a.asInstanceOf[T[Any]], b.asInstanceOf[T[Any]]) }
        .asInstanceOf[T[A]]

  }
  object DerivableUnion {

    trait Auto[T[_]] extends DerivableUnion[T] {
      final inline implicit def deriveUnionAuto[A](implicit ev: UnionGeneric[A]): T[A] = deriveUnion[A]
    }

  }

  trait DerivableIntersection[T[_]] {

    protected inline def foldIntersection[A, B](ta: T[A], tb: T[B]): T[A & B]

    final inline def deriveIntersection[A](implicit ev: IntersectionGeneric[A]): T[A] =
      summonFlatFieldInstances[ev.ElementTypes, T]
        .reduceLeft { (a, b) => foldIntersection(a.asInstanceOf[T[Any]], b.asInstanceOf[T[Any]]) }
        .asInstanceOf[T[A]]

  }
  object DerivableIntersection {

    trait Auto[T[_]] extends DerivableIntersection[T] {
      final inline implicit def deriveIntersectionAuto[A](implicit ev: IntersectionGeneric[A]): T[A] = deriveIntersection[A]
    }

  }

}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy