2017-02-13 92 views
1

我有一些控制器定義爲服務,我需要從路由名稱中獲得我的控制器的類名。Symfony2:從服務名稱或路由對象獲取控制器作爲服務類名稱

對於非服務控制器我可以得到路由集合與路由器服務:

$route = $this->router->getRouteCollection()->get($routeName); 
//Retrieve an object like that: 

Route { 
    -path: "/admin/dashboard" 
    -host: "" 
    -schemes: [] 
    -methods: [] 
    -defaults: array:1 [ 
    "_controller" => "AppBundle\Controller\Admin\AdminController::dashboardAction" 
    ] 
    -requirements: [] 
    -options: array:1 [] 
    -compiled: null 
    -condition: "" 
} 

我可以$route["defaults"]["_controller"]訪問控制器類名,所以這是很好。

的問題是我的控制器作爲服務的_controller屬性是服務,而不是控制器類的名稱(如app.controller.admin.user:listAction)我有服務的名字,但我需要有類名(AppBundle\Controller\Admin\UserController

我想出的唯一解決方案是從Container獲得服務,並在服務上使用get_class(),但它只會對檢索控制器/服務的類有巨大的性能影響。

有沒有其他解決方案?

+0

我相信沒有任何其他的選擇可以更高效。你需要怎麼處理類名? – Gerry

+0

我想重現本教程:https://www.trisoft.ro/blog/6-symfony2-advanced-menus我需要className才能讀取元數據:$ this-> metadataReader-> loadMetadataForClass(new \ ReflectionClass($類)); – iBadGamer

+0

我想你可以添加一個_controller_classname參數給你的路線。但是需要控制器類名稱來生成菜單似乎不是理想的設計。 – Cerad

回答

0

按照https://github.com/FriendsOfSymfony/FOSUserBundle/issues/2751的建議,我實現了一個緩存映射,以便將路由名解析爲控制器類和方法。

<?php 
// src/Cache/RouteClassMapWarmer.php 
namespace App\Cache; 

use Symfony\Component\Cache\Simple\PhpFilesCache; 
use Symfony\Component\DependencyInjection\ContainerInterface; 
use Symfony\Component\HttpKernel\CacheWarmer\CacheWarmerInterface; 
use Symfony\Component\Routing\RouterInterface; 

class RouteClassMapWarmer implements CacheWarmerInterface 
{ 
    /** @var ContainerInterface */ 
    protected $container; 
    /** @var RouterInterface */ 
    protected $router; 

    public function __construct(ContainerInterface $container, RouterInterface $router) 
    { 
     $this->container = $container; 
     $this->router = $router; 
    } 

    public function warmUp($cacheDirectory) 
    { 
     $cache = new PhpFilesCache('route_class_map', 0, $cacheDirectory); 
     $controllers = []; 
     foreach ($this->router->getRouteCollection() as $routeName => $route) { 
      $controller = $route->getDefault('_controller'); 
      if (false === strpos($controller, '::')) { 
       list($controllerClass, $controllerMethod) = explode(':', $controller, 2); 
       // service_id gets resolved here 
       $controllerClass = get_class($this->container->get($controllerClass)); 
      } 
      else { 
       list($controllerClass, $controllerMethod) = explode('::', $controller, 2); 
      } 
      $controllers[$routeName] = ['class' => $controllerClass, 'method' => $controllerMethod]; 
     } 
     unset($controller); 
     unset($route); 
     $cache->set('route_class_map', $controllers); 
    } 

    public function isOptional() 
    { 
     return false; 
    } 
} 

而且在我RouteHelper,閱讀本實施看起來是這樣的

$cache = new PhpFilesCache('route_class_map', 0, $this->cacheDirectory); 
    $controllers = $cache->get('route_class_map'); 
    if (!isset($controllers[$routeName])) { 
     throw new CacheException('No entry for route ' . $routeName . ' forund in RouteClassMap cache, please warmup first.'); 
    } 

    if (null !== $securityAnnotation = $this->annotationReader->getMethodAnnotation((new \ReflectionClass($controllers[$routeName]['class']))->getMethod($controllers[$routeName]['method']), Security::class)) 
    { 
     return $this->securityExpressionHelper->evaluate($securityAnnotation->getExpression(), ['myParameter' => $myParameter]); 
    } 

這應該是比獲得routeCollection快得多和解決的service_id:方法譜寫_controller的屬性對每個容器請求。