无法将kernel.view EventListener绑定到FOSUserBundle

时间:2015-05-31 10:18:24

标签: symfony fosuserbundle event-listener hwioauthbundle

我在kernel.view事件监听器的帮助下分离了移动和网络请求。

逻辑的工作原理如下:

  • 如果请求来自移动设备,请加载xxx.mobile.twig
  • 如果请求来自网络,请加载xxx.html.twig

这对我的CustomBundle没有任何问题。除此之外,我正在使用FOSUserBundle和HWIOAuthBundle以及它们的一些路线。我检查了var/logs/dev.log,我看不到有关这些捆绑路由的kernel.view个事件,最终我的监听器无法使用这些捆绑包。

您能否告诉我如何为这些捆绑包绑定kernel.view事件?

/**
 * @param GetResponseForControllerResultEvent $event
 * @return bool
 */
public function onKernelView(GetResponseForControllerResultEvent $event)
{
    if (!$this->isMobileRequest($event->getRequest()->headers->get('user-agent'))) {
        return false;
    }

    $template = $event->getRequest()->attributes->get('_template');
    if (!$template) {
        return false;
    }

    $templateReference = $this->templateNameParser->parse($template);

    if ($templateReference->get('format') == 'html' && $templateReference->get('bundle') == 'CustomBundle') {

        $mobileTemplate = sprintf(
            '%s:%s:%s.mobile.twig',
            $templateReference->get('bundle'),
            $templateReference->get('controller'),
            $templateReference->get('name')
        );

        if ($this->templating->exists($mobileTemplate)) {
            $templateReference->set('format', 'mobile');
            $event->getRequest()->attributes->set('_template', $templateReference);
        }
    }
}

2 个答案:

答案 0 :(得分:2)

在Symfony2上调试与事件相关的问题时,您应该考虑以下几点。

  1. 可以停止事件传播

    • 可能是另一个侦听器正在侦听同一个事件,并通过调用$event->stopPropagation()来停止事件传播。在这种情况下,请确保首先执行您的侦听器(请参阅下面的第2点)。
  2. 事件监听器具有优先级

    • 定义侦听器时,您可以设置其优先级,如下所示:

    view_response_listener:
        class: AppBundle\EventListener\ViewResponseListener
        tags:
            # The highest the priority, the earlier a listener is executed
            # @see http://symfony.com/doc/2.7/cookbook/event_dispatcher/event_listener.html#creating-an-event-listener
            - { name: kernel.event_listener, event: kernel.view, method: onKernelView, priority: 101 }
    
      

    另一个可选标记属性称为优先级,默认为0,它控制执行侦听器的顺序(优先级最高,执行侦听器的时间越早)。当您需要保证一个侦听器在另一个侦听器之前执行时,这非常有用。内部Symfony侦听器的优先级通常在-255到255之间,但您自己的侦听器可以使用任何正整数或负整数。

    来源http://symfony.com/doc/2.7/cookbook/event_dispatcher/event_listener.html#creating-an-event-listener

  3. 通常,这些事件的分派是在引导程序文件

    中完成的
    • 确保重新生成您的引导程序文件(并清除缓存,因为您正在使用它!),尤其是在您正在使用优先级和/或配置时。

    composer run-script post-update-cmd
    php app/console cache:clear --env=dev
    

    作曲家post-update-cmd将重新生成你的引导程序文件,但它也会做其他事情,比如重新安装资产,这可能是你不需要的东西。要重新生成引导程序文件,请检查我的答案here

答案 1 :(得分:0)

我找到了解决方案,但它有点解决方法,现在正常工作。

我将以下功能放到我的MobileTemplateListener.php文件中。 更多细节在这里 - > http://www.99bugs.com/handling-mobile-template-switching-in-symfony2/

    /**
 * @param GetResponseEvent $event
 */
public function onKernelRequest(GetResponseEvent $event)
{
    $request = $event->getRequest();

    if ($this->isMobileRequest($request->headers->get('user-agent')))
    {
        //ONLY AFFECT HTML REQUESTS
        //THIS ENSURES THAT YOUR JSON REQUESTS TO E.G. REST API, DO NOT GET SERVED TEXT/HTML CONTENT-TYPE
        if ($request->getRequestFormat() == "html")
        {
            $request->setRequestFormat('mobile');
        }
    }
}
/**
 * Returns true if request is from mobile device, otherwise false
 * @return boolean mobileUA
 */
private function isMobileRequest($userAgent)
{
    if (preg_match('/(android|blackberry|iphone|ipad|phone|playbook|mobile)/i', $userAgent)) {
        return true;
    }

    return false;
}

kernel.request事件侦听器开始处理时,它正在设置值为mobile的格式

我通过子捆绑使用FOSUserBundle来操作我的需求。您可以在此处找到更多详细信息 - > https://github.com/FriendsOfSymfony/FOSUserBundle/blob/master/Resources/doc/overriding_controllers.md

例如

:我们假设SecurityController.php 我在UserBundle下创建了一个名为SecurityController.php的文件。它看起来像是在跟随。

<?php

namespace Acme\UserBundle\Controller;

use Symfony\Component\HttpFoundation\RedirectResponse;
use FOS\UserBundle\Controller\SecurityController as BaseController;
use Symfony\Component\HttpFoundation\Request;
use Acme\UserBundle\Overrides\ControllerOverrideRenderTrait;

class SecurityController extends BaseController
{
    use ControllerOverrideRenderTrait;

    public function loginAction(Request $request)
    {
        $securityContext = $this->container->get('security.context');
        $router = $this->container->get('router');

        if ($securityContext->isGranted('IS_AUTHENTICATED_FULLY')) {
            return new RedirectResponse($router->generate('my_profile_dashboard'), 307);
        }
        return parent::loginAction($request);
    }
}

对于所有其他FOS控制器,我必须覆盖渲染功能。这是由smyfony Symfony\Bundle\FrameworkBundle\Controller

提供的

但是我的子包已经扩展了FOSUserBundle的控制器,在没有重复代码的情况下覆盖它的唯一方法就是使用特征。

我创建了一个特征如下。

    <?php
namespace Acme\UserBundle\Overrides;

use Symfony\Component\HttpFoundation\Response;

trait ControllerOverrideRenderTrait {
    /**
     * This overrides vendor/symfony/symfony/src/Symfony/Bundle/FrameworkBundle/Controller/Controller.php
     * Renders a view.
     *
     * @param string   $view       The view name
     * @param array    $parameters An array of parameters to pass to the view
     * @param Response $response   A response instance
     *
     * @return Response A Response instance
     */
    public function render($view, array $parameters = array(), Response $response = null)
    {
        $format = $this->getRequest()->getRequestFormat();
        $view = str_replace('.html.', '.' . $format . '.', $view);
        return $this->container->get('templating')->renderResponse($view, $parameters, $response);
    }
}

symfony提供的原始功能如下。

    /**
 * Renders a view.
 *
 * @param string   $view       The view name
 * @param array    $parameters An array of parameters to pass to the view
 * @param Response $response   A response instance
 *
 * @return Response A Response instance
 */
public function render($view, array $parameters = array(), Response $response = null)
{
    return $this->container->get('templating')->renderResponse($view, $parameters, $response);
}

基本上,我的更改通过提供'.html.'通过$format EventListener设置来替换模板名称中的onKernelRequest部分。

不要忘记在services.yml

中添加服务定义
services:
    acme.frontend.listener.mobile_template_listener:
        class: Acme\FrontendBundle\EventListener\MobileTemplateListener
        arguments: ['@templating', '@templating.name_parser']
        tags:
            - { name: kernel.event_listener, event: kernel.view, method: onKernelView }
            - { name: kernel.event_listener, event: kernel.request, method: onKernelRequest }