我想测试用户是否经过身份验证,并根据状态显示“登录”或“注销”链接。
到目前为止我已经
了$maybe muid <- maybeAuthId
<a href=@{AuthR LogoutR} >Logout
$nothing
<a href=@{AuthR LoginR} >Login
但是我收到了错误:
Couldn't match expected type `Maybe v0'
with actual type `GHandler s0 m0 (Maybe (AuthId m0))'
In the first argument of `Text.Hamlet.maybeH', namely `maybeAuthId'
答案 0 :(得分:6)
maybeAuthId
是一个执行数据库和会话相关操作的monadic操作。在Hamlet模板的定义中不能有monadic动作。想象一下如果你写这个(一个类似的monadic动作)会发生什么:
$maybe a <- liftIO (putStrLn "Hello World") >> return (Just "Hi")
<p>Just #{a}
$nothing
<p>Nothing
该行动应多久执行一次;每次渲染模板?什么时候加载?如果除了将“Hello World”打印到终端之外,它可能会变得非常混乱,即使这样也不是很安全 - 你希望你的模板文件能够打印到终端,启动核武器或窃取你的信用卡信息?
这就是为什么在所有莎士比亚模板中只允许纯值的原因。你需要这样做:
getMyHandlerR :: Handler RepHtml
getMyHandlerR = do
muid <- maybeAuthId
$(widgetFile "foo")
(foo.hamlet:)
$maybe uid <- muid
<p>Foo
$nothing
<p>Bar
如您所见,maybeAuthId
函数将在模板外执行,结果在模板中匹配。这样,您可以确保在您可以确定的特定时间点检查您的会话/数据库,并确保您的模板没有注入病毒,因为您的设计师没有得到足够的报酬并且他的报复就是你。
顺便说一句,您可能希望使用Bool
来指示用户是否已登录并使用$if
语句。您可能希望使用isJust
模块中的Data.Maybe
函数。