{-# LANGUAGE DeriveFunctor #-} {-# LANGUAGE ExplicitForAll #-} {-# LANGUAGE FlexibleInstances #-} {-# LANGUAGE GADTs #-} {-# LANGUAGE MultiParamTypeClasses #-} {-# LANGUAGE TypeOperators #-} {-# LANGUAGE UndecidableInstances #-} {- | A carrier for the 'State' effect. It evaluates its inner state strictly, which is the correct choice for the majority of use cases. Note that the parameter order in 'runState', 'evalState', and 'execState' is reversed compared the equivalent functions provided by @transformers@. This is an intentional decision made to enable the composition of effect handlers with '.' without invoking 'flip'. @since 1.0.0.0 -} module Control.Carrier.State.Strict ( -- * Strict state carrier runState , evalState , execState , StateC(..) -- * State effect , module Control.Effect.State ) where import Control.Algebra import Control.Applicative (Alternative(..)) import Control.Effect.State import Control.Monad (MonadPlus) import Control.Monad.Fail as Fail import Control.Monad.Fix import Control.Monad.IO.Class import Control.Monad.Trans.Class -- | Run a 'State' effect starting from the passed value. -- -- @ -- 'runState' s ('pure' a) = 'pure' (s, a) -- @ -- @ -- 'runState' s 'get' = 'pure' (s, s) -- @ -- @ -- 'runState' s ('put' t) = 'pure' (t, ()) -- @ -- -- @since 1.0.0.0 runState :: s -> StateC s m a -> m (s, a) runState s (StateC runStateC) = runStateC s {-# INLINE[3] runState #-} -- | Run a 'State' effect, yielding the result value and discarding the final state. -- -- @ -- 'evalState' s m = 'fmap' 'snd' ('runState' s m) -- @ -- -- @since 1.0.0.0 evalState :: forall s m a . Functor m => s -> StateC s m a -> m a evalState s = fmap snd . runState s {-# INLINE[3] evalState #-} -- | Run a 'State' effect, yielding the final state and discarding the return value. -- -- @ -- 'execState' s m = 'fmap' 'fst' ('runState' s m) -- @ -- -- @since 1.0.0.0 execState :: forall s m a . Functor m => s -> StateC s m a -> m s execState s = fmap fst . runState s {-# INLINE[3] execState #-} -- | @since 1.0.0.0 newtype StateC s m a = StateC (s -> m (s, a)) deriving (Functor) instance Monad m => Applicative (StateC s m) where pure a = StateC (\ s -> pure (s, a)) {-# INLINE pure #-} StateC f <*> StateC a = StateC $ \ s -> do (s', f') <- f s (s'', a') <- a s' pure (s'', f' a') {-# INLINE (<*>) #-} m *> k = m >>= const k {-# INLINE (*>) #-} instance (Alternative m, Monad m) => Alternative (StateC s m) where empty = StateC (const empty) {-# INLINE empty #-} StateC l <|> StateC r = StateC (\ s -> l s <|> r s) {-# INLINE (<|>) #-} instance Monad m => Monad (StateC s m) where StateC m >>= f = StateC $ \ s -> do (s', a) <- m s runState s' (f a) {-# INLINE (>>=) #-} instance Fail.MonadFail m => Fail.MonadFail (StateC s m) where fail s = StateC (const (Fail.fail s)) {-# INLINE fail #-} instance MonadFix m => MonadFix (StateC s m) where mfix f = StateC (\ s -> mfix (runState s . f . snd)) {-# INLINE mfix #-} instance MonadIO m => MonadIO (StateC s m) where liftIO io = StateC (\ s -> (,) s <$> liftIO io) {-# INLINE liftIO #-} instance (Alternative m, Monad m) => MonadPlus (StateC s m) instance MonadTrans (StateC s) where lift m = StateC (\ s -> (,) s <$> m) {-# INLINE lift #-} instance Algebra sig m => Algebra (State s :+: sig) (StateC s m) where alg hdl sig ctx = StateC $ \ s -> case sig of L Get -> pure (s, s <$ ctx) L (Put s) -> pure (s, ctx) R other -> thread (uncurry runState ~<~ hdl) other (s, ctx) {-# INLINE alg #-}