Symfony 2如何动态添加具有对请求对象的访问权限的路由

时间:2014-09-17 02:35:55

标签: php symfony event-listener symfony-2.5

背景

我正在尝试根据请求主机有条件地加载路由。我有一个数据库设置,其中包含映射到模板的主机。如果用户从主机A进入并使用模板TA,我想加载该模板的路由。如果他们从主机B进来,则加载该模板的路由(TB)。

我必须这样做的原因是因为每个模板将共享许多路由。但是,给定模板有一些独特的路线。

将每个模板路由限制到给定的主机是可以的,除了实际上有1000个主机。

我尝试过的事情:

我已经尝试了一个自定义路由加载器,如文档中所述:

http://symfony.com/doc/current/cookbook/routing/custom_route_loader.html

然而,当我配置服务并尝试注入"@request"时,构造函数失败,因为$ request为null

services:
    acme_demo.routing_loader:
        class: Acme\DemoBundle\Routing\ExtraLoader
        arguments: ["@request"]
        tags:
            - { name: routing.loader }

类别:

<?php
namespace: Acme\DemoBundle\Routing;
use Symfony\Component\HttpFoundation\Request;
class ExtraLoader
{
    protected $request;

    public function __construct(Request $request)
    {
        $this->request = $request;
    }
    // ...
}

如果我尝试为"@request"切换"@service_container"然后调用

,这也无效
$this->container->get('request');

我最接近的工作是按照此处的指南进行操作:

http://marcjschmidt.de/blog/2013/11/30/symfony-custom-dynamic-router.html

我遇到的问题是我试图在控制器上使用annotation而我似乎无法使Symfony\Component\Routing\Loader\AnnotationFileLoader正常工作。

1 个答案:

答案 0 :(得分:2)

好的,我终于想出了一个有效的解决方案,它是上面几个指南的混合。

我正在使用内核请求侦听器:

services:
    website_listener:
        class: NameSpace\Bundle\EventListener\WebsiteListener
        arguments:
            - "@website_service"
            - "@template_service"
            - "@sensio_framework_extra.routing.loader.annot_dir"
            - "@router"
            - "%admin_domain%"
        tags:
            - { name: kernel.event_listener, event: kernel.request, method: onKernelRequest, priority: 33 }

听众:

<?php
namespace NameSpace\WebsiteBundle\EventListener;

use NameSpace\TemplateBundle\Service\TemplateService;
use NameSpace\WebsiteBundle\Service\WebsiteService;
use Symfony\Component\HttpKernel\Event\GetResponseEvent;
use Symfony\Component\HttpKernel\Exception\NotFoundHttpException;
use Symfony\Component\Routing\Loader\AnnotationDirectoryLoader;
use Symfony\Component\HttpKernel\EventListener\RouterListener;
use Symfony\Component\Routing\Matcher\UrlMatcher;
use Symfony\Component\Routing\RequestContext;
use Symfony\Component\Routing\RouteCollection;
use Symfony\Bundle\FrameworkBundle\Routing\Router;
use Symfony\Component\PropertyAccess\PropertyAccess;

class WebsiteListener
{

    /**
     * @var WebsiteService
     */
    protected $websiteService;

    /**
     * @var TemplateService
     */
    protected $templateService;

    /**
     * @var AnnotationDirectoryLoader
     */
    protected $loader;

    /**
     * @var Router
     */
    protected $router;

    /**
     * @var string
     */
    protected $adminDomain;

    public function __construct(WebsiteService $websiteService, TemplateService $templateService, AnnotationDirectoryLoader $loader, Router $router, $adminDomain)
    {
        $this->websiteService = $websiteService;
        $this->templateService = $templateService;
        $this->loader = $loader;
        $this->router = $router;
        $this->adminDomain = $adminDomain;
    }

    public function loadRoutes()
    {
        $template = $this->templateService->getTemplateByAlias($this->websiteService->getWebsite()->getTemplate());

        $routes = $this->loader->load($template['routes'],'annotation');
        $allRoutes = $this->router->getRouteCollection();
        $allRoutes->addCollection($routes);
    }

    public function onKernelRequest(GetResponseEvent $event)
    {
        try {
            $this->websiteService->handleRequest($event->getRequest());
            $this->loadRoutes();
        } catch(NotFoundHttpException $e) {
            if($event->getRequest()->getHost() !== $this->adminDomain){
                throw $e;
            }
        }

    }
}

这方面的关键部分是:

  • Loader - 我在源代码中找到了"@sensio_framework_extra.routing.loader.annot_dir"。 symfony默认使用的注释目录加载器也是我想要使用的那个。但是如果你想使用不同的装载机,还有其他装载机。
  • Router - 这就是我用来获取所有当前路线的方法。 请注意 $allRoutes->addCollection($routes)来电是单独的。我不知道为什么它会产生影响但是把它全部称为1,就像没有工作一样。
  • $template['routes']只是一个名称空间控制器引用,就像您在routing.yml中添加路由一样。类似于:"@NamespaceBundle/Controller"