module Web.Fn (
RequestContext(..)
, toWAI
, Req
, route
, fallthrough
, (==>)
, (//)
, (/?)
, path
, end
, segment
, FromParam(..)
, param
, paramOptional
, paramPresent
, okText
, okHtml
, errText
, errHtml
, notFoundText
, notFoundHtml
, redirect
) where
import qualified Blaze.ByteString.Builder.Char.Utf8 as B
import Data.ByteString (ByteString)
import Data.List (find)
import Data.Maybe (fromJust)
import Data.Monoid ((<>))
import Data.Text (Text)
import qualified Data.Text as T
import qualified Data.Text.Encoding as T
import Data.Text.Read (decimal, double)
import Network.HTTP.Types
import Network.Wai
data Store b a = Store b (b -> a)
instance Functor (Store b) where
fmap f (Store b h) = Store b (f . h)
class RequestContext ctxt where
requestLens :: Functor f => (Request -> f Request) -> ctxt -> f ctxt
requestLens f c = setRequest c <$> f (getRequest c)
getRequest :: ctxt -> Request
getRequest c =
let (Store r _) = requestLens (`Store` id) c
in r
setRequest :: ctxt -> Request -> ctxt
setRequest c r =
let (Store _ b) = requestLens (`Store` id) c
in b r
toWAI :: RequestContext ctxt => ctxt -> (ctxt -> IO Response) -> Application
toWAI ctxt f req cont = let ctxt' = setRequest ctxt req
in f ctxt' >>= cont
fallthrough :: IO (Maybe Response) -> IO Response -> IO Response
fallthrough a ft =
do response <- a
case response of
Nothing -> ft
Just r -> return r
route :: RequestContext ctxt =>
ctxt ->
[ctxt -> Maybe (IO (Maybe Response))] ->
IO (Maybe Response)
route _ [] = return Nothing
route ctxt (x:xs) =
case x ctxt of
Nothing -> route ctxt xs
Just action ->
do resp <- action
case resp of
Nothing -> route ctxt xs
Just response -> return (Just response)
type Req = ([Text], Query)
(==>) :: RequestContext ctxt =>
(Req -> k -> Maybe (Req, a)) ->
(ctxt -> k) ->
ctxt -> Maybe a
(match ==> handle) ctxt =
let r = getRequest ctxt
x = (pathInfo r, queryString r)
in case match x (handle ctxt) of
Nothing -> Nothing
Just (_, action) -> Just action
(//) :: (r -> k -> Maybe (r, k')) ->
(r -> k' -> Maybe (r, a)) ->
r ->
k -> Maybe (r, a)
(match1 // match2) req k =
case match1 req k of
Nothing -> Nothing
Just (req', k') -> match2 req' k'
(/?) :: (r -> k -> Maybe (r, k')) ->
(r -> k' -> Maybe (r, a)) ->
r ->
k -> Maybe (r, a)
(/?) = (//)
path :: Text -> Req -> a -> Maybe (Req, a)
path s req k =
case fst req of
(x:xs) | x == s -> Just ((xs, snd req), k)
_ -> Nothing
end :: Req -> a -> Maybe (Req, a)
end req k =
case fst req of
[] -> Just (req, k)
_ -> Nothing
segment :: FromParam p => Req -> (p -> a) -> Maybe (Req, a)
segment req k =
case fst req of
(x:xs) -> case fromParam x of
Left _ -> Nothing
Right p -> Just ((xs, snd req), k p)
_ -> Nothing
class FromParam a where
fromParam :: Text -> Either Text a
instance FromParam Text where
fromParam = Right
instance FromParam Int where
fromParam t = case decimal t of
Left msg -> Left (T.pack msg)
Right m | snd m /= "" ->
Left ("Incomplete match: " <> T.pack (show m))
Right (v, _) -> Right v
instance FromParam Double where
fromParam t = case double t of
Left msg -> Left (T.pack msg)
Right m | snd m /= "" ->
Left ("Incomplete match: " <> T.pack (show m))
Right (v, _) -> Right v
param :: FromParam p => Text -> Req -> (p -> a) -> Maybe (Req, a)
param n req k =
let match = find ((== T.encodeUtf8 n) . fst) (snd req)
in case (maybe "" T.decodeUtf8 . snd) <$> match of
Nothing -> Nothing
Just p -> case fromParam p of
Left _ -> Nothing
Right p' -> Just (req, k p')
paramOptional :: FromParam p =>
Text ->
Req ->
(Either Text p -> a) ->
Maybe (Req, a)
paramOptional n req k =
let match = find ((== T.encodeUtf8 n) . fst) (snd req)
p = ((maybe "" T.decodeUtf8 . snd) <$> match)
in case p of
Nothing -> Just (req, k (Left "param missing"))
Just p' -> Just (req, k (fromParam p'))
paramPresent :: FromParam p =>
Text ->
Req ->
(Either Text p -> a) ->
Maybe (Req, a)
paramPresent n req k =
let match = find ((== T.encodeUtf8 n) . fst) (snd req)
p = ((maybe "" T.decodeUtf8 . snd) <$> match)
in case p of
Nothing -> Nothing
Just p' -> Just (req, k (fromParam p'))
returnText :: Text -> Status -> ByteString -> IO (Maybe Response)
returnText text status content =
return $ Just $
responseBuilder status
[(hContentType, content)]
(B.fromText text)
plainText :: ByteString
plainText = "text/plain; charset=utf-8"
html :: ByteString
html = "text/html; charset=utf-8"
okText :: Text -> IO (Maybe Response)
okText t = returnText t status200 plainText
okHtml :: Text -> IO (Maybe Response)
okHtml t = returnText t status200 html
errText :: Text -> IO (Maybe Response)
errText t = returnText t status500 plainText
errHtml :: Text -> IO (Maybe Response)
errHtml t = returnText t status500 html
notFoundText :: Text -> IO Response
notFoundText t = fromJust <$> returnText t status404 plainText
notFoundHtml :: Text -> IO Response
notFoundHtml t = fromJust <$> returnText t status404 html
redirect :: Text -> IO (Maybe Response)
redirect target =
return $ Just $
responseBuilder status303
[(hLocation, T.encodeUtf8 target)]
(B.fromText "")