aboutsummaryrefslogtreecommitdiff
path: root/server/src
diff options
context:
space:
mode:
Diffstat (limited to 'server/src')
-rw-r--r--server/src/Thermoprint/Server/Printer.hs65
1 files changed, 58 insertions, 7 deletions
diff --git a/server/src/Thermoprint/Server/Printer.hs b/server/src/Thermoprint/Server/Printer.hs
index e66afff..0db98a0 100644
--- a/server/src/Thermoprint/Server/Printer.hs
+++ b/server/src/Thermoprint/Server/Printer.hs
@@ -1,29 +1,80 @@
1{-# LANGUAGE RankNTypes #-} 1{-# LANGUAGE RankNTypes #-}
2{-# LANGUAGE MultiParamTypeClasses #-}
3{-# LANGUAGE FlexibleContexts #-}
4{-# LANGUAGE RecordWildCards #-}
5{-# LANGUAGE OverloadedStrings #-}
6{-# LANGUAGE TemplateHaskell #-}
7{-# LANGUAGE StandaloneDeriving #-}
8{-# LANGUAGE DeriveGeneric, DeriveAnyClass #-}
2 9
3module Thermoprint.Server.Printer 10module Thermoprint.Server.Printer
4 ( Printer 11 ( Printer(..)
12 , Queue(..)
13 , runPrinter
5 ) where 14 ) where
6 15
7import Thermoprint.API (PrintingError(..), Printout) 16import Thermoprint.API (PrintingError(..), Printout)
17import qualified Thermoprint.API as API (JobStatus(..))
8 18
9import Thermoprint.Server.Database 19import Thermoprint.Server.Database
10 20
11import Database.Persist 21import Database.Persist
12import Database.Persist.Sql 22import Database.Persist.Sql
13 23
14import Data.Sequence (Seq, ViewL(..)) 24import Data.Sequence (Seq, ViewL(..), viewl, (<|))
15import qualified Data.Sequence as Seq 25import qualified Data.Sequence as Seq
16import Data.Map (Map) 26import Data.Map (Map)
17import qualified Data.Map as Map 27import qualified Data.Map as Map
18 28
19import Control.Monad.IO.Class (MonadIO) 29import qualified Data.Text as T (pack)
30
31import Data.Typeable (Typeable)
32import GHC.Generics (Generic)
33import Control.DeepSeq
34
35import Control.Monad.Trans.Resource
36import Control.Monad.IO.Class
37import Control.Monad.Logger
38import Control.Monad.Reader
39
40import Control.Monad (forever)
20 41
21import Control.Concurrent.STM 42import Control.Concurrent.STM
22 43
23data Printer = Printer 44data Printer = Printer
24 { print :: forall m. (MonadIO m) => Printout -> m (Maybe PrintingError) 45 { print :: forall m. (MonadIO m) => Printout -> m (Maybe PrintingError)
25 , queue :: TVar (Seq JobId) 46 , queue :: TVar Queue
26 } 47 }
27 48
28runPrinter :: Printer -> IO () 49-- | Zipper for 'Seq JobId'
29runPrinter = undefined 50data Queue = Queue
51 { pending :: Seq JobId -- ^ Pending jobs, closest first
52 , current :: Maybe JobId
53 , history :: Seq (JobId, Maybe PrintingError) -- ^ Completed jobs, closest first
54 }
55 deriving (Typeable, Generic, NFData)
56
57atomically' :: MonadIO m => STM a -> m a
58atomically' = liftIO . atomically
59
60runPrinter :: ( MonadReader ConnectionPool m
61 , MonadIO m
62 , MonadLogger m
63 , MonadBaseControl IO m
64 ) => Printer -> m ()
65-- ^ Loop 'forever' pushing entries from the 'Queue' associated to a 'Printer' into its 'print'-method
66runPrinter Printer{..} = forever $ do
67 jobId <- atomically' $ do
68 (Queue queuePending Nothing history) <- readTVar queue
69 case viewl queuePending of
70 EmptyL -> retry
71 (jobId :< remaining) -> do
72 writeTVar queue $!! Queue remaining (Just jobId) history
73 return jobId
74 job <- runSqlPool (get jobId) =<< ask
75 case job of
76 Nothing -> $(logWarn) "Nonexistent job id in printer queue"
77 Just job -> do
78 printReturn <- print (jobContent job)
79 maybe (return ()) ($(logWarn) . T.pack . (("Error while printing " ++ show jobId ++ ": ") ++) . show) $ printReturn
80 atomically' $ modifyTVar' queue (\Queue{..} -> force . Queue pending Nothing $ (jobId, printReturn) <| history)