{-# LANGUAGE LambdaCase #-}
{-# LANGUAGE FlexibleInstances #-}
{-# LANGUAGE MultiParamTypeClasses #-}
{-# LANGUAGE FlexibleContexts #-}
{-# LANGUAGE TypeFamilies #-}
{-# LANGUAGE UndecidableInstances #-}
{-# LANGUAGE ScopedTypeVariables #-}
{-# language Safe #-}
module Control.Monad.Trans.Can
(
CanT(..)
, mapCanT
) where
import Data.Can
import Control.Applicative (liftA2)
import Control.Monad.Writer
import Control.Monad.Reader
import Control.Monad.State.Class
import Control.Monad.Except
import Control.Monad.RWS
newtype CanT a m b = CanT { CanT a m b -> m (Can a b)
runCanT :: m (Can a b) }
mapCanT :: (m (Can a b) -> n (Can c d)) -> CanT a m b -> CanT c n d
mapCanT :: (m (Can a b) -> n (Can c d)) -> CanT a m b -> CanT c n d
mapCanT m (Can a b) -> n (Can c d)
f = n (Can c d) -> CanT c n d
forall a (m :: * -> *) b. m (Can a b) -> CanT a m b
CanT (n (Can c d) -> CanT c n d)
-> (CanT a m b -> n (Can c d)) -> CanT a m b -> CanT c n d
forall b c a. (b -> c) -> (a -> b) -> a -> c
. m (Can a b) -> n (Can c d)
f (m (Can a b) -> n (Can c d))
-> (CanT a m b -> m (Can a b)) -> CanT a m b -> n (Can c d)
forall b c a. (b -> c) -> (a -> b) -> a -> c
. CanT a m b -> m (Can a b)
forall a (m :: * -> *) b. CanT a m b -> m (Can a b)
runCanT
instance Functor f => Functor (CanT a f) where
fmap :: (a -> b) -> CanT a f a -> CanT a f b
fmap a -> b
f = f (Can a b) -> CanT a f b
forall a (m :: * -> *) b. m (Can a b) -> CanT a m b
CanT (f (Can a b) -> CanT a f b)
-> (CanT a f a -> f (Can a b)) -> CanT a f a -> CanT a f b
forall b c a. (b -> c) -> (a -> b) -> a -> c
. (Can a a -> Can a b) -> f (Can a a) -> f (Can a b)
forall (f :: * -> *) a b. Functor f => (a -> b) -> f a -> f b
fmap ((a -> b) -> Can a a -> Can a b
forall (f :: * -> *) a b. Functor f => (a -> b) -> f a -> f b
fmap a -> b
f) (f (Can a a) -> f (Can a b))
-> (CanT a f a -> f (Can a a)) -> CanT a f a -> f (Can a b)
forall b c a. (b -> c) -> (a -> b) -> a -> c
. CanT a f a -> f (Can a a)
forall a (m :: * -> *) b. CanT a m b -> m (Can a b)
runCanT
instance (Semigroup a, Applicative f) => Applicative (CanT a f) where
pure :: a -> CanT a f a
pure = f (Can a a) -> CanT a f a
forall a (m :: * -> *) b. m (Can a b) -> CanT a m b
CanT (f (Can a a) -> CanT a f a)
-> (a -> f (Can a a)) -> a -> CanT a f a
forall b c a. (b -> c) -> (a -> b) -> a -> c
. Can a a -> f (Can a a)
forall (f :: * -> *) a. Applicative f => a -> f a
pure (Can a a -> f (Can a a)) -> (a -> Can a a) -> a -> f (Can a a)
forall b c a. (b -> c) -> (a -> b) -> a -> c
. a -> Can a a
forall (f :: * -> *) a. Applicative f => a -> f a
pure
CanT f (Can a (a -> b))
f <*> :: CanT a f (a -> b) -> CanT a f a -> CanT a f b
<*> CanT f (Can a a)
a = f (Can a b) -> CanT a f b
forall a (m :: * -> *) b. m (Can a b) -> CanT a m b
CanT (f (Can a b) -> CanT a f b) -> f (Can a b) -> CanT a f b
forall a b. (a -> b) -> a -> b
$ (Can a (a -> b) -> Can a a -> Can a b)
-> f (Can a (a -> b)) -> f (Can a a) -> f (Can a b)
forall (f :: * -> *) a b c.
Applicative f =>
(a -> b -> c) -> f a -> f b -> f c
liftA2 Can a (a -> b) -> Can a a -> Can a b
forall (f :: * -> *) a b. Applicative f => f (a -> b) -> f a -> f b
(<*>) f (Can a (a -> b))
f f (Can a a)
a
instance (Semigroup a, Monad m) => Monad (CanT a m) where
return :: a -> CanT a m a
return = a -> CanT a m a
forall (f :: * -> *) a. Applicative f => a -> f a
pure
CanT m (Can a a)
m >>= :: CanT a m a -> (a -> CanT a m b) -> CanT a m b
>>= a -> CanT a m b
k = m (Can a b) -> CanT a m b
forall a (m :: * -> *) b. m (Can a b) -> CanT a m b
CanT (m (Can a b) -> CanT a m b) -> m (Can a b) -> CanT a m b
forall a b. (a -> b) -> a -> b
$ do
Can a a
c <- m (Can a a)
m
case Can a a
c of
Eno a
a -> CanT a m b -> m (Can a b)
forall a (m :: * -> *) b. CanT a m b -> m (Can a b)
runCanT (CanT a m b -> m (Can a b)) -> CanT a m b -> m (Can a b)
forall a b. (a -> b) -> a -> b
$ a -> CanT a m b
k a
a
Two a
a a
b -> do
Can a b
c' <- CanT a m b -> m (Can a b)
forall a (m :: * -> *) b. CanT a m b -> m (Can a b)
runCanT (CanT a m b -> m (Can a b)) -> CanT a m b -> m (Can a b)
forall a b. (a -> b) -> a -> b
$ a -> CanT a m b
k a
b
Can a b -> m (Can a b)
forall (m :: * -> *) a. Monad m => a -> m a
return (Can a b -> m (Can a b)) -> Can a b -> m (Can a b)
forall a b. (a -> b) -> a -> b
$ case Can a b
c' of
Eno b
b' -> a -> b -> Can a b
forall a b. a -> b -> Can a b
Two a
a b
b'
Two a
a' b
b' -> a -> b -> Can a b
forall a b. a -> b -> Can a b
Two (a
a a -> a -> a
forall a. Semigroup a => a -> a -> a
<> a
a') b
b'
Can a b
_ -> Can a b
c'
One a
a -> Can a b -> m (Can a b)
forall (m :: * -> *) a. Monad m => a -> m a
return (Can a b -> m (Can a b)) -> Can a b -> m (Can a b)
forall a b. (a -> b) -> a -> b
$ a -> Can a b
forall a b. a -> Can a b
One a
a
Can a a
Non -> Can a b -> m (Can a b)
forall (m :: * -> *) a. Monad m => a -> m a
return Can a b
forall a b. Can a b
Non
instance (Semigroup a, MonadWriter w m) => MonadWriter w (CanT a m) where
tell :: w -> CanT a m ()
tell = m () -> CanT a m ()
forall (t :: (* -> *) -> * -> *) (m :: * -> *) a.
(MonadTrans t, Monad m) =>
m a -> t m a
lift (m () -> CanT a m ()) -> (w -> m ()) -> w -> CanT a m ()
forall b c a. (b -> c) -> (a -> b) -> a -> c
. w -> m ()
forall w (m :: * -> *). MonadWriter w m => w -> m ()
tell
listen :: CanT a m a -> CanT a m (a, w)
listen (CanT m (Can a a)
m) = m (Can a (a, w)) -> CanT a m (a, w)
forall a (m :: * -> *) b. m (Can a b) -> CanT a m b
CanT (m (Can a (a, w)) -> CanT a m (a, w))
-> m (Can a (a, w)) -> CanT a m (a, w)
forall a b. (a -> b) -> a -> b
$ (Can a a, w) -> Can a (a, w)
forall a a b. (Can a a, b) -> Can a (a, b)
go ((Can a a, w) -> Can a (a, w))
-> m (Can a a, w) -> m (Can a (a, w))
forall (f :: * -> *) a b. Functor f => (a -> b) -> f a -> f b
<$> m (Can a a) -> m (Can a a, w)
forall w (m :: * -> *) a. MonadWriter w m => m a -> m (a, w)
listen m (Can a a)
m where
go :: (Can a a, b) -> Can a (a, b)
go (Can a a
c,b
w) = case Can a a
c of
Can a a
Non -> Can a (a, b)
forall a b. Can a b
Non
One a
a -> a -> Can a (a, b)
forall a b. a -> Can a b
One a
a
Eno a
b -> (a, b) -> Can a (a, b)
forall a b. b -> Can a b
Eno (a
b,b
w)
Two a
a a
b -> a -> (a, b) -> Can a (a, b)
forall a b. a -> b -> Can a b
Two a
a (a
b, b
w)
pass :: CanT a m (a, w -> w) -> CanT a m a
pass (CanT m (Can a (a, w -> w))
m) = m (Can a a) -> CanT a m a
forall a (m :: * -> *) b. m (Can a b) -> CanT a m b
CanT (m (Can a a) -> CanT a m a) -> m (Can a a) -> CanT a m a
forall a b. (a -> b) -> a -> b
$ m (Can a a, w -> w) -> m (Can a a)
forall w (m :: * -> *) a. MonadWriter w m => m (a, w -> w) -> m a
pass (Can a (a, w -> w) -> (Can a a, w -> w)
forall a b a. Can a (b, a -> a) -> (Can a b, a -> a)
go (Can a (a, w -> w) -> (Can a a, w -> w))
-> m (Can a (a, w -> w)) -> m (Can a a, w -> w)
forall (f :: * -> *) a b. Functor f => (a -> b) -> f a -> f b
<$> m (Can a (a, w -> w))
m) where
go :: Can a (b, a -> a) -> (Can a b, a -> a)
go = \case
Can a (b, a -> a)
Non -> (Can a b
forall a b. Can a b
Non, a -> a
forall a. a -> a
id)
One a
a -> (a -> Can a b
forall a b. a -> Can a b
One a
a, a -> a
forall a. a -> a
id)
Eno (b
a,a -> a
f) -> (b -> Can a b
forall a b. b -> Can a b
Eno b
a, a -> a
f)
Two a
w (b
a,a -> a
f) -> (a -> b -> Can a b
forall a b. a -> b -> Can a b
Two a
w b
a, a -> a
f)
instance (Semigroup a, MonadReader r m) => MonadReader r (CanT a m) where
ask :: CanT a m r
ask = m r -> CanT a m r
forall (t :: (* -> *) -> * -> *) (m :: * -> *) a.
(MonadTrans t, Monad m) =>
m a -> t m a
lift m r
forall r (m :: * -> *). MonadReader r m => m r
ask
local :: (r -> r) -> CanT a m a -> CanT a m a
local r -> r
f (CanT m (Can a a)
m) = m (Can a a) -> CanT a m a
forall a (m :: * -> *) b. m (Can a b) -> CanT a m b
CanT ((r -> r) -> m (Can a a) -> m (Can a a)
forall r (m :: * -> *) a. MonadReader r m => (r -> r) -> m a -> m a
local r -> r
f m (Can a a)
m)
instance (MonadState s m, Semigroup t) => MonadState s (CanT t m) where
get :: CanT t m s
get = m s -> CanT t m s
forall (t :: (* -> *) -> * -> *) (m :: * -> *) a.
(MonadTrans t, Monad m) =>
m a -> t m a
lift m s
forall s (m :: * -> *). MonadState s m => m s
get
put :: s -> CanT t m ()
put = m () -> CanT t m ()
forall (t :: (* -> *) -> * -> *) (m :: * -> *) a.
(MonadTrans t, Monad m) =>
m a -> t m a
lift (m () -> CanT t m ()) -> (s -> m ()) -> s -> CanT t m ()
forall b c a. (b -> c) -> (a -> b) -> a -> c
. s -> m ()
forall s (m :: * -> *). MonadState s m => s -> m ()
put
instance (Semigroup t, MonadRWS r w s m) => MonadRWS r w s (CanT t m)
instance MonadTrans (CanT a) where
lift :: m a -> CanT a m a
lift = m (Can a a) -> CanT a m a
forall a (m :: * -> *) b. m (Can a b) -> CanT a m b
CanT (m (Can a a) -> CanT a m a)
-> (m a -> m (Can a a)) -> m a -> CanT a m a
forall b c a. (b -> c) -> (a -> b) -> a -> c
. (a -> Can a a) -> m a -> m (Can a a)
forall (f :: * -> *) a b. Functor f => (a -> b) -> f a -> f b
fmap a -> Can a a
forall a b. b -> Can a b
Eno
instance (MonadError e m, Semigroup e) => MonadError e (CanT e m) where
throwError :: e -> CanT e m a
throwError = m a -> CanT e m a
forall (t :: (* -> *) -> * -> *) (m :: * -> *) a.
(MonadTrans t, Monad m) =>
m a -> t m a
lift (m a -> CanT e m a) -> (e -> m a) -> e -> CanT e m a
forall b c a. (b -> c) -> (a -> b) -> a -> c
. e -> m a
forall e (m :: * -> *) a. MonadError e m => e -> m a
throwError
catchError :: CanT e m a -> (e -> CanT e m a) -> CanT e m a
catchError (CanT m (Can e a)
m) e -> CanT e m a
f = m (Can e a) -> CanT e m a
forall a (m :: * -> *) b. m (Can a b) -> CanT a m b
CanT (m (Can e a) -> CanT e m a) -> m (Can e a) -> CanT e m a
forall a b. (a -> b) -> a -> b
$ m (Can e a) -> (e -> m (Can e a)) -> m (Can e a)
forall e (m :: * -> *) a.
MonadError e m =>
m a -> (e -> m a) -> m a
catchError m (Can e a)
m (CanT e m a -> m (Can e a)
forall a (m :: * -> *) b. CanT a m b -> m (Can a b)
runCanT (CanT e m a -> m (Can e a))
-> (e -> CanT e m a) -> e -> m (Can e a)
forall b c a. (b -> c) -> (a -> b) -> a -> c
. e -> CanT e m a
f)