 1ac0f42a58
			
		
	
	1ac0f42a58
	
	
	
		
			
			Travis config update Removed HHVM script as Laravel no longer support HHVM after releasing 5.3
		
			
				
	
	
		
			409 lines
		
	
	
		
			15 KiB
		
	
	
	
		
			PHP
		
	
	
	
	
	
			
		
		
	
	
			409 lines
		
	
	
		
			15 KiB
		
	
	
	
		
			PHP
		
	
	
	
	
	
| <?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\Routing\Route;
 | |
| use Symfony\Component\Routing\RouteCompiler;
 | |
| 
 | |
| class RouteCompilerTest extends TestCase
 | |
| {
 | |
|     /**
 | |
|      * @dataProvider provideCompileData
 | |
|      */
 | |
|     public function testCompile($name, $arguments, $prefix, $regex, $variables, $tokens)
 | |
|     {
 | |
|         $r = new \ReflectionClass('Symfony\\Component\\Routing\\Route');
 | |
|         $route = $r->newInstanceArgs($arguments);
 | |
| 
 | |
|         $compiled = $route->compile();
 | |
|         $this->assertEquals($prefix, $compiled->getStaticPrefix(), $name.' (static prefix)');
 | |
|         $this->assertEquals($regex, $compiled->getRegex(), $name.' (regex)');
 | |
|         $this->assertEquals($variables, $compiled->getVariables(), $name.' (variables)');
 | |
|         $this->assertEquals($tokens, $compiled->getTokens(), $name.' (tokens)');
 | |
|     }
 | |
| 
 | |
|     public function provideCompileData()
 | |
|     {
 | |
|         return array(
 | |
|             array(
 | |
|                 'Static route',
 | |
|                 array('/foo'),
 | |
|                 '/foo', '#^/foo$#sD', array(), array(
 | |
|                     array('text', '/foo'),
 | |
|                 ),
 | |
|             ),
 | |
| 
 | |
|             array(
 | |
|                 'Route with a variable',
 | |
|                 array('/foo/{bar}'),
 | |
|                 '/foo', '#^/foo/(?P<bar>[^/]++)$#sD', array('bar'), array(
 | |
|                     array('variable', '/', '[^/]++', 'bar'),
 | |
|                     array('text', '/foo'),
 | |
|                 ),
 | |
|             ),
 | |
| 
 | |
|             array(
 | |
|                 'Route with a variable that has a default value',
 | |
|                 array('/foo/{bar}', array('bar' => 'bar')),
 | |
|                 '/foo', '#^/foo(?:/(?P<bar>[^/]++))?$#sD', array('bar'), array(
 | |
|                     array('variable', '/', '[^/]++', 'bar'),
 | |
|                     array('text', '/foo'),
 | |
|                 ),
 | |
|             ),
 | |
| 
 | |
|             array(
 | |
|                 'Route with several variables',
 | |
|                 array('/foo/{bar}/{foobar}'),
 | |
|                 '/foo', '#^/foo/(?P<bar>[^/]++)/(?P<foobar>[^/]++)$#sD', array('bar', 'foobar'), array(
 | |
|                     array('variable', '/', '[^/]++', 'foobar'),
 | |
|                     array('variable', '/', '[^/]++', 'bar'),
 | |
|                     array('text', '/foo'),
 | |
|                 ),
 | |
|             ),
 | |
| 
 | |
|             array(
 | |
|                 'Route with several variables that have default values',
 | |
|                 array('/foo/{bar}/{foobar}', array('bar' => 'bar', 'foobar' => '')),
 | |
|                 '/foo', '#^/foo(?:/(?P<bar>[^/]++)(?:/(?P<foobar>[^/]++))?)?$#sD', array('bar', 'foobar'), array(
 | |
|                     array('variable', '/', '[^/]++', 'foobar'),
 | |
|                     array('variable', '/', '[^/]++', 'bar'),
 | |
|                     array('text', '/foo'),
 | |
|                 ),
 | |
|             ),
 | |
| 
 | |
|             array(
 | |
|                 'Route with several variables but some of them have no default values',
 | |
|                 array('/foo/{bar}/{foobar}', array('bar' => 'bar')),
 | |
|                 '/foo', '#^/foo/(?P<bar>[^/]++)/(?P<foobar>[^/]++)$#sD', array('bar', 'foobar'), array(
 | |
|                     array('variable', '/', '[^/]++', 'foobar'),
 | |
|                     array('variable', '/', '[^/]++', 'bar'),
 | |
|                     array('text', '/foo'),
 | |
|                 ),
 | |
|             ),
 | |
| 
 | |
|             array(
 | |
|                 'Route with an optional variable as the first segment',
 | |
|                 array('/{bar}', array('bar' => 'bar')),
 | |
|                 '', '#^/(?P<bar>[^/]++)?$#sD', array('bar'), array(
 | |
|                     array('variable', '/', '[^/]++', 'bar'),
 | |
|                 ),
 | |
|             ),
 | |
| 
 | |
|             array(
 | |
|                 'Route with a requirement of 0',
 | |
|                 array('/{bar}', array('bar' => null), array('bar' => '0')),
 | |
|                 '', '#^/(?P<bar>0)?$#sD', array('bar'), array(
 | |
|                     array('variable', '/', '0', 'bar'),
 | |
|                 ),
 | |
|             ),
 | |
| 
 | |
|             array(
 | |
|                 'Route with an optional variable as the first segment with requirements',
 | |
|                 array('/{bar}', array('bar' => 'bar'), array('bar' => '(foo|bar)')),
 | |
|                 '', '#^/(?P<bar>(?:foo|bar))?$#sD', array('bar'), array(
 | |
|                     array('variable', '/', '(?:foo|bar)', 'bar'),
 | |
|                 ),
 | |
|             ),
 | |
| 
 | |
|             array(
 | |
|                 'Route with only optional variables',
 | |
|                 array('/{foo}/{bar}', array('foo' => 'foo', 'bar' => 'bar')),
 | |
|                 '', '#^/(?P<foo>[^/]++)?(?:/(?P<bar>[^/]++))?$#sD', array('foo', 'bar'), array(
 | |
|                     array('variable', '/', '[^/]++', 'bar'),
 | |
|                     array('variable', '/', '[^/]++', 'foo'),
 | |
|                 ),
 | |
|             ),
 | |
| 
 | |
|             array(
 | |
|                 'Route with a variable in last position',
 | |
|                 array('/foo-{bar}'),
 | |
|                 '/foo-', '#^/foo\-(?P<bar>[^/]++)$#sD', array('bar'), array(
 | |
|                     array('variable', '-', '[^/]++', 'bar'),
 | |
|                     array('text', '/foo'),
 | |
|                 ),
 | |
|             ),
 | |
| 
 | |
|             array(
 | |
|                 'Route with nested placeholders',
 | |
|                 array('/{static{var}static}'),
 | |
|                 '/{static', '#^/\{static(?P<var>[^/]+)static\}$#sD', array('var'), array(
 | |
|                     array('text', 'static}'),
 | |
|                     array('variable', '', '[^/]+', 'var'),
 | |
|                     array('text', '/{static'),
 | |
|                 ),
 | |
|             ),
 | |
| 
 | |
|             array(
 | |
|                 'Route without separator between variables',
 | |
|                 array('/{w}{x}{y}{z}.{_format}', array('z' => 'default-z', '_format' => 'html'), array('y' => '(y|Y)')),
 | |
|                 '', '#^/(?P<w>[^/\.]+)(?P<x>[^/\.]+)(?P<y>(?:y|Y))(?:(?P<z>[^/\.]++)(?:\.(?P<_format>[^/]++))?)?$#sD', array('w', 'x', 'y', 'z', '_format'), array(
 | |
|                     array('variable', '.', '[^/]++', '_format'),
 | |
|                     array('variable', '', '[^/\.]++', 'z'),
 | |
|                     array('variable', '', '(?:y|Y)', 'y'),
 | |
|                     array('variable', '', '[^/\.]+', 'x'),
 | |
|                     array('variable', '/', '[^/\.]+', 'w'),
 | |
|                 ),
 | |
|             ),
 | |
| 
 | |
|             array(
 | |
|                 'Route with a format',
 | |
|                 array('/foo/{bar}.{_format}'),
 | |
|                 '/foo', '#^/foo/(?P<bar>[^/\.]++)\.(?P<_format>[^/]++)$#sD', array('bar', '_format'), array(
 | |
|                     array('variable', '.', '[^/]++', '_format'),
 | |
|                     array('variable', '/', '[^/\.]++', 'bar'),
 | |
|                     array('text', '/foo'),
 | |
|                 ),
 | |
|             ),
 | |
| 
 | |
|             array(
 | |
|                 'Static non UTF-8 route',
 | |
|                 array("/fo\xE9"),
 | |
|                 "/fo\xE9", "#^/fo\xE9$#sD", array(), array(
 | |
|                     array('text', "/fo\xE9"),
 | |
|                 ),
 | |
|             ),
 | |
| 
 | |
|             array(
 | |
|                 'Route with an explicit UTF-8 requirement',
 | |
|                 array('/{bar}', array('bar' => null), array('bar' => '.'), array('utf8' => true)),
 | |
|                 '', '#^/(?P<bar>.)?$#sDu', array('bar'), array(
 | |
|                     array('variable', '/', '.', 'bar', true),
 | |
|                 ),
 | |
|             ),
 | |
|         );
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * @dataProvider provideCompileImplicitUtf8Data
 | |
|      * @expectedException \LogicException
 | |
|      */
 | |
|     public function testCompileImplicitUtf8Data($name, $arguments, $prefix, $regex, $variables, $tokens, $deprecationType)
 | |
|     {
 | |
|         $r = new \ReflectionClass('Symfony\\Component\\Routing\\Route');
 | |
|         $route = $r->newInstanceArgs($arguments);
 | |
| 
 | |
|         $compiled = $route->compile();
 | |
|         $this->assertEquals($prefix, $compiled->getStaticPrefix(), $name.' (static prefix)');
 | |
|         $this->assertEquals($regex, $compiled->getRegex(), $name.' (regex)');
 | |
|         $this->assertEquals($variables, $compiled->getVariables(), $name.' (variables)');
 | |
|         $this->assertEquals($tokens, $compiled->getTokens(), $name.' (tokens)');
 | |
|     }
 | |
| 
 | |
|     public function provideCompileImplicitUtf8Data()
 | |
|     {
 | |
|         return array(
 | |
|             array(
 | |
|                 'Static UTF-8 route',
 | |
|                 array('/foé'),
 | |
|                 '/foé', '#^/foé$#sDu', array(), array(
 | |
|                     array('text', '/foé'),
 | |
|                 ),
 | |
|                 'patterns',
 | |
|             ),
 | |
| 
 | |
|             array(
 | |
|                 'Route with an implicit UTF-8 requirement',
 | |
|                 array('/{bar}', array('bar' => null), array('bar' => 'é')),
 | |
|                 '', '#^/(?P<bar>é)?$#sDu', array('bar'), array(
 | |
|                     array('variable', '/', 'é', 'bar', true),
 | |
|                 ),
 | |
|                 'requirements',
 | |
|             ),
 | |
| 
 | |
|             array(
 | |
|                 'Route with a UTF-8 class requirement',
 | |
|                 array('/{bar}', array('bar' => null), array('bar' => '\pM')),
 | |
|                 '', '#^/(?P<bar>\pM)?$#sDu', array('bar'), array(
 | |
|                     array('variable', '/', '\pM', 'bar', true),
 | |
|                 ),
 | |
|                 'requirements',
 | |
|             ),
 | |
| 
 | |
|             array(
 | |
|                 'Route with a UTF-8 separator',
 | |
|                 array('/foo/{bar}§{_format}', array(), array(), array('compiler_class' => Utf8RouteCompiler::class)),
 | |
|                 '/foo', '#^/foo/(?P<bar>[^/§]++)§(?P<_format>[^/]++)$#sDu', array('bar', '_format'), array(
 | |
|                     array('variable', '§', '[^/]++', '_format', true),
 | |
|                     array('variable', '/', '[^/§]++', 'bar', true),
 | |
|                     array('text', '/foo'),
 | |
|                 ),
 | |
|                 'patterns',
 | |
|             ),
 | |
|         );
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * @expectedException \LogicException
 | |
|      */
 | |
|     public function testRouteWithSameVariableTwice()
 | |
|     {
 | |
|         $route = new Route('/{name}/{name}');
 | |
| 
 | |
|         $compiled = $route->compile();
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * @expectedException \LogicException
 | |
|      */
 | |
|     public function testRouteCharsetMismatch()
 | |
|     {
 | |
|         $route = new Route("/\xE9/{bar}", array(), array('bar' => '.'), array('utf8' => true));
 | |
| 
 | |
|         $compiled = $route->compile();
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * @expectedException \LogicException
 | |
|      */
 | |
|     public function testRequirementCharsetMismatch()
 | |
|     {
 | |
|         $route = new Route('/foo/{bar}', array(), array('bar' => "\xE9"), array('utf8' => true));
 | |
| 
 | |
|         $compiled = $route->compile();
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * @expectedException \InvalidArgumentException
 | |
|      */
 | |
|     public function testRouteWithFragmentAsPathParameter()
 | |
|     {
 | |
|         $route = new Route('/{_fragment}');
 | |
| 
 | |
|         $compiled = $route->compile();
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * @dataProvider getVariableNamesStartingWithADigit
 | |
|      * @expectedException \DomainException
 | |
|      */
 | |
|     public function testRouteWithVariableNameStartingWithADigit($name)
 | |
|     {
 | |
|         $route = new Route('/{'.$name.'}');
 | |
|         $route->compile();
 | |
|     }
 | |
| 
 | |
|     public function getVariableNamesStartingWithADigit()
 | |
|     {
 | |
|         return array(
 | |
|            array('09'),
 | |
|            array('123'),
 | |
|            array('1e2'),
 | |
|         );
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * @dataProvider provideCompileWithHostData
 | |
|      */
 | |
|     public function testCompileWithHost($name, $arguments, $prefix, $regex, $variables, $pathVariables, $tokens, $hostRegex, $hostVariables, $hostTokens)
 | |
|     {
 | |
|         $r = new \ReflectionClass('Symfony\\Component\\Routing\\Route');
 | |
|         $route = $r->newInstanceArgs($arguments);
 | |
| 
 | |
|         $compiled = $route->compile();
 | |
|         $this->assertEquals($prefix, $compiled->getStaticPrefix(), $name.' (static prefix)');
 | |
|         $this->assertEquals($regex, str_replace(array("\n", ' '), '', $compiled->getRegex()), $name.' (regex)');
 | |
|         $this->assertEquals($variables, $compiled->getVariables(), $name.' (variables)');
 | |
|         $this->assertEquals($pathVariables, $compiled->getPathVariables(), $name.' (path variables)');
 | |
|         $this->assertEquals($tokens, $compiled->getTokens(), $name.' (tokens)');
 | |
|         $this->assertEquals($hostRegex, str_replace(array("\n", ' '), '', $compiled->getHostRegex()), $name.' (host regex)');
 | |
|         $this->assertEquals($hostVariables, $compiled->getHostVariables(), $name.' (host variables)');
 | |
|         $this->assertEquals($hostTokens, $compiled->getHostTokens(), $name.' (host tokens)');
 | |
|     }
 | |
| 
 | |
|     public function provideCompileWithHostData()
 | |
|     {
 | |
|         return array(
 | |
|             array(
 | |
|                 'Route with host pattern',
 | |
|                 array('/hello', array(), array(), array(), 'www.example.com'),
 | |
|                 '/hello', '#^/hello$#sD', array(), array(), array(
 | |
|                     array('text', '/hello'),
 | |
|                 ),
 | |
|                 '#^www\.example\.com$#sDi', array(), array(
 | |
|                     array('text', 'www.example.com'),
 | |
|                 ),
 | |
|             ),
 | |
|             array(
 | |
|                 'Route with host pattern and some variables',
 | |
|                 array('/hello/{name}', array(), array(), array(), 'www.example.{tld}'),
 | |
|                 '/hello', '#^/hello/(?P<name>[^/]++)$#sD', array('tld', 'name'), array('name'), array(
 | |
|                     array('variable', '/', '[^/]++', 'name'),
 | |
|                     array('text', '/hello'),
 | |
|                 ),
 | |
|                 '#^www\.example\.(?P<tld>[^\.]++)$#sDi', array('tld'), array(
 | |
|                     array('variable', '.', '[^\.]++', 'tld'),
 | |
|                     array('text', 'www.example'),
 | |
|                 ),
 | |
|             ),
 | |
|             array(
 | |
|                 'Route with variable at beginning of host',
 | |
|                 array('/hello', array(), array(), array(), '{locale}.example.{tld}'),
 | |
|                 '/hello', '#^/hello$#sD', array('locale', 'tld'), array(), array(
 | |
|                     array('text', '/hello'),
 | |
|                 ),
 | |
|                 '#^(?P<locale>[^\.]++)\.example\.(?P<tld>[^\.]++)$#sDi', array('locale', 'tld'), array(
 | |
|                     array('variable', '.', '[^\.]++', 'tld'),
 | |
|                     array('text', '.example'),
 | |
|                     array('variable', '', '[^\.]++', 'locale'),
 | |
|                 ),
 | |
|             ),
 | |
|             array(
 | |
|                 'Route with host variables that has a default value',
 | |
|                 array('/hello', array('locale' => 'a', 'tld' => 'b'), array(), array(), '{locale}.example.{tld}'),
 | |
|                 '/hello', '#^/hello$#sD', array('locale', 'tld'), array(), array(
 | |
|                     array('text', '/hello'),
 | |
|                 ),
 | |
|                 '#^(?P<locale>[^\.]++)\.example\.(?P<tld>[^\.]++)$#sDi', array('locale', 'tld'), array(
 | |
|                     array('variable', '.', '[^\.]++', 'tld'),
 | |
|                     array('text', '.example'),
 | |
|                     array('variable', '', '[^\.]++', 'locale'),
 | |
|                 ),
 | |
|             ),
 | |
|         );
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * @expectedException \DomainException
 | |
|      */
 | |
|     public function testRouteWithTooLongVariableName()
 | |
|     {
 | |
|         $route = new Route(sprintf('/{%s}', str_repeat('a', RouteCompiler::VARIABLE_MAXIMUM_LENGTH + 1)));
 | |
|         $route->compile();
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * @dataProvider provideRemoveCapturingGroup
 | |
|      */
 | |
|     public function testRemoveCapturingGroup($regex, $requirement)
 | |
|     {
 | |
|         $route = new Route('/{foo}', array(), array('foo' => $requirement));
 | |
| 
 | |
|         $this->assertSame($regex, $route->compile()->getRegex());
 | |
|     }
 | |
| 
 | |
|     public function provideRemoveCapturingGroup()
 | |
|     {
 | |
|         yield array('#^/(?P<foo>a(?:b|c)(?:d|e)f)$#sD', 'a(b|c)(d|e)f');
 | |
|         yield array('#^/(?P<foo>a\(b\)c)$#sD', 'a\(b\)c');
 | |
|         yield array('#^/(?P<foo>(?:b))$#sD', '(?:b)');
 | |
|         yield array('#^/(?P<foo>(?(b)b))$#sD', '(?(b)b)');
 | |
|         yield array('#^/(?P<foo>(*F))$#sD', '(*F)');
 | |
|         yield array('#^/(?P<foo>(?:(?:foo)))$#sD', '((foo))');
 | |
|     }
 | |
| }
 | |
| 
 | |
| class Utf8RouteCompiler extends RouteCompiler
 | |
| {
 | |
|     const SEPARATORS = '/§';
 | |
| }
 |