module Magicbane.App (
module X
, module Magicbane.App
) where
import ClassyPrelude
import Control.Monad.Trans.Except as X
import Control.Monad.Except as X (MonadError, throwError)
import Data.Proxy as X
import Data.Has as X
import Servant as X hiding (And)
newtype MagicbaneApp β α = MagicbaneApp {
unMagicbaneApp ∷ ReaderT β (ExceptT ServantErr IO) α
} deriving (Functor, Applicative, Monad, MonadIO, MonadBase IO,
MonadThrow, MonadCatch, MonadError ServantErr, MonadReader β)
instance MonadBaseControl IO (MagicbaneApp β) where
type StM (MagicbaneApp β) α = StM (ReaderT β (ExceptT ServantErr IO)) α
liftBaseWith f = MagicbaneApp $ liftBaseWith $ \x → f $ x . unMagicbaneApp
restoreM = MagicbaneApp . restoreM
runMagicbaneExcept ∷ β → MagicbaneApp β α → ExceptT ServantErr IO α
runMagicbaneExcept ctx a = ExceptT $ liftIO $ runExceptT $ runReaderT (unMagicbaneApp a) ctx
magicbaneToExcept ∷ β → MagicbaneApp β :~> ExceptT ServantErr IO
#if MIN_VERSION_servant_server(0,10,0)
magicbaneToExcept ctx = NT $ runMagicbaneExcept ctx
#else
magicbaneToExcept ctx = Nat $ runMagicbaneExcept ctx
#endif
magicbaneApp api sctx ctx actions = serveWithContext api sctx $ srv ctx
where srv c = enter (magicbaneToExcept c) actions
askObj ∷ (Has β α, MonadReader α μ) ⇒ μ β
askObj = asks getter
askOpt ∷ (Has β α, MonadReader α μ) ⇒ (β → ψ) → μ ψ
askOpt f = asks $ f . getter