Symfony 5 更新后 Laravel 7 电子邮件异常中断

cli*_*mbd 5 php email exception symfony laravel-7

我已经升级到 Laravel 7.1,现在使用 Symfony 5 这些类不再存在:

use Symfony\Component\Debug\Exception\FlattenException;
use Symfony\Component\Debug\ExceptionHandler as SymfonyExceptionHandler;
Run Code Online (Sandbox Code Playgroud)

我在我的 app\Exceptions\Handler.php 文件中使用它们在抛出异常时发送电子邮件通知,并且它们在 Laravel 6 中运行良好,但是当我从 6.x 升级到 7.1.2 时坏了,7.1.2 也升级到了 Symfony 5。

我用这些替换了上述类:

use Symfony\Component\ErrorHandler\ErrorRenderer\HtmlErrorRenderer;
use Symfony\Component\ErrorHandler\Exception\FlattenException;
Run Code Online (Sandbox Code Playgroud)

然后替换了这个:

$e = FlattenException::create($exception);
$handler = new SymfonyExceptionHandler();
$html = $handler->getHtml($e);
Run Code Online (Sandbox Code Playgroud)

有了这个:

$e = FlattenException::create($exception);
$handler = new HtmlErrorRenderer();
$content = $handler->getBody($e);
Run Code Online (Sandbox Code Playgroud)

这行得通,但现在不是像以前那样在电子邮件中获取调试内容,而是收到了更基本的错误消息,因为它是面向公众的。

您可以在此处查看不同格式的示例:https : //symfony.com/doc/current/controller/error_pages.html

我确信我遗漏了一些简单的东西,但我还没有想出如何让它向我发送详细的异常数据,就像我在升级之前得到的那样。

有什么建议?

cli*_*mbd 13

下面是我最终用来在异常通知电子邮件中获得我想要的结果的代码。我之前遗漏的主要部分是我没有将真正的值传递给 HtmlErrorRender 类来引发调试标志。更正后的行如下所示:

new HtmlErrorRenderer(true);
Run Code Online (Sandbox Code Playgroud)

这是我现在用于 app/Exceptions/Handler.php 文件的完整代码

new HtmlErrorRenderer(true);
Run Code Online (Sandbox Code Playgroud)

$css 和 $content 被传递到 resources/views/emails/exception.blade.php 的视图中。我在该文件中的代码如下:

<?php

namespace App\Exceptions;

use Illuminate\Foundation\Exceptions\Handler as ExceptionHandler;
use Log;
use Throwable;
use Symfony\Component\ErrorHandler\ErrorRenderer\HtmlErrorRenderer;
use Symfony\Component\ErrorHandler\Exception\FlattenException;

class Handler extends ExceptionHandler
{
    /**
     * A list of the exception types that are not reported.
     *
     * @var array
     */
    protected $dontReport = [
    ];

    /**
     * A list of the inputs that are never flashed for validation exceptions.
     *
     * @var array
     */
    protected $dontFlash = [
        'password',
        'password_confirmation',
    ];

    /**
     * Report or log an exception.
     *
     * @param  \Throwable  $exception
     * @return void
     *
     * @throws \Exception
     */
    public function report(Throwable $exception)
    {
        if ($this->shouldReport($exception)) {
            $this->sendEmail($exception); // sends an email
        }
        parent::report($exception);
    }

    /**
     * Render an exception into an HTTP response.
     *
     * @param  \Illuminate\Http\Request  $request
     * @param  \Throwable  $exception
     * @return \Symfony\Component\HttpFoundation\Response
     *
     * @throws \Throwable
     */
    public function render($request, Throwable $exception)
    {
        if ($exception instanceof \Illuminate\Session\TokenMismatchException) {  //https://gist.github.com/jrmadsen67/bd0f9ad0ef1ed6bb594e
            return redirect()
                ->back()
                ->withInput($request->except('password'))
                ->with('errorMessage', 'This form has expired due to inactivity. Please try again.');
        }

        return parent::render($request, $exception);
    }

    /**
     * Sends an email to the developer about the exception.
     *
     * @return void
     */
    public function sendEmail(Throwable $exception)
    {
        try {
            $e = FlattenException::create($exception);
            $handler = new HtmlErrorRenderer(true); // boolean, true raises debug flag...
            $css = $handler->getStylesheet();
            $content = $handler->getBody($e);

            \Mail::send('emails.exception', compact('css','content'), function ($message) {
                $message
                    ->to('youremailhere@gmail.com')
                    ->subject('Exception: ' . \Request::fullUrl())
                ;
            });
        } catch (Throwable $ex) {
            Log::error($ex);
        }
    }
}
Run Code Online (Sandbox Code Playgroud)

  • 正如@Androi_Begin在下面指出的那样,我得到了“TypeError:参数1传递给Symfony \ Component \ ErrorHandler \ Exception \ FlattenException :: create()必须是异常的实例,给定错误的实例”...我更改了“创建” ()` 到 `createFromThrowable()` 现在正在工作。 (2认同)

小智 6

测试答案后,我收到一些错误“解析到异常类”,因此如果您使用的是 Laravel 7,则需要使用“create”的“createFromThrowable”来与 Throwable 对象兼容。