{-# LANGUAGE FlexibleContexts #-}
{-# LANGUAGE DeriveDataTypeable #-}
{-# LANGUAGE NoMonomorphismRestriction #-}
{-# LANGUAGE Rank2Types #-}
{-# LANGUAGE RankNTypes #-}
module Data.Conduit.Cereal ( GetException
, sinkGet
, conduitGet
, conduitGet2
, sourcePut
, conduitPut
) where
import Control.Exception.Base
import Control.Monad.Trans.Resource (MonadThrow, throwM)
import qualified Data.ByteString as BS
import qualified Data.ByteString.Lazy as LBS
import Data.Conduit (ConduitT, leftover, await, yield)
import qualified Data.Conduit.List as CL
import Data.Serialize hiding (get, put)
import Data.Typeable
import Data.Conduit.Cereal.Internal
data GetException = GetException String
deriving (Int -> GetException -> ShowS
[GetException] -> ShowS
GetException -> String
(Int -> GetException -> ShowS)
-> (GetException -> String)
-> ([GetException] -> ShowS)
-> Show GetException
forall a.
(Int -> a -> ShowS) -> (a -> String) -> ([a] -> ShowS) -> Show a
showList :: [GetException] -> ShowS
$cshowList :: [GetException] -> ShowS
show :: GetException -> String
$cshow :: GetException -> String
showsPrec :: Int -> GetException -> ShowS
$cshowsPrec :: Int -> GetException -> ShowS
Show, Typeable)
instance Exception GetException
conduitGet :: MonadThrow m => Get o -> ConduitT BS.ByteString o m ()
conduitGet :: Get o -> ConduitT ByteString o m ()
conduitGet = ConduitErrorHandler m o -> Get o -> ConduitT ByteString o m ()
forall (m :: * -> *) o.
Monad m =>
ConduitErrorHandler m o -> Get o -> ConduitT ByteString o m ()
mkConduitGet ConduitErrorHandler m o
forall (m :: * -> *) a. MonadThrow m => String -> m a
errorHandler
where errorHandler :: String -> m a
errorHandler msg :: String
msg = GetException -> m a
forall (m :: * -> *) e a. (MonadThrow m, Exception e) => e -> m a
throwM (GetException -> m a) -> GetException -> m a
forall a b. (a -> b) -> a -> b
$ String -> GetException
GetException String
msg
{-# DEPRECATED conduitGet "Please switch to conduitGet2, see comment on that function" #-}
sinkGet :: MonadThrow m => Get r -> ConduitT BS.ByteString o m r
sinkGet :: Get r -> ConduitT ByteString o m r
sinkGet = SinkErrorHandler m r
-> SinkTerminationHandler m r -> Get r -> ConduitT ByteString o m r
forall (m :: * -> *) r o.
Monad m =>
SinkErrorHandler m r
-> SinkTerminationHandler m r -> Get r -> ConduitT ByteString o m r
mkSinkGet SinkErrorHandler m r
forall (m :: * -> *) a. MonadThrow m => String -> m a
errorHandler SinkTerminationHandler m r
forall (m :: * -> *) a o.
MonadThrow m =>
(ByteString -> Result a) -> ConduitT ByteString o m a
terminationHandler
where errorHandler :: String -> m a
errorHandler msg :: String
msg = GetException -> m a
forall (m :: * -> *) e a. (MonadThrow m, Exception e) => e -> m a
throwM (GetException -> m a) -> GetException -> m a
forall a b. (a -> b) -> a -> b
$ String -> GetException
GetException String
msg
terminationHandler :: (ByteString -> Result a) -> ConduitT ByteString o m a
terminationHandler f :: ByteString -> Result a
f = case ByteString -> Result a
f ByteString
BS.empty of
Fail msg :: String
msg _ -> GetException -> ConduitT ByteString o m a
forall (m :: * -> *) e a. (MonadThrow m, Exception e) => e -> m a
throwM (GetException -> ConduitT ByteString o m a)
-> GetException -> ConduitT ByteString o m a
forall a b. (a -> b) -> a -> b
$ String -> GetException
GetException String
msg
Done r :: a
r lo :: ByteString
lo -> ByteString -> ConduitT ByteString o m ()
forall i o (m :: * -> *). i -> ConduitT i o m ()
leftover ByteString
lo ConduitT ByteString o m ()
-> ConduitT ByteString o m a -> ConduitT ByteString o m a
forall (m :: * -> *) a b. Monad m => m a -> m b -> m b
>> a -> ConduitT ByteString o m a
forall (m :: * -> *) a. Monad m => a -> m a
return a
r
Partial _ -> GetException -> ConduitT ByteString o m a
forall (m :: * -> *) e a. (MonadThrow m, Exception e) => e -> m a
throwM (GetException -> ConduitT ByteString o m a)
-> GetException -> ConduitT ByteString o m a
forall a b. (a -> b) -> a -> b
$ String -> GetException
GetException "Failed reading: Internal error: unexpected Partial."
sourcePut :: Monad m => Put -> ConduitT i BS.ByteString m ()
sourcePut :: Put -> ConduitT i ByteString m ()
sourcePut put :: Put
put = [ByteString] -> ConduitT i ByteString m ()
forall (m :: * -> *) a i. Monad m => [a] -> ConduitT i a m ()
CL.sourceList ([ByteString] -> ConduitT i ByteString m ())
-> [ByteString] -> ConduitT i ByteString m ()
forall a b. (a -> b) -> a -> b
$ ByteString -> [ByteString]
LBS.toChunks (ByteString -> [ByteString]) -> ByteString -> [ByteString]
forall a b. (a -> b) -> a -> b
$ Put -> ByteString
runPutLazy Put
put
conduitPut :: Monad m => Putter a -> ConduitT a BS.ByteString m ()
conduitPut :: Putter a -> ConduitT a ByteString m ()
conduitPut p :: Putter a
p = (a -> ByteString) -> ConduitT a ByteString m ()
forall (m :: * -> *) a b. Monad m => (a -> b) -> ConduitT a b m ()
CL.map ((a -> ByteString) -> ConduitT a ByteString m ())
-> (a -> ByteString) -> ConduitT a ByteString m ()
forall a b. (a -> b) -> a -> b
$ Put -> ByteString
runPut (Put -> ByteString) -> Putter a -> a -> ByteString
forall b c a. (b -> c) -> (a -> b) -> a -> c
. Putter a
p
conduitGet2 :: MonadThrow m => Get o -> ConduitT BS.ByteString o m ()
conduitGet2 :: Get o -> ConduitT ByteString o m ()
conduitGet2 get :: Get o
get =
ConduitT ByteString o m ByteString
forall (m :: * -> *) o.
Monad m =>
ConduitT ByteString o m ByteString
awaitNE ConduitT ByteString o m ByteString
-> (ByteString -> ConduitT ByteString o m ())
-> ConduitT ByteString o m ()
forall (m :: * -> *) a b. Monad m => m a -> (a -> m b) -> m b
>>= ByteString -> ConduitT ByteString o m ()
forall (m :: * -> *).
MonadThrow m =>
ByteString -> ConduitT ByteString o m ()
start
where
awaitNE :: ConduitT ByteString o m ByteString
awaitNE =
ConduitT ByteString o m ByteString
forall (m :: * -> *) o.
Monad m =>
ConduitT ByteString o m ByteString
loop
where
loop :: ConduitT ByteString o m ByteString
loop = ConduitT ByteString o m (Maybe ByteString)
forall (m :: * -> *) i. Monad m => Consumer i m (Maybe i)
await ConduitT ByteString o m (Maybe ByteString)
-> (Maybe ByteString -> ConduitT ByteString o m ByteString)
-> ConduitT ByteString o m ByteString
forall (m :: * -> *) a b. Monad m => m a -> (a -> m b) -> m b
>>= ConduitT ByteString o m ByteString
-> (ByteString -> ConduitT ByteString o m ByteString)
-> Maybe ByteString
-> ConduitT ByteString o m ByteString
forall b a. b -> (a -> b) -> Maybe a -> b
maybe (ByteString -> ConduitT ByteString o m ByteString
forall (m :: * -> *) a. Monad m => a -> m a
return ByteString
BS.empty) ByteString -> ConduitT ByteString o m ByteString
check
check :: ByteString -> ConduitT ByteString o m ByteString
check bs :: ByteString
bs
| ByteString -> Bool
BS.null ByteString
bs = ConduitT ByteString o m ByteString
loop
| Bool
otherwise = ByteString -> ConduitT ByteString o m ByteString
forall (m :: * -> *) a. Monad m => a -> m a
return ByteString
bs
start :: ByteString -> ConduitT ByteString o m ()
start bs :: ByteString
bs
| ByteString -> Bool
BS.null ByteString
bs = () -> ConduitT ByteString o m ()
forall (m :: * -> *) a. Monad m => a -> m a
return ()
| Bool
otherwise = Result o -> ConduitT ByteString o m ()
result (Get o -> ByteString -> Result o
forall a. Get a -> ByteString -> Result a
runGetPartial Get o
get ByteString
bs)
result :: Result o -> ConduitT ByteString o m ()
result (Fail msg :: String
msg _) = GetException -> ConduitT ByteString o m ()
forall (m :: * -> *) e a. (MonadThrow m, Exception e) => e -> m a
throwM (String -> GetException
GetException String
msg)
result (Partial f :: ByteString -> Result o
f) = ConduitT ByteString o m ByteString
forall (m :: * -> *) o.
Monad m =>
ConduitT ByteString o m ByteString
awaitNE ConduitT ByteString o m ByteString
-> (ByteString -> ConduitT ByteString o m ())
-> ConduitT ByteString o m ()
forall (m :: * -> *) a b. Monad m => m a -> (a -> m b) -> m b
>>= Result o -> ConduitT ByteString o m ()
result (Result o -> ConduitT ByteString o m ())
-> (ByteString -> Result o)
-> ByteString
-> ConduitT ByteString o m ()
forall b c a. (b -> c) -> (a -> b) -> a -> c
. ByteString -> Result o
f
result (Done x :: o
x rest :: ByteString
rest) = do
o -> ConduitT ByteString o m ()
forall (m :: * -> *) o i. Monad m => o -> ConduitT i o m ()
yield o
x
if ByteString -> Bool
BS.null ByteString
rest
then ConduitT ByteString o m ByteString
forall (m :: * -> *) o.
Monad m =>
ConduitT ByteString o m ByteString
awaitNE ConduitT ByteString o m ByteString
-> (ByteString -> ConduitT ByteString o m ())
-> ConduitT ByteString o m ()
forall (m :: * -> *) a b. Monad m => m a -> (a -> m b) -> m b
>>= ByteString -> ConduitT ByteString o m ()
start
else ByteString -> ConduitT ByteString o m ()
start ByteString
rest