mirror of
				https://github.com/dannyvankooten/AltoRouter.git
				synced 2025-10-25 21:26:21 +02:00 
			
		
		
		
	
		
			
				
	
	
		
			575 lines
		
	
	
		
			16 KiB
		
	
	
	
		
			PHP
		
	
	
	
	
	
			
		
		
	
	
			575 lines
		
	
	
		
			16 KiB
		
	
	
	
		
			PHP
		
	
	
	
	
	
| <?php
 | |
| 
 | |
| require 'AltoRouter.php';
 | |
| 
 | |
| class AltoRouterDebug extends AltoRouter
 | |
| {
 | |
|     public function getNamedRoutes()
 | |
|     {
 | |
|         return $this->namedRoutes;
 | |
|     }
 | |
| 
 | |
|     public function getBasePath()
 | |
|     {
 | |
|         return $this->basePath;
 | |
|     }
 | |
| }
 | |
| 
 | |
| class SimpleTraversable implements Iterator
 | |
| {
 | |
| 
 | |
|     protected $_position = 0;
 | |
| 
 | |
|     protected $_data = [
 | |
|         ['GET', '/foo', 'foo_action', null],
 | |
|         ['POST', '/bar', 'bar_action', 'second_route']
 | |
|     ];
 | |
| 
 | |
|     public function current()
 | |
|     {
 | |
|         return $this->_data[$this->_position];
 | |
|     }
 | |
|     public function key()
 | |
|     {
 | |
|         return $this->_position;
 | |
|     }
 | |
|     public function next()
 | |
|     {
 | |
|         ++$this->_position;
 | |
|     }
 | |
|     public function rewind()
 | |
|     {
 | |
|         $this->_position = 0;
 | |
|     }
 | |
|     public function valid()
 | |
|     {
 | |
|         return isset($this->_data[$this->_position]);
 | |
|     }
 | |
| }
 | |
| 
 | |
| class AltoRouterTest extends PHPUnit\Framework\TestCase
 | |
| {
 | |
|     /**
 | |
|      * @var AltoRouter
 | |
|      */
 | |
|     protected $router;
 | |
| 
 | |
|     /**
 | |
|      * Sets up the fixture, for example, opens a network connection.
 | |
|      * This method is called before a test is executed.
 | |
|      */
 | |
|     protected function setUp()
 | |
|     {
 | |
|         $this->router = new AltoRouterDebug;
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Tears down the fixture, for example, closes a network connection.
 | |
|      * This method is called after a test is executed.
 | |
|      */
 | |
|     protected function tearDown()
 | |
|     {
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * @covers AltoRouter::getRoutes
 | |
|      */
 | |
|     public function testGetRoutes()
 | |
|     {
 | |
|         $method = 'POST';
 | |
|         $route = '/[:controller]/[:action]';
 | |
|         $target = static function () {
 | |
|         };
 | |
| 
 | |
|         $this->assertInternalType('array', $this->router->getRoutes());
 | |
|         // $this->assertIsArray($this->router->getRoutes()); // for phpunit 7.x
 | |
|         $this->router->map($method, $route, $target);
 | |
|         $this->assertEquals([[$method, $route, $target, null]], $this->router->getRoutes());
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * @covers AltoRouter::addRoutes
 | |
|      */
 | |
|     public function testAddRoutes()
 | |
|     {
 | |
|         $method = 'POST';
 | |
|         $route = '/[:controller]/[:action]';
 | |
|         $target = static function () {
 | |
|         };
 | |
|         
 | |
|         $this->router->addRoutes([
 | |
|         [$method, $route, $target],
 | |
|         [$method, $route, $target, 'second_route']
 | |
|         ]);
 | |
|         
 | |
|         $routes = $this->router->getRoutes();
 | |
|         
 | |
|         $this->assertEquals([$method, $route, $target, null], $routes[0]);
 | |
|         $this->assertEquals([$method, $route, $target, 'second_route'], $routes[1]);
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * @covers AltoRouter::addRoutes
 | |
|      */
 | |
|     public function testAddRoutesAcceptsTraverable()
 | |
|     {
 | |
|         $traversable = new SimpleTraversable();
 | |
|         $this->router->addRoutes($traversable);
 | |
|         
 | |
|         $traversable->rewind();
 | |
|         
 | |
|         $first = $traversable->current();
 | |
|         $traversable->next();
 | |
|         $second = $traversable->current();
 | |
|         
 | |
|         $routes = $this->router->getRoutes();
 | |
|         
 | |
|         $this->assertEquals($first, $routes[0]);
 | |
|         $this->assertEquals($second, $routes[1]);
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * @covers AltoRouter::addRoutes
 | |
|      * @expectedException Exception
 | |
|      */
 | |
|     public function testAddRoutesThrowsExceptionOnInvalidArgument()
 | |
|     {
 | |
|         $this->router->addRoutes(new stdClass);
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * @covers AltoRouter::setBasePath
 | |
|      */
 | |
|     public function testSetBasePath()
 | |
|     {
 | |
|         $basePath = $this->router->setBasePath('/some/path');
 | |
|         $this->assertEquals('/some/path', $this->router->getBasePath());
 | |
|         
 | |
|         $basePath = $this->router->setBasePath('/some/path');
 | |
|         $this->assertEquals('/some/path', $this->router->getBasePath());
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * @covers AltoRouter::map
 | |
|      */
 | |
|     public function testMap()
 | |
|     {
 | |
|         $method = 'POST';
 | |
|         $route = '/[:controller]/[:action]';
 | |
|         $target = static function () {
 | |
|         };
 | |
|         
 | |
|         $this->router->map($method, $route, $target);
 | |
|         
 | |
|         $routes = $this->router->getRoutes();
 | |
|         
 | |
|         $this->assertEquals([$method, $route, $target, null], $routes[0]);
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * @covers AltoRouter::map
 | |
|      */
 | |
|     public function testMapWithName()
 | |
|     {
 | |
|         $method = 'POST';
 | |
|         $route = '/[:controller]/[:action]';
 | |
|         $target = static function () {
 | |
|         };
 | |
|         $name = 'myroute';
 | |
|         
 | |
|         $this->router->map($method, $route, $target, $name);
 | |
|         
 | |
|         $routes = $this->router->getRoutes();
 | |
|         $this->assertEquals([$method, $route, $target, $name], $routes[0]);
 | |
|         
 | |
|         $named_routes = $this->router->getNamedRoutes();
 | |
|         $this->assertEquals($route, $named_routes[$name]);
 | |
|         
 | |
|         try {
 | |
|             $this->router->map($method, $route, $target, $name);
 | |
|             $this->fail('Should not be able to add existing named route');
 | |
|         } catch (Exception $e) {
 | |
|             $this->assertEquals("Can not redeclare route '{$name}'", $e->getMessage());
 | |
|         }
 | |
|     }
 | |
| 
 | |
| 
 | |
|     /**
 | |
|      * @covers AltoRouter::generate
 | |
|      */
 | |
|     public function testGenerate()
 | |
|     {
 | |
|         $params =[
 | |
|             'controller' => 'test',
 | |
|             'action' => 'someaction'
 | |
|         ];
 | |
|         
 | |
|         $this->router->map('GET', '/[:controller]/[:action]', static function () {
 | |
|         }, 'foo_route');
 | |
|         
 | |
|         $this->assertEquals(
 | |
|             '/test/someaction',
 | |
|             $this->router->generate('foo_route', $params)
 | |
|         );
 | |
|         
 | |
|         $params = [
 | |
|             'controller' => 'test',
 | |
|             'action' => 'someaction',
 | |
|             'type' => 'json'
 | |
|         ];
 | |
|         
 | |
|         $this->assertEquals(
 | |
|             '/test/someaction',
 | |
|             $this->router->generate('foo_route', $params)
 | |
|         );
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * @covers AltoRouter::generate
 | |
|      */
 | |
|     public function testGenerateWithOptionalUrlParts()
 | |
|     {
 | |
|         $this->router->map('GET', '/[:controller]/[:action].[:type]?', static function () {
 | |
|         }, 'bar_route');
 | |
|         
 | |
|         $params = [
 | |
|             'controller' => 'test',
 | |
|             'action' => 'someaction'
 | |
|         ];
 | |
|         
 | |
|         $this->assertEquals(
 | |
|             '/test/someaction',
 | |
|             $this->router->generate('bar_route', $params)
 | |
|         );
 | |
|         
 | |
|         $params = [
 | |
|             'controller' => 'test',
 | |
|             'action' => 'someaction',
 | |
|             'type' => 'json'
 | |
|         ];
 | |
|         
 | |
|         $this->assertEquals(
 | |
|             '/test/someaction.json',
 | |
|             $this->router->generate('bar_route', $params)
 | |
|         );
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * GitHub #98
 | |
|      * @covers AltoRouter::generate
 | |
|      */
 | |
|     public function testGenerateWithOptionalPartOnBareUrl()
 | |
|     {
 | |
|         $this->router->map('GET', '/[i:page]?', static function () {
 | |
|         }, 'bare_route');
 | |
|         
 | |
|         $params = [
 | |
|             'page' => 1
 | |
|         ];
 | |
|         
 | |
|         $this->assertEquals(
 | |
|             '/1',
 | |
|             $this->router->generate('bare_route', $params)
 | |
|         );
 | |
|         
 | |
|         $params = [];
 | |
|         
 | |
|         $this->assertEquals(
 | |
|             '/',
 | |
|             $this->router->generate('bare_route', $params)
 | |
|         );
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * @covers AltoRouter::generate
 | |
|      */
 | |
|     public function testGenerateWithNonexistingRoute()
 | |
|     {
 | |
|         try {
 | |
|             $this->router->generate('nonexisting_route');
 | |
|             $this->fail('Should trigger an exception on nonexisting named route');
 | |
|         } catch (Exception $e) {
 | |
|             $this->assertEquals("Route 'nonexisting_route' does not exist.", $e->getMessage());
 | |
|         }
 | |
|     }
 | |
|     
 | |
|     /**
 | |
|      * @covers AltoRouter::match
 | |
|      * @covers AltoRouter::compileRoute
 | |
|      */
 | |
|     public function testMatch()
 | |
|     {
 | |
|         $this->router->map('GET', '/foo/[:controller]/[:action]', 'foo_action', 'foo_route');
 | |
|         
 | |
|         $this->assertEquals([
 | |
|             'target' => 'foo_action',
 | |
|             'params' => [
 | |
|                 'controller' => 'test',
 | |
|                 'action' => 'do'
 | |
|             ],
 | |
|             'name' => 'foo_route'
 | |
|         ], $this->router->match('/foo/test/do', 'GET'));
 | |
|         
 | |
|         $this->assertFalse($this->router->match('/foo/test/do', 'POST'));
 | |
|         
 | |
|         $this->assertEquals([
 | |
|             'target' => 'foo_action',
 | |
|             'params' => [
 | |
|                 'controller' => 'test',
 | |
|                 'action' => 'do'
 | |
|             ],
 | |
|             'name' => 'foo_route'
 | |
|         ], $this->router->match('/foo/test/do?param=value', 'GET'));
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * @covers AltoRouter::match
 | |
|      */
 | |
|     public function testMatchWithNonRegex()
 | |
|     {
 | |
|         $this->router->map('GET', '/about-us', 'PagesController#about', 'about_us');
 | |
| 
 | |
|         $this->assertEquals([
 | |
|             'target' => 'PagesController#about',
 | |
|             'params' => [],
 | |
|             'name' => 'about_us'
 | |
|         ], $this->router->match('/about-us', 'GET'));
 | |
| 
 | |
|         $this->assertFalse($this->router->match('/about-us', 'POST'));
 | |
|         $this->assertFalse($this->router->match('/about', 'GET'));
 | |
|         $this->assertFalse($this->router->match('/about-us-again', 'GET'));
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * @covers AltoRouter::match
 | |
|      */
 | |
|     public function testMatchWithFixedParamValues()
 | |
|     {
 | |
|         $this->router->map('POST', '/users/[i:id]/[delete|update:action]', 'usersController#doAction', 'users_do');
 | |
|         
 | |
|         $this->assertEquals([
 | |
|             'target' => 'usersController#doAction',
 | |
|             'params' => [
 | |
|                 'id' => 1,
 | |
|                 'action' => 'delete'
 | |
|             ],
 | |
|             'name' => 'users_do'
 | |
|         ], $this->router->match('/users/1/delete', 'POST'));
 | |
|         
 | |
|         $this->assertFalse($this->router->match('/users/1/delete', 'GET'));
 | |
|         $this->assertFalse($this->router->match('/users/abc/delete', 'POST'));
 | |
|         $this->assertFalse($this->router->match('/users/1/create', 'GET'));
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * @covers AltoRouter::match
 | |
|      */
 | |
|     public function testMatchWithPlainRoute()
 | |
|     {
 | |
|         $router = $this->getMockBuilder('AltoRouterDebug')
 | |
|             ->setMethods(['compileRoute'])
 | |
|             ->getMock();
 | |
|         
 | |
|         // this should prove that compileRoute is not called when the route doesn't
 | |
|         // have any params in it, but this doesn't work because compileRoute is private.
 | |
|         $router->expects($this->never())
 | |
|             ->method('compileRoute');
 | |
| 
 | |
|         $router->map('GET', '/contact', 'website#contact', 'contact');
 | |
| 
 | |
|         // exact match, so no regex compilation necessary
 | |
|         $this->assertEquals([
 | |
|             'target' => 'website#contact',
 | |
|             'params' => [],
 | |
|             'name' => 'contact'
 | |
|         ], $router->match('/contact', 'GET'));
 | |
| 
 | |
|         $router->map('GET', '/page/[:id]', 'pages#show', 'page');
 | |
| 
 | |
|         // no prefix match, so no regex compilation necessary
 | |
|         $this->assertFalse($router->match('/page1', 'GET'));
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * @covers AltoRouter::match
 | |
|      */
 | |
|     public function testMatchWithServerVars()
 | |
|     {
 | |
|         $this->router->map('GET', '/foo/[:controller]/[:action]', 'foo_action', 'foo_route');
 | |
|         
 | |
|         $_SERVER['REQUEST_URI'] = '/foo/test/do';
 | |
|         $_SERVER['REQUEST_METHOD'] = 'GET';
 | |
|         
 | |
|         $this->assertEquals([
 | |
|             'target' => 'foo_action',
 | |
|             'params' => [
 | |
|                 'controller' => 'test',
 | |
|                 'action' => 'do'
 | |
|             ],
 | |
|             'name' => 'foo_route'
 | |
|         ], $this->router->match());
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * @covers AltoRouter::match
 | |
|      */
 | |
|     public function testMatchWithOptionalUrlParts()
 | |
|     {
 | |
|         $this->router->map('GET', '/bar/[:controller]/[:action].[:type]?', 'bar_action', 'bar_route');
 | |
|         
 | |
|         $this->assertEquals([
 | |
|             'target' => 'bar_action',
 | |
|             'params' => [
 | |
|                 'controller' => 'test',
 | |
|                 'action' => 'do',
 | |
|                 'type' => 'json'
 | |
|             ],
 | |
|             'name' => 'bar_route'
 | |
|         ], $this->router->match('/bar/test/do.json', 'GET'));
 | |
|         
 | |
|         $this->assertEquals([
 | |
|             'target' => 'bar_action',
 | |
|             'params' => [
 | |
|                 'controller' => 'test',
 | |
|                 'action' => 'do'
 | |
|             ],
 | |
|             'name' => 'bar_route'
 | |
|         ], $this->router->match('/bar/test/do', 'GET'));
 | |
|     }
 | |
|     
 | |
|     /**
 | |
|      * GitHub #98
 | |
|      * @covers AltoRouter::match
 | |
|      */
 | |
|     public function testMatchWithOptionalPartOnBareUrl()
 | |
|     {
 | |
|         $this->router->map('GET', '/[i:page]?', 'bare_action', 'bare_route');
 | |
|         
 | |
|         $this->assertEquals([
 | |
|             'target' => 'bare_action',
 | |
|             'params' => [
 | |
|                 'page' => 1
 | |
|             ],
 | |
|             'name' => 'bare_route'
 | |
|         ], $this->router->match('/1', 'GET'));
 | |
|         
 | |
|         $this->assertEquals([
 | |
|             'target' => 'bare_action',
 | |
|             'params' => [],
 | |
|             'name' => 'bare_route'
 | |
|         ], $this->router->match('/', 'GET'));
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * @covers AltoRouter::match
 | |
|      */
 | |
|     public function testMatchWithWildcard()
 | |
|     {
 | |
|         $this->router->map('GET', '/a', 'foo_action', 'foo_route');
 | |
|         $this->router->map('GET', '*', 'bar_action', 'bar_route');
 | |
|         
 | |
|         $this->assertEquals([
 | |
|             'target' => 'bar_action',
 | |
|             'params' => [],
 | |
|             'name' => 'bar_route'
 | |
|         ], $this->router->match('/everything', 'GET'));
 | |
|     }
 | |
|     /**
 | |
|      * @covers AltoRouter::match
 | |
|      */
 | |
|     public function testMatchWithCustomRegexp()
 | |
|     {
 | |
|         $this->router->map('GET', '@^/[a-z]*$', 'bar_action', 'bar_route');
 | |
|         
 | |
|         $this->assertEquals([
 | |
|             'target' => 'bar_action',
 | |
|             'params' => [],
 | |
|             'name' => 'bar_route'
 | |
|         ], $this->router->match('/everything', 'GET'));
 | |
|         
 | |
|         $this->assertFalse($this->router->match('/some-other-thing', 'GET'));
 | |
|     }
 | |
|     /**
 | |
|      * @covers AltoRouter::match
 | |
|      */
 | |
|     public function testMatchWithUnicodeRegex()
 | |
|     {
 | |
|         $pattern = '/(?<path>[^';
 | |
|         // Arabic characters
 | |
|         $pattern .= '\x{0600}-\x{06FF}';
 | |
|         $pattern .= '\x{FB50}-\x{FDFD}';
 | |
|         $pattern .= '\x{FE70}-\x{FEFF}';
 | |
|         $pattern .= '\x{0750}-\x{077F}';
 | |
|         // Alphanumeric, /, _, - and space characters
 | |
|         $pattern .= 'a-zA-Z0-9\/_\-\s';
 | |
|         // 'ZERO WIDTH NON-JOINER'
 | |
|         $pattern .= '\x{200C}';
 | |
|         $pattern .= ']+)';
 | |
|         
 | |
|         $this->router->map('GET', '@' . $pattern, 'unicode_action', 'unicode_route');
 | |
|         
 | |
|         $this->assertEquals([
 | |
|             'target' => 'unicode_action',
 | |
|             'name' => 'unicode_route',
 | |
|             'params' => [
 | |
|                 'path' => '大家好'
 | |
|             ]
 | |
|         ], $this->router->match('/大家好', 'GET'));
 | |
|         
 | |
|         $this->assertFalse($this->router->match('/﷽', 'GET'));
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * @covers AltoRouter::addMatchTypes
 | |
|      */
 | |
|     public function testMatchWithCustomNamedRegex()
 | |
|     {
 | |
|         $this->router->addMatchTypes(['cId' => '[a-zA-Z]{2}[0-9](?:_[0-9]++)?']);
 | |
|         $this->router->map('GET', '/bar/[cId:customId]', 'bar_action', 'bar_route');
 | |
|         
 | |
|         $this->assertEquals([
 | |
|             'target' => 'bar_action',
 | |
|             'params' => [
 | |
|                 'customId' => 'AB1',
 | |
|             ],
 | |
|             'name' => 'bar_route'
 | |
|         ], $this->router->match('/bar/AB1', 'GET'));
 | |
| 
 | |
|         $this->assertEquals([
 | |
|             'target' => 'bar_action',
 | |
|             'params' => [
 | |
|                 'customId' => 'AB1_0123456789',
 | |
|             ],
 | |
|             'name' => 'bar_route'
 | |
|         ], $this->router->match('/bar/AB1_0123456789', 'GET'));
 | |
|         
 | |
|         $this->assertFalse($this->router->match('/some-other-thing', 'GET'));
 | |
|     }
 | |
|     /**
 | |
|      * @covers AltoRouter::addMatchTypes
 | |
|      */
 | |
|     public function testMatchWithCustomNamedUnicodeRegex()
 | |
|     {
 | |
|         $pattern = '[^';
 | |
|         // Arabic characters
 | |
|         $pattern .= '\x{0600}-\x{06FF}';
 | |
|         $pattern .= '\x{FB50}-\x{FDFD}';
 | |
|         $pattern .= '\x{FE70}-\x{FEFF}';
 | |
|         $pattern .= '\x{0750}-\x{077F}';
 | |
|         $pattern .= ']+';
 | |
|         
 | |
|         $this->router->addMatchTypes(['nonArabic' => $pattern]);
 | |
|         $this->router->map('GET', '/bar/[nonArabic:string]', 'non_arabic_action', 'non_arabic_route');
 | |
|         
 | |
|         $this->assertEquals([
 | |
|             'target' => 'non_arabic_action',
 | |
|             'name' => 'non_arabic_route',
 | |
|             'params' => [
 | |
|                 'string' => 'some-path'
 | |
|             ]
 | |
|         ], $this->router->match('/bar/some-path', 'GET'));
 | |
|         
 | |
|         $this->assertFalse($this->router->match('/﷽', 'GET'));
 | |
|     }
 | |
| }
 |