| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617 |
- <?php
- /**
- * CakePHP(tm) : Rapid Development Framework (http://cakephp.org)
- * Copyright (c) Cake Software Foundation, Inc. (http://cakefoundation.org)
- *
- * Licensed under The MIT License
- * For full copyright and license information, please see the LICENSE.txt
- * Redistributions of files must retain the above copyright notice.
- *
- * @copyright Copyright (c) Cake Software Foundation, Inc. (http://cakefoundation.org)
- * @link http://cakephp.org CakePHP(tm) Project
- * @since 3.0.0
- * @license http://www.opensource.org/licenses/mit-license.php MIT License
- */
- namespace Cake\Test\TestCase\Routing;
- use Cake\Routing\RouteBuilder;
- use Cake\Routing\RouteCollection;
- use Cake\Routing\Router;
- use Cake\Routing\Route\Route;
- use Cake\TestSuite\TestCase;
- /**
- * RouteBuilder test case
- */
- class RouteBuilderTest extends TestCase
- {
- /**
- * Setup method
- *
- * @return void
- */
- public function setUp()
- {
- parent::setUp();
- $this->collection = new RouteCollection();
- }
- /**
- * Test path()
- *
- * @return void
- */
- public function testPath()
- {
- $routes = new RouteBuilder($this->collection, '/some/path');
- $this->assertEquals('/some/path', $routes->path());
- $routes = new RouteBuilder($this->collection, '/:book_id');
- $this->assertEquals('/', $routes->path());
- $routes = new RouteBuilder($this->collection, '/path/:book_id');
- $this->assertEquals('/path/', $routes->path());
- $routes = new RouteBuilder($this->collection, '/path/book:book_id');
- $this->assertEquals('/path/book', $routes->path());
- }
- /**
- * Test params()
- *
- * @return void
- */
- public function testParams()
- {
- $routes = new RouteBuilder($this->collection, '/api', ['prefix' => 'api']);
- $this->assertEquals(['prefix' => 'api'], $routes->params());
- }
- /**
- * Test getting connected routes.
- *
- * @return void
- */
- public function testRoutes()
- {
- $routes = new RouteBuilder($this->collection, '/l');
- $routes->connect('/:controller', ['action' => 'index']);
- $routes->connect('/:controller/:action/*');
- $all = $this->collection->routes();
- $this->assertCount(2, $all);
- $this->assertInstanceOf('Cake\Routing\Route\Route', $all[0]);
- $this->assertInstanceOf('Cake\Routing\Route\Route', $all[1]);
- }
- /**
- * Test setting default route class
- *
- * @return void
- */
- public function testRouteClass()
- {
- $routes = new RouteBuilder(
- $this->collection,
- '/l',
- [],
- ['routeClass' => 'InflectedRoute']
- );
- $routes->connect('/:controller', ['action' => 'index']);
- $routes->connect('/:controller/:action/*');
- $all = $this->collection->routes();
- $this->assertInstanceOf('Cake\Routing\Route\InflectedRoute', $all[0]);
- $this->assertInstanceOf('Cake\Routing\Route\InflectedRoute', $all[1]);
- $this->collection = new RouteCollection();
- $routes = new RouteBuilder($this->collection, '/l');
- $routes->routeClass('TestApp\Routing\Route\DashedRoute');
- $routes->connect('/:controller', ['action' => 'index']);
- $all = $this->collection->routes();
- $this->assertInstanceOf('TestApp\Routing\Route\DashedRoute', $all[0]);
- }
- /**
- * Test connecting an instance routes.
- *
- * @return void
- */
- public function testConnectInstance()
- {
- $routes = new RouteBuilder($this->collection, '/l', ['prefix' => 'api']);
- $route = new Route('/:controller');
- $this->assertNull($routes->connect($route));
- $result = $this->collection->routes()[0];
- $this->assertSame($route, $result);
- }
- /**
- * Test connecting basic routes.
- *
- * @return void
- */
- public function testConnectBasic()
- {
- $routes = new RouteBuilder($this->collection, '/l', ['prefix' => 'api']);
- $this->assertNull($routes->connect('/:controller'));
- $route = $this->collection->routes()[0];
- $this->assertInstanceOf('Cake\Routing\Route\Route', $route);
- $this->assertEquals('/l/:controller', $route->template);
- $expected = ['prefix' => 'api', 'action' => 'index', 'plugin' => null];
- $this->assertEquals($expected, $route->defaults);
- }
- /**
- * Test that compiling a route results in an trailing / optional pattern.
- *
- * @return void
- */
- public function testConnectTrimTrailingSlash()
- {
- $routes = new RouteBuilder($this->collection, '/articles', ['controller' => 'Articles']);
- $routes->connect('/', ['action' => 'index']);
- $expected = ['plugin' => null, 'controller' => 'Articles', 'action' => 'index', 'pass' => []];
- $this->assertEquals($expected, $this->collection->parse('/articles'));
- $this->assertEquals($expected, $this->collection->parse('/articles/'));
- }
- /**
- * Test extensions being connected to routes.
- *
- * @return void
- */
- public function testConnectExtensions()
- {
- $routes = new RouteBuilder(
- $this->collection,
- '/l',
- [],
- ['extensions' => ['json']]
- );
- $this->assertEquals(['json'], $routes->extensions());
- $routes->connect('/:controller');
- $route = $this->collection->routes()[0];
- $this->assertEquals(['json'], $route->options['_ext']);
- $routes->extensions(['xml', 'json']);
- $routes->connect('/:controller/:action');
- $new = $this->collection->routes()[1];
- $this->assertEquals(['json'], $route->options['_ext']);
- $this->assertEquals(['xml', 'json'], $new->options['_ext']);
- }
- /**
- * test that extensions() accepts a string.
- *
- * @return void
- */
- public function testExtensionsString()
- {
- $routes = new RouteBuilder($this->collection, '/l');
- $routes->extensions('json');
- $this->assertEquals(['json'], $routes->extensions());
- }
- /**
- * Test error on invalid route class
- *
- * @expectedException \InvalidArgumentException
- * @expectedExceptionMessage Route class not found, or route class is not a subclass of
- * @return void
- */
- public function testConnectErrorInvalidRouteClass()
- {
- $routes = new RouteBuilder(
- $this->collection,
- '/l',
- [],
- ['extensions' => ['json']]
- );
- $routes->connect('/:controller', [], ['routeClass' => '\StdClass']);
- }
- /**
- * Test conflicting parameters raises an exception.
- *
- * @expectedException \BadMethodCallException
- * @expectedExceptionMessage You cannot define routes that conflict with the scope.
- * @return void
- */
- public function testConnectConflictingParameters()
- {
- $routes = new RouteBuilder($this->collection, '/admin', ['prefix' => 'admin']);
- $routes->connect('/', ['prefix' => 'manager', 'controller' => 'Dashboard', 'action' => 'view']);
- }
- /**
- * Test connecting redirect routes.
- *
- * @return void
- */
- public function testRedirect()
- {
- $routes = new RouteBuilder($this->collection, '/');
- $routes->redirect('/p/:id', ['controller' => 'posts', 'action' => 'view'], ['status' => 301]);
- $route = $this->collection->routes()[0];
- $this->assertInstanceOf('Cake\Routing\Route\RedirectRoute', $route);
- $routes->redirect('/old', '/forums', ['status' => 301]);
- $route = $this->collection->routes()[1];
- $this->assertInstanceOf('Cake\Routing\Route\RedirectRoute', $route);
- $this->assertEquals('/forums', $route->redirect[0]);
- }
- /**
- * Test creating sub-scopes with prefix()
- *
- * @return void
- */
- public function testPrefix()
- {
- $routes = new RouteBuilder($this->collection, '/path', ['key' => 'value']);
- $res = $routes->prefix('admin', function ($r) {
- $this->assertInstanceOf('Cake\Routing\RouteBuilder', $r);
- $this->assertCount(0, $this->collection->routes());
- $this->assertEquals('/path/admin', $r->path());
- $this->assertEquals(['prefix' => 'admin', 'key' => 'value'], $r->params());
- });
- $this->assertNull($res);
- }
- /**
- * Test creating sub-scopes with prefix()
- *
- * @return void
- */
- public function testNestedPrefix()
- {
- $routes = new RouteBuilder($this->collection, '/admin', ['prefix' => 'admin']);
- $res = $routes->prefix('api', function ($r) {
- $this->assertEquals('/admin/api', $r->path());
- $this->assertEquals(['prefix' => 'admin/api'], $r->params());
- });
- $this->assertNull($res);
- }
- /**
- * Test creating sub-scopes with plugin()
- *
- * @return void
- */
- public function testNestedPlugin()
- {
- $routes = new RouteBuilder($this->collection, '/b', ['key' => 'value']);
- $res = $routes->plugin('Contacts', function ($r) {
- $this->assertEquals('/b/contacts', $r->path());
- $this->assertEquals(['plugin' => 'Contacts', 'key' => 'value'], $r->params());
- $r->connect('/:controller');
- $route = $this->collection->routes()[0];
- $this->assertEquals(
- ['key' => 'value', 'plugin' => 'Contacts', 'action' => 'index'],
- $route->defaults
- );
- });
- $this->assertNull($res);
- }
- /**
- * Test creating sub-scopes with plugin() + path option
- *
- * @return void
- */
- public function testNestedPluginPathOption()
- {
- $routes = new RouteBuilder($this->collection, '/b', ['key' => 'value']);
- $routes->plugin('Contacts', ['path' => '/people'], function ($r) {
- $this->assertEquals('/b/people', $r->path());
- $this->assertEquals(['plugin' => 'Contacts', 'key' => 'value'], $r->params());
- });
- }
- /**
- * Test connecting resources.
- *
- * @return void
- */
- public function testResources()
- {
- $routes = new RouteBuilder($this->collection, '/api', ['prefix' => 'api']);
- $routes->resources('Articles', ['_ext' => 'json']);
- $all = $this->collection->routes();
- $this->assertCount(5, $all);
- $this->assertEquals('/api/articles', $all[0]->template);
- $this->assertEquals(
- ['controller', 'action', '_method', 'prefix', 'plugin'],
- array_keys($all[0]->defaults)
- );
- $this->assertEquals('json', $all[0]->options['_ext']);
- $this->assertEquals('Articles', $all[0]->defaults['controller']);
- }
- /**
- * Test connecting resources with the inflection option
- *
- * @return void
- */
- public function testResourcesInflection()
- {
- $routes = new RouteBuilder($this->collection, '/api', ['prefix' => 'api']);
- $routes->resources('BlogPosts', ['_ext' => 'json', 'inflect' => 'dasherize']);
- $all = $this->collection->routes();
- $this->assertCount(5, $all);
- $this->assertEquals('/api/blog-posts', $all[0]->template);
- $this->assertEquals(
- ['controller', 'action', '_method', 'prefix', 'plugin'],
- array_keys($all[0]->defaults)
- );
- $this->assertEquals('BlogPosts', $all[0]->defaults['controller']);
- }
- /**
- * Test connecting resources with additional mappings
- *
- * @return void
- */
- public function testResourcesMappings()
- {
- $routes = new RouteBuilder($this->collection, '/api', ['prefix' => 'api']);
- $routes->resources('Articles', [
- '_ext' => 'json',
- 'map' => [
- 'delete_all' => ['action' => 'deleteAll', 'method' => 'DELETE'],
- 'update_many' => ['action' => 'updateAll', 'method' => 'DELETE', 'path' => '/updateAll'],
- ]
- ]);
- $all = $this->collection->routes();
- $this->assertCount(7, $all);
- $this->assertEquals('/api/articles/delete_all', $all[5]->template, 'Path defaults to key name.');
- $this->assertEquals(
- ['controller', 'action', '_method', 'prefix', 'plugin'],
- array_keys($all[5]->defaults)
- );
- $this->assertEquals('Articles', $all[5]->defaults['controller']);
- $this->assertEquals('deleteAll', $all[5]->defaults['action']);
- $this->assertEquals('/api/articles/updateAll', $all[6]->template, 'Explicit path option');
- $this->assertEquals(
- ['controller', 'action', '_method', 'prefix', 'plugin'],
- array_keys($all[6]->defaults)
- );
- $this->assertEquals('Articles', $all[6]->defaults['controller']);
- $this->assertEquals('updateAll', $all[6]->defaults['action']);
- }
- /**
- * Test connecting resources.
- *
- * @return void
- */
- public function testResourcesInScope()
- {
- Router::scope('/api', ['prefix' => 'api'], function ($routes) {
- $routes->extensions(['json']);
- $routes->resources('Articles');
- });
- $url = Router::url([
- 'prefix' => 'api',
- 'controller' => 'Articles',
- 'action' => 'edit',
- '_method' => 'PUT',
- 'id' => 99
- ]);
- $this->assertEquals('/api/articles/99', $url);
- $url = Router::url([
- 'prefix' => 'api',
- 'controller' => 'Articles',
- 'action' => 'edit',
- '_method' => 'PUT',
- '_ext' => 'json',
- 'id' => 99
- ]);
- $this->assertEquals('/api/articles/99.json', $url);
- }
- /**
- * Test resource parsing.
- *
- * @return void
- */
- public function testResourcesParsing()
- {
- $routes = new RouteBuilder($this->collection, '/');
- $routes->resources('Articles');
- $_SERVER['REQUEST_METHOD'] = 'GET';
- $result = $this->collection->parse('/articles');
- $this->assertEquals('Articles', $result['controller']);
- $this->assertEquals('index', $result['action']);
- $this->assertEquals([], $result['pass']);
- $result = $this->collection->parse('/articles/1');
- $this->assertEquals('Articles', $result['controller']);
- $this->assertEquals('view', $result['action']);
- $this->assertEquals([1], $result['pass']);
- $_SERVER['REQUEST_METHOD'] = 'POST';
- $result = $this->collection->parse('/articles');
- $this->assertEquals('Articles', $result['controller']);
- $this->assertEquals('add', $result['action']);
- $this->assertEquals([], $result['pass']);
- $_SERVER['REQUEST_METHOD'] = 'PUT';
- $result = $this->collection->parse('/articles/1');
- $this->assertEquals('Articles', $result['controller']);
- $this->assertEquals('edit', $result['action']);
- $this->assertEquals([1], $result['pass']);
- $_SERVER['REQUEST_METHOD'] = 'DELETE';
- $result = $this->collection->parse('/articles/1');
- $this->assertEquals('Articles', $result['controller']);
- $this->assertEquals('delete', $result['action']);
- $this->assertEquals([1], $result['pass']);
- }
- /**
- * Test the only option of RouteBuilder.
- *
- * @return void
- */
- public function testResourcesOnlyString()
- {
- $routes = new RouteBuilder($this->collection, '/');
- $routes->resources('Articles', ['only' => 'index']);
- $result = $this->collection->routes();
- $this->assertCount(1, $result);
- $this->assertEquals('/articles', $result[0]->template);
- }
- /**
- * Test the only option of RouteBuilder.
- *
- * @return void
- */
- public function testResourcesOnlyArray()
- {
- $routes = new RouteBuilder($this->collection, '/');
- $routes->resources('Articles', ['only' => ['index', 'delete']]);
- $result = $this->collection->routes();
- $this->assertCount(2, $result);
- $this->assertEquals('/articles', $result[0]->template);
- $this->assertEquals('index', $result[0]->defaults['action']);
- $this->assertEquals('GET', $result[0]->defaults['_method']);
- $this->assertEquals('/articles/:id', $result[1]->template);
- $this->assertEquals('delete', $result[1]->defaults['action']);
- $this->assertEquals('DELETE', $result[1]->defaults['_method']);
- }
- /**
- * Test the actions option of RouteBuilder.
- *
- * @return void
- */
- public function testResourcesActions()
- {
- $routes = new RouteBuilder($this->collection, '/');
- $routes->resources('Articles', [
- 'only' => ['index', 'delete'],
- 'actions' => ['index' => 'showList']
- ]);
- $result = $this->collection->routes();
- $this->assertCount(2, $result);
- $this->assertEquals('/articles', $result[0]->template);
- $this->assertEquals('showList', $result[0]->defaults['action']);
- $this->assertEquals('/articles/:id', $result[1]->template);
- $this->assertEquals('delete', $result[1]->defaults['action']);
- }
- /**
- * Test nesting resources
- *
- * @return void
- */
- public function testResourcesNested()
- {
- $routes = new RouteBuilder($this->collection, '/api', ['prefix' => 'api']);
- $routes->resources('Articles', function ($routes) {
- $this->assertEquals('/api/articles/', $routes->path());
- $this->assertEquals(['prefix' => 'api'], $routes->params());
- $routes->resources('Comments');
- $route = $this->collection->routes()[6];
- $this->assertEquals('/api/articles/:article_id/comments', $route->template);
- });
- }
- /**
- * Test connecting fallback routes.
- *
- * @return void
- */
- public function testFallbacks()
- {
- $routes = new RouteBuilder($this->collection, '/api', ['prefix' => 'api']);
- $routes->fallbacks();
- $all = $this->collection->routes();
- $this->assertEquals('/api/:controller', $all[0]->template);
- $this->assertEquals('/api/:controller/:action/*', $all[1]->template);
- $this->assertInstanceOf('Cake\Routing\Route\Route', $all[0]);
- }
- /**
- * Test connecting fallback routes with specific route class
- *
- * @return void
- */
- public function testFallbacksWithClass()
- {
- $routes = new RouteBuilder($this->collection, '/api', ['prefix' => 'api']);
- $routes->fallbacks('InflectedRoute');
- $all = $this->collection->routes();
- $this->assertEquals('/api/:controller', $all[0]->template);
- $this->assertEquals('/api/:controller/:action/*', $all[1]->template);
- $this->assertInstanceOf('Cake\Routing\Route\InflectedRoute', $all[0]);
- }
- /**
- * Test connecting fallback routes after setting default route class.
- *
- * @return void
- */
- public function testDefaultRouteClassFallbacks()
- {
- $routes = new RouteBuilder($this->collection, '/api', ['prefix' => 'api']);
- $routes->routeClass('TestApp\Routing\Route\DashedRoute');
- $routes->fallbacks();
- $all = $this->collection->routes();
- $this->assertInstanceOf('TestApp\Routing\Route\DashedRoute', $all[0]);
- }
- /**
- * Test adding a scope.
- *
- * @return void
- */
- public function testScope()
- {
- $routes = new RouteBuilder($this->collection, '/api', ['prefix' => 'api']);
- $routes->scope('/v1', ['version' => 1], function ($routes) {
- $this->assertEquals('/api/v1', $routes->path());
- $this->assertEquals(['prefix' => 'api', 'version' => 1], $routes->params());
- });
- $routes = new RouteBuilder($this->collection, '/api', ['prefix' => 'api']);
- $routes->scope('/v1', function ($routes) {
- $this->assertEquals('/api/v1', $routes->path());
- $this->assertEquals(['prefix' => 'api'], $routes->params());
- });
- }
- }
|