使用yesod-form在POST / Redirect / GET后保持不正确的用户输入

时间:2013-03-28 13:49:43

标签: forms haskell yesod post-redirect-get

最近我偶然发现了以下问题:
使用yesod我想

  1. 显示一个应用表单,让用户将其数据发布到特殊的处理程序
  2. FormFailure 浏览器重定向到上一页
  3. 打印错误消息...
  4. ...并使用错误的用户提供的数据重新显示 已填充 表单
  5. 就我而言,这就是POST/Redirect/GET的全部内容。

    而点数< 1-3>需要一个简单而直接的实现,我发现无法实现点< 4>!
    yesod-form 包自动处理此问题,但不允许在表单解析< 2>之间进行任何重定向。和错误处理< 3,4>正如我所希望的那样。

    对迈克尔斯诺曼答案的回应

    您建议我应该序列化提交的数据,并在重定向后以某种方式将其注入表单。这导致了更详细的问题:

    1. 如何获取序列化数据?

      我知道我可以使用runRequestBody :: GHandler s m RequestBodyContents,但哪些是相关信息(字段的name会自动生成)?

    2. 如何将数据注入表单?

      如果你看一下例如aopt

      aopt :: Field sub master a -> FieldSettings master -> Maybe (Maybe a) -> AForm sub master (Maybe a)
      

      你会发现它要求默认值与Field的类型相同,因此无法重新插入可能无法正确解析的用户提供的数据。

      示例:用户在intField中键入“A”。现在我希望能够在重定向后在同一个字段中显示“A”,但API不允许我这样做。

    3. 我该如何处理这个问题?

2 个答案:

答案 0 :(得分:2)

我个人认为返回带有POST请求的填充表单是可以接受的,这是yesod-form API的优化。如果你想强制重定向表单提交失败,你需要序列化提交的数据并将其存储在某个地方,例如:

  1. 在数据库中。
  2. 在用户会话中。
  3. 作为您重定向到的网址的查询字符串参数的一部分。请注意,此方法不适用于敏感数据,因为任何中间代理都会缓存表单数据。

答案 1 :(得分:2)

老问题,但我今天需要这个,所以不妨将其发布给遇到同样问题的其他人。

基本上,正如迈克尔建议的那样,我们可以将数据序列化到会话中。这样做很棘手,加上一个表格甚至更棘手。我不得不从postEnv翻录postHelperYesod.Form.Functions,因为它们不会被导出但是需要这样做。

然后,您可以在重定向之前在处理程序中使用setLastInvalidPost,然后在目标处理程序中使用generateFormFromLastPost

请注意,使用类似Data.Serialize的内容进行序列化可能会更好;但是,Show / Read个实例足以满足我的需求(并且更简单)。

这里有好东西。如果您需要完整的工作代码段you can check out my gist

-- Create a form from last post data in the session if exists, otherwise create a blank form.
generateFormFromLastPost :: (RenderMessage (HandlerSite m) FormMessage, MonadHandler m) =>
                            (Markup -> MForm m (FormResult a, xml)) -> m (xml, Enctype)
generateFormFromLastPost form = do
    env <- getLastInvalidPost
    case env of
        Nothing -> generateFormPost form
        Just _ -> first snd <$> postHelper form env

lastInvalidPostSessionKey :: Text
lastInvalidPostSessionKey = "lastInvalidPost"

-- Sets the post data retreived from postEnv, ignoring the FileEnv.
setLastInvalidPost :: MonadHandler m => Maybe (Env, FileEnv) -> m ()
setLastInvalidPost Nothing = return ()
setLastInvalidPost (Just (env, _)) = sessionSetter lastInvalidPostSessionKey env

-- Retrieves the previous post data to be passed to postHelper.
getLastInvalidPost :: MonadHandler m => m (Maybe (Env, FileEnv))
getLastInvalidPost = do
    result <- sessionGetter lastInvalidPostSessionKey
    return $ case result of
        Nothing -> Nothing
        Just env -> Just (env, Map.fromList [])

sessionSetter :: (MonadHandler m, Show a) => Text -> a -> m ()
sessionSetter key = setSession key . pack . show

sessionGetter :: (MonadHandler m, Read b) => Text -> m (Maybe b)
sessionGetter key = do
    m <- lookupSession key
    return $ readMaybe . unpack =<< m