【问题标题】:How to correctly validate forms with Play! in Scala?如何使用 Play 正确验证表单!在斯卡拉?
【发布时间】:2013-11-06 11:47:49
【问题描述】:

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

我面临的问题是关于表单验证。

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

case class User(
  val id: Long,
  val username: String,
  val password: String,
  val email: String
  val created: Date)

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

我希望我的用户能够使用表单在我的网站上创建一个帐户,并且我希望 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.")
    })
}

显然,我不希望用户在表单中自己填写 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")
}

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

我已经阅读了许多教程和“Play for Scala”一书,但在我发现的唯一示例中,模型完全被表格填充......我真的很喜欢 Play!到目前为止,但看起来文档中经常缺少示例...

非常感谢您的回答!

【问题讨论】:

  • 我不使用您描述的任何类型的过渡模型。因此,对于用户注册表单的示例,id 字段可以是任何内容,因为我的 User.create 函数无论如何都会忽略它。 (数据库中的自动增量)。我对我的用户编辑表单(与创建表单相同)做同样的事情,除了我在表单绑定之后使用我的身份验证系统验证 id。

标签: forms validation scala playframework playframework-2.0


【解决方案1】:

你有几个选择:

首先,您可以将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)
)

我认为这是合乎逻辑的,因为带有None id 的User 表明它尚未保存。当您想使用相同的表单映射来更新现有记录时,这很有效。

或者,您可以将ignored 映射与一些任意占位符数据一起使用:

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

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

最后,不要忘记您的表单映射是由函数绑定/填充的,它们分别将元组转换为对象,将对象转换为元组。使用 case 类 User.applyUser.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))
}

然后使用 Form 对象中的那些:

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

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

【讨论】:

  • 非常感谢您非常详细的回答!我特别喜欢使用 apply / unapply 功能的解决方案。拥有一些类似应用的功能是一种不好的做法,还是我仍然缺少一些东西? (通过某些功能,我的意思是 2 到 3...)
  • @Martin:想不出为什么这是不好的做法。 companion object 中以不同方式构造类实例的工厂方法是完全有效的。
猜你喜欢
  • 1970-01-01
  • 2012-11-15
  • 2018-01-25
  • 2021-11-30
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
相关资源
最近更新 更多