higherkindness.droste.basis.scala Maven / Gradle / Ivy
Go to download
Show more of this group Show more artifacts with this name
Show all versions of droste-core_sjs1_2.12 Show documentation
Show all versions of droste-core_sjs1_2.12 Show documentation
recursion schemes for cats; to iterate is human, to recurse, divine
The newest version!
package higherkindness.droste
import higherkindness.droste.data.Attr
import higherkindness.droste.data.AttrF
import higherkindness.droste.data.Coattr
import higherkindness.droste.data.CoattrF
import higherkindness.droste.data.Fix
import higherkindness.droste.util.newtypes._
import cats.Eval
import cats.Eq
import cats.Foldable
import cats.Functor
import cats.Monad
import cats.Monoid
import cats.free.Trampoline
import higherkindness.droste.data.list.ListF
import higherkindness.droste.data.list.NilF
import higherkindness.droste.data.list.ConsF
import higherkindness.droste.data.prelude._
trait Embed[F[_], R] {
def algebra: Algebra[F, R]
}
object Embed extends FloatingBasisInstances[Embed] {
def apply[F[_], R](implicit ev: Embed[F, R]): Embed[F, R] = ev
}
trait Project[F[_], R] { self =>
def coalgebra: Coalgebra[F, R]
implicit val tc: Project[F, R] = self
def all(r: R)(p: R => Boolean)(implicit F: Foldable[F]): Boolean =
Project[F, R].all(r)(p)
def any(r: R)(p: R => Boolean)(implicit F: Foldable[F]): Boolean =
Project.any[F, R](r)(p)
def collect[U: Monoid, B](r: R)(
pf: PartialFunction[R, B]
)(implicit U: Basis[ListF[B, *], U], F: Foldable[F]): U =
Project.collect[F, R, U, B](r)(pf)
def contains(r: R, c: R)(implicit R: Eq[R], F: Foldable[F]): Boolean =
Project.contains[F, R](r, c)(tc, R, F)
def foldMap[Z: Monoid](r: R)(f: R => Z)(implicit F: Foldable[F]): Z =
Project.foldMap[F, R, Z](r)(f)
def foldMapM[M[_], Z](r: R)(
f: R => M[Z]
)(implicit M: Monad[M], Z: Monoid[Z], F: Foldable[F]): M[Z] =
Project.foldMapM[F, M, R, Z](r)(f)(tc, M, Z, F)
}
object Project extends FloatingBasisInstances[Project] {
def apply[F[_], R](implicit ev: Project[F, R]): Project[F, R] = ev
def all[F[_], R](r: R)(
p: R => Boolean
)(implicit P: Project[F, R], F: Foldable[F]): Boolean =
foldMap[F, R, Boolean @@ Tags.Conjunction](r)(p(_).conjunction).unwrap
def any[F[_], R](r: R)(
p: R => Boolean
)(implicit P: Project[F, R], F: Foldable[F]): Boolean =
foldMap[F, R, Boolean @@ Tags.Disjunction](r)(p(_).disjunction).unwrap
def collect[F[_], R, U: Monoid, B](r: R)(
pf: PartialFunction[R, B]
)(implicit P: Project[F, R], U: Basis[ListF[B, *], U], F: Foldable[F]): U =
foldMap[F, R, U](r)(
pf.lift(_)
.foldRight[U](U.algebra(NilF))((a, b) => U.algebra(ConsF(a, b)))
)
def contains[F[_], R](r: R, c: R)(implicit
P: Project[F, R],
R: Eq[R],
F: Foldable[F]
): Boolean =
any(r)(R.eqv(c, _))(P, F)
def foldMap[F[_], R, Z: Monoid](r: R)(
f: R => Z
)(implicit P: Project[F, R], F: Foldable[F]): Z =
foldMapM[F, Trampoline, R, Z](r)(x => Trampoline.done(f(x)))(
P,
Monad[Trampoline],
Monoid[Z],
F
).run
def foldMapM[F[_], M[_], R, Z](r: R)(f: R => M[Z])(implicit
P: Project[F, R],
M: Monad[M],
Z: Monoid[Z],
F: Foldable[F]
): M[Z] = {
def loop(z0: Z, term: R): M[Z] =
M.flatMap(f(term)) { z1 =>
F.foldLeftM(P.coalgebra(term), Z.combine(z0, z1))(loop(_, _))
}
loop(Z.empty, r)
}
}
sealed trait Basis[F[_], R] extends Embed[F, R] with Project[F, R]
object Basis extends FloatingBasisInstances[Basis] {
def apply[F[_], R](implicit ev: Basis[F, R]): Basis[F, R] = ev
final case class Default[F[_], R](
algebra: Algebra[F, R],
coalgebra: Coalgebra[F, R]
) extends Basis[F, R]
sealed trait Solve[PR[_[_]]] {
type PatF[F[_], A]
type PatR[F[_]] = PR[F]
}
object Solve extends FloatingBasisSolveInstances {
type Aux[PR[_[_]], PF[_[_], _]] = Solve[PR] {
type PatF[F[_], A] = PF[F, A]
}
}
}
private[droste] sealed trait FloatingBasisInstances[H[F[_], A] >: Basis[F, A]]
extends FloatingBasisInstances0[H] {
implicit def drosteBasisForListF[A]: H[ListF[A, *], List[A]] =
Basis.Default[ListF[A, *], List[A]](
ListF.toScalaListAlgebra,
ListF.fromScalaListCoalgebra
)
implicit def drosteBasisForAttr[F[_], A]: H[AttrF[F, A, *], Attr[F, A]] =
Basis.Default[AttrF[F, A, *], Attr[F, A]](Attr.algebra, Attr.coalgebra)
implicit def drosteBasisForCoattr[F[_], A]: H[
CoattrF[F, A, *],
Coattr[F, A]
] =
Basis
.Default[CoattrF[F, A, *], Coattr[F, A]](Coattr.algebra, Coattr.coalgebra)
}
private[droste] sealed trait FloatingBasisInstances0[
H[F[_], A] >: Basis[F, A]
] {
implicit def drosteBasisForFix[F[_]]: H[F, Fix[F]] =
Basis.Default[F, Fix[F]](Fix.algebra, Fix.coalgebra)
implicit def drosteBasisForCatsCofree[F[_], A]: H[
AttrF[F, A, *],
cats.free.Cofree[F, A]
] =
Basis.Default[AttrF[F, A, *], cats.free.Cofree[F, A]](
Algebra(fa => cats.free.Cofree(fa.ask, Eval.now(fa.lower))),
Coalgebra(a => AttrF(a.head, a.tailForced))
)
implicit def drosteBasisForCatsFree[F[_]: Functor, A]: H[
CoattrF[F, A, *],
cats.free.Free[F, A]
] =
Basis.Default[CoattrF[F, A, *], cats.free.Free[F, A]](
Algebra {
CoattrF.un(_).fold(cats.free.Free.pure, cats.free.Free.roll)
},
Coalgebra {
_.fold[CoattrF[F, A, cats.free.Free[F, A]]](CoattrF.pure, CoattrF.roll)
}
)
}
private[droste] sealed trait FloatingBasisSolveInstances {
import Basis.Solve
implicit val drosteSolveFix: Solve.Aux[Fix, ({ type L[F[_], A] = F[A] })#L] =
null
implicit def drosteSolveAttr[A]: Solve.Aux[
({ type L[F[_]] = Attr[F, A] })#L,
({ type L[F[_], B] = AttrF[F, A, B] })#L
] =
null
implicit def drosteSolveCatsCofree[A]: Solve.Aux[
({ type L[F[_]] = cats.free.Cofree[F, A] })#L,
({ type L[F[_], B] = AttrF[F, A, B] })#L
] =
null
implicit def drosteSolveCatsFree[A]: Solve.Aux[
({ type L[F[_]] = cats.free.Free[F, A] })#L,
({ type L[F[_], B] = CoattrF[F, A, B] })#L
] =
null
}