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

Posted

技术标签:

【中文标题】Symfony 5 更新后 Laravel 7 电子邮件异常中断【英文标题】:Laravel 7 email exceptions broke after Symfony 5 update 【发布时间】:2020-06-26 06:24:05 【问题描述】:

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

use Symfony\Component\Debug\Exception\FlattenException;
use Symfony\Component\Debug\ExceptionHandler as SymfonyExceptionHandler;

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

我用这些替换了上述类:

use Symfony\Component\ErrorHandler\ErrorRenderer\htmlErrorRenderer;
use Symfony\Component\ErrorHandler\Exception\FlattenException;

然后替换这个:

$e = FlattenException::create($exception);
$handler = new SymfonyExceptionHandler();
$html = $handler->getHtml($e);

用这个:

$e = FlattenException::create($exception);
$handler = new HtmlErrorRenderer();
$content = $handler->getBody($e);

这可行,但现在我不再像以前那样在电子邮件中获取调试内容,而是收到一条更基本的错误消息,因为它是为公众准备的。

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

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

有什么建议吗?

【问题讨论】:

【参考方案1】:

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

new HtmlErrorRenderer(true);

这是我现在用于 app/Exceptions/Handler.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);
        
    

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

<!DOCTYPE html>
<html>
    <head>
        <meta charset="UTF-8" />
        <style>!! $css ?? '' !!</style>
    </head>
    <body>
        !! $content ?? '' !!
    </body>
</html>

【讨论】:

正如@Androi_Begin 在下面指出的那样,我得到了TypeError: Argument 1 passed to Symfony\Component\ErrorHandler\Exception\FlattenException::create() must be an instance of Exception, instance of Error given ...我将create() 更改为createFromThrowable(),现在它正在工作。 请将@Androi_Begin 更改添加到您的代码中,以便其他人发现此解决方案!【参考方案2】:

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

【讨论】:

这并没有提供问题的答案。一旦你有足够的reputation,你就可以comment on any post;相反,provide answers that don't require clarification from the asker。 - From Review 这应该是对已接受答案的评论。 我还没有达到 50 的声誉,无法将其放入已接受的答案评论中 感谢@Androi_Begin - 这解决了我的问题!我将为您将评论添加到已接受的答案中,希望没关系 很高兴为您提供帮助【参考方案3】:

要获得完整的 html 响应,只需使用:

$html = ExceptionHandler::convertExceptionToResponse($e);

这是一个完整的 Handler.php 代码

<?php

namespace App\Exceptions;

use Log;
use Mail;
use Exception;
use Throwable;
use App\Mail\ErrorNotification;
use Illuminate\Database\Eloquent\ModelNotFoundException;
use Symfony\Component\HttpKernel\Exception\HttpException;
use Illuminate\Foundation\Exceptions\Handler as ExceptionHandler;
use Symfony\Component\HttpKernel\Exception\NotFoundHttpException;


class Handler extends ExceptionHandler


    /**
     * A list of the exception types that should not be reported.
     *
     * @var array
     */
    protected $dontReport = [
        HttpException::class,
        ModelNotFoundException::class,
    ];

    /**
     * Report or log an exception.
     *
     * This is a great spot to send exceptions to Sentry, Bugsnag, etc.
     *
     * @param  \Exception  $e
     * @return void
     */
    public function report(Throwable $e)
    
        if ($this->shouldReport($e)) 
            $this->sendEmail($e);
        

        return parent::report($e);
    

    /**
     * Render an exception into an HTTP response.
     *
     * @param  \Illuminate\Http\Request  $request
     * @param  \Exception  $e
     * @return \Illuminate\Http\Response
     */
    public function render($request, Throwable $e)
    
        if ($e instanceof ModelNotFoundException) 
            $e = new NotFoundHttpException($e->getMessage(), $e);
        

        return parent::render($request, $e);
    


    public function sendEmail(Throwable $e)
    
        try 

            $html = ExceptionHandler::convertExceptionToResponse($e);

            Mail::to('youremailhere@gmail.com')->send(new ErrorNotification($html));

         catch (Exception $ex) 
            Log::error($ex);
        
    


【讨论】:

谢谢。虽然这提供了一些我正在寻找的数据,但这并不是我想要的。我发布了我自己的问题的答案,显示了我正在使用的代码。 预期类型 'Illuminate\Contracts\Mail\Mailable'。找到 'App\Mail\ErrorNotification'.intelephense(1006)| laravel 8

以上是关于Symfony 5 更新后 Laravel 7 电子邮件异常中断的主要内容,如果未能解决你的问题,请参考以下文章

从 5.4 升级后的 Laravel 419 发布请求

Laravel 5.2 |测试 UploadedFile 在发布后错过了 $test 值。漏洞?

Symfony 2.1.7 - 用户通过身份验证后更新安全令牌设置特定角色

升级到 Symfony 5.3 并更新 flex 配方后出错 (symfony:recipes:install --force)

更新到 php 7 后如何修复 symfony 1.4 错误“Empty response header name, aborting request”

Symfony \ Component \ HttpKernel \ Exception \ MethodNotAllowedHttpException on laravel 5.6