scalaz.FreeT.scala Maven / Gradle / Ivy
package scalaz
import annotation.tailrec
import FreeT._
object FreeT extends FreeTInstances {
/** Suspend the computation with the given suspension. */
private case class Suspend[S[_], M[_], A](a: M[A \/ S[A]]) extends FreeT[S, M, A]
/** Call a subroutine and continue with the given function. */
private case class Gosub[S[_], M[_], A0, B](a0: FreeT[S, M, A0], f0: A0 => FreeT[S, M, B]) extends FreeT[S, M, B] {
type A = A0
def a: FreeT[S, M, A] = a0
def f: A => FreeT[S, M, B] = f0
}
/** Return the given value in the free monad. */
def point[S[_], M[_], A](value: A)(implicit M: Applicative[M]): FreeT[S, M, A] = Suspend(M.point(-\/(value)))
def suspend[S[_], M[_], A](a: M[A \/ S[FreeT[S, M, A]]])(implicit M: Applicative[M]): FreeT[S, M, A] =
liftM(a).flatMap({
case -\/(a) => point(a)
case \/-(s) => roll(s)
})
def tailrecM[S[_], M[_]: Applicative, A, B](a: A)(f: A => FreeT[S, M, A \/ B]): FreeT[S, M, B] =
f(a).flatMap {
case -\/(a0) => tailrecM(a0)(f)
case \/-(b) => point[S, M, B](b)
}
def liftM[S[_], M[_], A](value: M[A])(implicit M: Functor[M]): FreeT[S, M, A] =
Suspend(M.map(value)(\/.left))
/** A version of `liftM` that infers the nested type constructor. */
def liftMU[S[_], MA](value: MA)(implicit M: Unapply[Functor, MA]): FreeT[S, M.M, M.A] =
liftM[S, M.M, M.A](M(value))(M.TC)
/** Suspends a value within a functor in a single step. Monadic unit for a higher-order monad. */
def liftF[S[_], M[_], A](value: S[A])(implicit M: Applicative[M]): FreeT[S, M, A] =
Suspend(M.point(\/-(value)))
def roll[S[_], M[_], A](value: S[FreeT[S, M, A]])(implicit M: Applicative[M]): FreeT[S, M, A] =
liftF[S, M, FreeT[S, M, A]](value).flatMap(identity)
import Isomorphism._
def isoFree[S[_]]: FreeT[S, Id.Id, *] <~> Free[S, *] =
new IsoFunctorTemplate[FreeT[S, Id.Id, *], Free[S, *]] {
override def to_[A](fa: FreeT[S, Id.Id, A]) = fa match {
case Suspend(\/-(a)) =>
Free.liftF(a)
case Suspend(-\/(a)) =>
Free.point(a)
case a @ Gosub(_, _) =>
to_(a.a).flatMap(a.f.andThen(to_(_)))
}
override def from_[A](ga: Free[S, A]) =
ga.toFreeT
}
}
sealed abstract class FreeT[S[_], M[_], A] {
final def map[B](f: A => B)(implicit M: Applicative[M]): FreeT[S, M, B] =
flatMap(a => point(f(a)))
/** Binds the given continuation to the result of this computation. */
final def flatMap[B](f: A => FreeT[S, M, B]): FreeT[S, M, B] =
Gosub(this, f)
/**
* Changes the underlying `Monad` for this `FreeT`, ie.
* turning this `FreeT[S, M, A]` into a `FreeT[S, N, A]`.
*/
def hoist[N[_]](mn: M ~> N): FreeT[S, N, A] =
step match {
case e @ Gosub(_, _) =>
Gosub(e.a.hoist(mn), e.f.andThen(_.hoist(mn)))
case Suspend(m) =>
Suspend(mn(m))
}
/** Change the base functor `S` for a `FreeT` action. */
def interpret[T[_]](st: S ~> T)(implicit M: Functor[M]): FreeT[T, M, A] =
step match {
case e @ Gosub(_, _) =>
Gosub(e.a.interpret(st), e.f.andThen(_.interpret(st)))
case Suspend(m) =>
Suspend(M.map(m)(_.map(s => st(s))))
}
/**
* Runs to completion, mapping the suspension with the given transformation
* at each step and accumulating into the monad `M`.
*/
def foldMap(f: S ~> M)(implicit M0: BindRec[M], M1: Applicative[M]): M[A] = {
@tailrec
def go(ft: FreeT[S, M, A]): M[FreeT[S, M, A] \/ A] =
ft match {
case Suspend(ma) => M0.bind(ma) {
case -\/(a) => M1.point(\/-(a))
case \/-(sa) => M0.map(f(sa))(\/.right)
}
case g @ Gosub(_, _) => g.a match {
case Suspend(mx) => M0.bind(mx) {
case -\/(x) => M1.point(-\/(g.f(x)))
case \/-(sx) => M0.map(f(sx))(g.f andThen \/.left)
}
case g0 @ Gosub(_, _) => go(g0.a.flatMap(g0.f(_).flatMap(g.f)))
}
}
M0.tailrecM(this)(go)
}
/** Evaluates a single layer of the free monad **/
def resume(implicit S: Functor[S], M0: BindRec[M], M1: Applicative[M]): M[S[FreeT[S, M, A]] \/ A] =
M0.map(resumeC)(_.leftMap(_.run))
def resumeC(implicit M0: BindRec[M], M1: Applicative[M]): M[Coyoneda[S, FreeT[S, M, A]] \/ A] = {
@tailrec
def go(ft: FreeT[S, M, A]): M[FreeT[S, M, A] \/ (Coyoneda[S, FreeT[S, M, A]] \/ A)] =
ft match {
case Suspend(f) => M0.map(f) {
case -\/(a) => \/-(\/-(a))
case \/-(sa) => \/-(-\/(Coyoneda(sa)(point(_))))
}
case g1 @ Gosub(_, _) => g1.a match {
case Suspend(m1) => M0.map(m1) {
case -\/(a) => -\/(g1.f(a))
case \/-(fc) => \/-(-\/(Coyoneda(fc)(g1.f(_))))
}
case g2 @ Gosub(_, _) => go(g2.a.flatMap(g2.f(_).flatMap(g1.f)))
}
}
M0.tailrecM(this)(go)
}
/**
* Runs to completion, using a function that maps the resumption from `S` to a monad `M`.
*/
def runM(interp: S[FreeT[S, M, A]] => M[FreeT[S, M, A]])(implicit S: Functor[S], M0: BindRec[M], M1: Applicative[M]): M[A] =
M0.tailrecM(this)(ft => M0.bind(ft.resume) {
case a @ \/-(_) => M1.point(a.coerceLeft)
case -\/(fc) => M0.map(interp(fc))(\/.left)
})
/**
* Perform recursive binds on `M` until first suspension is reached.
*/
private[scalaz] final def toM(implicit M0: BindRec[M], M: Applicative[M]): M[FreeT[S, M, A]] =
M0.tailrecM(this)(_.step match {
case Suspend(m) => M.map(m) {
case -\/(a) => \/-(point(a))
case \/-(s) => \/-(liftF(s))
}
case g1 @ Gosub(_, _) => g1.a match {
case Suspend(m) => M.map(m) {
case -\/(a) => -\/(g1.f(a))
case \/-(s) => \/-(liftF[S, M, g1.A](s).flatMap(g1.f))
}
case g0 @ Gosub(_, _) => sys.error("Unreachable code: `Gosub` returned from `step` has `Suspend` on the left")
}
})
@tailrec
private def step: FreeT[S, M, A] =
this match {
case g @ Gosub(_, _) => g.a match {
case g0 @ Gosub(_, _) => g0.a.flatMap(a => g0.f(a).flatMap(g.f)).step
case _ => g
}
case x => x
}
}
sealed abstract class FreeTInstances6 {
implicit def freeTMonadTell[S[_], M[_], E](implicit M1: MonadTell[M, E]): MonadTell[FreeT[S, M, *], E] =
new MonadTell[FreeT[S, M, *], E] with FreeTMonad[S, M] {
override def M = implicitly
override def writer[A](w: E, v: A) =
FreeT.liftM(M1.writer(w, v))
}
}
sealed abstract class FreeTInstances5 extends FreeTInstances6 {
implicit def freeTMonadReader[S[_], M[_], E](implicit M1: MonadReader[M, E]): MonadReader[FreeT[S, M, *], E] =
new MonadReader[FreeT[S, M, *], E] with FreeTMonad[S, M] {
override def M = implicitly
override def ask =
FreeT.liftM(M1.ask)
override def local[A](f: E => E)(fa: FreeT[S, M, A]) =
fa.hoist(new (M ~> M){
def apply[A](a: M[A]) = M1.local(f)(a)
})
}
}
sealed abstract class FreeTInstances4 extends FreeTInstances5 {
implicit def freeTMonadState[S[_], M[_], E](implicit M1: MonadState[M, E]): MonadState[FreeT[S, M, *], E] =
new MonadState[FreeT[S, M, *], E] with FreeTMonad[S, M] {
override def M = implicitly
override def get =
FreeT.liftM(M1.get)
override def put(s: E) =
FreeT.liftM(M1.put(s))
}
}
sealed abstract class FreeTInstances3 extends FreeTInstances4 {
implicit def freeTMonadError[S[_], M[_]: BindRec, E](implicit E: MonadError[M, E]): MonadError[FreeT[S, M, *], E] =
new MonadError[FreeT[S, M, *], E] with FreeTMonad[S, M] {
override def M = implicitly
override def handleError[A](fa: FreeT[S, M, A])(f: E => FreeT[S, M, A]) =
FreeT.liftM[S, M, FreeT[S, M, A]](E.handleError(fa.toM)(f.andThen(_.toM)))(M).flatMap(identity)
override def raiseError[A](e: E) =
FreeT.liftM(E.raiseError[A](e))(M)
}
}
sealed abstract class FreeTInstances2 extends FreeTInstances3 {
implicit def freeTBind[S[_], M[_]](implicit M0: Applicative[M]): Bind[FreeT[S, M, *]] =
new FreeTBind[S, M] {
implicit def M: Applicative[M] = M0
}
implicit def freeTHoist[S[_]]: Hoist[({type l[a[_], b] = FreeT[S, a, b]})#l] =
new Hoist[({type l[a[_], b] = FreeT[S, a, b]})#l] {
def hoist[M[_]: Monad, N[_]](f: M ~> N): FreeT[S, M, *] ~> FreeT[S, N, *] =
new (FreeT[S, M, *] ~> FreeT[S, N, *]) {
def apply[A](fa: FreeT[S, M, A]) = fa.hoist(f)
}
def liftM[G[_]: Monad, A](a: G[A]) =
FreeT.liftM(a)
def apply[G[_]: Monad] =
Monad[FreeT[S, G, *]]
}
implicit def freeTFoldable[S[_]: Foldable, M[_]: Foldable: Applicative: BindRec]: Foldable[FreeT[S, M, *]] =
new FreeTFoldable[S, M] {
override def F = implicitly
override def M = implicitly
override def M1 = implicitly
override def M2 = implicitly
}
}
sealed abstract class FreeTInstances1 extends FreeTInstances2 {
implicit def freeTTraverse[S[_]: Traverse, M[_]: Traverse: Applicative: BindRec]: Traverse[FreeT[S, M, *]] =
new FreeTTraverse[S, M] {
override def F = implicitly
override def M = implicitly
override def M1 = implicitly
override def M2 = implicitly
}
}
sealed abstract class FreeTInstances0 extends FreeTInstances1 {
implicit def freeTMonad[S[_], M[_]](implicit M0: Applicative[M]): Monad[FreeT[S, M, *]] & BindRec[FreeT[S, M, *]] =
new FreeTMonad[S, M] {
def M = M0
}
implicit def freeTPlus[S[_], M[_]: Applicative: BindRec: Plus]: Plus[FreeT[S, M, *]] =
new FreeTPlus[S, M] {
override def M = implicitly
override def M1 = implicitly
override def M2 = implicitly
}
}
sealed abstract class FreeTInstances extends FreeTInstances0 {
implicit def freeTMonadPlus[S[_], M[_]: ApplicativePlus: BindRec]: MonadPlus[FreeT[S, M, *]] & Alt[FreeT[S, M, *]] =
new MonadPlus[FreeT[S, M, *]] with Alt[FreeT[S, M, *]] with FreeTPlus[S, M] with FreeTMonad[S, M] {
override def M = implicitly
override def M1 = implicitly
override def M2 = implicitly
override def empty[A] = FreeT.liftM[S, M, A](PlusEmpty[M].empty[A])(M)
override def alt[A](a1: => FreeT[S, M, A], a2: => FreeT[S, M, A]): FreeT[S, M, A] =
plus(a1, a2)
}
}
private trait FreeTBind[S[_], M[_]] extends Bind[FreeT[S, M, *]] {
implicit def M: Applicative[M]
override final def map[A, B](fa: FreeT[S, M, A])(f: A => B): FreeT[S, M, B] = fa.map(f)
def bind[A, B](fa: FreeT[S, M, A])(f: A => FreeT[S, M, B]): FreeT[S, M, B] = fa.flatMap(f)
}
private trait FreeTMonad[S[_], M[_]] extends Monad[FreeT[S, M, *]] with BindRec[FreeT[S, M, *]] with FreeTBind[S, M] {
implicit def M: Applicative[M]
override final def point[A](a: => A) =
FreeT.point[S, M, A](a)
override final def tailrecM[A, B](a: A)(f: A => FreeT[S, M, A \/ B]) =
FreeT.tailrecM(a)(f)
}
private trait FreeTPlus[S[_], M[_]] extends Plus[FreeT[S, M, *]] {
implicit def M: Applicative[M]
implicit def M1: BindRec[M]
def M2: Plus[M]
override final def plus[A](a: FreeT[S, M, A], b: => FreeT[S, M, A]) =
FreeT.liftM(M2.plus(a.toM, b.toM))(M).flatMap(identity)
}
private trait FreeTFoldable[S[_], M[_]] extends Foldable[FreeT[S, M, *]] with Foldable.FromFoldMap[FreeT[S, M, *]] {
implicit def M: Applicative[M]
implicit def M1: BindRec[M]
def F: Foldable[S]
def M2: Foldable[M]
override final def foldMap[A, B: Monoid](fa: FreeT[S, M, A])(f: A => B): B =
M2.foldMap(fa.resumeC){
case -\/(a) =>
F.foldMap(a.fi)(i => foldMap(a.k(i))(f))
case \/-(a) =>
f(a)
}
}
private trait FreeTTraverse[S[_], M[_]] extends Traverse[FreeT[S, M, *]] with FreeTFoldable[S, M] with FreeTBind[S, M] {
override implicit def F: Traverse[S]
override def M2: Traverse[M]
override implicit def M: Applicative[M]
override implicit def M1: BindRec[M]
override final def traverseImpl[G[_], A, B](fa: FreeT[S, M, A])(f: A => G[B])(implicit G: Applicative[G]) =
G.map(
M2.traverseImpl(fa.resume){
case -\/(a) =>
G.map(F.traverseImpl(a)(traverseImpl(_)(f)))(FreeT.roll(_)(M))
case \/-(a) =>
G.map(f(a))(FreeT.point[S, M, B])
}
)(FreeT.liftM(_)(M).flatMap(identity))
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy