aboutsummaryrefslogtreecommitdiff
path: root/server/src
diff options
context:
space:
mode:
Diffstat (limited to 'server/src')
-rw-r--r--server/src/Thermoprint/Server.hs24
-rw-r--r--server/src/Thermoprint/Server/QMConfig.hs129
2 files changed, 17 insertions, 136 deletions
diff --git a/server/src/Thermoprint/Server.hs b/server/src/Thermoprint/Server.hs
index c2a4972..446c63e 100644
--- a/server/src/Thermoprint/Server.hs
+++ b/server/src/Thermoprint/Server.hs
@@ -4,6 +4,7 @@
4{-# LANGUAGE TypeOperators #-} 4{-# LANGUAGE TypeOperators #-}
5{-# LANGUAGE FlexibleContexts #-} 5{-# LANGUAGE FlexibleContexts #-}
6{-# LANGUAGE ImpredicativeTypes #-} 6{-# LANGUAGE ImpredicativeTypes #-}
7{-# LANGUAGE ExistentialQuantification #-}
7{-# LANGUAGE ViewPatterns #-} 8{-# LANGUAGE ViewPatterns #-}
8{-# LANGUAGE DataKinds #-} 9{-# LANGUAGE DataKinds #-}
9 10
@@ -14,8 +15,7 @@ module Thermoprint.Server
14 , module Data.Default.Class 15 , module Data.Default.Class
15 , module Servant.Server.Internal.Enter 16 , module Servant.Server.Internal.Enter
16 , module Thermoprint.Server.Printer 17 , module Thermoprint.Server.Printer
17 , module Thermoprint.Server.QMConfig 18 , module Thermoprint.Server.Queue
18 , Queue(..), QueueEntry(..)
19 ) where 19 ) where
20 20
21import Data.Default.Class 21import Data.Default.Class
@@ -78,8 +78,7 @@ import Thermoprint.Server.Push
78 78
79import Thermoprint.Server.Database 79import Thermoprint.Server.Database
80import Thermoprint.Server.Printer 80import Thermoprint.Server.Printer
81import Thermoprint.Server.Queue hiding (intersection, idQM, union, nullQM, runQM) 81import Thermoprint.Server.Queue
82import Thermoprint.Server.QMConfig
83import qualified Thermoprint.Server.API as API (thermoprintServer) 82import qualified Thermoprint.Server.API as API (thermoprintServer)
84import Thermoprint.Server.API hiding (thermoprintServer) 83import Thermoprint.Server.API hiding (thermoprintServer)
85 84
@@ -89,9 +88,17 @@ import Debug.Trace
89data Config m = Config { dyreError :: Maybe String -- ^ Set by 'Dyre' -- sent to log as an error 88data Config m = Config { dyreError :: Maybe String -- ^ Set by 'Dyre' -- sent to log as an error
90 , warpSettings :: Warp.Settings -- ^ Configure 'Warp's behaviour 89 , warpSettings :: Warp.Settings -- ^ Configure 'Warp's behaviour
91 , printers :: Map API.PrinterId Printer 90 , printers :: Map API.PrinterId Printer
92 , queueManagers :: API.PrinterId -> QMConfig 91 , queueManagers :: API.PrinterId -> QMConfig m
93 } 92 }
94 93
94data QMConfig m = forall t. ( MonadTrans t
95 , MFunctor t
96 , Monad (t STM)
97 , MonadIO (t IO)
98 ) => QMConfig { manager :: QueueManager t
99 , collapse :: (t IO) :~> m
100 }
101
95instance MonadIO m => Default (Config m) where 102instance MonadIO m => Default (Config m) where
96 def = Config { dyreError = Nothing 103 def = Config { dyreError = Nothing
97 , warpSettings = Warp.defaultSettings 104 , warpSettings = Warp.defaultSettings
@@ -99,7 +106,10 @@ instance MonadIO m => Default (Config m) where
99 , queueManagers = const def 106 , queueManagers = const def
100 } 107 }
101 108
102withPrinters :: MonadResource m => Config m -> [(m PrinterMethod, QMConfig)] -> m (Config m) 109instance MonadIO m => Default (QMConfig m) where
110 def = QMConfig idQM $ Nat (liftIO . runIdentityT)
111
112withPrinters :: MonadResource m => Config m -> [(m PrinterMethod, QMConfig m)] -> m (Config m)
103-- ^ Add a list of printers to a 'Config' 113-- ^ Add a list of printers to a 'Config'
104withPrinters cfg = fmap updateCfg . foldlM mapInsert (Map.mapWithKey (\k a -> (a, queueManagers cfg k)) $ printers cfg) 114withPrinters cfg = fmap updateCfg . foldlM mapInsert (Map.mapWithKey (\k a -> (a, queueManagers cfg k)) $ printers cfg)
105 where 115 where
@@ -140,7 +150,7 @@ thermoprintServer dyre io = Dyre.wrapMain $ Dyre.defaultParams
140 gcChan <- liftIO newTChanIO 150 gcChan <- liftIO newTChanIO
141 fork tMgr $ jobGC gcChan 151 fork tMgr $ jobGC gcChan
142 let 152 let
143 runQM' = liftIO . runQM gcChan . queueManagers 153 runQM' (queueManagers -> QMConfig qm nat) printer = unNat nat $ runQM gcChan qm printer
144 mapM_ (fork tMgr . uncurry runQM') $ Map.toList printers 154 mapM_ (fork tMgr . uncurry runQM') $ Map.toList printers
145 nChan <- liftIO $ newBroadcastTChanIO 155 nChan <- liftIO $ newBroadcastTChanIO
146 let 156 let
diff --git a/server/src/Thermoprint/Server/QMConfig.hs b/server/src/Thermoprint/Server/QMConfig.hs
deleted file mode 100644
index 7255c8c..0000000
--- a/server/src/Thermoprint/Server/QMConfig.hs
+++ /dev/null
@@ -1,129 +0,0 @@
1{-# LANGUAGE GADTs #-}
2{-# LANGUAGE FlexibleContexts #-}
3{-# LANGUAGE ImpredicativeTypes #-}
4{-# LANGUAGE TypeOperators #-}
5{-# LANGUAGE RecordWildCards #-}
6{-# LANGUAGE ViewPatterns #-}
7{-# LANGUAGE ConstraintKinds #-}
8{-# LANGUAGE MultiParamTypeClasses #-}
9{-# LANGUAGE KindSignatures #-}
10{-# LANGUAGE FlexibleInstances #-}
11{-# LANGUAGE UndecidableInstances #-}
12{-# LANGUAGE IncoherentInstances #-}
13
14module Thermoprint.Server.QMConfig
15 ( QMConfig(..)
16 , nullQM
17 , idQM
18 , standardSleep
19 , limitHistorySize
20 , limitHistoryAge
21 ) where
22
23import Control.Monad.Trans.Resource
24import Control.Monad.Trans.Control
25import Control.Monad.Trans.Identity
26import Control.Monad.Logger
27import Control.Monad.Reader
28import Control.Monad.IO.Class
29import Control.Monad.Morph
30import Control.Monad.Trans.Compose
31import Control.Monad.State
32import Control.Category
33import Servant.Server.Internal.Enter
34
35import Data.Constraint
36import Data.Constraint.Forall
37
38import Data.Default.Class
39
40import Data.Sequence (Seq)
41import qualified Data.Sequence as Seq
42import Data.Time
43import Data.Foldable
44
45import Control.Concurrent.STM
46
47import Thermoprint.Server.Queue hiding (idQM, intersection, nullQM, union, runQM)
48import qualified Thermoprint.Server.Queue as Q
49
50import Thermoprint.Server.Database (JobId)
51
52import Prelude hiding (length, maximum, id, (.))
53
54type QMTrans t = ( MonadTrans t
55 , MFunctor t
56 , Monad (t STM)
57 , MonadIO (t IO)
58 , Monad (QueueManagerM t)
59 , MonadState Queue (QueueManagerM t)
60 )
61
62class QMTrans1 (t :: (* -> *) -> * -> *) (t' :: (* -> *) -> * -> *)
63instance QMTrans (ComposeT t t') => QMTrans1 t t'
64
65type QMTrans' t = (QMTrans t, Forall (QMTrans1 t))
66
67data QMConfig where
68 QMConfig :: QMTrans' t => QueueManager t -> (forall m. (t m) :~> m) -> QMConfig
69 QMConfig' :: ( forall t . QMTrans' t => QueueManager t
70 ) -> QMConfig
71
72instance Default QMConfig where
73 def = idQM
74
75intersection :: Foldable f => f QMConfig -> QMConfig
76intersection = foldr' (qmCombine Q.intersection) idQM
77
78idQM :: QMConfig
79idQM = QMConfig' Q.idQM
80
81union :: Foldable f => f QMConfig -> QMConfig
82union = foldr' (qmCombine Q.union) idQM
83
84nullQM :: QMConfig
85nullQM = QMConfig' Q.nullQM
86
87qmCombine :: (forall f t. (Foldable f, MonadState Queue (QueueManagerM t)) => f (QueueManager t) -> QueueManager t) -> QMConfig -> QMConfig -> QMConfig
88qmCombine qmCombine' (QMConfig a natA) (QMConfig b natB) = (QMConfig (qmCombine' [a', b']) (natComp natA natB)) \\ (inst :: Forall (QMTrans1 t) :- QMTrans (ComposeT t t'))
89 where
90 a' = mapComposeT (hoist $ ComposeT . hoist lift) a
91 b' = mapComposeT (hoist $ ComposeT . lift) b
92
93natComp :: (forall m. t m :~> m)
94 -> (forall m. t' m :~> m)
95 -> (forall m. (MFunctor t
96 , Monad (t' m)
97 ) => ComposeT t t' m :~> m
98 )
99natComp natA natB = natA . hoistNat natB . Nat getComposeT
100
101runQM :: ( HasQueue q ) => TChan JobId -> QMConfig -> q -> IO ()
102runQM gcChan (QMConfig qm nat) q = unNat nat $ Q.runQM gcChan qm q
103runQM gcChan (QMConfig' qm) q = runIdentityT $ Q.runQM gcChan qm q
104
105standardSleep :: Monad (QueueManagerM t) => QueueManager t
106-- ^ Instruct 'runQM' to sleep some standard amount of time
107--
108-- /TODO/: Investigate implementing a smarter algorithm (PID-controller)
109standardSleep = return $ 20
110
111limitHistorySize :: Int -> QMConfig
112-- ^ Limit a 'Queue's 'history's size to some number of 'QueueEntry's
113limitHistorySize max = QMConfig' $ modify' limitSize >> standardSleep
114 where
115 limitSize :: Queue -> Queue
116 limitSize q@Queue{..} = q { history = Seq.take max history }
117
118limitHistoryAge :: NominalDiffTime -- ^ Maximum age relative to the youngest 'QueueEntry'
119 -> QMConfig
120-- ^ Limit a 'Queue's 'history's contents to 'QueueEntry's below a certain age
121limitHistoryAge maxAge = QMConfig' $ modify' limitAge >> standardSleep
122 where
123 limitAge :: Queue -> Queue
124 limitAge q@Queue{..} = q { history = Seq.filter (filterAge . created . fst) history}
125 where
126 youngest :: UTCTime
127 youngest = maximum $ created . fst <$> history
128 filterAge :: UTCTime -> Bool
129 filterAge time = not $ (youngest `diffUTCTime` time) > maxAge