101 lines
3.6 KiB
Haskell
101 lines
3.6 KiB
Haskell
module Examples.Middleware.Main where
|
|
|
|
import Prelude hiding ((/))
|
|
|
|
import Data.Either (Either(..))
|
|
import Data.Generic.Rep (class Generic)
|
|
import Data.Maybe (Maybe(..))
|
|
import Effect.Class (liftEffect)
|
|
import Effect.Console (log)
|
|
import HTTPurple (type (<+>), Request, ResponseM, ServerM, fullPath, header, ok, ok', serve, (<+>))
|
|
import Record as Record
|
|
import Routing.Duplex as RD
|
|
import Routing.Duplex.Generic as RG
|
|
import Routing.Duplex.Generic.Syntax ((/))
|
|
import Type.Prelude (Proxy(..))
|
|
|
|
data Middleware = Middleware
|
|
|
|
derive instance Generic Middleware _
|
|
|
|
middlewareRoute :: RD.RouteDuplex' Middleware
|
|
middlewareRoute = RD.root $ RG.sum
|
|
{ "Middleware": "middleware" / RG.noArgs
|
|
}
|
|
|
|
data SayHello = SayHello
|
|
|
|
derive instance Generic SayHello _
|
|
|
|
sayHelloRoute :: RD.RouteDuplex' SayHello
|
|
sayHelloRoute = RD.root $ RG.sum
|
|
{ "SayHello": RG.noArgs
|
|
}
|
|
|
|
-- | A middleware that logs at the beginning and end of each request
|
|
loggingMiddleware ::
|
|
forall route.
|
|
(Request route -> ResponseM) ->
|
|
Request route ->
|
|
ResponseM
|
|
loggingMiddleware router request = do
|
|
liftEffect $ log $ "Request starting for " <> path
|
|
response <- router request
|
|
liftEffect $ log $ "Request ending for " <> path
|
|
pure response
|
|
where
|
|
path = fullPath request
|
|
|
|
-- | A middleware that adds the X-Middleware header to the response, if it
|
|
-- | wasn't already in the response
|
|
headerMiddleware ::
|
|
forall route.
|
|
(Request route -> ResponseM) ->
|
|
Request route ->
|
|
ResponseM
|
|
headerMiddleware router request = do
|
|
response@{ headers } <- router request
|
|
pure $ response { headers = header' <> headers }
|
|
where
|
|
header' = header "X-Middleware" "middleware"
|
|
|
|
-- | A middleware that sends the body "Middleware!" instead of running the
|
|
-- | router when requesting /middleware
|
|
pathMiddleware ::
|
|
forall route.
|
|
(Request route -> ResponseM) ->
|
|
Request (Middleware <+> route) ->
|
|
ResponseM
|
|
pathMiddleware _ { route: Left Middleware } = ok "Middleware!"
|
|
pathMiddleware router request@{ route: Right r } = router $ Record.set (Proxy :: _ "route") r request
|
|
|
|
-- | Say 'hello' when run, and add a default value to the X-Middleware header
|
|
sayHello :: Request SayHello -> ResponseM
|
|
sayHello _ = ok' (header "X-Middleware" "router") "hello"
|
|
|
|
-- | The stack of middlewares to use for the server
|
|
middlewareStack :: forall route. (Request route -> ResponseM) -> Request (Either Middleware route) -> ResponseM
|
|
middlewareStack = loggingMiddleware <<< headerMiddleware <<< pathMiddleware
|
|
|
|
-- | Boot up the server
|
|
main :: ServerM
|
|
main =
|
|
serve { hostname: "localhost", port: 8080, onStarted } { route: middlewareRoute <+> sayHelloRoute, router: middlewareStack sayHello }
|
|
where
|
|
onStarted = do
|
|
log " ┌───────────────────────────────────────┐"
|
|
log " │ Server now up on port 8080 │"
|
|
log " │ │"
|
|
log " │ To test, run: │"
|
|
log " │ > curl -v localhost:8080 │"
|
|
log " │ # => ... │"
|
|
log " │ # => ...< X-Middleware: router │"
|
|
log " │ # => ... │"
|
|
log " │ # => hello │"
|
|
log " │ > curl -v localhost:8080/middleware │"
|
|
log " │ # => ... │"
|
|
log " │ # => ...< X-Middleware: middleware │"
|
|
log " │ # => ... │"
|
|
log " │ # => Middleware! │"
|
|
log " └───────────────────────────────────────┘"
|