Laravel自定义异常配置

时间:2017-06-30 11:53:57

标签: php exception error-handling exception-handling laravel-5.3

我一直试图在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中输入日志代码。

谢谢!

2 个答案:

答案 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());
}