gpt4 book ai didi

scala - Play Framework : How to implement proper error handling

转载 作者:行者123 更新时间:2023-12-04 22:30:11 27 4
gpt4 key购买 nike

我有一个带有几个模块的 Play 应用程序,每个模块都有自己的异常集。以下是三个例子:

模块 common :

package services.common

trait CommonErrors {

final case class NotFound(id: String) extends Exception(s"object $id not found")
final case class InvalidId(id: String) extends Exception(s"$id is an invalid id")
...

// `toJson` is just an extension method that converts an exception to JSON
def toResult(e: Exception): Result = e match {
case NotFound => Results.NotFound(e.toJson)
case InvalidId => Results.BadRequest(e.toJson)
case _ => Results.InternalError(e.toJson)
}
}

模块 auth :
package services.auth

trait AuthErrors {

final case class UserNotFound(e: NotFound) extends Exception(s"user ${e.id} not found")
final case class UserAlreadyExists(email: String) extends Exception(s"user identified by $email already exists")
...

// `toJson` is just an extension method that converts an exception to JSON
def toResult(e: Exception): Result = e match {
case UserNotFound => Results.NotFound(e.toJson)
case UserAlreadyExists => Results.BadRequest(e.toJson)
case _ => Results.InternalError(e.toJson)
}
}

模块 other :
trait OtherErrors {

final case class AnotherError(s: String) extends Exception(s"another error: $s")
...

// `toJson` is just an extension method that converts an exception to JSON
def toResult(e: Exception): Result = e match {
case AnotherError => Results.BadRequest(e.toJson)
...
case _ => Results.InternalError(e.toJson)
}
}

如您所见,每个 trait 都定义了一组异常,并提供了一种将该异常转换为 JSON 响应的方法,如下所示:
{
"status": 404,
"code": "not_found",
"message": "user 123456789123456789123456 not found",
"request": "https://myhost.com/users/123456789123456789123456"
}

我想要实现的是让每个模块定义其异常,重用 common 模块中定义的异常,并根据需要混合异常特征:
object Users extends Controller {

val errors = new CommonErrors with AuthErrors with OtherErrors {
// here I have to override `toResult` to make the compiler happy
override def toResult(e: Exception) = super.toResult
}

def find(id: String) = Action { request =>
userService.find(id).map { user =>
Ok(success(user.toJson))
}.recover { case e =>
errors.toResult(e) // this returns the appropriate result
}
}
}

如果你看看我是如何覆盖 toResult 的,我总是返回 super.toResult ,它对应于 trait OtherErrors 中包含的实现......并且这个实现可能会错过一些预期会在 CommonErrors.toResult 中找到的模式。

当然我错过了一些东西......所以问题是:解决 toResult 多个实现问题的设计模式是什么?

最佳答案

您可以使用 Stackable Trait图案。我会替换你的 .toJson.getMessage为简化起见:

定义基本特征:

trait ErrorsStack {
def toResult(e: Exception): Result = e match {
case _ => Results.InternalServerError(e.getMessage)
}
}

和可堆叠的特征:
trait CommonErrors extends ErrorsStack {
case class NotFound(id: String) extends Exception(s"object $id not found")
case class InvalidId(id: String) extends Exception(s"$id is an invalid id")

override def toResult(e: Exception): Result = e match {
case e: NotFound => Results.NotFound(e.getMessage)
case e: InvalidId => Results.BadRequest(e.getMessage)
case _ => super.toResult(e)
}
}

trait AuthErrors extends ErrorsStack {
case class UserNotFound(id: String) extends Exception(s"user $id not found")
case class UserAlreadyExists(email: String) extends Exception(s"user identified by $email already exists")

override def toResult(e: Exception): Result = e match {
case e: UserNotFound => Results.NotFound(e.getMessage)
case e: UserAlreadyExists => Results.BadRequest(e.getMessage)
case _ => super.toResult(e)
}
}

trait OtherErrors extends ErrorsStack {
case class AnotherError(s: String) extends Exception(s"another error: $s")

override def toResult(e: Exception): Result = e match {
case e: AnotherError => Results.BadRequest(e.getMessage)

case _ => super.toResult(e)
}
}

所以如果我们有一些堆栈
val errors = new CommonErrors with AuthErrors with OtherErrors

并定义了一些助手
import java.nio.charset.StandardCharsets.UTF_8
import play.api.libs.iteratee.Iteratee
import concurrent.duration._
import scala.concurrent.Await

def getResult(ex: Exception) = {
val res = errors.toResult(ex)
val body = new String(Await.result(res.body.run(Iteratee.consume()), 5 seconds), UTF_8)
(res.header.status, body)
}

以下代码
import java.security.GeneralSecurityException

getResult(errors.UserNotFound("Riddle"))
getResult(errors.UserAlreadyExists("Weasley"))
getResult(errors.NotFound("Gryffindor sword"))
getResult(errors.AnotherError("Snape's death"))
getResult(new GeneralSecurityException("Marauders's map"))

会产生合理的输出
res0: (Int, String) = (404,user Riddle not found)
res1: (Int, String) = (400,user identified by Weasley already exists)
res2: (Int, String) = (404,object Gryffindor sword not found)
res3: (Int, String) = (400,another error: Snape's death)
res4: (Int, String) = (500,Marauders's map)

我们也可以重构这段代码,从特征中提取案例类,并使函数更具可组合性:
type Resolver = PartialFunction[Exception, Result]

object ErrorsStack {
val resolver: Resolver = {
case e => Results.InternalServerError(e.getMessage)
}
}

trait ErrorsStack {
def toResult: Resolver = ErrorsStack.resolver
}

object CommonErrors {
case class NotFound(id: String) extends Exception(s"object $id not found")
case class InvalidId(id: String) extends Exception(s"$id is an invalid id")
val resolver: Resolver = {
case e: NotFound => Results.NotFound(e.getMessage)
case e: InvalidId => Results.BadRequest(e.getMessage)
}
}

trait CommonErrors extends ErrorsStack {
override def toResult = CommonErrors.resolver orElse super.toResult
}

object AuthErrors {
case class UserNotFound(id: String) extends Exception(s"user $id not found")
case class UserAlreadyExists(email: String) extends Exception(s"user identified by $email already exists")
val resolver: Resolver = {
case e: UserNotFound => Results.NotFound(e.getMessage)
case e: UserAlreadyExists => Results.BadRequest(e.getMessage)
}
}

trait AuthErrors extends ErrorsStack {
override def toResult = AuthErrors.resolver orElse super.toResult
}

object OtherErrors {
case class AnotherError(s: String) extends Exception(s"another error: $s")

val resolver: Resolver = {
case e: AnotherError => Results.BadRequest(e.getMessage)
}
}

trait OtherErrors extends ErrorsStack {
override def toResult = OtherErrors.resolver orElse super.toResult
}

关于scala - Play Framework : How to implement proper error handling,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/30351481/

27 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com