{-# OPTIONS_GHC -fno-warn-name-shadowing #-} -- Deactivate warning because it is painful to refactor functions with two -- rebinded-do with different bind functions. Such as in the 'run' -- function. Which is a good argument for having support for F#-style builders. {-# LANGUAGE DerivingVia #-} {-# LANGUAGE GADTs #-} {-# LANGUAGE InstanceSigs #-} {-# LANGUAGE LinearTypes #-} {-# LANGUAGE RankNTypes #-} {-# LANGUAGE QualifiedDo #-} {-# LANGUAGE RecordWildCards #-} -- | This module defines an IO monad for linearly working with system resources -- like files. It provides tools to take resources that are currently -- unsafely accessible from "System.IO" and use them in this monad. -- -- Import this module qualified to avoid name clashes. -- -- To use this RIO monad, create some @RIO@ computation, -- run it to get a "System.IO" computation. -- -- = A simple example -- >>> :set -XLinearTypes -- >>> :set -XQualifiedDo -- >>> :set -XNoImplicitPrelude -- >>> import qualified System.IO.Resource as Linear -- >>> import qualified Control.Functor.Linear as Control -- >>> import qualified Data.Text as Text -- >>> import Prelude.Linear -- >>> import qualified Prelude -- >>> :{ -- linearWriteToFile :: IO () -- linearWriteToFile = Linear.run Prelude.$ Control.do -- handle1 <- Linear.openFile "/home/user/test.txt" Linear.WriteMode -- handle2 <- Linear.hPutStrLn handle1 (Text.pack "hello there") -- () <- Linear.hClose handle2 -- Control.return (Ur ()) -- :} -- -- To enable do notation, `QualifiedDo` extension is used. But since QualifiedDo -- only modifies the desugaring of binds, we still need to qualify `Control.return`. module System.IO.Resource ( -- * The Resource I/O Monad RIO , run -- * Using Resource Handles -- $monad -- $files , Handle -- ** File I/O , openFile , System.IOMode (..) -- ** Working with Handles , hClose , hIsEOF , hGetChar , hPutChar , hGetLine , hPutStr , hPutStrLn -- * Creating new types of resources -- $new-resources , UnsafeResource , unsafeRelease , unsafeAcquire , unsafeFromSystemIOResource , unsafeFromSystemIOResource_ ) where import Control.Exception (onException, mask, finally) import qualified Control.Monad as Ur (fmap) import qualified Data.Functor.Linear as Data import qualified Control.Functor.Linear as Control import Data.Coerce import qualified Data.IORef as System import Data.IORef (IORef) import qualified Data.IntMap.Strict as IntMap import Data.IntMap.Strict (IntMap) import Data.Text (Text) import qualified Data.Text.IO as Text import Prelude.Linear hiding (IO) import qualified Prelude import qualified System.IO.Linear as Linear import qualified System.IO as System -- XXX: This would be better as a multiplicity-parametric relative monad, but -- until we have multiplicity polymorphism, we use a linear monad. newtype ReleaseMap = ReleaseMap (IntMap (Linear.IO ())) -- | The resource-aware I/O monad. This monad guarantees that acquired resources -- are always released. newtype RIO a = RIO (IORef ReleaseMap -> Linear.IO a) deriving (Data.Functor, Data.Applicative) via (Control.Data RIO) unRIO :: RIO a %1-> IORef ReleaseMap -> Linear.IO a unRIO (RIO action) = action -- | Take a @RIO@ computation with a value @a@ that is not linearly bound and -- make it a "System.IO" computation. run :: RIO (Ur a) -> System.IO a run (RIO action) = do rrm <- System.newIORef (ReleaseMap IntMap.empty) mask (\restore -> onException (restore (Linear.withLinearIO (action rrm))) (do -- release stray resources ReleaseMap releaseMap <- System.readIORef rrm safeRelease Prelude.$ Ur.fmap snd Prelude.$ IntMap.toList releaseMap)) -- Remarks: resources are guaranteed to be released on non-exceptional -- return. So, contrary to a standard bracket/ResourceT implementation, we -- only release exceptions in the release map upon exception. where safeRelease :: [Linear.IO ()] -> System.IO () safeRelease [] = Prelude.return () safeRelease (finalizer:fs) = Linear.withLinearIO (moveLinearIO finalizer) `finally` safeRelease fs -- Should be just an application of a linear `(<$>)`. moveLinearIO :: Movable a => Linear.IO a %1-> Linear.IO (Ur a) moveLinearIO action' = Control.do result <- action' Control.return $ move result -- | Should not be applied to a function that acquires or releases resources. unsafeFromSystemIO :: System.IO a %1-> RIO a unsafeFromSystemIO action = RIO (\ _ -> Linear.fromSystemIO action) -- $monad instance Control.Functor RIO where fmap f (RIO action) = RIO $ \releaseMap -> Control.fmap f (action releaseMap) instance Control.Applicative RIO where pure a = RIO $ \_releaseMap -> Control.pure a (<*>) = Control.ap instance Control.Monad RIO where x >>= f = RIO $ \releaseMap -> Control.do a <- unRIO x releaseMap unRIO (f a) releaseMap x >> y = RIO $ \releaseMap -> Control.do unRIO x releaseMap unRIO y releaseMap -- $files -- Remark: Handle needs to be private otherwise `Data.Coerce.coerce` could wreak -- Havoc on the abstraction. But we could provide a smart constructor/view to -- unsafely convert to file handles in order for the Handle API to be -- extensible. newtype Handle = Handle (UnsafeResource System.Handle) -- | See 'System.IO.openFile' openFile :: FilePath -> System.IOMode -> RIO Handle openFile path mode = Control.do h <- unsafeAcquire (Linear.fromSystemIOU Prelude.$ System.openFile path mode) (\h -> Linear.fromSystemIO $ System.hClose h) Control.return $ Handle h hClose :: Handle %1-> RIO () hClose (Handle h) = unsafeRelease h hIsEOF :: Handle %1-> RIO (Ur Bool, Handle) hIsEOF = coerce (unsafeFromSystemIOResource System.hIsEOF) hGetChar :: Handle %1-> RIO (Ur Char, Handle) hGetChar = coerce (unsafeFromSystemIOResource System.hGetChar) hPutChar :: Handle %1-> Char -> RIO Handle hPutChar h c = flipHPutChar c h -- needs a multiplicity polymorphic flip where flipHPutChar :: Char -> Handle %1-> RIO Handle flipHPutChar c = coerce (unsafeFromSystemIOResource_ (\h' -> System.hPutChar h' c)) hGetLine :: Handle %1-> RIO (Ur Text, Handle) hGetLine = coerce (unsafeFromSystemIOResource Text.hGetLine) hPutStr :: Handle %1-> Text -> RIO Handle hPutStr h s = flipHPutStr s h -- needs a multiplicity polymorphic flip where flipHPutStr :: Text -> Handle %1-> RIO Handle flipHPutStr s = coerce (unsafeFromSystemIOResource_ (\h' -> Text.hPutStr h' s)) hPutStrLn :: Handle %1-> Text -> RIO Handle hPutStrLn h s = flipHPutStrLn s h -- needs a multiplicity polymorphic flip where flipHPutStrLn :: Text -> Handle %1-> RIO Handle flipHPutStrLn s = coerce (unsafeFromSystemIOResource_ (\h' -> Text.hPutStrLn h' s)) -- $new-resources -- | The type of system resources. To create and use resources, you need to -- use the API since the constructor is not released. data UnsafeResource a where UnsafeResource :: Int -> a -> UnsafeResource a -- Note that both components are unrestricted. -- | Given an unsafe resource, release it with the linear IO action provided -- when the resrouce was acquired. unsafeRelease :: UnsafeResource a %1-> RIO () unsafeRelease (UnsafeResource key _) = RIO (\st -> Linear.mask_ (releaseWith key st)) where releaseWith key rrm = Control.do Ur (ReleaseMap releaseMap) <- Linear.readIORef rrm () <- releaseMap IntMap.! key Linear.writeIORef rrm (ReleaseMap (IntMap.delete key releaseMap)) -- | Given a resource in the "System.IO.Linear.IO" monad, and -- given a function to release that resource, provides that resource in -- the @RIO@ monad. For example, releasing a @Handle@ from "System.IO" -- would be done with @fromSystemIO hClose@. Because this release function -- is an input, and could be wrong, this function is unsafe. unsafeAcquire :: Linear.IO (Ur a) -> (a -> Linear.IO ()) -> RIO (UnsafeResource a) unsafeAcquire acquire release = RIO $ \rrm -> Linear.mask_ (Control.do Ur resource <- acquire Ur (ReleaseMap releaseMap) <- Linear.readIORef rrm () <- Linear.writeIORef rrm (ReleaseMap (IntMap.insert (releaseKey releaseMap) (release resource) releaseMap)) Control.return $ UnsafeResource (releaseKey releaseMap) resource) where releaseKey releaseMap = case IntMap.null releaseMap of True -> 0 False -> fst (IntMap.findMax releaseMap) + 1 -- | Given a "System.IO" computation on an unsafe resource, -- lift it to @RIO@ computaton on the acquired resource. -- That is function of type @a -> IO b@ turns into a function of type -- @UnsafeResource a %1-> RIO (Ur b)@ -- along with threading the @UnsafeResource a@. -- -- Note that the result @b@ can be used non-linearly. unsafeFromSystemIOResource :: (a -> System.IO b) -> (UnsafeResource a %1-> RIO (Ur b, UnsafeResource a)) unsafeFromSystemIOResource action (UnsafeResource key resource) = unsafeFromSystemIO (do c <- action resource Prelude.return (Ur c, UnsafeResource key resource)) unsafeFromSystemIOResource_ :: (a -> System.IO ()) -> (UnsafeResource a %1-> RIO (UnsafeResource a)) unsafeFromSystemIOResource_ action resource = Control.do (Ur _, resource) <- unsafeFromSystemIOResource action resource Control.return resource