module Katip.Monadic
(
logFM
, logTM
, logItemM
, logExceptionM
, KatipContext(..)
, AnyLogContext
, LogContexts
, liftPayload
, KatipContextT(..)
, runKatipContextT
, KatipContextTState(..)
) where
import Control.Applicative
import Control.Monad.Base
import Control.Monad.Catch
import Control.Monad.Error.Class
import Control.Monad.IO.Class
import Control.Monad.Reader
import Control.Monad.State
import Control.Monad.Trans.Control
import Control.Monad.Trans.Either (EitherT)
import Control.Monad.Trans.Except (ExceptT)
import Control.Monad.Trans.Identity (IdentityT)
import Control.Monad.Trans.List (ListT)
import Control.Monad.Trans.Maybe (MaybeT)
import Control.Monad.Trans.Resource (ResourceT)
import Control.Monad.Trans.RWS (RWST)
import qualified Control.Monad.Trans.RWS.Strict as Strict (RWST)
import qualified Control.Monad.Trans.State.Strict as Strict (StateT)
import qualified Control.Monad.Trans.Writer.Strict as Strict (WriterT)
import Control.Monad.Writer
import Data.Aeson
import qualified Data.HashMap.Strict as HM
import Data.Monoid as M
import Data.Text (Text)
import Language.Haskell.TH
import Katip.Core
data AnyLogContext where
AnyLogContext :: (LogItem a) => a -> AnyLogContext
newtype LogContexts = LogContexts [AnyLogContext] deriving (Monoid)
instance ToJSON LogContexts where
toJSON (LogContexts cs) =
Object $ mconcat $ map (\(AnyLogContext v) -> toObject v) cs
instance ToObject LogContexts
instance LogItem LogContexts where
payloadKeys verb (LogContexts vs) = mconcat $ map payloadKeys' vs
where
payloadKeys' (AnyLogContext v) = case payloadKeys verb v of
AllKeys -> SomeKeys $ HM.keys $ toObject v
x -> x
liftPayload :: (LogItem a) => a -> LogContexts
liftPayload = LogContexts . (:[]) . AnyLogContext
class Katip m => KatipContext m where
getKatipContext :: m LogContexts
getKatipNamespace :: m Namespace
#define TRANS(T) \
instance (KatipContext m, Katip (T m)) => KatipContext (T m) where \
getKatipContext = lift getKatipContext; \
getKatipNamespace = lift getKatipNamespace
#define TRANS_CTX(CTX, T) \
instance (CTX, KatipContext m, Katip (T m)) => KatipContext (T m) where \
getKatipContext = lift getKatipContext; \
getKatipNamespace = lift getKatipNamespace
TRANS(IdentityT)
TRANS(MaybeT)
TRANS(EitherT e)
TRANS(ListT)
TRANS(ReaderT r)
TRANS(ResourceT)
TRANS(Strict.StateT s)
TRANS(StateT s)
TRANS(ExceptT s)
TRANS_CTX(Monoid w, Strict.WriterT w)
TRANS_CTX(Monoid w, WriterT w)
TRANS_CTX(Monoid w, Strict.RWST r w s)
TRANS_CTX(Monoid w, RWST r w s)
deriving instance (Monad m, KatipContext m) => KatipContext (KatipT m)
logItemM
:: (Applicative m, KatipContext m, Katip m)
=> Maybe Loc
-> Severity
-> LogStr
-> m ()
logItemM loc sev msg = do
ctx <- getKatipContext
ns <- getKatipNamespace
logItem ctx ns loc sev msg
logFM
:: (Applicative m, KatipContext m, Katip m)
=> Severity
-> LogStr
-> m ()
logFM sev msg = do
ctx <- getKatipContext
ns <- getKatipNamespace
logF ctx ns sev msg
logTM :: ExpQ
logTM = [| logItemM (Just $(getLoc)) |]
logExceptionM
:: (KatipContext m, MonadCatch m, Applicative m)
=> m a
-> Severity
-> m a
logExceptionM action sev = action `catchAll` \e -> f e >> throwM e
where
f e = logFM sev (msg e)
msg e = ls ("An exception has occured: " :: Text) M.<> showLS e
newtype KatipContextT m a = KatipContextT {
unKatipContextT :: ReaderT KatipContextTState m a
} deriving ( Functor
, Applicative
, Monad
, MonadIO
, MonadThrow
, MonadCatch
, MonadMask
, MonadBase b
, MonadState s
, MonadWriter w
, MonadError e
, MonadPlus
, Alternative
, MonadFix
, MonadTrans
)
data KatipContextTState = KatipContextTState {
ltsLogEnv :: !LogEnv
, ltsContext :: !LogContexts
, ltsNamespace :: !Namespace
}
instance MonadTransControl KatipContextT where
type StT KatipContextT a = StT (ReaderT KatipContextTState) a
liftWith = defaultLiftWith KatipContextT unKatipContextT
restoreT = defaultRestoreT KatipContextT
instance (MonadBaseControl b m) => MonadBaseControl b (KatipContextT m) where
type StM (KatipContextT m) a = ComposeSt KatipContextT m a
liftBaseWith = defaultLiftBaseWith
restoreM = defaultRestoreM
instance (MonadReader r m) => MonadReader r (KatipContextT m) where
ask = lift ask
local f (KatipContextT (ReaderT m)) = KatipContextT $ ReaderT $ \r ->
local f (m r)
instance (MonadIO m) => Katip (KatipContextT m) where
getLogEnv = KatipContextT $ ReaderT $ \lts -> return (ltsLogEnv lts)
instance (MonadIO m) => KatipContext (KatipContextT m) where
getKatipContext = KatipContextT $ ReaderT $ \lts -> return (ltsContext lts)
getKatipNamespace = KatipContextT $ ReaderT $ \lts -> return (ltsNamespace lts)
runKatipContextT :: (LogItem c) => LogEnv -> c -> Namespace -> KatipContextT m a -> m a
runKatipContextT le ctx ns = flip runReaderT lts . unKatipContextT
where
lts = KatipContextTState le (liftPayload ctx) ns