aboutsummaryrefslogtreecommitdiff
path: root/server/test/Thermoprint/ServerSpec.hs
blob: 334f78501f7a9631adaf2f622b4fc0f2d38e90b2 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
{-# LANGUAGE OverloadedStrings  #-}
{-# LANGUAGE OverloadedLists  #-}
{-# LANGUAGE ImpredicativeTypes #-}
{-# LANGUAGE RecordWildCards #-}
{-# LANGUAGE StandaloneDeriving #-}

module Thermoprint.ServerSpec (spec) where

import           Test.Hspec
import           Test.Hspec.QuickCheck
import qualified Test.Hspec as Hspec

import           Test.QuickCheck

import           Thermoprint.API
import qualified Thermoprint.Server as S
import           Thermoprint.Client

import           Data.Monoid
import           Data.Function
import           Control.Monad
import           Control.Monad.Logger
import           Control.Monad.Reader
import           Control.Monad.State
import           Control.Monad.Trans.Identity
import           Control.Monad.Trans.Resource
import           Control.Monad.Catch (finally)

import           Database.Persist.Sqlite

import           Control.Concurrent
import           Control.Concurrent.STM
import           Control.Concurrent.STM.TSem
import           Control.Concurrent.Async

import           Control.Exception
import           System.IO.Error

import           System.IO
import           System.IO.Temp

import qualified Data.Text as T

import           Network.Wai.Handler.Warp (defaultSettings, setBeforeMainLoop)
import           Network.HTTP.Types.Status (Status(..))

import qualified Data.Map as Map

deriving instance Eq PrintingError
deriving instance Eq JobStatus
deriving instance Eq PrinterStatus

-- Equality via cotext on Block
instance Eq Block where
  (==) = (==) `on` cotext
-- Structural equality for Chunk 
deriving instance Eq Chunk

data TestPrinter = TestPrinter
  { outputChan :: TMVar Printout
  , failSwitch :: TMVar PrintingError
  }

data TestManager = TestManager
  { manage :: TMVar (S.QueueManager IdentityT)
  , ran :: TSem
  }

data RunningServer = RunningServer
  { thread :: ThreadId
  , termination :: MVar (Either SomeException ())
  , startup :: QSem
  , printer :: TestPrinter
  , manager :: TestManager
  }

setup :: QSem -> IO RunningServer
setup startup = withSystemTempFile "thermoprint.sqlite" $ \fp h -> hClose h >> do
  tPrinter <- TestPrinter <$> newEmptyTMVarIO <*> newEmptyTMVarIO
  tManager <- TestManager <$> newEmptyTMVarIO <*> atomically (newTSem 0)
  term <- newEmptyMVar
  startSem <- newQSem 0
  let
    runSqlite :: ReaderT ConnectionPool (LoggingT IO) a -> IO a
    runSqlite = runStderrLoggingT . filterLogger (const (> LevelDebug)) . withSqlitePool (T.pack fp) 1 . runReaderT

    printers = [ ( pure $ S.PM tPM
                 , S.QMConfig ((join . lift $ takeTMVar (manage tManager) <* signalTSem (ran tManager)) >> return 0) (Nat $ liftIO . runIdentityT)
                 )
               ]

    tPM :: MonadIO m => Printout -> m (Maybe PrintingError)
    tPM printout = liftIO . atomically $ putTMVar (outputChan tPrinter) printout >> tryTakeTMVar (failSwitch tPrinter)
  RunningServer <$> forkFinally (S.thermoprintServer False (Nat runSqlite) $ def' `S.withPrinters` printers) (putMVar term) <*> pure term <*> pure startSem <*> pure tPrinter <*> pure tManager
  where
    def' :: MonadIO m => S.Config m
    def' = S.def { S.warpSettings = setBeforeMainLoop (signalQSem startup) $ defaultSettings }

withSetup :: SpecWith RunningServer -> Spec
withSetup = beforeAll setup' . afterAll teardown
  where
    setup' = do
      startup <- newQSem 0
      r <- setup startup
      waitStartup <- async $ waitQSem startup
      waitTermination <- async . readMVar $ termination r
      wait <- waitEitherCancel waitTermination waitStartup
      case wait of
        Left (Left err) -> throwIO err
        Left (Right _) -> throwIO $ userError "Server thread terminated early"
        Right _ -> return r
    teardown RunningServer{..} = killThread thread >> void (readMVar termination)

spec :: Spec
spec = withSetup $ do
  it "Reports initial server state" $ \RunningServer{..} -> do
    printers                     `shouldReturn` [(0, Available)]
    jobs Nothing Nothing Nothing `shouldReturn` []
    drafts                       `shouldReturn` []
  it "Reports printing errors" $ \RunningServer{..} -> do
    let
      err = IOError "test"
    atomically $ putTMVar (failSwitch printer) err
    jId <- jobCreate Nothing mempty
    atomically . takeTMVar $ outputChan printer
    jobStatus jId `shouldReturn` (Failed err)
  -- it "Queues any Printout" $ \RunningServer{..} -> property $ \p -> do
  --   jId <- jobCreate (Just 0) p
  --   (atomically . takeTMVar . outputChan $ printer) `shouldReturn` p
  it "Reports qualitative queue position" $ \RunningServer{..} -> do
    jids <- replicateM 3 $ jobCreate (Just 0) mempty
    zipWithM_ (\jid ret -> jobStatus jid `shouldReturn` ret) jids
      [ Done
      , Printing 0
      , Queued 0
      ]
    replicateM_ (length jids) . atomically . takeTMVar . outputChan $ printer
    zipWithM_ (\jid ret -> jobStatus jid `shouldReturn` ret) jids (replicate (length jids) Done)
  it "Keeps history" $ \RunningServer{..} -> do
    jobs Nothing Nothing Nothing `shouldNotReturn` []
  it "Runs queue managers" $ \RunningServer{..} -> do
    atomically $ putTMVar (manage manager) S.nullQM
    atomically $ waitTSem (ran manager)
    jobs Nothing Nothing Nothing `shouldReturn` []
  it "Handles drafts" $ \RunningServer{..} -> do
    drafts `shouldReturn` []
    dId <- draftCreate Nothing mempty
    draft dId `shouldReturn` (Nothing, mempty)
    drafts `shouldReturn` [(dId, Nothing :: Maybe DraftTitle)]
    p <- generate arbitrary
    draftReplace dId (Just "Title") p
    draft dId `shouldReturn` (Just "Title", p)
    jId <- draftPrint dId Nothing
    (atomically . takeTMVar $ outputChan printer) `shouldReturn` p
    draftDelete dId
    drafts `shouldReturn` []
    draftReplace dId Nothing p `shouldThrow` is404
  where
    Client{..} = mkClient' $ BaseUrl Http "localhost" 3000 ""
    is404 :: ServantError -> Bool
    is404 e@(FailureResponse {}) = statusCode (responseStatus e) == 404
    is404 _ = False