123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364 |
- <?php
- /*
- * This file is part of the Symfony package.
- *
- * (c) Fabien Potencier <fabien@symfony.com>
- *
- * For the full copyright and license information, please view the LICENSE
- * file that was distributed with this source code.
- */
- namespace Symfony\Component\Routing\Tests;
- use PHPUnit\Framework\TestCase;
- use Symfony\Component\Config\FileLocator;
- use Symfony\Component\Config\Resource\FileResource;
- use Symfony\Component\Routing\Loader\YamlFileLoader;
- use Symfony\Component\Routing\Route;
- use Symfony\Component\Routing\RouteCollection;
- use Symfony\Component\Routing\RouteCollectionBuilder;
- class RouteCollectionBuilderTest extends TestCase
- {
- public function testImport()
- {
- $resolvedLoader = $this->getMockBuilder('Symfony\Component\Config\Loader\LoaderInterface')->getMock();
- $resolver = $this->getMockBuilder('Symfony\Component\Config\Loader\LoaderResolverInterface')->getMock();
- $resolver->expects($this->once())
- ->method('resolve')
- ->with('admin_routing.yml', 'yaml')
- ->will($this->returnValue($resolvedLoader));
- $originalRoute = new Route('/foo/path');
- $expectedCollection = new RouteCollection();
- $expectedCollection->add('one_test_route', $originalRoute);
- $expectedCollection->addResource(new FileResource(__DIR__.'/Fixtures/file_resource.yml'));
- $resolvedLoader
- ->expects($this->once())
- ->method('load')
- ->with('admin_routing.yml', 'yaml')
- ->will($this->returnValue($expectedCollection));
- $loader = $this->getMockBuilder('Symfony\Component\Config\Loader\LoaderInterface')->getMock();
- $loader->expects($this->any())
- ->method('getResolver')
- ->will($this->returnValue($resolver));
- // import the file!
- $routes = new RouteCollectionBuilder($loader);
- $importedRoutes = $routes->import('admin_routing.yml', '/', 'yaml');
- // we should get back a RouteCollectionBuilder
- $this->assertInstanceOf('Symfony\Component\Routing\RouteCollectionBuilder', $importedRoutes);
- // get the collection back so we can look at it
- $addedCollection = $importedRoutes->build();
- $route = $addedCollection->get('one_test_route');
- $this->assertSame($originalRoute, $route);
- // should return file_resource.yml, which is in the original collection
- $this->assertCount(1, $addedCollection->getResources());
- // make sure the routes were imported into the top-level builder
- $routeCollection = $routes->build();
- $this->assertCount(1, $routes->build());
- $this->assertCount(1, $routeCollection->getResources());
- }
- public function testImportAddResources()
- {
- $routeCollectionBuilder = new RouteCollectionBuilder(new YamlFileLoader(new FileLocator([__DIR__.'/Fixtures/'])));
- $routeCollectionBuilder->import('file_resource.yml');
- $routeCollection = $routeCollectionBuilder->build();
- $this->assertCount(1, $routeCollection->getResources());
- }
- /**
- * @expectedException \BadMethodCallException
- */
- public function testImportWithoutLoaderThrowsException()
- {
- $collectionBuilder = new RouteCollectionBuilder();
- $collectionBuilder->import('routing.yml');
- }
- public function testAdd()
- {
- $collectionBuilder = new RouteCollectionBuilder();
- $addedRoute = $collectionBuilder->add('/checkout', 'AppBundle:Order:checkout');
- $addedRoute2 = $collectionBuilder->add('/blogs', 'AppBundle:Blog:list', 'blog_list');
- $this->assertInstanceOf('Symfony\Component\Routing\Route', $addedRoute);
- $this->assertEquals('AppBundle:Order:checkout', $addedRoute->getDefault('_controller'));
- $finalCollection = $collectionBuilder->build();
- $this->assertSame($addedRoute2, $finalCollection->get('blog_list'));
- }
- public function testFlushOrdering()
- {
- $importedCollection = new RouteCollection();
- $importedCollection->add('imported_route1', new Route('/imported/foo1'));
- $importedCollection->add('imported_route2', new Route('/imported/foo2'));
- $loader = $this->getMockBuilder('Symfony\Component\Config\Loader\LoaderInterface')->getMock();
- // make this loader able to do the import - keeps mocking simple
- $loader->expects($this->any())
- ->method('supports')
- ->will($this->returnValue(true));
- $loader
- ->expects($this->once())
- ->method('load')
- ->will($this->returnValue($importedCollection));
- $routes = new RouteCollectionBuilder($loader);
- // 1) Add a route
- $routes->add('/checkout', 'AppBundle:Order:checkout', 'checkout_route');
- // 2) Import from a file
- $routes->mount('/', $routes->import('admin_routing.yml'));
- // 3) Add another route
- $routes->add('/', 'AppBundle:Default:homepage', 'homepage');
- // 4) Add another route
- $routes->add('/admin', 'AppBundle:Admin:dashboard', 'admin_dashboard');
- // set a default value
- $routes->setDefault('_locale', 'fr');
- $actualCollection = $routes->build();
- $this->assertCount(5, $actualCollection);
- $actualRouteNames = array_keys($actualCollection->all());
- $this->assertEquals([
- 'checkout_route',
- 'imported_route1',
- 'imported_route2',
- 'homepage',
- 'admin_dashboard',
- ], $actualRouteNames);
- // make sure the defaults were set
- $checkoutRoute = $actualCollection->get('checkout_route');
- $defaults = $checkoutRoute->getDefaults();
- $this->assertArrayHasKey('_locale', $defaults);
- $this->assertEquals('fr', $defaults['_locale']);
- }
- public function testFlushSetsRouteNames()
- {
- $collectionBuilder = new RouteCollectionBuilder();
- // add a "named" route
- $collectionBuilder->add('/admin', 'AppBundle:Admin:dashboard', 'admin_dashboard');
- // add an unnamed route
- $collectionBuilder->add('/blogs', 'AppBundle:Blog:list')
- ->setMethods(['GET']);
- // integer route names are allowed - they don't confuse things
- $collectionBuilder->add('/products', 'AppBundle:Product:list', 100);
- $actualCollection = $collectionBuilder->build();
- $actualRouteNames = array_keys($actualCollection->all());
- $this->assertEquals([
- 'admin_dashboard',
- 'GET_blogs',
- '100',
- ], $actualRouteNames);
- }
- public function testFlushSetsDetailsOnChildrenRoutes()
- {
- $routes = new RouteCollectionBuilder();
- $routes->add('/blogs/{page}', 'listAction', 'blog_list')
- // unique things for the route
- ->setDefault('page', 1)
- ->setRequirement('id', '\d+')
- ->setOption('expose', true)
- // things that the collection will try to override (but won't)
- ->setDefault('_format', 'html')
- ->setRequirement('_format', 'json|xml')
- ->setOption('fooBar', true)
- ->setHost('example.com')
- ->setCondition('request.isSecure()')
- ->setSchemes(['https'])
- ->setMethods(['POST']);
- // a simple route, nothing added to it
- $routes->add('/blogs/{id}', 'editAction', 'blog_edit');
- // configure the collection itself
- $routes
- // things that will not override the child route
- ->setDefault('_format', 'json')
- ->setRequirement('_format', 'xml')
- ->setOption('fooBar', false)
- ->setHost('symfony.com')
- ->setCondition('request.query.get("page")==1')
- // some unique things that should be set on the child
- ->setDefault('_locale', 'fr')
- ->setRequirement('_locale', 'fr|en')
- ->setOption('niceRoute', true)
- ->setSchemes(['http'])
- ->setMethods(['GET', 'POST']);
- $collection = $routes->build();
- $actualListRoute = $collection->get('blog_list');
- $this->assertEquals(1, $actualListRoute->getDefault('page'));
- $this->assertEquals('\d+', $actualListRoute->getRequirement('id'));
- $this->assertTrue($actualListRoute->getOption('expose'));
- // none of these should be overridden
- $this->assertEquals('html', $actualListRoute->getDefault('_format'));
- $this->assertEquals('json|xml', $actualListRoute->getRequirement('_format'));
- $this->assertTrue($actualListRoute->getOption('fooBar'));
- $this->assertEquals('example.com', $actualListRoute->getHost());
- $this->assertEquals('request.isSecure()', $actualListRoute->getCondition());
- $this->assertEquals(['https'], $actualListRoute->getSchemes());
- $this->assertEquals(['POST'], $actualListRoute->getMethods());
- // inherited from the main collection
- $this->assertEquals('fr', $actualListRoute->getDefault('_locale'));
- $this->assertEquals('fr|en', $actualListRoute->getRequirement('_locale'));
- $this->assertTrue($actualListRoute->getOption('niceRoute'));
- $actualEditRoute = $collection->get('blog_edit');
- // inherited from the collection
- $this->assertEquals('symfony.com', $actualEditRoute->getHost());
- $this->assertEquals('request.query.get("page")==1', $actualEditRoute->getCondition());
- $this->assertEquals(['http'], $actualEditRoute->getSchemes());
- $this->assertEquals(['GET', 'POST'], $actualEditRoute->getMethods());
- }
- /**
- * @dataProvider providePrefixTests
- */
- public function testFlushPrefixesPaths($collectionPrefix, $routePath, $expectedPath)
- {
- $routes = new RouteCollectionBuilder();
- $routes->add($routePath, 'someController', 'test_route');
- $outerRoutes = new RouteCollectionBuilder();
- $outerRoutes->mount($collectionPrefix, $routes);
- $collection = $outerRoutes->build();
- $this->assertEquals($expectedPath, $collection->get('test_route')->getPath());
- }
- public function providePrefixTests()
- {
- $tests = [];
- // empty prefix is of course ok
- $tests[] = ['', '/foo', '/foo'];
- // normal prefix - does not matter if it's a wildcard
- $tests[] = ['/{admin}', '/foo', '/{admin}/foo'];
- // shows that a prefix will always be given the starting slash
- $tests[] = ['0', '/foo', '/0/foo'];
- // spaces are ok, and double slahses at the end are cleaned
- $tests[] = ['/ /', '/foo', '/ /foo'];
- return $tests;
- }
- public function testFlushSetsPrefixedWithMultipleLevels()
- {
- $loader = $this->getMockBuilder('Symfony\Component\Config\Loader\LoaderInterface')->getMock();
- $routes = new RouteCollectionBuilder($loader);
- $routes->add('homepage', 'MainController::homepageAction', 'homepage');
- $adminRoutes = $routes->createBuilder();
- $adminRoutes->add('/dashboard', 'AdminController::dashboardAction', 'admin_dashboard');
- // embedded collection under /admin
- $adminBlogRoutes = $routes->createBuilder();
- $adminBlogRoutes->add('/new', 'BlogController::newAction', 'admin_blog_new');
- // mount into admin, but before the parent collection has been mounted
- $adminRoutes->mount('/blog', $adminBlogRoutes);
- // now mount the /admin routes, above should all still be /blog/admin
- $routes->mount('/admin', $adminRoutes);
- // add a route after mounting
- $adminRoutes->add('/users', 'AdminController::userAction', 'admin_users');
- // add another sub-collection after the mount
- $otherAdminRoutes = $routes->createBuilder();
- $otherAdminRoutes->add('/sales', 'StatsController::indexAction', 'admin_stats_sales');
- $adminRoutes->mount('/stats', $otherAdminRoutes);
- // add a normal collection and see that it is also prefixed
- $importedCollection = new RouteCollection();
- $importedCollection->add('imported_route', new Route('/foo'));
- // make this loader able to do the import - keeps mocking simple
- $loader->expects($this->any())
- ->method('supports')
- ->will($this->returnValue(true));
- $loader
- ->expects($this->any())
- ->method('load')
- ->will($this->returnValue($importedCollection));
- // import this from the /admin route builder
- $adminRoutes->import('admin.yml', '/imported');
- $collection = $routes->build();
- $this->assertEquals('/admin/dashboard', $collection->get('admin_dashboard')->getPath(), 'Routes before mounting have the prefix');
- $this->assertEquals('/admin/users', $collection->get('admin_users')->getPath(), 'Routes after mounting have the prefix');
- $this->assertEquals('/admin/blog/new', $collection->get('admin_blog_new')->getPath(), 'Sub-collections receive prefix even if mounted before parent prefix');
- $this->assertEquals('/admin/stats/sales', $collection->get('admin_stats_sales')->getPath(), 'Sub-collections receive prefix if mounted after parent prefix');
- $this->assertEquals('/admin/imported/foo', $collection->get('imported_route')->getPath(), 'Normal RouteCollections are also prefixed properly');
- }
- public function testAutomaticRouteNamesDoNotConflict()
- {
- $routes = new RouteCollectionBuilder();
- $adminRoutes = $routes->createBuilder();
- // route 1
- $adminRoutes->add('/dashboard', '');
- $accountRoutes = $routes->createBuilder();
- // route 2
- $accountRoutes->add('/dashboard', '')
- ->setMethods(['GET']);
- // route 3
- $accountRoutes->add('/dashboard', '')
- ->setMethods(['POST']);
- $routes->mount('/admin', $adminRoutes);
- $routes->mount('/account', $accountRoutes);
- $collection = $routes->build();
- // there are 2 routes (i.e. with non-conflicting names)
- $this->assertCount(3, $collection->all());
- }
- public function testAddsThePrefixOnlyOnceWhenLoadingMultipleCollections()
- {
- $firstCollection = new RouteCollection();
- $firstCollection->add('a', new Route('/a'));
- $secondCollection = new RouteCollection();
- $secondCollection->add('b', new Route('/b'));
- $loader = $this->getMockBuilder('Symfony\Component\Config\Loader\LoaderInterface')->getMock();
- $loader->expects($this->any())
- ->method('supports')
- ->will($this->returnValue(true));
- $loader
- ->expects($this->any())
- ->method('load')
- ->will($this->returnValue([$firstCollection, $secondCollection]));
- $routeCollectionBuilder = new RouteCollectionBuilder($loader);
- $routeCollectionBuilder->import('/directory/recurse/*', '/other/', 'glob');
- $routes = $routeCollectionBuilder->build()->all();
- $this->assertCount(2, $routes);
- $this->assertEquals('/other/a', $routes['a']->getPath());
- $this->assertEquals('/other/b', $routes['b']->getPath());
- }
- }
|