module Control.Monad.Catana (
Catana(..),
consume,
top,
push,
produce,
stop,
evalCatana,
evalCatana',
parallelB,
parallelE,
serial
) where
import Control.Applicative
import Control.Monad
import Control.Monad.Cont
import Data.Either
import Data.Maybe
data Catana i o b a = Catana { runCatana :: (a -> Step i o b) -> Step i o b }
data Step i o a = Yield o (Step i o a)
| Waiting (i -> Step i o a)
| Done a
instance Functor (Catana i o b) where
fmap f (Catana l) = Catana $ l . (. f)
instance Applicative (Catana i o b) where
Catana fl <*> x = Catana $ \k ->
fl $ \f ->
runCatana (fmap f x) k
pure x = Catana $ \k -> k x
instance Monad (Catana i o b) where
return = pure
Catana l >>= f = Catana $ \k ->
l $ \x ->
runCatana (f x) k
instance MonadCont (Catana i o b) where
callCC f = Catana $ \k -> runCatana (f $ Catana . const . k) k
consume :: Catana i o a i
consume = Catana Waiting
top :: Catana i o a i
top = Catana $ \k -> Waiting (\i -> feed i (k i))
feed :: i -> Step i o a -> Step i o a
feed i (Yield o s) = Yield o (feed i s)
feed i (Waiting s) = s i
feed _ (Done x) = Done x
stop :: b -> Catana i o b a
stop = Catana . const . Done
push :: i -> Catana i o a ()
push x = Catana $ feed x . ($())
produce :: o -> Catana i o a ()
produce x = Catana $ Yield x . ($())
evalCatana :: Catana i o a a -> [i] -> [o]
evalCatana c = runSteps (runCatana c Done)
runSteps :: Step i o a -> [i] -> [o]
runSteps (Yield o s) i = o : runSteps s i
runSteps (Waiting f) (i:is) = runSteps (f i) is
runSteps (Waiting _) [] = []
runSteps (Done x) _ = []
evalCatana' :: Catana i o a a -> [i] -> (Maybe a, [o])
evalCatana' c i = (listToMaybe x, o)
where (x, o) = partitionEithers $ runSteps' (runCatana c Done) i
runSteps' :: Step i o a -> [i] -> [Either a o]
runSteps' (Yield o s) i = Right o : runSteps' s i
runSteps' (Waiting f) (i:is) = runSteps' (f i) is
runSteps' (Waiting _) [] = []
runSteps' (Done x) _ = [Left x]
parStepsB :: Step i o a -> Step i o b -> ((a,b) -> Step i o c) -> Step i o c
parStepsB (Yield oA sA) (Yield oB sB) k = Yield oA . Yield oB $ parStepsB sA sB k
parStepsB (Yield oA sA) sB k = Yield oA $ parStepsB sA sB k
parStepsB sA (Yield oB sB) k = Yield oB $ parStepsB sA sB k
parStepsB (Waiting fA) (Waiting fB) k = Waiting $ \i -> parStepsB (fA i) (fB i) k
parStepsB (Waiting fA) sB k = Waiting $ \i -> parStepsB (fA i) sB k
parStepsB sA (Waiting fB) k = Waiting $ \i -> parStepsB sA (fB i) k
parStepsB (Done xA) (Done xB) k = k (xA, xB)
parallelB :: Catana i o a a -> Catana i o b b -> Catana i o c (a, b)
parallelB a b = Catana $ parStepsB (runCatana a Done) (runCatana b Done)
parStepsE :: Step i o a -> Step i o b -> (Either a b -> Step i o c) -> Step i o c
parStepsE (Yield oA sA) (Yield oB sB) k = Yield oA . Yield oB $ parStepsE sA sB k
parStepsE (Yield oA sA) sB k = Yield oA $ parStepsE sA sB k
parStepsE sA (Yield oB sB) k = Yield oB $ parStepsE sA sB k
parStepsE (Done xA) _ k = k (Left xA)
parStepsE _ (Done xB) k = k (Right xB)
parStepsE (Waiting fA) (Waiting fB) k = Waiting $ \i -> parStepsE (fA i) (fB i) k
parallelE :: Catana i o a a -> Catana i o b b -> Catana i o c (Either a b)
parallelE a b = Catana $ parStepsE (runCatana a Done) (runCatana b Done)
serSteps :: Step io o a -> Step i io b -> (Either a b -> Step i o c) -> Step i o c
serSteps (Yield oA sA) sB k = Yield oA $ serSteps sA sB k
serSteps (Done xA) _ k = k (Left xA)
serSteps _ (Done xB) k = k (Right xB)
serSteps (Waiting fA) (Yield oB sB) k = serSteps (fA oB) sB k
serSteps sA (Waiting fB) k = Waiting $ \i -> serSteps sA (fB i) k
serial :: Catana io o a a -> Catana i io b b -> Catana i o c (Either a b)
serial a b = Catana $ serSteps (runCatana a Done) (runCatana b Done)