{-# LANGUAGE UnicodeSyntax , NoImplicitPrelude , CPP , RankNTypes #-} ------------------------------------------------------------------------------- -- | -- Module : System.IO.SaferFileHandles -- Copyright : (c) 2010 Bas van Dijk -- License : BSD3 (see the file LICENSE) -- Maintainer : Bas van Dijk -- -- The contributions of this module are as follows: -- -- * This module extends the @regions@ library with the ability to open files in -- a 'RegionT'. When the region terminates, all opened resources will be closed -- automatically. The main advantage of regions is that the handles to the -- opened resources can not be returned from the region. This ensures no I/O -- with closed resources is possible. The primary technique used in @regions@ is -- called \"Lightweight monadic regions\" which was invented by Oleg Kiselyov -- and Chung-chieh Shan. See: -- -- -- * Secondly this module provides all the file operations of @System.IO@ lifted -- to the region monad. -- -- * Third, filepaths that are given to functions like 'openFile' are created -- and manipulated in a type-safe way using the @pathtype@ package. -- -- * The final contribution of this module is that file handles are -- parameterised with the IOMode in which the file was opened. This can be -- either 'ReadMode', 'WriteMode', 'AppendMode' or 'ReadWriteMode'. All -- operations on files explicitly specify the needed IOMode using the -- 'ReadModes' and 'WriteModes' type classes. This way it is impossible to read -- from a write-only handle or write to a read-only handle for example. -- -- See the @safer-file-handles-examples@ package for examples how to use this -- package: -- -- @darcs get @ -- -- /NOTE:/ This module also provides functions from @System.IO@ which don't -- directly work with file handles like 'putStrLn' or 'getLine' for -- example. These functions implicitly use the standard handles. I actually -- provide more general versions of these that work in any 'MonadIO'. It could -- be argued that these functions don't belong in this module because they don't -- have anything to do with regions and explicit IOModes. However I provide them -- as a convenience. But be warned that in the future these lifted functions may -- move to their own package! -- ------------------------------------------------------------------------------- module System.IO.SaferFileHandles ( -- * Regional file handles RegionalFileHandle -- ** IO Modes -- | Types that represent the IOMode an opened file can be in. , ReadMode , WriteMode , AppendMode , ReadWriteMode , ReadModes , WriteModes , IOMode(..) , MkIOMode(mkIOMode) -- ** Standard handles {-| These standard handles have concrete IOModes by default which work for the majority of cases. In the rare occasion that you know these handles have different IOModes you can 'cast' them. -} , stdin, stdout, stderr , cast -- ** Opening files in a region , openFile, withFile -- *** Opening files by inferring the IOMode , openFile', withFile' -- ** Regions {-| Note that this module re-exports the @Control.Monad.Trans.Region@ module from the @regions@ package which allows you to: * Run a region using 'runRegionT'. * Concurrently run a region inside another region using 'forkIOTopRegion'. * Duplicate a 'RegionalFileHandle' to a parent region using 'dup'. -} , module Control.Monad.Trans.Region -- * Operations on regional file handles -- ** Determining and changing the size of a file , hFileSize #ifdef __GLASGOW_HASKELL__ , hSetFileSize #endif -- ** Detecting the end of input , hIsEOF , isEOF -- ** Buffering operations , BufferMode(..) , hSetBuffering , hGetBuffering , hFlush -- ** Repositioning handles , hGetPosn , hSetPosn , HandlePosn , hSeek , SeekMode(..) #if !defined(__NHC__) , hTell #endif -- ** Handle properties , hIsOpen, hIsClosed , hIsReadable, hIsWritable , hIsSeekable -- ** Terminal operations (not portable: GHC/Hugs only) #if !defined(__NHC__) , hIsTerminalDevice , hSetEcho , hGetEcho #endif -- ** Showing handle state (not portable: GHC only) #ifdef __GLASGOW_HASKELL__ , hShow #endif -- * Text input and output -- ** Text input -- | Note that the following text input operations are polymorphic in the -- IOMode of the given handle. However the IOModes are restricted to -- 'ReadModes' only which can be either 'R' or 'RW'. , hWaitForInput , hReady , hGetChar , hGetLine , hLookAhead , hGetContents -- ** Text ouput -- | Note that the following text output operations are polymorphic in the -- IOMode of the given handle. However the IOModes are restricted to -- 'WriteModes' only which can be either 'W', 'A' or 'RW'. , hPutChar , hPutStr , hPutStrLn , hPrint -- ** Special cases for standard input and output , interact , putChar , putStr , putStrLn , print , getChar , getLine , getContents , readIO , readLn -- * Binary input and output , openBinaryFile, withBinaryFile -- ** Opening binary files by inferring the IOMode , openBinaryFile', withBinaryFile' -- ** Operations on binary handles , hSetBinaryMode , hPutBuf , hGetBuf #if !defined(__NHC__) && !defined(__HUGS__) , hPutBufNonBlocking , hGetBufNonBlocking #endif -- * Temporary files , Template , openTempFile , openBinaryTempFile #if MIN_VERSION_base(4,2,0) , openTempFileWithDefaultPermissions , openBinaryTempFileWithDefaultPermissions #endif #if MIN_VERSION_base(4,2,0) && !defined(__NHC__) && !defined(__HUGS__) -- * Unicode encoding/decoding , hSetEncoding , hGetEncoding -- ** Unicode encodings , TextEncoding , latin1 , utf8, utf8_bom , utf16, utf16le, utf16be , utf32, utf32le, utf32be , localeEncoding , mkTextEncoding -- * Newline conversion , hSetNewlineMode , Newline(..) , nativeNewline , NewlineMode(..) , noNewlineTranslation, universalNewlineMode, nativeNewlineMode #endif ) where ------------------------------------------------------------------------------- -- Imports ------------------------------------------------------------------------------- -- from base: import Prelude ( Integer ) import Control.Monad ( return, (>>=), fail, liftM ) import Data.Bool ( Bool(..) ) import Data.Function ( ($), flip ) import Data.Functor ( fmap ) import Data.Char ( Char, String ) import Data.Int ( Int ) import Data.Maybe ( Maybe(Nothing, Just) ) import Text.Show ( Show ) import Text.Read ( Read ) import Foreign.Ptr ( Ptr ) import qualified System.IO as SIO #ifdef __HADDOCK__ import System.IO.Error #endif -- from base-unicode-symbols: import Data.Function.Unicode ( (∘) ) -- from MonadCatchIO-transformers: import Control.Monad.CatchIO ( MonadCatchIO, block ) -- from transformers: import Control.Monad.IO.Class ( MonadIO, liftIO ) -- from regions: import Control.Monad.Trans.Region -- ( re-exported entirely ) import Control.Monad.Trans.Region.OnExit ( onExit ) -- from explicit-iomodes import System.IO.ExplicitIOModes ( IO , hClose , IOMode(..) , MkIOMode(mkIOMode) , ReadMode , WriteMode , AppendMode , ReadWriteMode , ReadModes , WriteModes , CheckMode , BufferMode(..) , HandlePosn , SeekMode(..) #if MIN_VERSION_base(4,2,0) && !defined(__NHC__) && !defined(__HUGS__) , TextEncoding , latin1 , utf8, utf8_bom , utf16, utf16le, utf16be , utf32, utf32le, utf32be , localeEncoding , Newline(..) , nativeNewline , NewlineMode(..) , noNewlineTranslation , universalNewlineMode , nativeNewlineMode #endif ) import qualified System.IO.ExplicitIOModes as E -- from pathtype: import System.Path ( DirPath, FilePath , AbsFile, RelFile , AbsRelClass , getPathString, asAbsFile ) -- from ourselves: import System.IO.SaferFileHandles.Internal ( RegionalFileHandle(RegionalFileHandle) ) import System.IO.SaferFileHandles.Unsafe ( wrap, wrap2, wrap3, sanitizeIOError ) ------------------------------------------------------------------------------- -- ** Standard handles ------------------------------------------------------------------------------- -- | Wraps: @System.IO.'SIO.stdin'@. stdin ∷ RegionalFileHandle ReadMode r stdin = RegionalFileHandle E.stdin Nothing -- | Wraps: @System.IO.'SIO.stdout'@. stdout ∷ RegionalFileHandle WriteMode r stdout = RegionalFileHandle E.stdout Nothing -- | Wraps: @System.IO.'SIO.stderr'@. stderr ∷ RegionalFileHandle WriteMode r stderr = RegionalFileHandle E.stderr Nothing {-| Cast the IOMode of a handle if the handle supports it. This function is primarily used in combination with the standard handles. When you know the IOMode of a handle is different from its default IOMode you can cast it to the right one. -} cast ∷ (pr `ParentOf` cr, MonadIO cr, CheckMode castedIOMode) ⇒ RegionalFileHandle anyIOMode pr → cr (Maybe (RegionalFileHandle castedIOMode pr)) cast (RegionalFileHandle h mbCloseHndl) = (liftM ∘ fmap) (flip RegionalFileHandle mbCloseHndl) $ liftIO $ E.cast h ------------------------------------------------------------------------------- -- ** Opening files in a region ------------------------------------------------------------------------------- {-| Computation 'openFile' @filePath ioMode@ allocates and returns a new, regional file handle to manage the file identified by @filePath@. It provides a safer replacement for @System.IO.'SIO.openFile'@. If the file does not exist and it is opened for output, it should be created as a new file. If @ioMode@ is 'WriteMode' and the file already exists, then it should be truncated to zero length. Some operating systems delete empty files, so there is no guarantee that the file will exist following an 'openFile' with @ioMode@ 'WriteMode' unless it is subsequently written to successfully. The handle is positioned at the end of the file if @ioMode@ is 'AppendMode', and otherwise at the beginning (in which case its internal position is 0). The initial buffer mode is implementation-dependent. Note that the returned regional file handle is parameterized by the region in which it was created. This ensures that handles can never escape their region. And it also allows operations on handles to be executed in a child region of the region in which the handle was created. Note that if you do wish to return a handle from the region in which it was created you have to duplicate the handle by applying 'dup' to it. Finally note that the returned regional file handle is also parameterized by the IOMode in which you opened the file. All operations on files explicitly specify the needed IOMode using the 'ReadModes' and 'WriteModes' type classes. This way it is impossible to read from a write-only handle or write to a read-only handle for example. This operation may fail with: * 'isAlreadyInUseError' if the file is already open and cannot be reopened; * 'isDoesNotExistError' if the file does not exist; or * 'isPermissionError' if the user does not have permission to open the file. Note: if you will be working with files containing binary data, you'll want to be using 'openBinaryFile'. -} openFile ∷ (MonadCatchIO pr, AbsRelClass ar) ⇒ FilePath ar → IOMode ioMode → RegionT s pr (RegionalFileHandle ioMode (RegionT s pr)) openFile = openNormal E.openFile openNormal ∷ (MonadCatchIO pr, AbsRelClass ar) ⇒ (E.FilePath → IOMode ioMode → IO (E.Handle ioMode)) → ( FilePath ar → IOMode ioMode → RegionT s pr (RegionalFileHandle ioMode (RegionT s pr)) ) openNormal open = \filePath ioMode → block $ do h ← liftIO $ open (getPathString filePath) ioMode ch ← onExit $ sanitizeIOError $ hClose h return $ RegionalFileHandle h $ Just ch {-| Convenience function which opens a file, applies the given continuation function to the resulting regional file handle and runs the resulting region. This provides a safer safer replacement for @System.IO.'SIO.withFile'@. -} withFile ∷ (MonadCatchIO pr, AbsRelClass ar) ⇒ FilePath ar → IOMode ioMode → (∀ s. RegionalFileHandle ioMode (RegionT s pr) → RegionT s pr α) → pr α withFile filePath ioMode f = runRegionT $ openFile filePath ioMode >>= f -- *** Opening files by inferring the IOMode -- | Open a file without explicitly specifying the IOMode. The IOMode is -- inferred from the type of the resulting 'RegionalFileHandle'. -- -- Note that: @openFile' fp = 'openFile' fp 'mkIOMode'@. openFile' ∷ (MonadCatchIO pr, AbsRelClass ar, MkIOMode ioMode) ⇒ FilePath ar → RegionT s pr (RegionalFileHandle ioMode (RegionT s pr)) openFile' filePath = openFile filePath mkIOMode -- | Note that: @withFile' filePath = 'withFile' filePath 'mkIOMode'@. withFile' ∷ (MonadCatchIO pr, AbsRelClass ar, MkIOMode ioMode) ⇒ FilePath ar → (∀ s. RegionalFileHandle ioMode (RegionT s pr) → RegionT s pr α) → pr α withFile' filePath = withFile filePath mkIOMode ------------------------------------------------------------------------------- -- * Operations on regional file handles ------------------------------------------------------------------------------- -- ** Determining and changing the size of a file -- | For a handle @hdl@ which attached to a physical file, -- 'hFileSize' @hdl@ returns the size of that file in 8-bit bytes. -- -- Wraps: @System.IO.'SIO.hFileSize'@. hFileSize ∷ (pr `ParentOf` cr, MonadIO cr) ⇒ RegionalFileHandle ioMode pr → cr Integer hFileSize = wrap E.hFileSize #ifdef __GLASGOW_HASKELL__ -- | 'hSetFileSize' @hdl@ @size@ truncates the physical file with handle @hdl@ -- to @size@ bytes. -- -- Wraps: @System.IO.'SIO.hSetFileSize'@. hSetFileSize ∷ (pr `ParentOf` cr, MonadIO cr) ⇒ RegionalFileHandle ioMode pr → Integer → cr () hSetFileSize = wrap2 E.hSetFileSize #endif -- ** Detecting the end of input -- | For a readable handle @hdl@, 'hIsEOF' @hdl@ returns -- 'True' if no further input can be taken from @hdl@ or for a -- physical file, if the current I\/O position is equal to the length of -- the file. Otherwise, it returns 'False'. -- -- NOTE: 'hIsEOF' may block, because it is the same as calling -- 'hLookAhead' and checking for an EOF exception. -- -- Wraps: @System.IO.'SIO.hIsEOF'@. hIsEOF ∷ (pr `ParentOf` cr, MonadIO cr, ReadModes ioMode) ⇒ RegionalFileHandle ioMode pr → cr Bool hIsEOF = wrap E.hIsEOF -- | Generalizes: @System.IO.'SIO.isEOF'@ to any 'MonadIO'. isEOF ∷ MonadIO m ⇒ m Bool isEOF = liftIO $ E.isEOF -- ** Buffering operations -- | Computation 'hSetBuffering' @hdl mode@ sets the mode of buffering for -- handle @hdl@ on subsequent reads and writes. -- -- If the buffer mode is changed from 'BlockBuffering' or -- 'LineBuffering' to 'NoBuffering', then -- -- * if @hdl@ is writable, the buffer is flushed as for 'hFlush'; -- -- * if @hdl@ is not writable, the contents of the buffer is discarded. -- -- This operation may fail with: -- -- * 'isPermissionError' if the handle has already been used for reading -- or writing and the implementation does not allow the buffering mode -- to be changed. -- -- Wraps: @System.IO.'SIO.hSetBuffering'@. hSetBuffering ∷ (pr `ParentOf` cr, MonadIO cr) ⇒ RegionalFileHandle ioMode pr → BufferMode → cr () hSetBuffering = wrap2 E.hSetBuffering -- | Computation 'hGetBuffering' @hdl@ returns the current buffering mode for -- @hdl@. -- -- Wraps: @System.IO.'SIO.hGetBuffering'@. hGetBuffering ∷ (pr `ParentOf` cr, MonadIO cr) ⇒ RegionalFileHandle ioMode pr → cr BufferMode hGetBuffering = wrap E.hGetBuffering -- | The action 'hFlush' @hdl@ causes any items buffered for output in handle -- @hdl@ to be sent immediately to the operating system. -- -- This operation may fail with: -- -- * 'isFullError' if the device is full; -- -- * 'isPermissionError' if a system resource limit would be exceeded. It is -- unspecified whether the characters in the buffer are discarded or retained -- under these circumstances. -- -- Wraps: @System.IO.'SIO.hFlush'@. hFlush ∷ (pr `ParentOf` cr, MonadIO cr, WriteModes ioMode) ⇒ RegionalFileHandle ioMode pr → cr () hFlush = wrap E.hFlush -- ** Repositioning handles -- | Computation 'hGetPosn' @hdl@ returns the current I\/O position of @hdl@ as -- a value of the abstract type 'HandlePosn'. -- -- Wraps: @System.IO.'SIO.hGetPosn'@. hGetPosn ∷ (pr `ParentOf` cr, MonadIO cr) ⇒ RegionalFileHandle ioMode pr → cr HandlePosn hGetPosn = wrap E.hGetPosn -- | If a call to 'hGetPosn' @hdl@ returns a position @p@, then computation -- 'hSetPosn' @p@ sets the position of @hdl@ to the position it held at the time -- of the call to 'hGetPosn'. -- -- This operation may fail with: -- -- * 'isPermissionError' if a system resource limit would be exceeded. -- -- Wraps: @System.IO.'SIO.hSetPosn'@. hSetPosn ∷ MonadIO m ⇒ HandlePosn → m () hSetPosn = liftIO ∘ E.hSetPosn -- | Computation 'hSeek' @hdl mode i@ sets the position of handle @hdl@ -- depending on @mode@. The offset @i@ is given in terms of 8-bit bytes. -- -- If @hdl@ is block- or line-buffered, then seeking to a position which is not -- in the current buffer will first cause any items in the output buffer to be -- written to the device, and then cause the input buffer to be discarded. Some -- handles may not be seekable (see 'hIsSeekable'), or only support a subset of -- the possible positioning operations (for instance, it may only be possible to -- seek to the end of a tape, or to a positive offset from the beginning or -- current position). -- It is not possible to set a negative I\/O position, or -- for a physical file, an I\/O position beyond the current end-of-file. -- -- This operation may fail with: -- -- * 'isPermissionError' if a system resource limit would be exceeded. -- -- Wraps: @System.IO.'SIO.hSeek'@. hSeek ∷ (pr `ParentOf` cr, MonadIO cr) ⇒ RegionalFileHandle ioMode pr → SeekMode → Integer → cr () hSeek = wrap3 E.hSeek #if !defined(__NHC__) -- | Wraps: @System.IO.'SIO.hTell'@. hTell ∷ (pr `ParentOf` cr, MonadIO cr) ⇒ RegionalFileHandle ioMode pr → cr Integer hTell = wrap E.hTell #endif -- ** Handle properties -- | Note that this operation should always return 'True' since the @regions@ -- framework ensures that handles are always open. This function is used only -- for testing the correctness of this library. -- -- Wraps: @System.IO.'SIO.hIsOpen'@. hIsOpen ∷ (pr `ParentOf` cr, MonadIO cr) ⇒ RegionalFileHandle ioMode pr → cr Bool hIsOpen = wrap E.hIsOpen -- | Note that this operation should always return 'False' since the @regions@ -- framework ensures that handles are never closed. This function is used only -- for testing the correctness of this library. -- -- Wraps: @System.IO.'SIO.hIsClosed'@. hIsClosed ∷ (pr `ParentOf` cr, MonadIO cr) ⇒ RegionalFileHandle ioMode pr → cr Bool hIsClosed = wrap E.hIsClosed -- | Note that this operation should always return 'True' for IOModes which have -- an instance for 'ReadModes'. This function is used only for testing the -- correctness of this library. -- -- Wraps: @System.IO.'SIO.hIsReadable'@. hIsReadable ∷ (pr `ParentOf` cr, MonadIO cr) ⇒ RegionalFileHandle ioMode pr → cr Bool hIsReadable = wrap E.hIsReadable -- | Note that this operation should always return 'True' for IOModes which have -- an instance for 'WriteModes'. This function is used only for testing the -- correctness of this library. -- -- Wraps: @System.IO.'SIO.hIsWritable'@. hIsWritable ∷ (pr `ParentOf` cr, MonadIO cr) ⇒ RegionalFileHandle ioMode pr → cr Bool hIsWritable = wrap E.hIsWritable -- | Wraps: @System.IO.'SIO.hIsSeekable'@. hIsSeekable ∷ (pr `ParentOf` cr, MonadIO cr) ⇒ RegionalFileHandle ioMode pr → cr Bool hIsSeekable = wrap E.hIsSeekable -- ** Terminal operations (not portable: GHC/Hugs only) #if !defined(__NHC__) -- | Is the handle connected to a terminal? -- -- Wraps: @System.IO.'SIO.hIsTerminalDevice'@. hIsTerminalDevice ∷ (pr `ParentOf` cr, MonadIO cr) ⇒ RegionalFileHandle ioMode pr → cr Bool hIsTerminalDevice = wrap E.hIsTerminalDevice -- | Set the echoing status of a handle connected to a terminal. -- -- Wraps: @System.IO.'SIO.hSetEcho'@. hSetEcho ∷ (pr `ParentOf` cr, MonadIO cr) ⇒ RegionalFileHandle ioMode pr → Bool → cr () hSetEcho = wrap2 E.hSetEcho -- | Get the echoing status of a handle connected to a terminal. -- -- Wraps: @System.IO.'SIO.hGetEcho'@. hGetEcho ∷ (pr `ParentOf` cr, MonadIO cr) ⇒ RegionalFileHandle ioMode pr → cr Bool hGetEcho = wrap E.hGetEcho #endif -- ** Showing handle state (not portable: GHC only) #ifdef __GLASGOW_HASKELL__ -- | Wraps: @System.IO.'SIO.hShow'@. hShow ∷ (pr `ParentOf` cr, MonadIO cr) ⇒ RegionalFileHandle ioMode pr → cr String hShow = wrap E.hShow #endif -------------------------------------------------------------------------------- -- * Text input and output -------------------------------------------------------------------------------- -- ** Text input -- | Computation 'hWaitForInput' @hdl t@ -- waits until input is available on handle @hdl@. -- It returns 'True' as soon as input is available on @hdl@, -- or 'False' if no input is available within @t@ milliseconds. -- -- If @t@ is less than zero, then @hWaitForInput@ waits indefinitely. -- -- This operation may fail with: -- -- * 'isEOFError' if the end of file has been reached. -- -- NOTE for GHC users: unless you use the @-threaded@ flag, -- @hWaitForInput t@ where @t >= 0@ will block all other Haskell -- threads for the duration of the call. It behaves like a -- @safe@ foreign call in this respect. -- -- Wraps: @System.IO.'SIO.hWaitForInput'@. hWaitForInput ∷ (pr `ParentOf` cr, MonadIO cr, ReadModes ioMode) ⇒ RegionalFileHandle ioMode pr → Int → cr Bool hWaitForInput = wrap2 E.hWaitForInput -- | Computation 'hReady' @hdl@ indicates whether at least one item is -- available for input from handle @hdl@. -- -- This operation may fail with: -- -- * 'isEOFError' if the end of file has been reached. -- -- Wraps: @System.IO.'SIO.hReady'@. hReady ∷ (pr `ParentOf` cr, MonadIO cr, ReadModes ioMode) ⇒ RegionalFileHandle ioMode pr → cr Bool hReady = wrap E.hReady -- | Computation 'hGetChar' @hdl@ reads a character from the file or -- channel managed by @hdl@, blocking until a character is available. -- -- This operation may fail with: -- -- * 'isEOFError' if the end of file has been reached. -- -- Wraps: @System.IO.'SIO.hGetChar'@. hGetChar ∷ (pr `ParentOf` cr, MonadIO cr, ReadModes ioMode) ⇒ RegionalFileHandle ioMode pr → cr Char hGetChar = wrap E.hGetChar -- | Computation 'hGetLine' @hdl@ reads a line from the file or -- channel managed by @hdl@. -- -- This operation may fail with: -- -- * 'isEOFError' if the end of file is encountered when reading -- the /first/ character of the line. -- -- If 'hGetLine' encounters end-of-file at any other point while reading -- in a line, it is treated as a line terminator and the (partial) -- line is returned. -- -- Wraps: @System.IO.'SIO.hGetLine'@. hGetLine ∷ (pr `ParentOf` cr, MonadIO cr, ReadModes ioMode) ⇒ RegionalFileHandle ioMode pr → cr String hGetLine = wrap E.hGetLine -- | Computation 'hLookAhead' returns the next character from the handle -- without removing it from the input buffer, blocking until a character -- is available. -- -- This operation may fail with: -- -- * 'isEOFError' if the end of file has been reached. -- -- Wraps: @System.IO.'SIO.hLookAhead'@. hLookAhead ∷ (pr `ParentOf` cr, MonadIO cr, ReadModes ioMode) ⇒ RegionalFileHandle ioMode pr → cr Char hLookAhead = wrap E.hLookAhead -- | Computation 'hGetContents' @hdl@ returns the list of characters -- corresponding to the unread portion of the channel or file managed -- by @hdl@, which is put into an intermediate state, /semi-closed/. -- In this state, @hdl@ is effectively closed, -- but items are read from @hdl@ on demand and accumulated in a special -- list returned by 'hGetContents' @hdl@. -- -- Any operation that fails because a handle is closed, -- also fails if a handle is semi-closed. -- A semi-closed handle becomes closed: -- -- * if its corresponding region terminates; -- -- * if an I\/O error occurs when reading an item from the handle; -- -- * or once the entire contents of the handle has been read. -- -- Once a semi-closed handle becomes closed, the contents of the -- associated list becomes fixed. The contents of this final list is -- only partially specified: it will contain at least all the items of -- the stream that were evaluated prior to the handle becoming closed. -- -- Any I\/O errors encountered while a handle is semi-closed are simply -- discarded. -- -- This operation may fail with: -- -- * 'isEOFError' if the end of file has been reached. -- -- Wraps: @System.IO.'SIO.hGetContents'@. hGetContents ∷ (pr `ParentOf` cr, MonadIO cr, ReadModes ioMode) ⇒ RegionalFileHandle ioMode pr → cr String hGetContents = wrap E.hGetContents -- ** Text ouput -- | Computation 'hPutChar' @hdl ch@ writes the character @ch@ to the -- file or channel managed by @hdl@. Characters may be buffered if -- buffering is enabled for @hdl@. -- -- This operation may fail with: -- -- * 'isFullError' if the device is full; or -- -- * 'isPermissionError' if another system resource limit would be exceeded. -- -- Wraps: @System.IO.'SIO.hPutChar'@. hPutChar ∷ (pr `ParentOf` cr, MonadIO cr, WriteModes ioMode) ⇒ RegionalFileHandle ioMode pr → Char → cr () hPutChar = wrap2 E.hPutChar -- | Computation 'hPutStr' @hdl s@ writes the string -- @s@ to the file or channel managed by @hdl@. -- -- This operation may fail with: -- -- * 'isFullError' if the device is full; or -- -- * 'isPermissionError' if another system resource limit would be exceeded. -- -- Wraps: @System.IO.'SIO.hPutStr'@. hPutStr ∷ (pr `ParentOf` cr, MonadIO cr, WriteModes ioMode) ⇒ RegionalFileHandle ioMode pr → String → cr () hPutStr = wrap2 E.hPutStr -- | The same as 'hPutStr', but adds a newline character. -- -- Wraps: @System.IO.'SIO.hPutStrLn'@. hPutStrLn ∷ (pr `ParentOf` cr, MonadIO cr, WriteModes ioMode) ⇒ RegionalFileHandle ioMode pr → String → cr () hPutStrLn = wrap2 E.hPutStrLn -- | Computation 'hPrint' @hdl t@ writes the string representation of @t@ -- given by the 'shows' function to the file or channel managed by @hdl@ -- and appends a newline. -- -- This operation may fail with: -- -- * 'isFullError' if the device is full; or -- -- * 'isPermissionError' if another system resource limit would be exceeded. -- -- Wraps: @System.IO.'SIO.hPrint'@. hPrint ∷ (pr `ParentOf` cr, MonadIO cr, WriteModes ioMode, Show α) ⇒ RegionalFileHandle ioMode pr → α → cr () hPrint = wrap2 E.hPrint -- ** Special cases for standard input and output -- | Generalizes @System.IO.'SIO.interact'@ to any 'MonadIO'. interact ∷ MonadIO m ⇒ (String → String) → m () interact f = liftIO $ SIO.interact f -- | Generalizes @System.IO.'SIO.putChar'@ to any 'MonadIO'. putChar ∷ MonadIO m ⇒ Char → m () putChar c = liftIO $ SIO.putChar c -- | Generalizes @System.IO.'SIO.putStr'@ to any 'MonadIO'. putStr ∷ MonadIO m ⇒ String → m () putStr s = liftIO $ SIO.putStr s -- | Generalizes @System.IO.'SIO.putStrLn'@ to any 'MonadIO'. putStrLn ∷ MonadIO m ⇒ String → m () putStrLn s = liftIO $ SIO.putStrLn s -- | Generalizes @System.IO.'SIO.print'@ to any 'MonadIO'. print ∷ (MonadIO m, Show α) ⇒ α → m () print x = liftIO $ SIO.print x -- | Generalizes @System.IO.'SIO.getChar'@ to any 'MonadIO'. getChar ∷ MonadIO m ⇒ m Char getChar = liftIO SIO.getChar -- | Generalizes @System.IO.'SIO.getLine'@ to any 'MonadIO'. getLine ∷ MonadIO m ⇒ m String getLine = liftIO SIO.getLine -- | Generalizes @System.IO.'SIO.getContents'@ to any 'MonadIO'. getContents ∷ MonadIO m ⇒ m String getContents = liftIO SIO.getContents -- | Generalizes @System.IO.'SIO.readIO'@ to any 'MonadIO'. readIO ∷ (MonadIO m, Read α) ⇒ String → m α readIO s = liftIO $ SIO.readIO s -- | Generalizes @System.IO.'SIO.readLn'@ to any 'MonadIO'. readLn ∷ (MonadIO m, Read α) ⇒ m α readLn = liftIO SIO.readLn -------------------------------------------------------------------------------- -- * Binary input and output -------------------------------------------------------------------------------- -- | Like 'openFile', but open the file in binary mode. -- -- On Windows, reading a file in text mode (which is the default) will translate -- CRLF to LF, and writing will translate LF to CRLF. This is usually what you -- want with text files. With binary files this is undesirable; also, as usual -- under Microsoft operating systems, text mode treats control-Z as EOF. Binary -- mode turns off all special treatment of end-of-line and end-of-file -- characters. (See also 'hSetBinaryMode'.) -- -- This provides a safer replacement for @System.IO.'SIO.openBinaryFile'@. openBinaryFile ∷ (MonadCatchIO pr, AbsRelClass ar) ⇒ FilePath ar → IOMode ioMode → RegionT s pr (RegionalFileHandle ioMode (RegionT s pr)) openBinaryFile = openNormal E.openBinaryFile {-| A convenience function which opens a file in binary mode, applies the given continuation function to the resulting regional file handle and runs the resulting region. This provides a safer replacement for @System.IO.'SIO.withBinaryFile'@. -} withBinaryFile ∷ (MonadCatchIO pr, AbsRelClass ar) ⇒ FilePath ar → IOMode ioMode → (∀ s. RegionalFileHandle ioMode (RegionT s pr) → RegionT s pr α) → pr α withBinaryFile filePath ioMode f = runRegionT $ openBinaryFile filePath ioMode >>= f -- ** Opening binary files by inferring the IOMode -- | Note that: @openBinaryFile' filePath = 'openBinaryFile' filePath 'mkIOMode'@. openBinaryFile' ∷ (MonadCatchIO pr, AbsRelClass ar, MkIOMode ioMode) ⇒ FilePath ar → RegionT s pr (RegionalFileHandle ioMode (RegionT s pr)) openBinaryFile' filePath = openBinaryFile filePath mkIOMode -- | Note that: @withBinaryFile' filePath = 'withBinaryFile' filePath 'mkIOMode'@. withBinaryFile' ∷ (MonadCatchIO pr, AbsRelClass ar, MkIOMode ioMode) ⇒ FilePath ar → (∀ s. RegionalFileHandle ioMode (RegionT s pr) → RegionT s pr α) → pr α withBinaryFile' filePath = withBinaryFile filePath mkIOMode -- ** Operations on binary handles -- | Select binary mode ('True') or text mode ('False') on a open handle. -- (See also 'openBinaryFile'.) -- -- This has the same effect as calling 'hSetEncoding' with 'latin1', together -- with 'hSetNewlineMode' with 'noNewlineTranslation'. -- -- Wraps: @System.IO.'SIO.hSetBinaryMode'@. hSetBinaryMode ∷ (pr `ParentOf` cr, MonadIO cr) ⇒ RegionalFileHandle ioMode pr → Bool → cr () hSetBinaryMode = wrap2 E.hSetBinaryMode -- | 'hPutBuf' @hdl buf count@ writes @count@ 8-bit bytes from the -- buffer @buf@ to the handle @hdl@. It returns (). -- -- 'hPutBuf' ignores any text encoding that applies to the 'RegionalFileHandle', -- writing the bytes directly to the underlying file or device. -- -- 'hPutBuf' ignores the prevailing 'TextEncoding' and -- 'NewlineMode' on the 'RegionalFileHandle', and writes bytes directly. -- -- This operation may fail with: -- -- * 'ResourceVanished' if the handle is a pipe or socket, and the -- reading end is closed. (If this is a POSIX system, and the program -- has not asked to ignore SIGPIPE, then a SIGPIPE may be delivered -- instead, whose default action is to terminate the program). -- -- Wraps: @System.IO.'SIO.hPutBuf'@. hPutBuf ∷ (pr `ParentOf` cr, MonadIO cr, WriteModes ioMode) ⇒ RegionalFileHandle ioMode pr → Ptr α → Int → cr () hPutBuf = wrap3 E.hPutBuf -- | 'hGetBuf' @hdl buf count@ reads data from the handle @hdl@ -- into the buffer @buf@ until either EOF is reached or -- @count@ 8-bit bytes have been read. -- It returns the number of bytes actually read. This may be zero if -- EOF was reached before any data was read (or if @count@ is zero). -- -- 'hGetBuf' ignores whatever 'TextEncoding' the 'RegionalFileHandle' is -- currently using, and reads bytes directly from the underlying IO device. -- -- 'hGetBuf' never raises an EOF exception, instead it returns a value -- smaller than @count@. -- -- If the handle is a pipe or socket, and the writing end -- is closed, 'hGetBuf' will behave as if EOF was reached. -- -- 'hGetBuf' ignores the prevailing 'TextEncoding' and 'NewlineMode' on the -- 'RegionalFileHandle', and reads bytes directly. -- -- Wraps: @System.IO.'SIO.hGetBuf'@. hGetBuf ∷ (pr `ParentOf` cr, MonadIO cr, ReadModes ioMode) ⇒ RegionalFileHandle ioMode pr → Ptr α → Int → cr Int hGetBuf = wrap3 E.hGetBuf #if !defined(__NHC__) && !defined(__HUGS__) -- | Wraps: @System.IO.'SIO.hPutBufNonBlocking'@. hPutBufNonBlocking ∷ (pr `ParentOf` cr, MonadIO cr, WriteModes ioMode) ⇒ RegionalFileHandle ioMode pr → Ptr α → Int → cr Int hPutBufNonBlocking = wrap3 E.hPutBufNonBlocking -- | 'hGetBufNonBlocking' @hdl buf count@ reads data from the handle @hdl@ -- into the buffer @buf@ until either EOF is reached, or -- @count@ 8-bit bytes have been read, or there is no more data available -- to read immediately. -- -- 'hGetBufNonBlocking' is identical to 'hGetBuf', except that it will -- never block waiting for data to become available, instead it returns -- only whatever data is available. To wait for data to arrive before -- calling 'hGetBufNonBlocking', use 'hWaitForInput'. -- -- 'hGetBufNonBlocking' ignores whatever 'TextEncoding' the 'RegionalFileHandle' -- is currently using, and reads bytes directly from the underlying IO device. -- -- If the handle is a pipe or socket, and the writing end -- is closed, 'hGetBufNonBlocking' will behave as if EOF was reached. -- -- 'hGetBufNonBlocking' ignores the prevailing 'TextEncoding' and 'NewlineMode' -- on the 'RegionalFileHandle', and reads bytes directly. -- -- Wraps: @System.IO.'SIO.hGetBufNonBlocking'@. hGetBufNonBlocking ∷ (pr `ParentOf` cr, MonadIO cr, ReadModes ioMode) ⇒ RegionalFileHandle ioMode pr → Ptr α → Int → cr Int hGetBufNonBlocking = wrap3 E.hGetBufNonBlocking #endif -------------------------------------------------------------------------------- -- * Temporary files -------------------------------------------------------------------------------- -- | The template of a temporary file path. -- -- If the template is \"foo.ext\" then the created file will be \"fooXXX.ext\" -- where XXX is some random number. type Template = RelFile openTemp ∷ (MonadCatchIO pr, AbsRelClass ar) ⇒ (E.FilePath → String → IO (E.FilePath, E.Handle ReadWriteMode)) → ( DirPath ar → Template → RegionT s pr ( AbsFile , RegionalFileHandle ReadWriteMode (RegionT s pr) ) ) openTemp open = \dirPath template → block $ do (fp, h) ← liftIO $ open (getPathString dirPath) (getPathString template) ch ← onExit $ sanitizeIOError $ hClose h return (asAbsFile fp, RegionalFileHandle h $ Just ch) -- | The function creates a temporary file in 'ReadWriteMode'. The created file -- isn\'t deleted automatically, so you need to delete it manually. -- -- The file is created with permissions such that only the current -- user can read\/write it. -- -- With some exceptions (see below), the file will be created securely in the -- sense that an attacker should not be able to cause 'openTempFile' to -- overwrite another file on the filesystem using your credentials, by putting -- symbolic links (on Unix) in the place where the temporary file is to be -- created. On Unix the @O_CREAT@ and @O_EXCL@ flags are used to prevent this -- attack, but note that @O_EXCL@ is sometimes not supported on NFS filesystems, -- so if you rely on this behaviour it is best to use local filesystems only. -- -- This provides a safer replacement for @System.IO.'SIO.openTempFile'@. openTempFile ∷ (MonadCatchIO pr, AbsRelClass ar) ⇒ DirPath ar -- ^ Directory in which to create the file. → Template -- ^ File name template. → RegionT s pr ( AbsFile , RegionalFileHandle ReadWriteMode (RegionT s pr) ) openTempFile = openTemp E.openTempFile -- | Like 'openTempFile', but opens the file in binary mode. See -- 'openBinaryFile' for more comments. -- -- This provides a safer replacement for @System.IO.'SIO.openBinaryTempFile'@. openBinaryTempFile ∷ (MonadCatchIO pr, AbsRelClass ar) ⇒ DirPath ar → Template → RegionT s pr ( AbsFile , RegionalFileHandle ReadWriteMode (RegionT s pr) ) openBinaryTempFile = openTemp E.openBinaryTempFile #if MIN_VERSION_base(4,2,0) -- | Like 'openTempFile', but uses the default file permissions. -- -- This provides a safer replacement for -- @System.IO.'SIO.openTempFileWithDefaultPermissions'@. openTempFileWithDefaultPermissions ∷ (MonadCatchIO pr, AbsRelClass ar) ⇒ DirPath ar → Template → RegionT s pr ( AbsFile , RegionalFileHandle ReadWriteMode (RegionT s pr) ) openTempFileWithDefaultPermissions = openTemp E.openTempFileWithDefaultPermissions -- | Like 'openBinaryTempFile', but uses the default file permissions. -- -- This provides a safer replacement for -- @System.IO.'SIO.openBinaryTempFileWithDefaultPermissions'@. openBinaryTempFileWithDefaultPermissions ∷ (MonadCatchIO pr, AbsRelClass ar) ⇒ DirPath ar → Template → RegionT s pr ( AbsFile , RegionalFileHandle ReadWriteMode (RegionT s pr) ) openBinaryTempFileWithDefaultPermissions = openTemp $ E.openBinaryTempFileWithDefaultPermissions #endif #if MIN_VERSION_base(4,2,0) && !defined(__NHC__) && !defined(__HUGS__) -------------------------------------------------------------------------------- -- * Unicode encoding/decoding -------------------------------------------------------------------------------- -- | The action 'hSetEncoding' @hdl@ @encoding@ changes the text encoding for -- the handle @hdl@ to @encoding@. The default encoding when a -- 'RegionalFileHandle' is created is 'localeEncoding', namely the default -- encoding for the current locale. -- -- To create a 'RegionalFileHandle' with no encoding at all, use -- 'openBinaryFile'. To stop further encoding or decoding on an existing -- 'RegionalFileHandle', use 'hSetBinaryMode'. -- -- 'hSetEncoding' may need to flush buffered data in order to change -- the encoding. -- -- Wraps: @System.IO.'SIO.hSetEncoding'@. hSetEncoding ∷ (pr `ParentOf` cr, MonadIO cr) ⇒ RegionalFileHandle ioMode pr → TextEncoding → cr () hSetEncoding = wrap2 E.hSetEncoding -- | Return the current 'TextEncoding' for the specified 'RegionalFileHandle', -- or 'Nothing' if the 'Handle' is in binary mode. -- -- Note that the 'TextEncoding' remembers nothing about the state of the -- encoder/decoder in use on this 'RegionalFileHandle'. For example, if the -- encoding in use is UTF-16, then using 'hGetEncoding' and 'hSetEncoding' to -- save and restore the encoding may result in an extra byte-order-mark being -- written to the file. -- -- Wraps: @System.IO.'SIO.hGetEncoding'@. hGetEncoding ∷ (pr `ParentOf` cr, MonadIO cr) ⇒ RegionalFileHandle ioMode pr → cr (Maybe TextEncoding) hGetEncoding = wrap E.hGetEncoding -- | Generalizes @System.IO.'SIO.mkTextEncoding'@ to any 'MonadIO'. mkTextEncoding ∷ MonadIO m ⇒ String → m TextEncoding mkTextEncoding = liftIO ∘ E.mkTextEncoding -------------------------------------------------------------------------------- -- * Newline conversion -------------------------------------------------------------------------------- -- | Set the 'NewlineMode' on the specified 'RegionalFileHandle'. All buffered -- data is flushed first. -- -- Wraps: @System.IO.'SIO.hSetNewlineMode'@. hSetNewlineMode ∷ (pr `ParentOf` cr, MonadIO cr) ⇒ RegionalFileHandle ioMode pr → NewlineMode → cr () hSetNewlineMode = wrap2 E.hSetNewlineMode #endif -- The End ---------------------------------------------------------------------