Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Variant of withResource that allows explicit resource destruction #9

Open
chris-martin opened this issue Jun 21, 2022 · 3 comments
Open

Comments

@chris-martin
Copy link

(Copied from bos/pool#40 because it seems that I put this issue in the wrong repository last year)

The resource-pool library assumes that a corrupted resource will cause an action that uses to it to throw an exception. I do not think this is universally true. I have an action involving a query on a database connection where, if the query throws an exception, the action can catch the exception and still return a meaningful value. However, I still want the connection resource to be destroyed when this happens.

I have written the following function for this, and I would like to propose the adding it to the library.

import Control.Exception (mask, onException)
import Control.Monad (unless)
import Control.Monad.IO.Class (MonadIO(..))
import Control.Monad.Trans.Control (MonadBaseControl, control)
import Data.IORef (atomicModifyIORef', newIORef, readIORef)
import Data.Pool (Pool, destroyResource, putResource, takeResource)

{- | Like 'withResource', but allows the action to explicitly destroy the resource

If the action throws an exception, the resource will be destroyed. If the action returns
normally and does not run the destroy action, the resource will be returned to the pool.
-}
withResource' :: (MonadBaseControl IO m, MonadIO m) =>
    Pool a
    -> (a -> m () -> m b) -- ^ The first argument is the resource; the second
                          --   argument is an action to destroy the resource
    -> m b
withResource' pool action = control $ \runInIO -> mask $ \restore ->
  do
    -- Acquire the resource
    (resource, local) <- takeResource pool

    -- Keep track of whether the resource has been destroyed, to avoid destroying it repeatedly
    destroyedRef <- newIORef False

    -- This action destroys the resource, if it has not already been destroyed
    let destroy = do alreadyDestroyed <- atomicModifyIORef' destroyedRef (\x -> (True, x))
                     unless alreadyDestroyed $ destroyResource pool local resource

    -- Run the user's action; if it throws an exception, destroy the resource
    result <- restore (runInIO (action resource (liftIO destroy))) `onException` destroy

    -- Return the resource to the pool, if it has not been destroyed
    destroyed <- readIORef destroyedRef
    unless destroyed $ putResource local resource

    -- Return the result from the user's action
    return result
@chris-martin
Copy link
Author

I see this is now outdated because the MonadBaseControl IO constraint on withResource has been concretized to IO. I think I can make the corresponding change to withResource', if this function is wanted at all.

@arybczak
Copy link
Contributor

arybczak commented Aug 17, 2022

The problem here is that this variant is useful to you, but another, slightly different variant might be useful to someone else. Do we include it then too?

I'd rather provide a single, simple withResource and make it easy for users to write variants that fit their specific use cases.

@chris-martin
Copy link
Author

chris-martin commented Aug 22, 2022

Yes, I would think including it would generally be the right thing to do. Why volunteer to maintain a library if not to help collect things that are useful to people?

The resource destruction mechanism in this function is more general than the one presently included in the library, which makes a more specific assumption (which is not true for all users) about the conditions under which a resource should be destroyed. If the growing size of this library's API is becoming a problem, then maybe the right solution would be to replace the present function with the one that makes fewer assumptions, instead of adding on.

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
None yet
Projects
None yet
Development

No branches or pull requests

2 participants