{-# LANGUAGE RankNTypes #-} -- | Please see the README.md file for information on using this -- package at <https://www.stackage.org/package/unliftio-core>. module Control.Monad.IO.Unlift ( MonadUnliftIO (..) , UnliftIO (..) , askRunInIO , withUnliftIO , toIO , MonadIO (..) ) where import Control.Monad (liftM) import Control.Monad.IO.Class import Control.Monad.Trans.Reader (ReaderT (..)) import Control.Monad.Trans.Identity (IdentityT (..)) -- | The ability to run any monadic action @m a@ as @IO a@. -- -- This is more precisely a natural transformation. We need to new -- datatype (instead of simply using a @forall@) due to lack of -- support in GHC for impredicative types. -- -- @since 0.1.0.0 newtype UnliftIO m = UnliftIO { unliftIO :: forall a. m a -> IO a } -- | Monads which allow their actions to be run in 'IO'. -- -- While 'MonadIO' allows an 'IO' action to be lifted into another -- monad, this class captures the opposite concept: allowing you to -- capture the monadic context. Note that, in order to meet the laws -- given below, the intuition is that a monad must have no monadic -- state, but may have monadic context. This essentially limits -- 'MonadUnliftIO' to 'ReaderT' and 'IdentityT' transformers on top of -- 'IO'. -- -- Laws. For any value @u@ returned by 'askUnliftIO', it must meet the -- monad transformer laws as reformulated for @MonadUnliftIO@: -- -- * @unliftIO u . return = return@ -- -- * @unliftIO u (m >>= f) = unliftIO u m >>= unliftIO u . f@ -- -- The third is a currently nameless law which ensures that the -- current context is preserved. -- -- * @askUnliftIO >>= (\u -> liftIO (unliftIO u m)) = m@ -- -- If you have a name for this, please submit it in a pull request for -- great glory. -- -- @since 0.1.0.0 class MonadIO m => MonadUnliftIO m where {-# MINIMAL askUnliftIO | withRunInIO #-} -- | Capture the current monadic context, providing the ability to -- run monadic actions in 'IO'. -- -- See 'UnliftIO' for an explanation of why we need a helper -- datatype here. -- @since 0.1.0.0 askUnliftIO :: m (UnliftIO m) askUnliftIO = withRunInIO (\run -> return (UnliftIO run)) {-# INLINE askUnliftIO #-} -- Would be better, but GHC hates us -- askUnliftIO :: m (forall a. m a -> IO a) -- | Convenience function for capturing the monadic context and running an 'IO' -- action with a runner function. The runner function is used to run a monadic -- action @m@ in @IO@. -- -- @since 0.1.0.0 {-# INLINE withRunInIO #-} withRunInIO :: ((forall a. m a -> IO a) -> IO b) -> m b withRunInIO inner = askUnliftIO >>= \u -> liftIO (inner (unliftIO u)) instance MonadUnliftIO IO where {-# INLINE askUnliftIO #-} askUnliftIO = return (UnliftIO id) {-# INLINE withRunInIO #-} withRunInIO inner = inner id instance MonadUnliftIO m => MonadUnliftIO (ReaderT r m) where {-# INLINE askUnliftIO #-} askUnliftIO = ReaderT $ \r -> withUnliftIO $ \u -> return (UnliftIO (unliftIO u . flip runReaderT r)) {-# INLINE withRunInIO #-} withRunInIO inner = ReaderT $ \r -> withRunInIO $ \run -> inner (run . flip runReaderT r) instance MonadUnliftIO m => MonadUnliftIO (IdentityT m) where {-# INLINE askUnliftIO #-} askUnliftIO = IdentityT $ withUnliftIO $ \u -> return (UnliftIO (unliftIO u . runIdentityT)) {-# INLINE withRunInIO #-} withRunInIO inner = IdentityT $ withRunInIO $ \run -> inner (run . runIdentityT) -- | Same ask 'askUnliftIO', but returns a monomorphic function -- instead of a polymorphic newtype wrapper. If you only need to apply -- the transformation on one concrete type, this function can be more -- convenient. -- -- @since 0.1.0.0 {-# INLINE askRunInIO #-} askRunInIO :: MonadUnliftIO m => m (m a -> IO a) -- withRunInIO return would be nice, but GHC 7.8.4 doesn't like it askRunInIO = withRunInIO (\run -> (return (\ma -> run ma))) -- | Convenience function for capturing the monadic context and running -- an 'IO' action. The 'UnliftIO' newtype wrapper is rarely needed, so -- prefer 'withRunInIO' to this function. -- -- @since 0.1.0.0 {-# INLINE withUnliftIO #-} withUnliftIO :: MonadUnliftIO m => (UnliftIO m -> IO a) -> m a withUnliftIO inner = askUnliftIO >>= liftIO . inner -- | Convert an action in @m@ to an action in @IO@. -- -- @since 0.1.0.0 {-# INLINE toIO #-} toIO :: MonadUnliftIO m => m a -> m (IO a) toIO m = withRunInIO $ \run -> return $ run m