add CombinatorImplementation

This commit is contained in:
Sönke Hahn 2016-10-23 14:26:14 -04:00
parent 16cffc7d69
commit 7177f0a729
2 changed files with 40 additions and 23 deletions

View file

@ -1,11 +1,16 @@
{-# LANGUAGE GADTs #-}
{-# LANGUAGE RankNTypes #-}
{-# LANGUAGE ScopedTypeVariables #-} {-# LANGUAGE ScopedTypeVariables #-}
{-# LANGUAGE TypeFamilies #-} {-# LANGUAGE TypeFamilies #-}
{-# LANGUAGE TypeOperators #-} {-# LANGUAGE TypeOperators #-}
module Servant.Server.CombinatorUtils ( module Servant.Server.CombinatorUtils (
RouteResult(..), CombinatorImplementation,
argumentCombinator, runCI,
captureCombinator, captureCombinator,
argumentCombinator,
-- * re-exports
RouteResult(..),
) where ) where
import Data.Proxy import Data.Proxy
@ -16,29 +21,41 @@ import Servant.API
import Servant.Server import Servant.Server
import Servant.Server.Internal import Servant.Server.Internal
argumentCombinator :: data CombinatorImplementation combinator arg api context where
forall api combinator arg context env . CI :: (forall env .
(ServerT (combinator :> api) Handler ~ (arg -> ServerT api Handler), Proxy (combinator :> api)
HasServer api context) => -> Context context
(Request -> RouteResult arg) -> Delayed env (arg -> Server api)
-> Proxy (combinator :> api) -> Router' env RoutingApplication)
-> Context context -> CombinatorImplementation combinator arg api context
-> Delayed env (Server (combinator :> api))
-> Router' env RoutingApplication
argumentCombinator getArg Proxy context delayed =
route (Proxy :: Proxy api) context $ addBodyCheck delayed
(DelayedIO (return ())) $ \ () ->
withRequest $ \ request -> liftRouteResult (getArg request)
captureCombinator :: runCI :: CombinatorImplementation combinator arg api context
forall api combinator arg context env .
(HasServer api context) =>
(Text -> RouteResult arg)
-> Proxy (combinator :> api) -> Proxy (combinator :> api)
-> Context context -> Context context
-> Delayed env (arg -> Server api) -> Delayed env (arg -> Server api)
-> Router' env RoutingApplication -> Router' env RoutingApplication
captureCombinator getArg Proxy context delayed = runCI (CI i) = i
captureCombinator ::
forall api combinator arg context .
(HasServer api context) =>
(Text -> RouteResult arg)
-> CombinatorImplementation combinator arg api context
captureCombinator getArg = CI $ \ Proxy context delayed ->
CaptureRouter $ CaptureRouter $
route (Proxy :: Proxy api) context $ addCapture delayed $ \ captured -> route (Proxy :: Proxy api) context $ addCapture delayed $ \ captured ->
(liftRouteResult (getArg captured)) (liftRouteResult (getArg captured))
argumentCombinator ::
forall api combinator arg context .
(ServerT (combinator :> api) Handler ~ (arg -> ServerT api Handler),
HasServer api context) =>
(Request -> RouteResult arg)
-> CombinatorImplementation combinator arg api context
argumentCombinator getArg = CI $ \ Proxy context delayed ->
route (Proxy :: Proxy api) context $
addBodyCheck delayed contentTypeCheck bodyCheck
where
contentTypeCheck = return ()
bodyCheck () = withRequest $ \ request ->
liftRouteResult (getArg request)

View file

@ -27,7 +27,7 @@ import Servant.Server.CombinatorUtils
runApp :: Application -> Request -> IO Response runApp :: Application -> Request -> IO Response
runApp app req = do runApp app req = do
mvar <- newMVar Nothing mvar <- newMVar Nothing
app req $ \ response -> do ResponseReceived <- app req $ \ response -> do
modifyMVar mvar $ \ Nothing -> modifyMVar mvar $ \ Nothing ->
return $ (Just response, ResponseReceived) return $ (Just response, ResponseReceived)
modifyMVar mvar $ \mResponse -> do modifyMVar mvar $ \mResponse -> do
@ -100,7 +100,7 @@ data FooHeader
instance HasServer api context => HasServer (FooHeader :> api) context where instance HasServer api context => HasServer (FooHeader :> api) context where
type ServerT (FooHeader :> api) m = String -> ServerT api m type ServerT (FooHeader :> api) m = String -> ServerT api m
route = argumentCombinator getCustom route = runCI $ argumentCombinator getCustom
getCustom :: Request -> RouteResult String getCustom :: Request -> RouteResult String
getCustom request = case lookup "FooHeader" (requestHeaders request) of getCustom request = case lookup "FooHeader" (requestHeaders request) of
@ -111,7 +111,7 @@ data StringCapture
instance HasServer api context => HasServer (StringCapture :> api) context where instance HasServer api context => HasServer (StringCapture :> api) context where
type ServerT (StringCapture :> api) m = String -> ServerT api m type ServerT (StringCapture :> api) m = String -> ServerT api m
route = captureCombinator getCapture route = runCI $ captureCombinator getCapture
getCapture :: Text -> RouteResult String getCapture :: Text -> RouteResult String
getCapture = Route . cs getCapture = Route . cs