{-# LANGUAGE ExistentialQuantification #-}
{-# LANGUAGE ScopedTypeVariables #-}
{-# LANGUAGE PatternGuards #-}
{-# LANGUAGE RecordWildCards #-}
{-# LANGUAGE EmptyDataDecls #-}
{-# LANGUAGE StandaloneDeriving #-}
{-# LANGUAGE AllowAmbiguousTypes #-}
module Control.Distributed.Process.ManagedProcess.Server.Priority
(
prioritiseCall
, prioritiseCall_
, prioritiseCast
, prioritiseCast_
, prioritiseInfo
, prioritiseInfo_
, setPriority
, check
, raw
, raw_
, api
, api_
, info
, info_
, refuse
, reject
, rejectApi
, store
, storeM
, crash
, ensure
, ensureM
, Filter()
, DispatchFilter()
, safe
, apiSafe
, safely
, Message()
, evalAfter
, currentTimeout
, processState
, processDefinition
, processFilters
, processUnhandledMsgPolicy
, setUserTimeout
, setProcessState
, GenProcess
, peek
, push
, addUserTimer
, act
, runAfter
) where
import Control.Distributed.Process hiding (call, Message)
import qualified Control.Distributed.Process as P (Message)
import Control.Distributed.Process.Extras
( ExitReason(..)
)
import Control.Distributed.Process.ManagedProcess.Internal.GenProcess
( addUserTimer
, currentTimeout
, processState
, processDefinition
, processFilters
, processUnhandledMsgPolicy
, setUserTimeout
, setProcessState
, GenProcess
, peek
, push
, evalAfter
, act
, runAfter
)
import Control.Distributed.Process.ManagedProcess.Internal.Types
import Control.Distributed.Process.Serializable
import Prelude hiding (init)
data RejectedByServer = RejectedByServer deriving (Show)
data FilterHandler s =
forall m . (Serializable m) =>
HandlePure
{
pureCheck :: s -> m -> Process Bool
, handler :: s -> m -> Process (Filter s)
}
| forall m b . (Serializable m, Serializable b) =>
HandleApi
{
apiCheck :: s -> m -> Process Bool
, apiHandler :: s -> Message m b -> Process (Filter s)
}
| HandleRaw
{
rawCheck :: s -> P.Message -> Process Bool
, rawHandler :: s -> P.Message -> Process (Maybe (Filter s))
}
| HandleState { stateHandler :: s -> Process (Maybe (Filter s)) }
| HandleSafe
{
safeCheck :: s -> P.Message -> Process Bool
}
check :: forall s . FilterHandler s -> DispatchFilter s
check h
| HandlePure{..} <- h = FilterAny $ \s m -> pureCheck s m >>= procUnless s m handler
| HandleRaw{..} <- h = FilterRaw $ \s m -> do
c <- rawCheck s m
if c then return $ Just $ FilterOk s
else rawHandler s m
| HandleState{..} <- h = FilterState stateHandler
| HandleApi{..} <- h = FilterApi $ \s m@(CallMessage m' _) -> do
c <- apiCheck s m'
if c then return $ FilterOk s
else apiHandler s m
| HandleSafe{..} <- h = FilterRaw $ \s m -> do
c <- safeCheck s m
let ctr = if c then FilterSafe else FilterOk
return $ Just $ ctr s
where
procUnless s _ _ True = return $ FilterOk s
procUnless s m h' False = h' s m
raw :: forall s .
(s -> P.Message -> Process Bool)
-> (s -> P.Message -> Process (Maybe (Filter s)))
-> FilterHandler s
raw = HandleRaw
raw_ :: forall s .
(P.Message -> Process Bool)
-> (s -> P.Message -> Process (Maybe (Filter s)))
-> FilterHandler s
raw_ c h = raw (const $ c) h
api :: forall s m b . (Serializable m, Serializable b)
=> (s -> m -> Process Bool)
-> (s -> Message m b -> Process (Filter s))
-> FilterHandler s
api = HandleApi
api_ :: forall m b s . (Serializable m, Serializable b)
=> (m -> Process Bool)
-> (s -> Message m b -> Process (Filter s))
-> FilterHandler s
api_ c h = api (const $ c) h
info :: forall s m . (Serializable m)
=> (s -> m -> Process Bool)
-> (s -> m -> Process (Filter s))
-> FilterHandler s
info = HandlePure
info_ :: forall s m . (Serializable m)
=> (m -> Process Bool)
-> (s -> m -> Process (Filter s))
-> FilterHandler s
info_ c h = info (const $ c) h
apiSafe :: forall s m b . (Serializable m, Serializable b)
=> (s -> m -> Maybe b -> Bool)
-> DispatchFilter s
apiSafe c = check $ HandleSafe (go c)
where
go c' s (i :: P.Message) = do
m <- unwrapMessage i :: Process (Maybe (Message m b))
case m of
Just (CallMessage m' _) -> return $ c' s m' Nothing
Just (CastMessage m') -> return $ c' s m' Nothing
Just (ChanMessage m' _) -> return $ c' s m' Nothing
Nothing -> return False
safe :: forall s m . (Serializable m)
=> (s -> m -> Bool)
-> DispatchFilter s
safe c = check $ HandleSafe (go c)
where
go c' s (i :: P.Message) = do
m <- unwrapMessage i :: Process (Maybe m)
case m of
Just m' -> return $ c' s m'
Nothing -> return False
safely :: forall s . (s -> P.Message -> Bool) -> DispatchFilter s
safely c = check $ HandleSafe $ \s m -> return (c s m)
reject :: forall s m r . (Show r)
=> r -> s -> m -> Process (Filter s)
reject r = \s _ -> do return $ FilterReject (show r) s
crash :: forall s . s -> ExitReason -> Process (Filter s)
crash s r = return $ FilterStop s r
rejectApi :: forall s m b r . (Show r, Serializable m, Serializable b)
=> r -> s -> Message m b -> Process (Filter s)
rejectApi r = \s m -> do let r' = show r
rejectToCaller m r'
return $ FilterSkip s
store :: (s -> s) -> DispatchFilter s
store f = FilterState $ return . Just . FilterOk . f
storeM :: forall s m . (Serializable m)
=> (s -> m -> Process s)
-> DispatchFilter s
storeM proc = check $ HandlePure (\_ _ -> return True)
(\s m -> proc s m >>= return . FilterOk)
refuse :: forall s m . (Serializable m)
=> (m -> Bool)
-> DispatchFilter s
refuse c = check $ info (const $ \m -> return $ c m) (reject RejectedByServer)
ensure :: forall s . (s -> Bool) -> DispatchFilter s
ensure c =
check $ HandleState { stateHandler = (\s -> if c s
then return $ Just $ FilterOk s
else return $ Just $ FilterStop s filterFail)
}
ensureM :: forall s m . (Serializable m) => (s -> m -> Process Bool) -> DispatchFilter s
ensureM c =
check $ HandlePure { pureCheck = c
, handler = (\s _ -> return $ FilterStop s filterFail) :: s -> m -> Process (Filter s)
}
filterFail :: ExitReason
filterFail = ExitOther "Control.Distributed.Process.ManagedProcess.Priority:FilterFailed"
setPriority :: Int -> Priority m
setPriority n
| n < 1 = Priority 0
| n > 100 = Priority 100
| otherwise = Priority n
prioritiseCall_ :: forall s a b . (Serializable a, Serializable b)
=> (a -> Priority b)
-> DispatchPriority s
prioritiseCall_ h = prioritiseCall (const h)
prioritiseCall :: forall s a b . (Serializable a, Serializable b)
=> (s -> a -> Priority b)
-> DispatchPriority s
prioritiseCall h = PrioritiseCall (unCall . h)
where
unCall :: (a -> Priority b) -> P.Message -> Process (Maybe (Int, P.Message))
unCall h' m = fmap (matchPrioritise m h') (unwrapMessage m)
matchPrioritise :: P.Message
-> (a -> Priority b)
-> Maybe (Message a b)
-> Maybe (Int, P.Message)
matchPrioritise msg p msgIn
| (Just a@(CallMessage m _)) <- msgIn
, True <- isEncoded msg = Just (getPrio $ p m, wrapMessage a)
| (Just (CallMessage m _)) <- msgIn
, False <- isEncoded msg = Just (getPrio $ p m, msg)
| otherwise = Nothing
prioritiseCast_ :: forall s a . (Serializable a)
=> (a -> Priority ())
-> DispatchPriority s
prioritiseCast_ h = prioritiseCast (const h)
prioritiseCast :: forall s a . (Serializable a)
=> (s -> a -> Priority ())
-> DispatchPriority s
prioritiseCast h = PrioritiseCast (unCast . h)
where
unCast :: (a -> Priority ()) -> P.Message -> Process (Maybe (Int, P.Message))
unCast h' m = fmap (matchPrioritise m h') (unwrapMessage m)
matchPrioritise :: P.Message
-> (a -> Priority ())
-> Maybe (Message a ())
-> Maybe (Int, P.Message)
matchPrioritise msg p msgIn
| (Just a@(CastMessage m)) <- msgIn
, True <- isEncoded msg = Just (getPrio $ p m, wrapMessage a)
| (Just (CastMessage m)) <- msgIn
, False <- isEncoded msg = Just (getPrio $ p m, msg)
| otherwise = Nothing
prioritiseInfo_ :: forall s a . (Serializable a)
=> (a -> Priority ())
-> DispatchPriority s
prioritiseInfo_ h = prioritiseInfo (const h)
prioritiseInfo :: forall s a . (Serializable a)
=> (s -> a -> Priority ())
-> DispatchPriority s
prioritiseInfo h = PrioritiseInfo (unMsg . h)
where
unMsg :: (a -> Priority ()) -> P.Message -> Process (Maybe (Int, P.Message))
unMsg h' m = fmap (matchPrioritise m h') (unwrapMessage m)
matchPrioritise :: P.Message
-> (a -> Priority ())
-> Maybe a
-> Maybe (Int, P.Message)
matchPrioritise msg p msgIn
| (Just m') <- msgIn
, True <- isEncoded msg = Just (getPrio $ p m', wrapMessage m')
| (Just m') <- msgIn
, False <- isEncoded msg = Just (getPrio $ p m', msg)
| otherwise = Nothing