在Silex中可以根据抛出的异常使用错误处理程序吗?
我知道这可以通过一个异常处理程序和一个关于抛出异常的类名的switch语句来实现,但对我而言,似乎“Silex方式”更清晰,但不起作用。
这就是我期望它的工作方式
<?php
// Handle access denied errors
$app->error(function (\App\Rest\Exception\AccessDenied $e) {
$message = $e->getMessage() ?: 'Access denied!';
return new Response($message, 403);
});
// Handle Resource not found errors
$app->error(function (\App\Rest\Exception\ResourceNotFound $e) {
$message = $e->getMessage() ?: 'Resource not found!';
return new Response($message, 404);
});
// Handle other exception as 500 errors
$app->error(function (\Exception $e, $code) {
return new Response($e->getMessage(), $code);
});
问题是当我在控制器中抛出ResourceNotFound异常时,会执行绑定到AccessDenied的错误处理程序
可捕获的致命错误:传递给{closure}()的参数1必须是App \ Rest \ Exception \ AccessDenied的实例,App \ Rest \ Exception \ ResourceNotFound的实例
这是以另一种方式实现的,还是应该只是处理与通用异常一起使用的处理程序中的所有内容并切换抛出的异常类型?
PS:我知道$app->abort()
方法,但更喜欢处理异常
答案 0 :(得分:5)
编辑:此功能现已成为Silex核心!
目前无法做到这一点。现在你必须拥有一个带有switch语句的处理程序,或者每个都有if ($e instanceof MyException)
的处理程序。
我确实喜欢这个想法,应该可以通过使用反射来实现它。如果您感兴趣的话,如果您可以创建新的票证on the tracker,甚至可以在补丁上工作,那将是非常棒的。
干杯!
答案 1 :(得分:0)
我在项目中使用的另一种解决方案:
class ProcessCallbackException extends Exception
{
public function __construct(\Closure $callback, $message = "", Exception $previous = null)
{
parent::__construct($message, 0, $previous);
$this->callback = $callback;
}
public $callback;
}
class AccessDeniedException extends ProcessCallbackException
{
public function __construct($message = null)
{
$f = function() {
return app()->redirect('/login');
};
parent::__construct($f, $message);
}
}
# Handle your special errors
$app->error(function (\Exception $e, $code) {
if ($e instanceof ProcessCallbackException)
{
/** @var ProcessCallbackException $callbackException */
$callbackException = $e;
return call_user_func($callbackException->callback);
}
else
return null;
});