2013-10-03 19:12:48 +04:00
|
|
|
{-# LANGUAGE OverloadedStrings #-}
|
|
|
|
|
|
|
|
module ExceptionSpec (main, spec) where
|
|
|
|
|
|
|
|
import Control.Applicative
|
|
|
|
import Control.Monad
|
|
|
|
import Control.Concurrent (forkIO, threadDelay)
|
|
|
|
import Network.HTTP
|
|
|
|
import Network.Stream
|
|
|
|
import Network.HTTP.Types hiding (Header)
|
|
|
|
import Network.Wai hiding (Response)
|
2013-10-10 13:51:16 +04:00
|
|
|
import Network.Wai.Internal (Request(..))
|
2013-10-03 19:12:48 +04:00
|
|
|
import Network.Wai.Handler.Warp
|
|
|
|
import System.IO.Unsafe (unsafePerformIO)
|
|
|
|
import Test.Hspec
|
2014-04-09 08:23:28 +04:00
|
|
|
import Control.Exception
|
|
|
|
import qualified Data.Streaming.Network as N
|
|
|
|
import Control.Concurrent.Async (withAsync)
|
|
|
|
import Network.Socket (sClose)
|
2013-10-03 19:12:48 +04:00
|
|
|
|
|
|
|
main :: IO ()
|
|
|
|
main = hspec spec
|
|
|
|
|
2014-04-09 08:23:28 +04:00
|
|
|
withTestServer :: (Int -> IO a) -> IO a
|
|
|
|
withTestServer inner = bracket
|
|
|
|
(N.bindRandomPortTCP "*4")
|
|
|
|
(sClose . snd)
|
|
|
|
$ \(port, lsocket) -> do
|
|
|
|
withAsync (runSettingsSocket defaultSettings lsocket testApp)
|
|
|
|
$ \_ -> inner port
|
2013-10-03 19:12:48 +04:00
|
|
|
|
|
|
|
testApp :: Application
|
2014-04-23 19:17:33 +04:00
|
|
|
testApp (Network.Wai.Internal.Request {pathInfo = [x]}) f
|
2013-10-03 19:12:48 +04:00
|
|
|
| x == "statusError" =
|
2014-04-23 19:17:33 +04:00
|
|
|
f $ responseLBS undefined [] "foo"
|
2013-10-03 19:12:48 +04:00
|
|
|
| x == "headersError" =
|
2014-04-23 19:17:33 +04:00
|
|
|
f $ responseLBS ok200 undefined "foo"
|
2013-10-03 19:12:48 +04:00
|
|
|
| x == "headerError" =
|
2014-04-23 19:17:33 +04:00
|
|
|
f $ responseLBS ok200 [undefined] "foo"
|
2013-10-03 19:12:48 +04:00
|
|
|
| x == "bodyError" =
|
2014-04-23 19:17:33 +04:00
|
|
|
f $ responseLBS ok200 [] undefined
|
2013-10-03 19:12:48 +04:00
|
|
|
| x == "ioException" = do
|
|
|
|
void $ fail "ioException"
|
2014-04-23 19:17:33 +04:00
|
|
|
f $ responseLBS ok200 [] "foo"
|
|
|
|
testApp _ f =
|
|
|
|
f $ responseLBS ok200 [] "foo"
|
2013-10-03 19:12:48 +04:00
|
|
|
|
|
|
|
spec :: Spec
|
2014-04-09 08:23:28 +04:00
|
|
|
spec = describe "responds even if there is an exception" $ do
|
2014-02-05 09:02:05 +04:00
|
|
|
{- Disabling these tests. We can consider forcing evaluation in Warp.
|
2013-10-03 19:12:48 +04:00
|
|
|
it "statusError" $ do
|
|
|
|
sc <- rspCode <$> sendGET "http://localhost:2345/statusError"
|
|
|
|
sc `shouldBe` (5,0,0)
|
|
|
|
it "headersError" $ do
|
|
|
|
sc <- rspCode <$> sendGET "http://localhost:2345/headersError"
|
|
|
|
sc `shouldBe` (5,0,0)
|
|
|
|
it "headerError" $ do
|
|
|
|
sc <- rspCode <$> sendGET "http://localhost:2345/headerError"
|
|
|
|
sc `shouldBe` (5,0,0)
|
|
|
|
it "bodyError" $ do
|
|
|
|
sc <- rspCode <$> sendGET "http://localhost:2345/bodyError"
|
|
|
|
sc `shouldBe` (5,0,0)
|
2014-02-05 09:02:05 +04:00
|
|
|
-}
|
2014-04-09 08:23:28 +04:00
|
|
|
it "ioException" $ withTestServer $ \port -> do
|
|
|
|
sc <- rspCode <$> sendGET (concat $ ["http://localhost:", show port, "/ioException"])
|
2013-10-03 19:12:48 +04:00
|
|
|
sc `shouldBe` (5,0,0)
|
|
|
|
|
|
|
|
----------------------------------------------------------------
|
|
|
|
|
|
|
|
sendGET :: String -> IO (Response String)
|
|
|
|
sendGET url = sendGETwH url []
|
|
|
|
|
|
|
|
sendGETwH :: String -> [Header] -> IO (Response String)
|
|
|
|
sendGETwH url hdr = unResult $ simpleHTTP $ (getRequest url) { rqHeaders = hdr }
|
|
|
|
|
|
|
|
unResult :: IO (Result (Response String)) -> IO (Response String)
|
|
|
|
unResult action = do
|
|
|
|
res <- action
|
|
|
|
case res of
|
|
|
|
Right rsp -> return rsp
|
|
|
|
Left _ -> error "Connection error"
|