module Network.Wai.Routing.Route
( Routes
, Meta (..)
, prepare
, route
, addRoute
, attach
, examine
, get
, Network.Wai.Routing.Route.head
, post
, put
, delete
, trace
, options
, connect
, patch
, Renderer
, renderer
) where
import Control.Applicative hiding (Const)
import Control.Monad
import Control.Monad.Trans.State.Strict hiding (get, put)
import Data.ByteString (ByteString)
import Data.CaseInsensitive (mk)
import Data.Either
import Data.Function
import Data.List hiding (head, delete)
import Data.Maybe (fromMaybe, mapMaybe, catMaybes)
import Data.Monoid
import Network.HTTP.Types
import Network.Wai (Request, Response, responseLBS, responseBuilder, rawPathInfo)
import Network.Wai.Predicate
import Network.Wai.Predicate.Request
import Network.Wai.Routing.Request
import qualified Data.ByteString.Char8 as C
import qualified Data.ByteString.Lazy as Lazy
import qualified Data.List as L
import qualified Network.Wai.Route.Tree as Tree
data Route a m = Route
{ _method :: !Method
, _path :: !ByteString
, _meta :: Maybe a
, _pred :: Pack m
}
data Handler m = Handler
{ _delta :: !Double
, _handler :: m Response
}
data Pack m where
Pack :: Predicate RoutingReq Error a -> (a -> m Response) -> Pack m
type Renderer = Error -> Maybe Lazy.ByteString
data Meta a = Meta
{ routeMethod :: !Method
, routePath :: !ByteString
, routeMeta :: a
}
renderer :: Renderer -> Routes a m ()
renderer f = Routes . modify $ \s -> s { renderfn = f }
defRenderer :: Renderer
defRenderer e =
let r = reason2str <$> reason e
s = source2str <$> source e
m = message2str <$> message e
l = labels2str . map Lazy.fromStrict $ labels e
x = case catMaybes [s, r, l] of
[] -> Nothing
xs -> Just (Lazy.intercalate " " xs)
in maybe x (\y -> (<> (" -- " <> y)) <$> x) m
where
reason2str NotAvailable = "not-available"
reason2str TypeError = "type-error"
source2str s = "'" <> Lazy.fromStrict s <> "'"
message2str s = Lazy.fromStrict s
labels2str [] = Nothing
labels2str xs = Just $ "[" <> Lazy.intercalate "," xs <> "]"
data St a m = St
{ routes :: [Route a m]
, renderfn :: Renderer
}
zero :: St a m
zero = St [] defRenderer
newtype Routes a m b = Routes { _unroutes :: State (St a m) b }
instance Functor (Routes a m) where
fmap = liftM
instance Applicative (Routes a m) where
pure = return
(<*>) = ap
instance Monad (Routes a m) where
return = Routes . return
m >>= f = Routes $ _unroutes m >>= _unroutes . f
addRoute :: Monad m
=> Method
-> ByteString
-> (a -> m Response)
-> Predicate RoutingReq Error a
-> Routes b m ()
addRoute m r x p = Routes . modify $ \s ->
s { routes = Route m r Nothing (Pack p x) : routes s }
get, head, post, put, delete, trace, options, connect, patch ::
Monad m
=> ByteString
-> (a -> m Response)
-> Predicate RoutingReq Error a
-> Routes b m ()
get = addRoute (renderStdMethod GET)
head = addRoute (renderStdMethod HEAD)
post = addRoute (renderStdMethod POST)
put = addRoute (renderStdMethod PUT)
delete = addRoute (renderStdMethod DELETE)
trace = addRoute (renderStdMethod TRACE)
options = addRoute (renderStdMethod OPTIONS)
connect = addRoute (renderStdMethod CONNECT)
patch = addRoute (renderStdMethod PATCH)
attach :: a -> Routes a m ()
attach a = Routes $ modify addToLast
where
addToLast s@(St [] _) = s
addToLast (St (r:rr) f) = St (r { _meta = Just a } : rr) f
examine :: Routes a m b -> [Meta a]
examine (Routes r) = let St rr _ = execState r zero in
mapMaybe (\x -> Meta (_method x) (_path x) <$> _meta x) rr
route :: Monad m => [(ByteString, RoutingReq -> m Response)] -> Request -> m Response
route rm rq = do
let tr = Tree.fromList rm
case Tree.lookup tr (Tree.segments $ rawPathInfo rq) of
Just (f, v) -> f (fromReq v (fromRequest rq))
Nothing -> return notFound
where
notFound = responseLBS status404 [] ""
prepare :: Monad m => Routes a m b -> [(ByteString, RoutingReq -> m Response)]
prepare (Routes rr) =
let s = execState rr zero in
map (\g -> (_path (L.head g), select (renderfn s) g)) (normalise (routes s))
normalise :: [Route a m] -> [[Route a m]]
normalise rr =
let rg = grouped . sorted $ rr
paths = map (namelessPath . L.head) rg
ambig = paths \\ nub paths
in if null ambig then rg else error (ambiguityMessage ambig)
where
sorted :: [Route a m] -> [Route a m]
sorted = sortBy (compare `on` _path)
grouped :: [Route a m] -> [[Route a m]]
grouped = groupBy ((==) `on` _path)
namelessPath :: Route a m -> ByteString
namelessPath =
let fun s = if s /= "" && C.head s == ':' then "<>" else s
in C.intercalate "/" . map fun . C.split '/' . _path
ambiguityMessage a =
"Paths differing only in variable names are not supported.\n" ++
"Problematic paths (with variable positions denoted by <>):\n" ++
show a
select :: Monad m => Renderer -> [Route a m] -> RoutingReq -> m Response
select render rr req = do
let ms = filter ((method req ==) . _method) rr
if null ms
then return $ respond render e405 [(allow, validMethods)]
else evalAll ms
where
allow :: HeaderName
allow = mk "Allow"
validMethods :: ByteString
validMethods = C.intercalate "," $ nub (C.pack . show . _method <$> rr)
evalAll :: Monad m => [Route a m] -> m Response
evalAll rs =
let (n, y) = partitionEithers $ foldl' evalSingle [] rs
in if null y
then return $ respond render (L.head n) []
else closest y
evalSingle :: Monad m => [Either Error (Handler m)] -> Route a m -> [Either Error (Handler m)]
evalSingle rs r =
case _pred r of
Pack p h -> case p req of
Fail m -> Left m : rs
Okay d v -> Right (Handler d (h v)) : rs
closest :: Monad m => [Handler m] -> m Response
closest hh = case map _handler . sortBy (compare `on` _delta) $ hh of
[] -> return $ responseBuilder status404 [] mempty
h:_ -> h
respond :: Renderer -> Error -> ResponseHeaders -> Response
respond f e h = responseLBS (status e) h (fromMaybe mempty (f e))