我在Play 2.6中提交了一个表单,其中大多数验证都无法预先执行。 Web应用程序将提交的表单数据发送到另一个项目的后端,这将为大多数用户错误抛出GrammarException。如何将错误消息和原始表单值传播回视图
这类似于How to access my forms properties when validation fails in the fold call?,但我需要成功的表单值。
form.bindFromRequest().fold(
formWithErrors => {
BadRequest(myView(newForm = formWithErrors)(request))
},
data => try {
val results = MyBackend.build(data) // time-consuming
Ok(views.html.myView(results)
} catch { // catches most user errors
case e: GrammarException =>
val submittedForm = ....? //
val formWithErrors = submittedForm.withGlobalError(e.getMessage)
BadRequest(myView(newForm = formWithErrors)(request))
}
)
答案 0 :(得分:1)
您已经拥有包含请求中所有数据的表单,因此您可以使用它。
val formWithBoundData = form.bindFromRequest()
formWithBoundData.fold(
formWithErrors => {
BadRequest(myView(newForm = formWithErrors)(request))
},
data => try {
val results = MyBackend.build(data) // time-consuming
Ok(views.html.myView(results)
} catch { // catches most user errors
case e: GrammarException =>
val formWithErrors = formWithBoundData.withGlobalError(e.getMessage)
BadRequest(myView(newForm = formWithErrors)(request))
}
)
答案 1 :(得分:1)
我建议在处理对另一个组件(模型,api等)的调用时不要在代码中使用try/catch
,这就是原因:
尽可能多地编写并发代码:在Scala / Play中,我们可以使用Future
这个奇妙的世界来编写并发代码。当方法具有某些内容的recover
时(例如Future
),我们无需“捕获”任何内容,而是使用Future[Unit]
。所以尽可能多地编写并发代码。 Try / catch是类似Java的代码,可能会对问题所在的位置产生误解;这让我想到了下一点。
关注点分离会发生什么?:如果我在控制器中尝试/ catch,而我在模型中调用/使用方法,那么事实就是{{1处理controller
错误的错误是什么?为什么?如果模型有错误,控制器应该知道它的确切类型吗?或者控制器应该只知道存在错误(哪种类型不是他/她的业务),并且例如向视图返回model
响应;还是InternalServerError
?
以下代码调用模型/后端,映射返回结果,如果出现错误,它会将BadRequesst
恢复为Future
:
BadRequest
上述方法的好处:
并发和关注点分离:已经解释了原因。
扩展到多个来电:稍后,如果您有新约束,您仍然可以使用上述方法使用form.bindFromRequest().fold(
formWithErrors => {
BadRequest(myView(formWithErrors)(request))
},
givenData =>
MyBackend.build(data).map{
_ => Ok(views.html.myView(results) //When case is successful
}.recover{
//Any other exception that may occur while calling the other components other than the controller.
case _ => {
val formWithErrors = ???
BadRequest(FormWithErrors)
}
//Could build on that if you want to match an exact exception like GrammerException
}
}
进行外部呼叫,flatMap
进行内部呼叫,映射他们的成功和失败。
更多例外:如果您需要知道确切的异常并提供最佳错误/警告消息,您可以在map
块中展开更多recover
类型观点。
答案 2 :(得分:0)
从我看到的情况来看,Form的case类有一个错误字段:https://github.com/playframework/playframework/blob/master/framework/src/play/src/main/scala/play/api/data/Form.scala#L37
您可以复制收到的表格,添加错误,然后将其退回,不是吗?