me gustaría capturar una excepción dentro runResourceT sin liberar el recurso, pero la captura función corre el cálculo interno en IO. ¿Hay alguna manera de detectar una excepción dentro de runResourceT, o cuál es la forma recomendada de refactorizar el código?Cómo coger una excepción dentro runResourceT
Gracias por su ayuda.
{-# LANGUAGE FlexibleContexts #-}
module Main where
import Control.Exception as EX
import Control.Monad.IO.Class
import Control.Monad.Trans.Resource
type Resource = String
allocResource :: IO Resource
allocResource = let r = "Resource"
in putStrLn (r ++ " opened.") >> return r
closeResource :: Resource -> IO()
closeResource r = putStrLn $ r ++ " closed."
withResource :: (MonadIO m
, MonadBaseControl IO m
, MonadThrow m
, MonadUnsafeIO m
) => (Resource -> ResourceT m a) -> m a
withResource f = runResourceT $ do
(_, r) <- allocate allocResource closeResource
f r
useResource :: (MonadIO m
, MonadBaseControl IO m
, MonadThrow m
, MonadUnsafeIO m
) => Resource -> ResourceT m Int
useResource r = liftIO $ putStrLn ("Using " ++ r) >> return 1
main :: IO()
main = do
putStrLn "Start..."
withResource $ \r -> do
x <- useResource r
{-- This does not compile as the catch computation runs inside IO
y <- liftIO $ EX.catch (useResource r)
(\e -> do putStrLn $ show (e::SomeException)
return 0)
--}
return()
putStrLn "Done."
Gracias por su respuesta y comentarios. Funciona como se esperaba y me ayuda a comprender mejor los transformadores de mónadas. – jedf