Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

309 handle application exceptions with 500 errors #954

Open
wants to merge 9 commits into
base: master
Choose a base branch
from
Open
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
1 change: 0 additions & 1 deletion servant-server/example/greet.hs
Original file line number Diff line number Diff line change
Expand Up @@ -6,7 +6,6 @@
{-# LANGUAGE TypeOperators #-}

import Data.Aeson
import Data.Monoid
import Data.Proxy
import Data.Text
import GHC.Generics
Expand Down
1 change: 1 addition & 0 deletions servant-server/servant-server.cabal
Original file line number Diff line number Diff line change
Expand Up @@ -87,6 +87,7 @@ library
build-depends:
base-compat >= 0.10.1 && < 0.11
, base64-bytestring >= 1.0.0.1 && < 1.1
, deepseq >= 1.4.3.0 && < 1.5
, exceptions >= 0.10.0 && < 0.11
, http-api-data >= 0.3.8.1 && < 0.4
, http-media >= 0.7.1.2 && < 0.8
Expand Down
2 changes: 1 addition & 1 deletion servant-server/src/Servant/Server.hs
Original file line number Diff line number Diff line change
Expand Up @@ -132,7 +132,7 @@ serve p = serveWithContext p EmptyContext
serveWithContext :: (HasServer api context)
=> Proxy api -> Context context -> Server api -> Application
serveWithContext p context server =
toApplication (runRouter (route p context (emptyDelayed (Route server))))
toApplication Force (runRouter (route p context (emptyDelayed (Route server))))
Copy link
Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

@phadej Should I alter serveWithContext to take in another param that will determine if we want to NF the response or not ? How would you like this to look for the end user in a sense do you want them to be able to control the response evaluation ?


-- | Hoist server implementation.
--
Expand Down
7 changes: 7 additions & 0 deletions servant-server/src/Servant/Server/Internal/Handler.hs
Original file line number Diff line number Diff line change
Expand Up @@ -46,3 +46,10 @@ instance MonadBaseControl IO Handler where

runHandler :: Handler a -> IO (Either ServantErr a)
runHandler = runExceptT . runHandler'

-- determins if response should be reduced to NF
data Evaluate =
Copy link
Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

@phadej Is this a good place for this type ?

Force
| Lazy
deriving (Show)

23 changes: 16 additions & 7 deletions servant-server/src/Servant/Server/Internal/RoutingApplication.hs
Original file line number Diff line number Diff line change
Expand Up @@ -10,6 +10,8 @@
{-# LANGUAGE UndecidableInstances #-}
module Servant.Server.Internal.RoutingApplication where

import Control.DeepSeq
(force)
import Control.Monad
(ap, liftM)
import Control.Monad.Base
Expand Down Expand Up @@ -93,13 +95,20 @@ instance MonadTransControl RouteResultT where
instance MonadThrow m => MonadThrow (RouteResultT m) where
throwM = lift . throwM

toApplication :: RoutingApplication -> Application
toApplication ra request respond = ra request routingRespond
where
routingRespond :: RouteResult Response -> IO ResponseReceived
routingRespond (Fail err) = respond $ responseServantErr err
routingRespond (FailFatal err) = respond $ responseServantErr err
routingRespond (Route v) = respond v
toApplication :: Evaluate -> RoutingApplication -> Application
toApplication fullyEvaluate ra request respond =
ra request (maybeEval routingRespond)
where
maybeEval :: (RouteResult Response -> IO ResponseReceived)
-> RouteResult Response -> IO ResponseReceived
maybeEval resp =
case fullyEvaluate of
Force -> force resp
Lazy -> resp
Comment on lines +104 to +107
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

still trying to figure out what's going on here, but this looks slightly more correct to me:

Suggested change
maybeEval resp =
case fullyEvaluate of
Force -> force resp
Lazy -> resp
maybeEval cont =
case fullyEvaluate of
Force -> \resp -> resp `deepseq` cont resp
Lazy -> cont

routingRespond :: RouteResult Response -> IO ResponseReceived
routingRespond (Fail err) = respond $ responseServantErr err
routingRespond (FailFatal err) = respond $ responseServantErr err
routingRespond (Route v) = respond v

-- | A 'Delayed' is a representation of a handler with scheduled
-- delayed checks that can trigger errors.
Expand Down
6 changes: 5 additions & 1 deletion servant-server/src/Servant/Server/Internal/ServantErr.hs
Original file line number Diff line number Diff line change
@@ -1,14 +1,18 @@
{-# LANGUAGE DeriveDataTypeable #-}
{-# LANGUAGE DeriveGeneric #-}
{-# LANGUAGE DeriveAnyClass #-}
{-# LANGUAGE OverloadedStrings #-}
{-# LANGUAGE RecordWildCards #-}
module Servant.Server.Internal.ServantErr where

import Control.Exception
(Exception)
import Control.DeepSeq (NFData)
import qualified Data.ByteString.Char8 as BS
import qualified Data.ByteString.Lazy as LBS
import Data.Typeable
(Typeable)
import GHC.Generics (Generic)
import qualified Network.HTTP.Types as HTTP
import Network.Wai
(Response, responseLBS)
Expand All @@ -17,7 +21,7 @@ data ServantErr = ServantErr { errHTTPCode :: Int
, errReasonPhrase :: String
, errBody :: LBS.ByteString
, errHeaders :: [HTTP.Header]
} deriving (Show, Eq, Read, Typeable)
} deriving (Show, Eq, Read, Typeable, Generic, NFData)

instance Exception ServantErr

Expand Down
2 changes: 1 addition & 1 deletion servant-server/test/Servant/Server/RouterSpec.hs
Original file line number Diff line number Diff line change
Expand Up @@ -31,7 +31,7 @@ spec = describe "Servant.Server.Internal.Router" $ do
routerSpec :: Spec
routerSpec = do
let app' :: Application
app' = toApplication $ runRouter router'
app' = toApplication Force $ runRouter router'

router', router :: Router ()
router' = tweakResponse (fmap twk) router
Expand Down