{-# LANGUAGE DeriveDataTypeable #-} {-# LANGUAGE IncoherentInstances #-} {-# LANGUAGE RoleAnnotations #-} {-# OPTIONS_GHC -Wno-unused-binds #-} {-# OPTIONS_GHC -Wno-redundant-constraints #-} -- | Checked exceptions module Hackage.Security.Util.Checked ( Throws , unthrow -- ** Base exceptions , throwChecked , catchChecked , handleChecked , tryChecked , checkIO , throwUnchecked , internalError ) where import Prelude import Control.Exception (Exception, IOException) import qualified Control.Exception as Base import Data.Typeable (Typeable) import GHC.Prim (coerce) {------------------------------------------------------------------------------- Basic infrastructure -------------------------------------------------------------------------------} -- | Checked exceptions class Throws e where type role Throws representational unthrow :: forall a e proxy . proxy e -> (Throws e => a) -> a unthrow _ x = unWrap (coerceWrap (Wrap x :: Wrap e a)) {------------------------------------------------------------------------------- Base exceptions -------------------------------------------------------------------------------} -- | Determine if an exception is asynchronous, based on its type. isAsync :: Exception e => e -> Bool isAsync e = case Base.fromException $ Base.toException e of Just Base.SomeAsyncException{} -> True Nothing -> False -- | 'Base.catch', but immediately rethrows asynchronous exceptions -- (as determined by 'isAsync'). catchSync :: Exception e => IO a -> (e -> IO a) -> IO a catchSync act onErr = act `Base.catch` \e -> if isAsync e then Base.throwIO e else onErr e -- | Wraps up an async exception as a synchronous exception. newtype SyncException = SyncException Base.SomeException deriving (Show, Typeable) instance Exception SyncException -- | Throw a checked exception throwChecked :: (Exception e, Throws e) => e -> IO a throwChecked e | isAsync e = Base.throwIO $ SyncException $ Base.toException e | otherwise = Base.throwIO e -- | Catch a checked exception catchChecked :: forall a e. Exception e => (Throws e => IO a) -> (e -> IO a) -> IO a catchChecked act = catchSync (unthrow (Proxy :: Proxy e) act) -- | 'catchChecked' with the arguments reversed handleChecked :: Exception e => (e -> IO a) -> (Throws e => IO a) -> IO a handleChecked act handler = catchChecked handler act -- | Like 'try', but for checked exceptions tryChecked :: Exception e => (Throws e => IO a) -> IO (Either e a) tryChecked act = catchChecked (Right <$> act) (return . Left) -- | Rethrow IO exceptions as checked exceptions checkIO :: Throws IOException => IO a -> IO a checkIO = Base.handle $ \(ex :: IOException) -> throwChecked ex -- | Throw an unchecked exception -- -- This is just an alias for 'throw', but makes it evident that this is a very -- intentional use of an unchecked exception. throwUnchecked :: Exception e => e -> IO a throwUnchecked = Base.throwIO -- | Variation on 'throwUnchecked' for internal errors internalError :: String -> IO a internalError = throwUnchecked . userError {------------------------------------------------------------------------------- Auxiliary definitions (not exported) -------------------------------------------------------------------------------} -- | Wrap an action that may throw a checked exception -- -- This is used internally in 'unthrow' to avoid impredicative -- instantiation of the type of 'coerce'/'unsafeCoerce'. newtype Wrap e a = Wrap { unWrap :: Throws e => a } coerceWrap :: Wrap e a -> Wrap (Catch e) a coerceWrap = coerce data Proxy a = Proxy newtype Catch a = Catch a instance Throws (Catch e) where