RouteTest.php 19 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373
  1. <?php
  2. /*
  3. * This file is part of the Symfony package.
  4. *
  5. * (c) Fabien Potencier <fabien@symfony.com>
  6. *
  7. * For the full copyright and license information, please view the LICENSE
  8. * file that was distributed with this source code.
  9. */
  10. namespace Symfony\Component\Routing\Tests;
  11. use PHPUnit\Framework\TestCase;
  12. use Symfony\Component\Routing\CompiledRoute;
  13. use Symfony\Component\Routing\Route;
  14. use Symfony\Component\Routing\Tests\Fixtures\CustomCompiledRoute;
  15. use Symfony\Component\Routing\Tests\Fixtures\CustomRouteCompiler;
  16. class RouteTest extends TestCase
  17. {
  18. public function testConstructor()
  19. {
  20. $route = new Route('/{foo}', ['foo' => 'bar'], ['foo' => '\d+'], ['foo' => 'bar'], '{locale}.example.com');
  21. $this->assertEquals('/{foo}', $route->getPath(), '__construct() takes a path as its first argument');
  22. $this->assertEquals(['foo' => 'bar'], $route->getDefaults(), '__construct() takes defaults as its second argument');
  23. $this->assertEquals(['foo' => '\d+'], $route->getRequirements(), '__construct() takes requirements as its third argument');
  24. $this->assertEquals('bar', $route->getOption('foo'), '__construct() takes options as its fourth argument');
  25. $this->assertEquals('{locale}.example.com', $route->getHost(), '__construct() takes a host pattern as its fifth argument');
  26. $route = new Route('/', [], [], [], '', ['Https'], ['POST', 'put'], 'context.getMethod() == "GET"');
  27. $this->assertEquals(['https'], $route->getSchemes(), '__construct() takes schemes as its sixth argument and lowercases it');
  28. $this->assertEquals(['POST', 'PUT'], $route->getMethods(), '__construct() takes methods as its seventh argument and uppercases it');
  29. $this->assertEquals('context.getMethod() == "GET"', $route->getCondition(), '__construct() takes a condition as its eight argument');
  30. $route = new Route('/', [], [], [], '', 'Https', 'Post');
  31. $this->assertEquals(['https'], $route->getSchemes(), '__construct() takes a single scheme as its sixth argument');
  32. $this->assertEquals(['POST'], $route->getMethods(), '__construct() takes a single method as its seventh argument');
  33. }
  34. public function testPath()
  35. {
  36. $route = new Route('/{foo}');
  37. $route->setPath('/{bar}');
  38. $this->assertEquals('/{bar}', $route->getPath(), '->setPath() sets the path');
  39. $route->setPath('');
  40. $this->assertEquals('/', $route->getPath(), '->setPath() adds a / at the beginning of the path if needed');
  41. $route->setPath('bar');
  42. $this->assertEquals('/bar', $route->getPath(), '->setPath() adds a / at the beginning of the path if needed');
  43. $this->assertEquals($route, $route->setPath(''), '->setPath() implements a fluent interface');
  44. $route->setPath('//path');
  45. $this->assertEquals('/path', $route->getPath(), '->setPath() does not allow two slashes "//" at the beginning of the path as it would be confused with a network path when generating the path from the route');
  46. $route->setPath('/path/{!foo}');
  47. $this->assertEquals('/path/{!foo}', $route->getPath(), '->setPath() keeps ! to pass important params');
  48. $route->setPath('/path/{bar<\w++>}');
  49. $this->assertEquals('/path/{bar}', $route->getPath(), '->setPath() removes inline requirements');
  50. $route->setPath('/path/{foo?value}');
  51. $this->assertEquals('/path/{foo}', $route->getPath(), '->setPath() removes inline defaults');
  52. $route->setPath('/path/{!bar<\d+>?value}');
  53. $this->assertEquals('/path/{!bar}', $route->getPath(), '->setPath() removes all inline settings');
  54. }
  55. public function testOptions()
  56. {
  57. $route = new Route('/{foo}');
  58. $route->setOptions(['foo' => 'bar']);
  59. $this->assertEquals(array_merge([
  60. 'compiler_class' => 'Symfony\\Component\\Routing\\RouteCompiler',
  61. ], ['foo' => 'bar']), $route->getOptions(), '->setOptions() sets the options');
  62. $this->assertEquals($route, $route->setOptions([]), '->setOptions() implements a fluent interface');
  63. $route->setOptions(['foo' => 'foo']);
  64. $route->addOptions(['bar' => 'bar']);
  65. $this->assertEquals($route, $route->addOptions([]), '->addOptions() implements a fluent interface');
  66. $this->assertEquals(['foo' => 'foo', 'bar' => 'bar', 'compiler_class' => 'Symfony\\Component\\Routing\\RouteCompiler'], $route->getOptions(), '->addDefaults() keep previous defaults');
  67. }
  68. public function testOption()
  69. {
  70. $route = new Route('/{foo}');
  71. $this->assertFalse($route->hasOption('foo'), '->hasOption() return false if option is not set');
  72. $this->assertEquals($route, $route->setOption('foo', 'bar'), '->setOption() implements a fluent interface');
  73. $this->assertEquals('bar', $route->getOption('foo'), '->setOption() sets the option');
  74. $this->assertTrue($route->hasOption('foo'), '->hasOption() return true if option is set');
  75. }
  76. public function testDefaults()
  77. {
  78. $route = new Route('/{foo}');
  79. $route->setDefaults(['foo' => 'bar']);
  80. $this->assertEquals(['foo' => 'bar'], $route->getDefaults(), '->setDefaults() sets the defaults');
  81. $this->assertEquals($route, $route->setDefaults([]), '->setDefaults() implements a fluent interface');
  82. $route->setDefault('foo', 'bar');
  83. $this->assertEquals('bar', $route->getDefault('foo'), '->setDefault() sets a default value');
  84. $route->setDefault('foo2', 'bar2');
  85. $this->assertEquals('bar2', $route->getDefault('foo2'), '->getDefault() return the default value');
  86. $this->assertNull($route->getDefault('not_defined'), '->getDefault() return null if default value is not set');
  87. $route->setDefault('_controller', $closure = function () { return 'Hello'; });
  88. $this->assertEquals($closure, $route->getDefault('_controller'), '->setDefault() sets a default value');
  89. $route->setDefaults(['foo' => 'foo']);
  90. $route->addDefaults(['bar' => 'bar']);
  91. $this->assertEquals($route, $route->addDefaults([]), '->addDefaults() implements a fluent interface');
  92. $this->assertEquals(['foo' => 'foo', 'bar' => 'bar'], $route->getDefaults(), '->addDefaults() keep previous defaults');
  93. }
  94. public function testRequirements()
  95. {
  96. $route = new Route('/{foo}');
  97. $route->setRequirements(['foo' => '\d+']);
  98. $this->assertEquals(['foo' => '\d+'], $route->getRequirements(), '->setRequirements() sets the requirements');
  99. $this->assertEquals('\d+', $route->getRequirement('foo'), '->getRequirement() returns a requirement');
  100. $this->assertNull($route->getRequirement('bar'), '->getRequirement() returns null if a requirement is not defined');
  101. $route->setRequirements(['foo' => '^\d+$']);
  102. $this->assertEquals('\d+', $route->getRequirement('foo'), '->getRequirement() removes ^ and $ from the path');
  103. $this->assertEquals($route, $route->setRequirements([]), '->setRequirements() implements a fluent interface');
  104. $route->setRequirements(['foo' => '\d+']);
  105. $route->addRequirements(['bar' => '\d+']);
  106. $this->assertEquals($route, $route->addRequirements([]), '->addRequirements() implements a fluent interface');
  107. $this->assertEquals(['foo' => '\d+', 'bar' => '\d+'], $route->getRequirements(), '->addRequirement() keep previous requirements');
  108. }
  109. public function testRequirement()
  110. {
  111. $route = new Route('/{foo}');
  112. $this->assertFalse($route->hasRequirement('foo'), '->hasRequirement() return false if requirement is not set');
  113. $route->setRequirement('foo', '^\d+$');
  114. $this->assertEquals('\d+', $route->getRequirement('foo'), '->setRequirement() removes ^ and $ from the path');
  115. $this->assertTrue($route->hasRequirement('foo'), '->hasRequirement() return true if requirement is set');
  116. }
  117. public function testRequirementAlternativeStartAndEndRegexSyntax()
  118. {
  119. $route = new Route('/{foo}');
  120. $route->setRequirement('foo', '\A\d+\z');
  121. $this->assertEquals('\d+', $route->getRequirement('foo'), '->setRequirement() removes \A and \z from the path');
  122. $this->assertTrue($route->hasRequirement('foo'));
  123. }
  124. /**
  125. * @dataProvider getInvalidRequirements
  126. */
  127. public function testSetInvalidRequirement($req)
  128. {
  129. $this->expectException(\InvalidArgumentException::class);
  130. $route = new Route('/{foo}');
  131. $route->setRequirement('foo', $req);
  132. }
  133. public static function getInvalidRequirements()
  134. {
  135. return [
  136. [''],
  137. ['^$'],
  138. ['^'],
  139. ['$'],
  140. ['\A\z'],
  141. ['\A'],
  142. ['\z'],
  143. ];
  144. }
  145. public function testHost()
  146. {
  147. $route = new Route('/');
  148. $route->setHost('{locale}.example.net');
  149. $this->assertEquals('{locale}.example.net', $route->getHost(), '->setHost() sets the host pattern');
  150. }
  151. public function testScheme()
  152. {
  153. $route = new Route('/');
  154. $this->assertEquals([], $route->getSchemes(), 'schemes is initialized with []');
  155. $this->assertFalse($route->hasScheme('http'));
  156. $route->setSchemes('hTTp');
  157. $this->assertEquals(['http'], $route->getSchemes(), '->setSchemes() accepts a single scheme string and lowercases it');
  158. $this->assertTrue($route->hasScheme('htTp'));
  159. $this->assertFalse($route->hasScheme('httpS'));
  160. $route->setSchemes(['HttpS', 'hTTp']);
  161. $this->assertEquals(['https', 'http'], $route->getSchemes(), '->setSchemes() accepts an array of schemes and lowercases them');
  162. $this->assertTrue($route->hasScheme('htTp'));
  163. $this->assertTrue($route->hasScheme('httpS'));
  164. }
  165. public function testMethod()
  166. {
  167. $route = new Route('/');
  168. $this->assertEquals([], $route->getMethods(), 'methods is initialized with []');
  169. $route->setMethods('gEt');
  170. $this->assertEquals(['GET'], $route->getMethods(), '->setMethods() accepts a single method string and uppercases it');
  171. $route->setMethods(['gEt', 'PosT']);
  172. $this->assertEquals(['GET', 'POST'], $route->getMethods(), '->setMethods() accepts an array of methods and uppercases them');
  173. }
  174. public function testCondition()
  175. {
  176. $route = new Route('/');
  177. $this->assertSame('', $route->getCondition());
  178. $route->setCondition('context.getMethod() == "GET"');
  179. $this->assertSame('context.getMethod() == "GET"', $route->getCondition());
  180. }
  181. public function testCompile()
  182. {
  183. $route = new Route('/{foo}');
  184. $this->assertInstanceOf(CompiledRoute::class, $compiled = $route->compile(), '->compile() returns a compiled route');
  185. $this->assertSame($compiled, $route->compile(), '->compile() only compiled the route once if unchanged');
  186. $route->setRequirement('foo', '.*');
  187. $this->assertNotSame($compiled, $route->compile(), '->compile() recompiles if the route was modified');
  188. }
  189. public function testSerialize()
  190. {
  191. $route = new Route('/prefix/{foo}', ['foo' => 'default'], ['foo' => '\d+']);
  192. $serialized = serialize($route);
  193. $unserialized = unserialize($serialized);
  194. $this->assertEquals($route, $unserialized);
  195. $this->assertNotSame($route, $unserialized);
  196. }
  197. public function testInlineDefaultAndRequirement()
  198. {
  199. $this->assertEquals((new Route('/foo/{bar}'))->setDefault('bar', null), new Route('/foo/{bar?}'));
  200. $this->assertEquals((new Route('/foo/{bar}'))->setDefault('bar', 'baz'), new Route('/foo/{bar?baz}'));
  201. $this->assertEquals((new Route('/foo/{bar}'))->setDefault('bar', 'baz<buz>'), new Route('/foo/{bar?baz<buz>}'));
  202. $this->assertEquals((new Route('/foo/{!bar}'))->setDefault('bar', 'baz<buz>'), new Route('/foo/{!bar?baz<buz>}'));
  203. $this->assertEquals((new Route('/foo/{bar}'))->setDefault('bar', 'baz'), new Route('/foo/{bar?}', ['bar' => 'baz']));
  204. $this->assertEquals((new Route('/foo/{bar}'))->setRequirement('bar', '.*'), new Route('/foo/{bar<.*>}'));
  205. $this->assertEquals((new Route('/foo/{bar}'))->setRequirement('bar', '>'), new Route('/foo/{bar<>>}'));
  206. $this->assertEquals((new Route('/foo/{bar}'))->setRequirement('bar', '\d+'), new Route('/foo/{bar<.*>}', [], ['bar' => '\d+']));
  207. $this->assertEquals((new Route('/foo/{bar}'))->setRequirement('bar', '[a-z]{2}'), new Route('/foo/{bar<[a-z]{2}>}'));
  208. $this->assertEquals((new Route('/foo/{!bar}'))->setRequirement('bar', '\d+'), new Route('/foo/{!bar<\d+>}'));
  209. $this->assertEquals((new Route('/foo/{bar}'))->setDefault('bar', null)->setRequirement('bar', '.*'), new Route('/foo/{bar<.*>?}'));
  210. $this->assertEquals((new Route('/foo/{bar}'))->setDefault('bar', '<>')->setRequirement('bar', '>'), new Route('/foo/{bar<>>?<>}'));
  211. $this->assertEquals((new Route('/{foo}/{!bar}'))->setDefaults(['bar' => '<>', 'foo' => '\\'])->setRequirements(['bar' => '\\', 'foo' => '.']), new Route('/{foo<.>?\}/{!bar<\>?<>}'));
  212. $this->assertEquals((new Route('/'))->setHost('{bar}')->setDefault('bar', null), (new Route('/'))->setHost('{bar?}'));
  213. $this->assertEquals((new Route('/'))->setHost('{bar}')->setDefault('bar', 'baz'), (new Route('/'))->setHost('{bar?baz}'));
  214. $this->assertEquals((new Route('/'))->setHost('{bar}')->setDefault('bar', 'baz<buz>'), (new Route('/'))->setHost('{bar?baz<buz>}'));
  215. $this->assertEquals((new Route('/'))->setHost('{bar}')->setDefault('bar', null), (new Route('/', ['bar' => 'baz']))->setHost('{bar?}'));
  216. $this->assertEquals((new Route('/'))->setHost('{bar}')->setRequirement('bar', '.*'), (new Route('/'))->setHost('{bar<.*>}'));
  217. $this->assertEquals((new Route('/'))->setHost('{bar}')->setRequirement('bar', '>'), (new Route('/'))->setHost('{bar<>>}'));
  218. $this->assertEquals((new Route('/'))->setHost('{bar}')->setRequirement('bar', '.*'), (new Route('/', [], ['bar' => '\d+']))->setHost('{bar<.*>}'));
  219. $this->assertEquals((new Route('/'))->setHost('{bar}')->setRequirement('bar', '[a-z]{2}'), (new Route('/'))->setHost('{bar<[a-z]{2}>}'));
  220. $this->assertEquals((new Route('/'))->setHost('{bar}')->setDefault('bar', null)->setRequirement('bar', '.*'), (new Route('/'))->setHost('{bar<.*>?}'));
  221. $this->assertEquals((new Route('/'))->setHost('{bar}')->setDefault('bar', '<>')->setRequirement('bar', '>'), (new Route('/'))->setHost('{bar<>>?<>}'));
  222. }
  223. /**
  224. * Tests that the compiled version is also serialized to prevent the overhead
  225. * of compiling it again after unserialize.
  226. */
  227. public function testSerializeWhenCompiled()
  228. {
  229. $route = new Route('/prefix/{foo}', ['foo' => 'default'], ['foo' => '\d+']);
  230. $route->setHost('{locale}.example.net');
  231. $route->compile();
  232. $serialized = serialize($route);
  233. $unserialized = unserialize($serialized);
  234. $this->assertEquals($route, $unserialized);
  235. $this->assertNotSame($route, $unserialized);
  236. }
  237. /**
  238. * Tests that unserialization does not fail when the compiled Route is of a
  239. * class other than CompiledRoute, such as a subclass of it.
  240. */
  241. public function testSerializeWhenCompiledWithClass()
  242. {
  243. $route = new Route('/', [], [], ['compiler_class' => CustomRouteCompiler::class]);
  244. $this->assertInstanceOf(CustomCompiledRoute::class, $route->compile(), '->compile() returned a proper route');
  245. $serialized = serialize($route);
  246. try {
  247. $unserialized = unserialize($serialized);
  248. $this->assertInstanceOf(CustomCompiledRoute::class, $unserialized->compile(), 'the unserialized route compiled successfully');
  249. } catch (\Exception $e) {
  250. $this->fail('unserializing a route which uses a custom compiled route class');
  251. }
  252. }
  253. /**
  254. * Tests that the serialized representation of a route in one symfony version
  255. * also works in later symfony versions, i.e. the unserialized route is in the
  256. * same state as another, semantically equivalent, route.
  257. */
  258. public function testSerializedRepresentationKeepsWorking()
  259. {
  260. $serialized = 'C:31:"Symfony\Component\Routing\Route":936:{a:8:{s:4:"path";s:13:"/prefix/{foo}";s:4:"host";s:20:"{locale}.example.net";s:8:"defaults";a:1:{s:3:"foo";s:7:"default";}s:12:"requirements";a:1:{s:3:"foo";s:3:"\d+";}s:7:"options";a:1:{s:14:"compiler_class";s:39:"Symfony\Component\Routing\RouteCompiler";}s:7:"schemes";a:0:{}s:7:"methods";a:0:{}s:8:"compiled";C:39:"Symfony\Component\Routing\CompiledRoute":571:{a:8:{s:4:"vars";a:2:{i:0;s:6:"locale";i:1;s:3:"foo";}s:11:"path_prefix";s:7:"/prefix";s:10:"path_regex";s:31:"{^/prefix(?:/(?P<foo>\d+))?$}sD";s:11:"path_tokens";a:2:{i:0;a:4:{i:0;s:8:"variable";i:1;s:1:"/";i:2;s:3:"\d+";i:3;s:3:"foo";}i:1;a:2:{i:0;s:4:"text";i:1;s:7:"/prefix";}}s:9:"path_vars";a:1:{i:0;s:3:"foo";}s:10:"host_regex";s:40:"{^(?P<locale>[^\.]++)\.example\.net$}sDi";s:11:"host_tokens";a:2:{i:0;a:2:{i:0;s:4:"text";i:1;s:12:".example.net";}i:1;a:4:{i:0;s:8:"variable";i:1;s:0:"";i:2;s:7:"[^\.]++";i:3;s:6:"locale";}}s:9:"host_vars";a:1:{i:0;s:6:"locale";}}}}}';
  261. $unserialized = unserialize($serialized);
  262. $route = new Route('/prefix/{foo}', ['foo' => 'default'], ['foo' => '\d+']);
  263. $route->setHost('{locale}.example.net');
  264. $route->compile();
  265. $this->assertEquals($route, $unserialized);
  266. $this->assertNotSame($route, $unserialized);
  267. }
  268. /**
  269. * @dataProvider provideNonLocalizedRoutes
  270. */
  271. public function testLocaleDefaultWithNonLocalizedRoutes(Route $route)
  272. {
  273. $this->assertNotSame('fr', $route->getDefault('_locale'));
  274. $route->setDefault('_locale', 'fr');
  275. $this->assertSame('fr', $route->getDefault('_locale'));
  276. }
  277. /**
  278. * @dataProvider provideLocalizedRoutes
  279. */
  280. public function testLocaleDefaultWithLocalizedRoutes(Route $route)
  281. {
  282. $expected = $route->getDefault('_locale');
  283. $this->assertIsString($expected);
  284. $this->assertNotSame('fr', $expected);
  285. $route->setDefault('_locale', 'fr');
  286. $this->assertSame($expected, $route->getDefault('_locale'));
  287. }
  288. /**
  289. * @dataProvider provideNonLocalizedRoutes
  290. */
  291. public function testLocaleRequirementWithNonLocalizedRoutes(Route $route)
  292. {
  293. $this->assertNotSame('fr', $route->getRequirement('_locale'));
  294. $route->setRequirement('_locale', 'fr');
  295. $this->assertSame('fr', $route->getRequirement('_locale'));
  296. }
  297. /**
  298. * @dataProvider provideLocalizedRoutes
  299. */
  300. public function testLocaleRequirementWithLocalizedRoutes(Route $route)
  301. {
  302. $expected = $route->getRequirement('_locale');
  303. $this->assertIsString($expected);
  304. $this->assertNotSame('fr', $expected);
  305. $route->setRequirement('_locale', 'fr');
  306. $this->assertSame($expected, $route->getRequirement('_locale'));
  307. }
  308. public static function provideNonLocalizedRoutes()
  309. {
  310. return [
  311. [new Route('/foo')],
  312. [(new Route('/foo'))->setDefault('_locale', 'en')],
  313. [(new Route('/foo'))->setDefault('_locale', 'en')->setDefault('_canonical_route', 'foo')],
  314. [(new Route('/foo'))->setDefault('_locale', 'en')->setDefault('_canonical_route', 'foo')->setRequirement('_locale', 'foobar')],
  315. ];
  316. }
  317. public static function provideLocalizedRoutes()
  318. {
  319. return [
  320. [(new Route('/foo'))->setDefault('_locale', 'en')->setDefault('_canonical_route', 'foo')->setRequirement('_locale', 'en')],
  321. ];
  322. }
  323. }