Пользовательские ограничения проверки JSON в Play Framework 2.3 (Scala)

Мне удалось реализовать проверку формы с пользовательскими ограничениями, но теперь я хочу сделать то же самое с данными JSON.

Как я могу применить пользовательские правила проверки к анализатору JSON?

Пример: POST-запрос клиента содержит имя пользователя (username) и я не только хочу убедиться, что этот параметр является непустым текстом, но и что этот пользователь действительно существует в базе данных.

// In the controller...

def postNew = Action { implicit request =>
    request.body.asJson.map { json =>
        json.validate[ExampleCaseClass] match {
            case success: JsSuccess[ExampleCaseClass] =>
                val obj: ExampleCaseClass = success.get
                // ...do something with obj...
                Ok("ok")
            case error: JsError =>
                BadRequest(JsError.toFlatJson(error))
        }
    } getOrElse(BadRequest(Json.obj("msg" -> "JSON request expected")))
}


// In ExampleCaseClass.scala...

case class ExampleCaseClass(username: String, somethingElse: String)

object ExampleCaseClass {
    // That's what I would use for a form:
    val userCheck: Mapping[String] = nonEmptyText.verifying(userExistsConstraint)

    implicit val exampleReads: Reads[ExampleCaseClass] = (
        (JsPath \ "username").read[String] and
        (JsPath \ "somethingElse").read[String]
    )(ExampleCaseClass.apply _)
}

Это насколько я понимаю, но это только гарантирует, чтоusername это строкаКак применить мое дополнительное пользовательское правило проверкинапример, проверить, существует ли данный пользователь на самом деле? Это вообще возможно?

Конечно, я мог бы взятьobj вcase success раздел в действии и выполнить дополнительные проверки там, но это не кажется очень элегантным, потому что тогда мне придется создать свое собственное сообщение об ошибке и может только пользовательJsError.toFlatJson(error) для некоторых случаев. После поисков и попыток в течение нескольких часов я не мог найти никаких примеров.

Для обычных форм я бы использовал что-то вроде этого:

// In the controller object...

val userValidConstraint: Constraint[String] = Constraint("constraints.uservalid")({ username =>
    if (User.find(username).isDefined) {
        Valid
    } else {
        val errors = Seq(ValidationError("User does not exist"))
        Invalid(errors)
    }
})

val userCheck: Mapping[String] = nonEmptyText.verifying(userValidConstraint)

val exampleForm = Form(
    mapping(
        "username" -> userCheck
        // ...and maybe some more fields...
    )(ExampleCaseClass.apply)(ExampleCaseClass.unapply)
)


// In the controller's action method...

exampleForm.bindFromRequest.fold(
    formWithErrors => {
        BadRequest("Example error message")
    },
    formData => {
        // do something
        Ok("Valid!")
    }
)

Но что, если данные представлены в формате JSON?

Ответы на вопрос(1)

Ваш ответ на вопрос