module Data.Iteratee.Iteratee (
Cont
,EnumerateeHandler
,throwErr
,throwRecoverableErr
,throwRec
,checkErr
,identity
,skipToEof
,isStreamFinished
,mapChunksM_
,foldChunksM
,getChunk
,getChunks
,mapChunks
,mapChunksM
,convStream
,unfoldConvStream
,joinI
,joinIM
,Enumerator
,Enumeratee
,enumChunk
,enumEof
,enumErr
,enumPure1Chunk
,enumList
,enumCheckIfDone
,enumFromCallback
,enumFromCallbackCatch
,(>>>)
,eneeCheckIfDone
,eneeCheckIfDoneHandle
,eneeCheckIfDonePass
,eneeCheckIfDoneIgnore
,mergeEnums
,($=)
,(=$)
,(><>)
,(<><)
,CBState (..)
,Callback
,seek
,FileOffset
,module Data.Iteratee.Base
)
where
import Prelude hiding (head, drop, dropWhile, take, break, foldl, foldl1, length, filter, sum, product)
import Data.Iteratee.IO.Base
import Data.Iteratee.Base
import Control.Arrow (first, (***))
import Control.Exception
import Control.Monad
import Control.Monad.Trans.Class
import Data.Maybe
import Data.Typeable
excDivergent :: SomeException
excDivergent = toException DivergentException
throwErr :: SomeException -> Iteratee s m a
throwErr e = ierr (throwErr e) e
throwRecoverableErr :: (Exception e) => e -> Iteratee s m a -> Iteratee s m a
throwRecoverableErr e i = ierr i (toException e)
throwRec :: (Exception e) => e -> Iteratee s m a -> Iteratee s m a
throwRec = throwRecoverableErr
checkErr :: Monad m => Iteratee s m a -> Iteratee s m (Either SomeException a)
checkErr iter = runIter iter (idone . Right) oc oe oR
where
oc k = icont (liftM (first checkErr) . k)
oe _ e = idone (Left e)
oR mb doB = ireq mb (checkErr . doB)
identity :: Iteratee s m ()
identity = idone ()
isStreamFinished :: (Nullable s, Monad m) => Iteratee s m (Maybe SomeException)
isStreamFinished = icontP check
where
check s@(Chunk xs)
| nullC xs = (isStreamFinished, s)
| otherwise = (idone Nothing, s)
check s@(EOF e) = (idone (Just $ fromMaybe (toException EofException) e), s)
skipToEof :: (NullPoint s, Monad m) => Iteratee s m ()
skipToEof = icontP check
where
check (Chunk _) = (skipToEof, Chunk empty)
check s = (idone (), s)
seek :: (NullPoint s) => FileOffset -> Iteratee s m ()
seek o = throwRec (SeekException o) identity
mapChunksM_ :: (Monad m, Nullable s, NullPoint s)
=> (s -> m b)
-> Iteratee s m ()
mapChunksM_ f = icont step
where
step s@(Chunk xs)
| nullC xs = return (icont step, s)
| otherwise = f xs >> return (icont step, Chunk empty)
step s@(EOF _) = return (idone (), s)
foldChunksM :: (Monad m, Nullable s, NullPoint s)
=> (a -> s -> m a)
-> a
-> Iteratee s m a
foldChunksM f = icont . go
where
go a (Chunk c) = f a c >>= \a' -> return (icont (go a'), Chunk empty)
go a e = return (idone a, e)
getChunk :: (Monad m, Nullable s, NullPoint s) => Iteratee s m s
getChunk = icontP step
where
step s@(Chunk xs)
| nullC xs = (icontP step, s)
| otherwise = (idone xs, Chunk empty)
step s@(EOF Nothing) = (throwRec EofException getChunk, s)
step s@(EOF (Just e)) = (throwRec e getChunk, s)
getChunks :: (Monad m, Nullable s, NullPoint s) => Iteratee s m [s]
getChunks = icontP (step id)
where
step acc s@(Chunk xs)
| nullC xs = (icontP (step acc), s)
| otherwise = (icontP (step $ acc . (xs:)), s)
step acc stream = (idone (acc []), stream)
type Enumeratee sFrom sTo (m :: * -> *) a =
Iteratee sTo m a
-> Iteratee sFrom m (Iteratee sTo m a)
eneeCheckIfDone ::
(Monad m, NullPoint elo) =>
(Cont eli m a -> Iteratee elo m (Iteratee eli m a))
-> Enumeratee elo eli m a
eneeCheckIfDone = eneeCheckIfDonePass
type Cont s m a = Stream s -> m (Iteratee s m a, Stream s)
type EnumerateeHandler eli elo m a =
Iteratee eli m a
-> SomeException
-> Iteratee elo m (Iteratee eli m a)
eneeCheckIfDoneHandle
:: forall m eli elo a. (NullPoint elo)
=> EnumerateeHandler eli elo m a
-> (Cont eli m a -> Iteratee elo m (Iteratee eli m a))
-> Enumeratee elo eli m a
eneeCheckIfDoneHandle h fc inner = worker inner
where
worker i = runIter i onDone fc h onReq
onDone x = idone (idone x)
onReq :: forall b. m b
-> (b -> Iteratee eli m a)
-> Iteratee elo m (Iteratee eli m a)
onReq mb doB = ireq mb (worker . doB)
eneeCheckIfDonePass
:: (NullPoint elo)
=> (Cont eli m a -> Iteratee elo m (Iteratee eli m a))
-> Enumeratee elo eli m a
eneeCheckIfDonePass f = worker
where
worker = eneeCheckIfDoneHandle handler f
handler i = ierr (worker i)
eneeCheckIfDoneIgnore
:: (NullPoint elo)
=> (Cont eli m a -> Iteratee elo m (Iteratee eli m a))
-> Enumeratee elo eli m a
eneeCheckIfDoneIgnore f = worker
where
worker = eneeCheckIfDoneHandle handler f
handler i _e = worker i
mapChunks :: (Monad m, NullPoint s) => (s -> s') -> Enumeratee s s' m a
mapChunks f = go
where
go = eneeCheckIfDonePass (icont . step)
step k (Chunk xs) = k (Chunk (f xs)) >>= \(i',_) ->
return (go i', Chunk empty)
step k (EOF mErr) = (idone *** const (EOF mErr)) `liftM` k (EOF mErr)
mapChunksM
:: (Monad m, NullPoint s, Nullable s)
=> (s -> m s')
-> Enumeratee s s' m a
mapChunksM f = go
where
go = eneeCheckIfDonePass (icont . step)
step k (Chunk xs) = f xs >>= k . Chunk >>= \(i', _str) ->
return (go i', Chunk empty)
step k (EOF mErr) = (idone *** const (EOF mErr)) `liftM` k (EOF mErr)
convStream :: forall s s' m a.
(Monad m, Nullable s) =>
Iteratee s m s'
-> Enumeratee s s' m a
convStream fi = go
where
go = eneeCheckIfDonePass check
check k = isStreamFinished >>= maybe (step k) (hndl k)
hndl k e = case fromException e of
Just EofException -> idone (icont k)
_ -> ierr (step k) e
step k = fi >>= lift . k . Chunk >>= go . fst
unfoldConvStream ::
(Monad m, Nullable s) =>
(acc -> Iteratee s m (acc, s'))
-> acc
-> Enumeratee s s' m a
unfoldConvStream fi acc0 = unfoldConvStreamCheck eneeCheckIfDonePass fi acc0
unfoldConvStreamCheck
:: (Monad m, Nullable elo)
=> ((Cont eli m a -> Iteratee elo m (Iteratee eli m a))
-> Enumeratee elo eli m a
)
-> (acc -> Iteratee elo m (acc, eli))
-> acc
-> Enumeratee elo eli m a
unfoldConvStreamCheck checkDone f acc0 = go acc0
where
go acc = checkDone (check acc)
check acc k = isStreamFinished >>= maybe (step acc k) (hndl acc k)
hndl acc k e = case fromException e of
Just EofException -> idone (icont k)
_ -> ierr (step acc k) e
step acc k = do
(acc', s') <- f acc
(i', _) <- lift . k $ Chunk s'
go acc' i'
joinI ::
(Monad m, Nullable s) =>
Iteratee s m (Iteratee s' m a)
-> Iteratee s m a
joinI i = runIter i onDone onCont onErr onR
where
onDone i' = ireq (tryRun i') (either throwErr return)
onCont k = icont $ \str -> first joinI `liftM` k str
onErr i' e = throwRec e (joinI i')
onR mb doB = lift mb >>= joinI . doB
joinIM :: (Monad m) => m (Iteratee s m a) -> Iteratee s m a
joinIM mIter = ireq mIter id
type Enumerator s m a = Iteratee s m a -> m (Iteratee s m a)
enumChunk :: (Monad m) => Stream s -> Enumerator s m a
enumChunk (Chunk xs) = enumPure1Chunk xs
enumChunk (EOF Nothing) = enumEof
enumChunk (EOF (Just e)) = enumErr e
enumEof :: (Monad m) => Enumerator s m a
enumEof iter = runIter iter idoneM onC ierrM onR
where
onC k = k (EOF Nothing) >>= \(i,_) -> runIter i idoneM onC' ierrM onR
onC' _k = return $ throwErr excDivergent
onR mb doB = mb >>= enumEof . doB
enumErr :: (Exception e, Monad m) => e -> Enumerator s m a
enumErr e iter = runIter iter idoneM onCont ierrM onR
where
onCont k = do
(i',_) <- k . EOF . Just $ toException e
runIter i' idoneM onCont' ierrM onR
onCont' _ = return $ throwErr excDivergent
onR mb doB = mb >>= enumErr e . doB
(>>>) :: (Monad m) => Enumerator s m a -> Enumerator s m a -> Enumerator s m a
(e1 >>> e2) i = e1 i >>= e2
infixr 0 =$
(=$)
:: (Nullable s, Nullable s', Monad m)
=> Enumeratee s s' m a
-> Iteratee s' m a
-> Iteratee s m a
(=$) = (.) joinI
infixl 1 $=
($=)
:: (Nullable s, Nullable s', Monad m)
=> (forall a. Enumerator s m a)
-> Enumeratee s s' m b
-> Enumerator s' m b
($=) enum enee iter = enum (enee iter) >>= run
(><>) ::
(Nullable s1, Monad m)
=> (forall x . Enumeratee s1 s2 m x)
-> Enumeratee s2 s3 m a
-> Enumeratee s1 s3 m a
f ><> g = joinI . f . g
(<><) ::
(Nullable s1, Monad m)
=> Enumeratee s2 s3 m a
-> (forall x. Enumeratee s1 s2 m x)
-> Enumeratee s1 s3 m a
f <>< g = joinI . g . f
mergeEnums ::
(Nullable s2, Nullable s1, Monad m)
=> Enumerator s1 m a
-> Enumerator s2 (Iteratee s1 m) a
-> Enumeratee s2 s1 (Iteratee s1 m) a
-> Enumerator s1 m a
mergeEnums e1 e2 etee i = e1 $ e2 (joinI . etee $ ilift lift i) >>= run
enumPure1Chunk :: (Monad m) => s -> Enumerator s m a
enumPure1Chunk str iter = runIter iter idoneM onC ierrM onR
where
onC k = fst `liftM` k (Chunk str)
onR mb doB = mb >>= enumPure1Chunk str . doB
enumList :: (Monad m) => [s] -> Enumerator s m a
enumList chunks = go chunks
where
go [] i = return i
go xs' i = runIter i idoneM (onCont xs') onErr (onReq xs')
where
onCont (x:xs) k = k (Chunk x) >>= go xs . fst
onCont [] k = return $ icont k
onErr iRes e = return $ throwRec e iRes
onReq xs mb doB = mb >>= go xs . doB
enumCheckIfDone :: (Monad m) => Iteratee s m a -> m (Bool, Iteratee s m a)
enumCheckIfDone iter = runIter iter onDone onCont onErr onReq
where
onDone x = return (True, idone x)
onCont k = return (False, icont k)
onErr i e = return (False, ierr i e)
onReq mb doB = mb >>= enumCheckIfDone . doB
enumFromCallback ::
(Monad m, NullPoint s) =>
Callback st m s
-> st
-> Enumerator s m a
enumFromCallback c =
enumFromCallbackCatch c (\NotAnException -> return Nothing)
data NotAnException = NotAnException
deriving (Show, Typeable)
instance Exception NotAnException where
instance IException NotAnException where
data CBState = HasMore | Finished deriving (Eq, Show, Ord, Enum)
type Callback st m s = st -> m (Either SomeException ((CBState, st), s))
enumFromCallbackCatch
:: forall e m s st a. (IException e, Monad m, NullPoint s)
=> Callback st m s
-> (e -> m (Maybe EnumException))
-> st
-> Enumerator s m a
enumFromCallbackCatch c handler = loop
where
loop st iter = runIter iter idoneM (onCont st) (onErr st) (onReq st)
onCont st k = c st >>= either (liftM fst . k . EOF . Just) (check k)
onErr st i e = case fromException e of
Just e' -> handler e' >>=
maybe (loop st i)
(return . ierr i) . fmap toException
Nothing -> return (ierr i e)
onReq :: st -> m x -> (x -> Iteratee s m a) -> m (Iteratee s m a)
onReq st mb doB = mb >>= loop st . doB
check :: (Stream s -> m (Iteratee s m a, Stream s))
-> ((CBState, st), s)
-> m (Iteratee s m a)
check k ((HasMore, st'), s) = k (Chunk s) >>= loop st' . fst
check k ((Finished,_st'), s) = fst `liftM` k (Chunk s)