我一直试图在Laravel 5.3中实现一些自定义异常而没有运气。
我创建了一个扩展Exception类的AppException。这个想法是我想将一个用户附加到我的例外,所以我知道是谁绊了它并且可以记录它。
所以我设置了这样的自定义异常:
class AppException extends Exception{
protected $userId;
public function __construct($message = '', $userId=0)
{
parent::__construct($message);
$this->userId = $userId;
}
}
现在我想要的是在我的控制器函数中使用try catch,然后捕获异常,然后重新抛出应用程序异常,以便我可以附加用户。像这样:
try{
<< code here >>
}
catch(Exception $e){
throw new AppException($e->getMessage, $request->user()->id);
}
我发现的是我无法获得良好的跟踪堆栈,因为我从异常中记录的行是来自catch中的rethrow的行,而不是来自实际的异常。
设置它的正确方法是什么?我试图这样做,我可以利用Laravel附带的内置Handler.php文件,而不必在每个try catch中输入日志代码。
谢谢!
答案 0 :(得分:0)
尝试将实际异常附加到抛出的异常,例如:
class AppException extends \Exception
{
/**
* @var int
*/
private $userId;
/**
* @param \Exception $exception
* @param int $userId
*
* @return self
*/
public static function fromExceptionAndUserId(\Exception $exception, $userId)
{
$instance = new self(
$exception->getMessage(),
0,
$exception
);
$instance->userId = $userId;
return $instance;
}
/**
* @return null|int
*/
public function userId()
{
return $this->userId;
}
}
然后在你的控制器中:
try {
// ...
} catch (\Exception $exception) {
throw AppException::fromExceptionAndUserId(
$exception,
$request->user()->id
);
}
这个想法不仅要使用实际的异常消息,还要将实际异常作为$previous
异常附加。
使用命名构造函数为您提供的优势是,您可以控制AppException
的异常消息的构造方式(同时还附加用户标识符),并且仍然可以使用原始{{1}构造函数。它使您可以自由地为不同的用例添加更多命名构造函数。
供参考,见
答案 1 :(得分:0)
Exception
类有一个previous
参数,您可以使用它来检索所有先前的异常。
class AppException extends Exception
{
private $userId;
public function __construct($userId, $message = '', $code = 0, Exception $previous = null)
{
$this->userId = $userId;
parent::__construct($message, $code, $previous);
}
public static function fromUserId($userId, Exception $previous = null)
{
return new static($userId, sprintf('Exception thrown from `%s` userid', $userId), 0, $previous);
}
public function getUserId()
{
return $this->userId;
}
}
或者只是简单地
class AppException extends Exception
{
public static function fromUserId($userId, Exception $previous = null)
{
return new static(sprintf('Exception thrown from `%s` userid', $userId), 0, $previous);
}
}
在捕获AppException
之后,您可以像这样迭代所有异常:
do {
printf("%s:%d %s (%d) [%s]\n", $e->getFile(), $e->getLine(), $e->getMessage(), $e->getCode(), get_class($e));
} while($e = $e->getPrevious());
示例:
try {
try{
throw new RuntimeException('Something bad happened.');
}
catch(Exception $e){
throw AppException::fromUserId(123, $e);
}
} catch(AppException $e) {
do {
printf("%s:%d %s (%d) [%s]\n", $e->getFile(), $e->getLine(), $e->getMessage(), $e->getCode(), get_class($e));
} while($e = $e->getPrevious());
}