如何使用Play正确验证表单!在斯卡拉?

Mar*_*tin 6 forms validation scala playframework playframework-2.0

我是Play!的新手,我正在尝试将现有网站从cakePHP迁移到Play!.

我面临的问题是表单验证.

我定义了一个案例类User,代表我网站的用户:

case class User(
  val id: Long,
  val username: String,
  val password: String,
  val email: String
  val created: Date)
Run Code Online (Sandbox Code Playgroud)

(还有一些字段,但这些字段足以解释我的问题)

我希望我的用户能够使用表单在我的网站上创建一个帐户,我希望这个表单能够通过Play!进行验证.

所以,我创建了以下操作:

def register = Action {
  implicit request =>

    val userForm = Form(
      mapping(
        "id" -> longNumber,
        "username" -> nonEmptyText(8),
        "password" -> nonEmptyText(5),
        "email" -> email,
        "created" -> date)(User.apply)(User.unapply))

    val processedForm = userForm.bindFromRequest
    processedForm.fold(hasErrors => BadRequest("Invalid submission"), success => {
      Ok("Account registered.")
    })
}
Run Code Online (Sandbox Code Playgroud)

显然,我不希望用户在表单中自己填写id或创建日期.所以我的问题是:我该怎么办?

我应该定义一个新的"转换模型",其中只包含实际提供给表单中用户的字段,并在将此中间模型插入到我的数据库之前将其转换为完整的模型吗?

也就是说,用这样的东西代替我的行动:

def register = Action {
  implicit request =>

    case class UserRegister(
      username: String,
      password: String,
      email: String)

    val userForm = Form(
      mapping(
        "username" -> nonEmptyText(8),
        "password" -> nonEmptyText(8),
        "email" -> email)(UserRegister.apply)(UserRegister.unapply)

    val processedForm = userForm.bindFromRequest
    processedForm.fold(hasErrors => BadRequest("Invalid submission"), success => {
      val user = User(nextID, success.username, success.password, success.email, new Date())
      // Register the user...
      Ok("Account created")
}
Run Code Online (Sandbox Code Playgroud)

或者是否有另一种更清洁的方式来做我想做的事情?

我已经阅读了很多教程和"Play for Scala"这本书,但在我发现的唯一例子中,模型完全由表格填充......我真的很喜欢Play!到目前为止,但看起来文档往往缺乏示例......

非常感谢您的回答!

Mik*_*ame 4

你有几个选择:

首先,您可以分别创建idcreated字段Option[Long]Option[Date]。然后使用如下映射:

val userForm = Form(
  mapping(
    "id" -> optional(longNumber),
    "username" -> nonEmptyText(8),
    "password" -> nonEmptyText(5),
    "email" -> email,
    "created" -> optional(date)
  )(User.apply)(User.unapply)
)
Run Code Online (Sandbox Code Playgroud)

我认为这是合乎逻辑的,因为User带有Noneid 的 a 表示它尚未保存。当您想要使用相同的表单映射来更新现有记录时,此方法非常有效。

或者,您可以使用ignored带有一些任意占位符数据的映射:

val userForm = Form(
  mapping(
    "id" -> ignored(-1L),
    "username" -> nonEmptyText(8),
    "password" -> nonEmptyText(5),
    "email" -> email,
    "created" -> ignored(new Date)
  )(User.apply)(User.unapply)
)
Run Code Online (Sandbox Code Playgroud)

当重用表单进行更新操作时,这不太好!

最后,不要忘记您的表单映射是由分别将元组转换为对象以及将对象转换为元组的函数绑定/填充的。User.apply使用案例类和方法只是一种方便的约定,User.unapply因为它们就是这样做的。您可以在对象上编写替代工厂方法User来处理表单实例化:

object User {
  def formApply(username: String, password: String, email: String): User = 
    new User(-1L, username, password, email, new Date)

  def formUnapply(user: User): Option[(String,String,String)] = 
    Some((user.username, user.password, user.email))
}
Run Code Online (Sandbox Code Playgroud)

然后在 Form 对象中使用它们:

val userForm = Form(
  mapping(
    "username" -> nonEmptyText(8),
    "password" -> nonEmptyText(5),
    "email" -> email
  )(User.formApply)(User.formUnapply)
)
Run Code Online (Sandbox Code Playgroud)

另外,值得注意的是,Scala 表单文档在 2.2.1 中将会变得更好(事实上它可能已经在这里推出)。