module Bio.Iteratee.IO.Handle(
enumHandle
,enumHandleCatch
,enumHandleRandom
,enumFile
,enumFileRandom
,fileDriverHandle
,fileDriverRandomHandle
)
where
import Bio.Iteratee.Iteratee
import Bio.Prelude
import Control.Monad.Catch as CIO
import Control.Monad.IO.Class
import Data.ByteString (packCStringLen)
import Foreign.Marshal.Alloc
import System.IO
makeHandleCallback ::
MonadIO m =>
Ptr Word8
-> Int
-> Handle
-> st
-> m (Either SomeException ((Bool, st), Bytes))
makeHandleCallback p bsize h st = do
n' <- liftIO (CIO.try $ hGetBuf h p bsize :: IO (Either SomeException Int))
case n' of
Left e -> return $ Left e
Right 0 -> return $ Right ((False, st), emptyP)
Right n -> liftM (\s -> Right ((True, st), s)) $
readFromPtr p (fromIntegral n)
where
readFromPtr buf l = liftIO $ packCStringLen (castPtr buf, l)
enumHandle ::
(MonadIO m, MonadMask m) =>
Int
-> Handle
-> Enumerator Bytes m a
enumHandle bufsize h i =
CIO.bracket (liftIO $ mallocBytes bufsize)
(liftIO . free)
(\p -> enumFromCallback (makeHandleCallback p bufsize h) () i)
enumHandleCatch
::
forall e m a.(IException e,
MonadIO m, MonadMask m) =>
Int
-> Handle
-> (e -> m (Maybe EnumException))
-> Enumerator Bytes m a
enumHandleCatch bufsize h handler i =
CIO.bracket (liftIO $ mallocBytes bufsize)
(liftIO . free)
(\p -> enumFromCallbackCatch (makeHandleCallback p bufsize h) handler () i)
enumHandleRandom ::
forall m a.(MonadIO m, MonadMask m) =>
Int
-> Handle
-> Enumerator Bytes m a
enumHandleRandom bs h i = enumHandleCatch bs h handler i
where
handler (SeekException off) =
liftM (either
(Just . EnumException :: IOException -> Maybe EnumException)
(const Nothing))
. liftIO . CIO.try $ hSeek h AbsoluteSeek $ fromIntegral off
enumFile' :: (MonadIO m, MonadMask m) =>
(Int -> Handle -> Enumerator s m a)
-> Int
-> FilePath
-> Enumerator s m a
enumFile' enumf bufsize filepath iter = CIO.bracket
(liftIO $ openBinaryFile filepath ReadMode)
(liftIO . hClose)
(flip (enumf bufsize) iter)
enumFile ::
(MonadIO m, MonadMask m)
=> Int
-> FilePath
-> Enumerator Bytes m a
enumFile = enumFile' enumHandle
enumFileRandom ::
(MonadIO m, MonadMask m)
=> Int
-> FilePath
-> Enumerator Bytes m a
enumFileRandom = enumFile' enumHandleRandom
fileDriverHandle
:: (MonadIO m, MonadMask m) =>
Int
-> Iteratee Bytes m a
-> FilePath
-> m a
fileDriverHandle bufsize iter filepath =
enumFile bufsize filepath iter >>= run
fileDriverRandomHandle
:: (MonadIO m, MonadMask m) =>
Int
-> Iteratee Bytes m a
-> FilePath
-> m a
fileDriverRandomHandle bufsize iter filepath =
enumFileRandom bufsize filepath iter >>= run