Route is a data family, part of RenderRoute class
This commit is contained in:
parent
d69ee53a17
commit
144b215a38
12
yesod-routes/Yesod/Routes/Class.hs
Normal file
12
yesod-routes/Yesod/Routes/Class.hs
Normal file
@ -0,0 +1,12 @@
|
|||||||
|
{-# LANGUAGE TypeFamilies #-}
|
||||||
|
{-# LANGUAGE FlexibleContexts #-}
|
||||||
|
module Yesod.Routes.Class
|
||||||
|
( RenderRoute (..)
|
||||||
|
) where
|
||||||
|
|
||||||
|
import Data.Text (Text)
|
||||||
|
|
||||||
|
class Eq (Route a) => RenderRoute a where
|
||||||
|
-- | The type-safe URLs associated with a site argument.
|
||||||
|
data Route a
|
||||||
|
renderRoute :: Route a -> ([Text], [(Text, Text)])
|
||||||
@ -5,14 +5,12 @@ module Yesod.Routes.TH
|
|||||||
, Piece (..)
|
, Piece (..)
|
||||||
, Dispatch (..)
|
, Dispatch (..)
|
||||||
-- * Functions
|
-- * Functions
|
||||||
-- ** Route data type
|
|
||||||
, mkRouteType
|
|
||||||
, mkRouteCons
|
|
||||||
-- ** RenderRoute
|
-- ** RenderRoute
|
||||||
, mkRenderRouteClauses
|
|
||||||
, mkRenderRouteInstance
|
, mkRenderRouteInstance
|
||||||
|
, mkRouteCons
|
||||||
|
, mkRenderRouteClauses
|
||||||
-- ** Dispatch
|
-- ** Dispatch
|
||||||
, mkDispatchClause
|
--, mkDispatchClause
|
||||||
) where
|
) where
|
||||||
|
|
||||||
import Language.Haskell.TH.Syntax
|
import Language.Haskell.TH.Syntax
|
||||||
@ -38,9 +36,11 @@ data Dispatch = Methods (Maybe Type) [String] | Subsite
|
|||||||
, subsiteFunc :: String
|
, subsiteFunc :: String
|
||||||
}
|
}
|
||||||
|
|
||||||
|
resourceMulti :: Resource -> Maybe Type
|
||||||
resourceMulti Resource { resourceDispatch = Methods (Just t) _ } = Just t
|
resourceMulti Resource { resourceDispatch = Methods (Just t) _ } = Just t
|
||||||
resourceMulti _ = Nothing
|
resourceMulti _ = Nothing
|
||||||
|
|
||||||
|
-- | Generate the constructors of a route data type.
|
||||||
mkRouteCons :: [Resource] -> [Con]
|
mkRouteCons :: [Resource] -> [Con]
|
||||||
mkRouteCons =
|
mkRouteCons =
|
||||||
map mkRouteCon
|
map mkRouteCon
|
||||||
@ -61,12 +61,7 @@ mkRouteCons =
|
|||||||
Subsite { subsiteType = typ } -> [ConT ''Route `AppT` typ]
|
Subsite { subsiteType = typ } -> [ConT ''Route `AppT` typ]
|
||||||
_ -> []
|
_ -> []
|
||||||
|
|
||||||
mkRouteType :: String -> [Resource] -> Dec
|
-- | Clauses for the 'renderRoute' method.
|
||||||
mkRouteType name res =
|
|
||||||
DataD [] (mkName name) [] (mkRouteCons res) clazzes
|
|
||||||
where
|
|
||||||
clazzes = [''Show, ''Eq, ''Read]
|
|
||||||
|
|
||||||
mkRenderRouteClauses :: [Resource] -> Q [Clause]
|
mkRenderRouteClauses :: [Resource] -> Q [Clause]
|
||||||
mkRenderRouteClauses =
|
mkRenderRouteClauses =
|
||||||
mapM go
|
mapM go
|
||||||
@ -83,9 +78,9 @@ mkRenderRouteClauses =
|
|||||||
_ -> return []
|
_ -> return []
|
||||||
let pat = ConP (mkName $ resourceName res) $ map VarP $ dyns ++ sub
|
let pat = ConP (mkName $ resourceName res) $ map VarP $ dyns ++ sub
|
||||||
|
|
||||||
pack <- [|pack|]
|
pack' <- [|pack|]
|
||||||
tsp <- [|toPathPiece|]
|
tsp <- [|toPathPiece|]
|
||||||
let piecesSingle = mkPieces (AppE pack . LitE . StringL) tsp (resourcePieces res) dyns
|
let piecesSingle = mkPieces (AppE pack' . LitE . StringL) tsp (resourcePieces res) dyns
|
||||||
|
|
||||||
piecesMulti <-
|
piecesMulti <-
|
||||||
case resourceMulti res of
|
case resourceMulti res of
|
||||||
@ -102,7 +97,7 @@ mkRenderRouteClauses =
|
|||||||
b <- newName "b"
|
b <- newName "b"
|
||||||
|
|
||||||
colon <- [|(:)|]
|
colon <- [|(:)|]
|
||||||
let cons a b = InfixE (Just a) colon (Just b)
|
let cons y ys = InfixE (Just y) colon (Just ys)
|
||||||
let pieces = foldr cons (VarE a) piecesSingle
|
let pieces = foldr cons (VarE a) piecesSingle
|
||||||
|
|
||||||
return $ LamE [TupP [VarP a, VarP b]] (TupE [pieces, VarE b]) `AppE` (rr `AppE` VarE x)
|
return $ LamE [TupP [VarP a, VarP b]] (TupE [pieces, VarE b]) `AppE` (rr `AppE` VarE x)
|
||||||
@ -116,14 +111,23 @@ mkRenderRouteClauses =
|
|||||||
mkPieces _ _ [] _ = []
|
mkPieces _ _ [] _ = []
|
||||||
mkPieces toText tsp (Static s:ps) dyns = toText s : mkPieces toText tsp ps dyns
|
mkPieces toText tsp (Static s:ps) dyns = toText s : mkPieces toText tsp ps dyns
|
||||||
mkPieces toText tsp (Dynamic{}:ps) (d:dyns) = tsp `AppE` VarE d : mkPieces toText tsp ps dyns
|
mkPieces toText tsp (Dynamic{}:ps) (d:dyns) = tsp `AppE` VarE d : mkPieces toText tsp ps dyns
|
||||||
|
mkPieces _ _ ((Dynamic _) : _) [] = error "mkPieces 120"
|
||||||
|
|
||||||
mkRenderRouteInstance :: String -> [Resource] -> Q Dec
|
-- | Generate the 'RenderRoute' instance.
|
||||||
mkRenderRouteInstance name ress = do
|
--
|
||||||
|
-- This includes both the 'Route' associated type and the 'renderRoute' method.
|
||||||
|
-- This function uses both 'mkRouteCons' and 'mkRenderRouteClasses'.
|
||||||
|
mkRenderRouteInstance :: Type -> [Resource] -> Q Dec
|
||||||
|
mkRenderRouteInstance typ ress = do
|
||||||
cls <- mkRenderRouteClauses ress
|
cls <- mkRenderRouteClauses ress
|
||||||
return $ InstanceD [] (ConT ''RenderRoute `AppT` ConT (mkName name))
|
return $ InstanceD [] (ConT ''RenderRoute `AppT` typ)
|
||||||
[ FunD (mkName "renderRoute") cls
|
[ DataInstD [] ''Route [typ] (mkRouteCons ress) clazzes
|
||||||
|
, FunD (mkName "renderRoute") cls
|
||||||
]
|
]
|
||||||
|
where
|
||||||
|
clazzes = [''Show, ''Eq, ''Read]
|
||||||
|
|
||||||
|
{- FIXME
|
||||||
mkDispatchClause :: [Resource]
|
mkDispatchClause :: [Resource]
|
||||||
-> Q Exp -- ^ convert handler to application
|
-> Q Exp -- ^ convert handler to application
|
||||||
-> Q Clause
|
-> Q Clause
|
||||||
@ -155,9 +159,9 @@ mkDispatchClause ress toApp = do
|
|||||||
let m = maybe [|False|] (const [|True|]) $ resourceMulti res
|
let m = maybe [|False|] (const [|True|]) $ resourceMulti res
|
||||||
case resourceDispatch res of
|
case resourceDispatch res of
|
||||||
Methods mmulti mds -> do
|
Methods mmulti mds -> do
|
||||||
let toPair m = do
|
let toPair m' = do
|
||||||
key <- [|pack $(lift m)|]
|
key <- [|pack $(lift m')|]
|
||||||
let value = VarE $ mkName $ map toLower m ++ resourceName res
|
let value = VarE $ mkName $ map toLower m' ++ resourceName res
|
||||||
return $ TupE [key, value]
|
return $ TupE [key, value]
|
||||||
let handler =
|
let handler =
|
||||||
if null mds
|
if null mds
|
||||||
@ -186,6 +190,7 @@ mkDispatchClause ress toApp = do
|
|||||||
toPiece (Static s) = [|D.Static $ pack $(lift s)|]
|
toPiece (Static s) = [|D.Static $ pack $(lift s)|]
|
||||||
toPiece Dynamic{} = [|D.Dynamic|]
|
toPiece Dynamic{} = [|D.Dynamic|]
|
||||||
|
|
||||||
|
mkTsPattern :: [Piece] -> Maybe t -> Q ([Name], Maybe Name, Pat)
|
||||||
mkTsPattern pieces mmulti = do
|
mkTsPattern pieces mmulti = do
|
||||||
end <-
|
end <-
|
||||||
case mmulti of
|
case mmulti of
|
||||||
@ -200,3 +205,4 @@ mkTsPattern pieces mmulti = do
|
|||||||
go Dynamic{} = do
|
go Dynamic{} = do
|
||||||
dyn <- newName "dyn"
|
dyn <- newName "dyn"
|
||||||
return (Just dyn, VarP dyn)
|
return (Just dyn, VarP dyn)
|
||||||
|
-}
|
||||||
|
|||||||
@ -56,10 +56,12 @@ overlap = toDispatch
|
|||||||
test :: Dispatch () Int -> [Text] -> Maybe Int
|
test :: Dispatch () Int -> [Text] -> Maybe Int
|
||||||
test dispatch ts = dispatch ts ()
|
test dispatch ts = dispatch ts ()
|
||||||
|
|
||||||
|
data MyApp = MyApp
|
||||||
|
|
||||||
data MySub = MySub
|
data MySub = MySub
|
||||||
data instance YRC.Route MySub = MySubRoute ([Text], [(Text, Text)])
|
instance RenderRoute MySub where
|
||||||
deriving (Show, Eq, Read)
|
data YRC.Route MySub = MySubRoute ([Text], [(Text, Text)])
|
||||||
instance RenderRoute (YRC.Route MySub) where
|
deriving (Show, Eq, Read)
|
||||||
renderRoute (MySubRoute x) = x
|
renderRoute (MySubRoute x) = x
|
||||||
|
|
||||||
dispatchHelper :: Either String (Map.Map Text String) -> Maybe String
|
dispatchHelper :: Either String (Map.Map Text String) -> Maybe String
|
||||||
@ -73,12 +75,11 @@ do
|
|||||||
, Resource "WikiR" [Static "wiki"] $ Methods (Just texts) []
|
, Resource "WikiR" [Static "wiki"] $ Methods (Just texts) []
|
||||||
, Resource "SubsiteR" [Static "subsite"] $ Subsite (ConT ''MySub) "getMySub"
|
, Resource "SubsiteR" [Static "subsite"] $ Subsite (ConT ''MySub) "getMySub"
|
||||||
]
|
]
|
||||||
rrinst <- mkRenderRouteInstance "MyAppRoute" ress
|
rrinst <- mkRenderRouteInstance (ConT ''MyApp) ress
|
||||||
dispatch <- mkDispatchClause ress [|error "FIXME" dispatchHelper|]
|
dispatch <- [|error "FIXME dispatch"|]
|
||||||
return
|
return
|
||||||
[ mkRouteType "MyAppRoute" ress
|
[ rrinst
|
||||||
, rrinst
|
, FunD (mkName "thDispatch") [Clause [] (NormalB dispatch) []]
|
||||||
, FunD (mkName "thDispatch") [dispatch]
|
|
||||||
]
|
]
|
||||||
|
|
||||||
main :: IO ()
|
main :: IO ()
|
||||||
|
|||||||
Loading…
Reference in New Issue
Block a user