{-# LANGUAGE DeriveFunctor, ExistentialQuantification, FlexibleContexts, FlexibleInstances, LambdaCase, MultiParamTypeClasses, RankNTypes, StandaloneDeriving, TypeOperators, UndecidableInstances #-}
module Control.Effect.Resource
( Resource(..)
, bracket
, bracketOnError
, finally
, onException
, runResource
, ResourceC(..)
) where
import Control.Effect.Carrier
import Control.Effect.Internal
import Control.Effect.Sum
import qualified Control.Exception as Exc
import Control.Monad.IO.Class
data Resource m k
= forall resource any output . Resource (m resource) (resource -> m any) (resource -> m output) (output -> k)
| forall resource any output . OnError (m resource) (resource -> m any) (resource -> m output) (output -> k)
deriving instance Functor (Resource m)
instance HFunctor Resource where
hmap f (Resource acquire release use k) = Resource (f acquire) (f . release) (f . use) k
hmap f (OnError acquire release use k) = OnError (f acquire) (f . release) (f . use) k
instance Effect Resource where
handle state handler (Resource acquire release use k) = Resource (handler (acquire <$ state)) (handler . fmap release) (handler . fmap use) (handler . fmap k)
handle state handler (OnError acquire release use k) = OnError (handler (acquire <$ state)) (handler . fmap release) (handler . fmap use) (handler . fmap k)
bracket :: (Member Resource sig, Carrier sig m)
=> m resource
-> (resource -> m any)
-> (resource -> m a)
-> m a
bracket acquire release use = send (Resource acquire release use ret)
bracketOnError :: (Member Resource sig, Carrier sig m)
=> m resource
-> (resource -> m any)
-> (resource -> m a)
-> m a
bracketOnError acquire release use = send (OnError acquire release use ret)
finally :: (Member Resource sig, Carrier sig m, Applicative m)
=> m a
-> m b
-> m a
finally act end = bracket (pure ()) (const end) (const act)
onException :: (Member Resource sig, Carrier sig m, Applicative m)
=> m a
-> m b
-> m a
onException act end = bracketOnError (pure ()) (const end) (const act)
runResource :: (Carrier sig m, MonadIO m)
=> (forall x . m x -> IO x)
-> Eff (ResourceC m) a
-> m a
runResource handler = runResourceC handler . interpret
newtype ResourceC m a = ResourceC ((forall x . m x -> IO x) -> m a)
runResourceC :: (forall x . m x -> IO x) -> ResourceC m a -> m a
runResourceC handler (ResourceC m) = m handler
instance (Carrier sig m, MonadIO m) => Carrier (Resource :+: sig) (ResourceC m) where
ret a = ResourceC (const (ret a))
eff op = ResourceC (\ handler -> handleSum
(eff . handlePure (runResourceC handler))
(\case
Resource acquire release use k -> liftIO (Exc.bracket
(handler (runResourceC handler acquire))
(handler . runResourceC handler . release)
(handler . runResourceC handler . use))
>>= runResourceC handler . k
OnError acquire release use k -> liftIO (Exc.bracketOnError
(handler (runResourceC handler acquire))
(handler . runResourceC handler . release)
(handler . runResourceC handler . use))
>>= runResourceC handler . k
) op)