php - Symfony 中的 ResponseListener 导致在开发模式下来自 apache 的重复 header 'Content-Type'

标签 php apache symfony error-handling mamp

我在 Mac 上的 MAMP PRO 设置上运行 Symfony 应用程序。在我的 symfony 应用程序中,我使用了一个包含以下函数的 ResponseListener:

/**
 * Handle OPTIONS calls and add Access-Control headers.
 *
 * @param FilterResponseEvent $event Filter response event
 */
public function onKernelResponse(FilterResponseEvent $event)
{
    // Don't do anything if it's not the master request.
    if (!$event->isMasterRequest()) {
        return;
    }

    $request = $event->getRequest();
    if ($request->getRealMethod() == Request::METHOD_OPTIONS) {
        $response = new Response();
        $response->headers->set('Access-Control-Allow-Origin', '*');
        $response->headers->set('Access-Control-Allow-Headers', 'Origin, X-Requested-With, X-Auth-Token, X-App-Version, Content-Type, Accept, Authorization');
        $response->headers->set('Access-Control-Allow-Methods', 'POST, GET, PUT');
        $event->setResponse($response);
    } else {
        $response = $event->getResponse();

        $response->headers->set('Access-Control-Allow-Origin', '*');
        $response->headers->set('Access-Control-Allow-Credentials', true);
        $response->headers->set('Access-Control-Allow-Headers', 'Origin, X-Requested-With, X-Auth-Token, X-App-Version, Content-Type, Accept, Authorization');
        $response->headers->set('Access-Control-Allow-Methods', 'POST, GET, PUT');
    }
}

这只是在浏览器中开发 Ionic 应用程序并处理来自应用程序的 OPTIONS 调用而不会出现错误的解决方案。但这只是一个例子,我对其他使用 ResponseListener 的 Symfony 应用程序也有同样的问题。

如果我在 prod 环境或 dev 环境中运行上面的示例,如果没有发生错误,则可以完美运行。但是一旦我收到 PHP 错误,就会导致以下结果:

Internal Server Error

The server encountered an internal error or misconfiguration and was unable to complete your request.

Please contact the server administrator, you@example.com and inform them of the time the error occurred, and anything you might have done that may have caused the error.

More information about this error may be available in the server error log.

Additionally, a 500 Internal Server Error error was encountered while trying to use an ErrorDocument to handle the request.

如果我查看 apache_error.log,它会说:

FastCGI: comm with server "/Applications/MAMP/fcgi-bin/php7.0.13.fcgi" aborted: error parsing headers: duplicate header 'Content-Type', referer: http://app.domain/app_dev.php/my-route

这似乎只发生在 PHP 错误中。例如,如果我用

抛出一个 NotFoundHttpException
throw new NotFoundHttpException("Test exception");

我得到了通常的 Symfony 错误页面。

如果我做类似的事情

new ClassDoesNotExist();

它会导致内部服务器错误。

我知道这是由于 ResponseListener 造成的,因为只要我将 exit; 放在函数顶部,就会显示 Symfony 错误页面,如下所示:

public function onKernelResponse(FilterResponseEvent $event)
{
    exit;

    // Don't do anything if it's not the master request.
    if (!$event->isMasterRequest()) {
        return;
    }
    ...

ResponseListener 有问题吗?是否有某种配置可以在 apache 上完成而不会引发此错误?

编辑:

这是 MAMP PRO 虚拟主机配置:

<VirtualHost *:80>
ServerName project.localhost

DocumentRoot "/Users/MyUser/Projekte/MyProject/web"

<IfModule xsendfile_module>
    XSendFilePath "/Users/MyUser/Projekte/MyProject/web"
</IfModule>

<Directory "/Users/MyUser/Projekte/MyProject/web">
    Options Includes FollowSymLinks ExecCGI
    AllowOverride All
    <IfModule authz_host_module>
        Order allow,deny
        Allow from all
    </IfModule>

</Directory>

WSGIDaemonProcess project.localhost processes=2 threads=15
WSGIProcessGroup project.localhost
WSGIScriptAlias /project.localhostWsgiApp "/Users/MyUser/Projekte/MyProject/web/wsgiapp.py"
AddHandler php-fastcgi .php
Action php-fastcgi "/fcgi-bin/php7.0.13.fcgi"

编辑 2:

正如@mickadoo 所建议的,另一个监听器可能正在添加重复的 header Content-Type。如果我停止传播,我会收到我想要的 Symfony 错误消息。但我的听众也是第一个被调用的,所以这将防止任何其他听众被触发。执行 debug:event-dispatcher 结果如下:

"kernel.response" event
-----------------------

 ------- -------------------------------------------------------------------------------------------- ----------
Order   Callable                                                                                     Priority
 ------- -------------------------------------------------------------------------------------------- ----------
  #1      MyCompany\Bundle\AppBundle\EventListener\ResponseListener::onKernelResponse()                   0
  #2      Sonata\BlockBundle\Cache\HttpCacheHandler::onKernelResponse()                                0
  #3      Symfony\Component\HttpKernel\EventListener\ResponseListener::onKernelResponse()              0
  #4      Symfony\Bundle\FrameworkBundle\DataCollector\RequestDataCollector::onKernelResponse()        0
  #5      Symfony\Component\Security\Http\RememberMe\ResponseListener::onKernelResponse()              0
  #6      Sensio\Bundle\FrameworkExtraBundle\EventListener\HttpCacheListener::onKernelResponse()       0
  #7      Symfony\Component\HttpKernel\EventListener\ProfilerListener::onKernelResponse()              -100
  #8      Symfony\Bundle\WebProfilerBundle\EventListener\WebDebugToolbarListener::onKernelResponse()   -128
  #9      Symfony\Component\HttpKernel\EventListener\SaveSessionListener::onKernelResponse()           -1000
  #10     Symfony\Component\HttpKernel\EventListener\StreamedResponseListener::onKernelResponse()      -1024
 ------- -------------------------------------------------------------------------------------------- --------

当我添加另一个优先级为 -2048 的响应监听器以使其成为最后一个触发时,并在那里使用以下方法:

/**
 * @param FilterResponseEvent $event Filter response event
 */
public function onKernelResponse(FilterResponseEvent $event)
{
    // Don't do anything if it's not the master request.
    if (!$event->isMasterRequest()) {
        return;
    }

    $response = $event->getResponse();

    echo "<pre>";
    print_r($response->headers->all());
    exit;
}

它返回:

Array
(
    [cache-control] => Array
    (
        [0] => no-cache, private
    )
    [access-control-allow-origin] => Array
    (
        [0] => *
    )
    [access-control-allow-credentials] => Array
    (
        [0] => 1
    )
    [access-control-allow-headers] => Array
    (
        [0] => Origin, X-Requested-With, X-Auth-Token, X-App-Version, Content-Type, Accept, Authorization
    )
    [access-control-allow-methods] => Array
    (
        [0] => POST, GET, PUT
    )
    [content-type] => Array
    (
        [0] => text/html; charset=UTF-8
    )
    [x-debug-token] => Array
    (
        [0] => 2f5dbb
    )
    [x-debug-token-link] => Array
    (
        [0] => http://app.project.localhost/app_dev.php/_profiler/2f5dbb
    )
)

但是还是没有额外的Content-Type。有什么想法吗?

编辑 3:

另一个奇怪的事情是:

如果我在我的方法末尾使用 $event->stopPropagation() 并添加优先级为 0 的监听器,一切正常(意味着显示 Symfony 错误页面),如果我设置它优先级 -2048 我收到 Apache 错误消息。但即使我在它的末尾输出所有标题,也没有重复的标题。我真的不明白这是怎么可能的。如何仅通过一个请求将多个相同的 header 返回给 apache,并且最后的所有 header 在输出中都是唯一的?

最佳答案

如何设置对事件的响应?

我的意思是,在 $event->getResponse() 给出的响应中,您添加了 header ,但您没有在其他部分添加 $event->setResponse($response)。

关于php - Symfony 中的 ResponseListener 导致在开发模式下来自 apache 的重复 header 'Content-Type',我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/42032435/

相关文章:

php - 如何从Curl获取指定数据?

php - 在 PHP 中获取下一个/上一个 ISO 周和年份

apache - 如何使.htaccess RewriteRule同时支持path和query_string

apache - 如何在子域中的反向代理后面运行PrimeFaces?

symfony - capifony 多阶段部署和每个阶段的数据库配置

PHP DOMDocument appendChild 到根目录末尾

php - Ajax 文件浏览器自动登录 Microsoft Office 身份验证

php - 无法启动 Apache(c :/php/php5apache2_4. dll 进入服务器:找不到指定的模块)

php - 在自定义服务 symfony2 中插入条令

php - 交响乐 : can't we have a hidden entity field?