2021-01-21 15:24:40 +03:00
|
|
|
||| Note: The difference to a 'strict' Writer implementation is
|
2021-01-19 16:34:15 +03:00
|
|
|
||| that accumulation of values does not happen in the
|
|
|
|
||| Applicative and Monad instances but when invoking `Writer`-specific
|
|
|
|
||| functions like `writer` or `listen`.
|
2021-01-19 13:45:54 +03:00
|
|
|
module Control.Monad.Writer.CPS
|
2021-01-19 16:34:15 +03:00
|
|
|
|
|
|
|
import Control.Monad.Identity
|
|
|
|
import Control.Monad.Trans
|
|
|
|
|
2021-06-09 01:05:10 +03:00
|
|
|
%default total
|
|
|
|
|
2021-01-19 16:34:15 +03:00
|
|
|
||| A writer monad parameterized by:
|
2021-01-19 19:53:42 +03:00
|
|
|
|||
|
2021-01-19 16:34:15 +03:00
|
|
|
||| @w the output to accumulate.
|
2021-01-19 19:53:42 +03:00
|
|
|
|||
|
2021-01-21 15:25:48 +03:00
|
|
|
||| @m The inner monad.
|
2021-01-19 19:53:42 +03:00
|
|
|
|||
|
2021-01-19 16:34:15 +03:00
|
|
|
||| The `pure` function produces the output `neutral`, while `>>=`
|
|
|
|
||| combines the outputs of the subcomputations using `<+>`.
|
|
|
|
public export
|
|
|
|
record WriterT (w : Type) (m : Type -> Type) (a : Type) where
|
|
|
|
constructor MkWriterT
|
|
|
|
unWriterT : w -> m (a, w)
|
|
|
|
|
|
|
|
||| Construct an writer computation from a (result,output) computation.
|
|
|
|
||| (The inverse of `runWriterT`.)
|
|
|
|
public export %inline
|
2021-10-17 01:32:16 +03:00
|
|
|
writerT : Semigroup w => Functor m => m (a, w) -> WriterT w m a
|
2021-01-19 16:34:15 +03:00
|
|
|
writerT f = MkWriterT $ \w => (\(a,w') => (a,w <+> w')) <$> f
|
|
|
|
|
|
|
|
||| Unwrap a writer computation.
|
|
|
|
||| (The inverse of 'writerT'.)
|
|
|
|
public export %inline
|
|
|
|
runWriterT : Monoid w => WriterT w m a -> m (a,w)
|
|
|
|
runWriterT m = unWriterT m neutral
|
|
|
|
|
|
|
|
||| Extract the output from a writer computation.
|
|
|
|
public export %inline
|
2021-10-17 01:32:16 +03:00
|
|
|
execWriterT : Monoid w => Functor m => WriterT w m a -> m w
|
2021-01-19 16:34:15 +03:00
|
|
|
execWriterT = map snd . runWriterT
|
|
|
|
|
|
|
|
||| Map both the return value and output of a computation using
|
|
|
|
||| the given function.
|
|
|
|
public export %inline
|
|
|
|
mapWriterT : (Functor n, Monoid w, Semigroup w')
|
|
|
|
=> (m (a, w) -> n (b, w')) -> WriterT w m a -> WriterT w' n b
|
2021-08-10 21:24:32 +03:00
|
|
|
mapWriterT f m = MkWriterT $ \w =>
|
2021-01-19 16:34:15 +03:00
|
|
|
(\(a,w') => (a,w <+> w')) <$> f (runWriterT m)
|
|
|
|
|
|
|
|
--------------------------------------------------------------------------------
|
|
|
|
-- Writer Functions
|
|
|
|
--------------------------------------------------------------------------------
|
|
|
|
|
|
|
|
||| The `return` function produces the output `neutral`, while `>>=`
|
|
|
|
||| combines the outputs of the subcomputations using `<+>`.
|
|
|
|
public export
|
|
|
|
Writer : Type -> Type -> Type
|
|
|
|
Writer w = WriterT w Identity
|
|
|
|
|
|
|
|
||| Unwrap a writer computation as a (result, output) pair.
|
|
|
|
public export %inline
|
|
|
|
runWriter : Monoid w => Writer w a -> (a, w)
|
|
|
|
runWriter = runIdentity . runWriterT
|
|
|
|
|
|
|
|
||| Extract the output from a writer computation.
|
|
|
|
public export %inline
|
|
|
|
execWriter : Monoid w => Writer w a -> w
|
|
|
|
execWriter = runIdentity . execWriterT
|
|
|
|
|
|
|
|
||| Map both the return value and output of a computation using
|
|
|
|
||| the given function.
|
|
|
|
public export %inline
|
|
|
|
mapWriter : (Monoid w, Semigroup w')
|
|
|
|
=> ((a, w) -> (b, w')) -> Writer w a -> Writer w' b
|
2021-08-10 21:24:32 +03:00
|
|
|
mapWriter f = mapWriterT $ \(Id p) => Id (f p)
|
2021-01-19 16:34:15 +03:00
|
|
|
|
|
|
|
--------------------------------------------------------------------------------
|
|
|
|
-- Implementations
|
|
|
|
--------------------------------------------------------------------------------
|
|
|
|
|
|
|
|
public export %inline
|
|
|
|
Functor m => Functor (WriterT w m) where
|
2021-08-10 21:24:32 +03:00
|
|
|
map f m = MkWriterT $ \w => (\(a,w') => (f a,w')) <$> unWriterT m w
|
2021-01-19 16:34:15 +03:00
|
|
|
|
|
|
|
public export %inline
|
|
|
|
Monad m => Applicative (WriterT w m) where
|
2021-08-10 21:24:32 +03:00
|
|
|
pure a = MkWriterT $ \w => pure (a,w)
|
2021-01-19 16:34:15 +03:00
|
|
|
MkWriterT mf <*> MkWriterT mx =
|
2021-08-10 21:24:32 +03:00
|
|
|
MkWriterT $ \w => do (f,w1) <- mf w
|
|
|
|
(a,w2) <- mx w1
|
|
|
|
pure (f a,w2)
|
2021-01-19 16:34:15 +03:00
|
|
|
|
|
|
|
public export %inline
|
|
|
|
(Monad m, Alternative m) => Alternative (WriterT w m) where
|
2021-08-10 21:24:32 +03:00
|
|
|
empty = MkWriterT $ \_ => empty
|
2021-08-10 14:44:51 +03:00
|
|
|
MkWriterT m <|> mn = MkWriterT $ \w => m w <|> unWriterT mn w
|
2021-01-19 16:34:15 +03:00
|
|
|
|
|
|
|
public export %inline
|
|
|
|
Monad m => Monad (WriterT w m) where
|
2021-08-10 21:24:32 +03:00
|
|
|
m >>= k = MkWriterT $ \w => do (a,w1) <- unWriterT m w
|
|
|
|
unWriterT (k a) w1
|
2021-01-19 16:34:15 +03:00
|
|
|
|
|
|
|
public export %inline
|
|
|
|
MonadTrans (WriterT w) where
|
2021-08-10 21:24:32 +03:00
|
|
|
lift m = MkWriterT $ \w => map (\a => (a,w)) m
|
2021-01-19 16:34:15 +03:00
|
|
|
|
|
|
|
public export %inline
|
|
|
|
HasIO m => HasIO (WriterT w m) where
|
|
|
|
liftIO = lift . liftIO
|