{-# LANGUAGE FlexibleContexts, FlexibleInstances, GeneralizedNewtypeDeriving, KindSignatures, MultiParamTypeClasses #-}
module Control.Effect.Lift
( Lift(..)
, sendM
, runM
, LiftC(..)
) where
import Control.Applicative (Alternative(..))
import Control.Effect.Carrier
import Control.Effect.Sum
import Control.Monad (MonadPlus(..))
import Control.Monad.Fail
import Control.Monad.IO.Class
import Control.Monad.IO.Unlift
import Control.Monad.Trans.Class
import Data.Coerce
newtype Lift sig (m :: * -> *) k = Lift { unLift :: sig k }
deriving (Functor)
instance Functor sig => HFunctor (Lift sig) where
hmap _ = coerce
{-# INLINE hmap #-}
instance Functor sig => Effect (Lift sig) where
handle state handler (Lift op) = Lift (fmap (handler . (<$ state)) op)
runM :: LiftC m a -> m a
runM = runLiftC
sendM :: (Member (Lift n) sig, Carrier sig m, Functor n) => n a -> m a
sendM = send . Lift . fmap pure
newtype LiftC m a = LiftC { runLiftC :: m a }
deriving (Alternative, Applicative, Functor, Monad, MonadFail, MonadIO, MonadPlus)
instance MonadTrans LiftC where
lift = LiftC
instance Monad m => Carrier (Lift m) (LiftC m) where
eff = LiftC . (>>= runLiftC) . unLift
instance MonadUnliftIO m => MonadUnliftIO (LiftC m) where
askUnliftIO = LiftC $ withUnliftIO $ \u -> return (UnliftIO (unliftIO u . runLiftC))
{-# INLINE askUnliftIO #-}
withRunInIO inner = LiftC $ withRunInIO $ \run -> inner (run . runLiftC)
{-# INLINE withRunInIO #-}