aboutsummaryrefslogtreecommitdiff
path: root/server/test/Thermoprint
diff options
context:
space:
mode:
Diffstat (limited to 'server/test/Thermoprint')
-rw-r--r--server/test/Thermoprint/Server/QueueSpec.hs13
-rw-r--r--server/test/Thermoprint/ServerSpec.hs126
2 files changed, 103 insertions, 36 deletions
diff --git a/server/test/Thermoprint/Server/QueueSpec.hs b/server/test/Thermoprint/Server/QueueSpec.hs
index 4a9297e..8a6bb7b 100644
--- a/server/test/Thermoprint/Server/QueueSpec.hs
+++ b/server/test/Thermoprint/Server/QueueSpec.hs
@@ -16,19 +16,6 @@ import Test.QuickCheck.Modifiers
16deriving instance (Eq PrintingError) 16deriving instance (Eq PrintingError)
17deriving instance (Eq Queue) 17deriving instance (Eq Queue)
18 18
19instance Arbitrary Queue where
20 arbitrary = Queue <$> arbitrary <*> arbitrary <*> arbitrary
21
22instance Arbitrary QueueEntry where
23 arbitrary = QueueEntry <$> arbitrary <*> arbitrary
24
25instance Arbitrary PrintingError where
26 arbitrary = oneof [ return (IOError "dummy")
27 ]
28
29instance Arbitrary JobId where
30 arbitrary = castId . getNonNegative <$> (arbitrary :: Gen (NonNegative Integer))
31
32spec :: Spec 19spec :: Spec
33spec = do 20spec = do
34 describe "queue morphisms" $ do 21 describe "queue morphisms" $ do
diff --git a/server/test/Thermoprint/ServerSpec.hs b/server/test/Thermoprint/ServerSpec.hs
index 495ad10..deb2b9c 100644
--- a/server/test/Thermoprint/ServerSpec.hs
+++ b/server/test/Thermoprint/ServerSpec.hs
@@ -1,66 +1,146 @@
1{-# LANGUAGE OverloadedStrings #-} 1{-# LANGUAGE OverloadedStrings #-}
2{-# LANGUAGE OverloadedLists #-}
2{-# LANGUAGE ImpredicativeTypes #-} 3{-# LANGUAGE ImpredicativeTypes #-}
4{-# LANGUAGE RecordWildCards #-}
5{-# LANGUAGE StandaloneDeriving #-}
3 6
4module Thermoprint.ServerSpec (spec) where 7module Thermoprint.ServerSpec (spec) where
5 8
6import Test.Hspec 9import Test.Hspec
10import Test.Hspec.QuickCheck
11import qualified Test.Hspec as Hspec
12
13import Test.QuickCheck
7 14
8import Thermoprint.API 15import Thermoprint.API
9import Thermoprint.Server 16import qualified Thermoprint.Server as S
17import Thermoprint.Client
10 18
19import Data.Monoid
20import Data.Function
11import Control.Monad 21import Control.Monad
12import Control.Monad.Logger 22import Control.Monad.Logger
13import Control.Monad.Reader 23import Control.Monad.Reader
24import Control.Monad.State
14import Control.Monad.Trans.Identity 25import Control.Monad.Trans.Identity
15import Control.Monad.Trans.Resource 26import Control.Monad.Trans.Resource
27import Control.Monad.Catch (finally)
16 28
17import Database.Persist.Sqlite 29import Database.Persist.Sqlite
18 30
19import Control.Concurrent 31import Control.Concurrent
20import Control.Concurrent.STM 32import Control.Concurrent.STM
33import Control.Concurrent.STM.TSem
21 34
22import System.IO 35import System.IO
23import System.IO.Temp 36import System.IO.Temp
24 37
25import qualified Data.Text as T 38import qualified Data.Text as T
26 39
27import Debug.Trace 40import Network.Wai.Handler.Warp (defaultSettings, setBeforeMainLoop)
41
42import qualified Data.Map as Map
43
44deriving instance Eq PrintingError
45deriving instance Eq JobStatus
46deriving instance Eq PrinterStatus
47
48-- Equality via cotext on Block
49instance Eq Block where
50 (==) = (==) `on` cotext
51-- Structural equality for Chunk
52deriving instance Eq Chunk
28 53
29data TestPrinter = TestPrinter 54data TestPrinter = TestPrinter
30 { outputChan :: TChan Printout 55 { outputChan :: TMVar Printout
31 , failSwitch :: TMVar PrintingError 56 , failSwitch :: TMVar PrintingError
32 } 57 }
33 58
34data TestManager = TestManager 59data TestManager = TestManager
35 { manage :: TMVar (QueueManager IdentityT) 60 { manage :: TMVar (S.QueueManager IdentityT)
61 , ran :: TSem
36 } 62 }
37 63
38setup :: IO (ThreadId, QSem, TestPrinter, TestManager) 64data RunningServer = RunningServer
39setup = withSystemTempFile "thermoprint.sqlite" $ \fp h -> hClose h >> do 65 { thread :: ThreadId
40 tPrinter <- TestPrinter <$> newTChanIO <*> newEmptyTMVarIO 66 , termination :: QSem
41 tManager <- TestManager <$> newEmptyTMVarIO 67 , printer :: TestPrinter
68 , manager :: TestManager
69 }
70
71setup :: QSem -> IO RunningServer
72setup startup = withSystemTempFile "thermoprint.sqlite" $ \fp h -> hClose h >> do
73 tPrinter <- TestPrinter <$> newEmptyTMVarIO <*> newEmptyTMVarIO
74 tManager <- TestManager <$> newEmptyTMVarIO <*> atomically (newTSem 0)
42 termSem <- newQSem 0 75 termSem <- newQSem 0
43 let 76 let
44 runSqlite :: ReaderT ConnectionPool (NoLoggingT IO) a -> IO a 77 runSqlite :: ReaderT ConnectionPool (LoggingT IO) a -> IO a
45 runSqlite = runNoLoggingT . withSqlitePool (T.pack fp) 1 . runReaderT 78 runSqlite = runStderrLoggingT . filterLogger (const (> LevelDebug)) . withSqlitePool (T.pack fp) 1 . runReaderT
46 79
47 printers = [ ( pure $ PM tPM 80 printers = [ ( pure $ S.PM tPM
48 , QMConfig (join . lift $ takeTMVar (manage tManager)) (Nat $ liftIO . runIdentityT) 81 , S.QMConfig ((join . lift $ takeTMVar (manage tManager) <* signalTSem (ran tManager)) >> return 0) (Nat $ liftIO . runIdentityT)
49 ) 82 )
50 ] 83 ]
51 84
52 tPM :: MonadIO m => Printout -> m (Maybe PrintingError) 85 tPM :: MonadIO m => Printout -> m (Maybe PrintingError)
53 tPM printout = liftIO . atomically $ writeTChan (outputChan tPrinter) printout >> tryTakeTMVar (failSwitch tPrinter) 86 tPM printout = liftIO . atomically $ putTMVar (outputChan tPrinter) printout >> tryTakeTMVar (failSwitch tPrinter)
54 (,,,) <$> forkFinally (thermoprintServer False (Nat runSqlite) $ def `withPrinters` printers) (const $ signalQSem termSem) <*> pure termSem <*> pure tPrinter <*> pure tManager 87 RunningServer <$> forkFinally (S.thermoprintServer False (Nat runSqlite) $ def' `S.withPrinters` printers) (const $ signalQSem termSem) <*> pure termSem <*> pure tPrinter <*> pure tManager
55 88 where
56withSetup :: SpecWith (ThreadId, QSem, TestPrinter, TestManager) -> Spec 89 def' :: MonadIO m => S.Config m
57withSetup = beforeAll setup . afterAll (\(tId, termSem, _, _) -> killThread tId >> waitQSem termSem) 90 def' = S.def { S.warpSettings = setBeforeMainLoop (signalQSem startup) $ defaultSettings }
91
92withSetup :: SpecWith RunningServer -> Spec
93withSetup = beforeAll setup' . afterAll teardown
94 where
95 setup' = do
96 startup <- newQSem 0
97 setup startup <* waitQSem startup
98 teardown RunningServer{..} = killThread thread >> waitQSem termination
58 99
59spec :: Spec 100spec :: Spec
60spec = withSetup $ do 101spec = withSetup $ do
61 describe "blubTests" $ do 102 it "Reports initial server state" $ \RunningServer{..} -> do
62 it "prints Blub." $ \(tId, _, _, _) -> do 103 printers `shouldReturn` [(0, Available)]
63 threadDelay 5000 104 jobs Nothing Nothing Nothing `shouldReturn` []
64 putStrLn "Blub." 105 drafts `shouldReturn` []
65 System.IO.print tId 106 it "Reports printing errors" $ \RunningServer{..} -> do
66 True `shouldSatisfy` id 107 let
108 err = IOError "test"
109 atomically $ putTMVar (failSwitch printer) err
110 jId <- jobCreate Nothing mempty
111 atomically . takeTMVar $ outputChan printer
112 jobStatus jId `shouldReturn` (Failed err)
113 -- it "Queues any Printout" $ \RunningServer{..} -> property $ \p -> do
114 -- jId <- jobCreate (Just 0) p
115 -- (atomically . takeTMVar . outputChan $ printer) `shouldReturn` p
116 it "Reports qualitative queue position" $ \RunningServer{..} -> do
117 jids <- replicateM 3 $ jobCreate (Just 0) mempty
118 zipWithM_ (\jid ret -> jobStatus jid `shouldReturn` ret) jids
119 [ Done
120 , Printing 0
121 , Queued 0
122 ]
123 replicateM_ (length jids) . atomically . takeTMVar . outputChan $ printer
124 zipWithM_ (\jid ret -> jobStatus jid `shouldReturn` ret) jids (replicate (length jids) Done)
125 it "Keeps history" $ \RunningServer{..} -> do
126 jobs Nothing Nothing Nothing `shouldNotReturn` []
127 it "Runs queue managers" $ \RunningServer{..} -> do
128 atomically $ putTMVar (manage manager) S.nullQM
129 atomically $ waitTSem (ran manager)
130 jobs Nothing Nothing Nothing `shouldReturn` []
131 it "Handles drafts" $ \RunningServer{..} -> do
132 drafts `shouldReturn` []
133 dId <- draftCreate Nothing mempty
134 draft dId `shouldReturn` (Nothing, mempty)
135 drafts `shouldReturn` [(dId, mempty)]
136 p <- generate arbitrary
137 draftReplace dId (Just "Title") p
138 draft dId `shouldReturn` (Just "Title", p)
139 jId <- draftPrint dId Nothing
140 (atomically . takeTMVar $ outputChan printer) `shouldReturn` p
141 draftDelete dId
142 drafts `shouldReturn` []
143 where
144 Client{..} = mkClient' $ BaseUrl Http "localhost" 3000
145
146