the whole shebang
This commit is contained in:
		
							
								
								
									
										118
									
								
								vendor/symfony/routing/Symfony/Component/Routing/Matcher/ApacheUrlMatcher.php
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										118
									
								
								vendor/symfony/routing/Symfony/Component/Routing/Matcher/ApacheUrlMatcher.php
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,118 @@ | ||||
| <?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\Matcher; | ||||
|  | ||||
| use Symfony\Component\Routing\Exception\MethodNotAllowedException; | ||||
|  | ||||
| /** | ||||
|  * ApacheUrlMatcher matches URL based on Apache mod_rewrite matching (see ApacheMatcherDumper). | ||||
|  * | ||||
|  * @author Fabien Potencier <fabien@symfony.com> | ||||
|  * @author Arnaud Le Blanc <arnaud.lb@gmail.com> | ||||
|  */ | ||||
| class ApacheUrlMatcher extends UrlMatcher | ||||
| { | ||||
|     /** | ||||
|      * Tries to match a URL based on Apache mod_rewrite matching. | ||||
|      * | ||||
|      * Returns false if no route matches the URL. | ||||
|      * | ||||
|      * @param string $pathinfo The pathinfo to be parsed | ||||
|      * | ||||
|      * @return array An array of parameters | ||||
|      * | ||||
|      * @throws MethodNotAllowedException If the current method is not allowed | ||||
|      */ | ||||
|     public function match($pathinfo) | ||||
|     { | ||||
|         $parameters = array(); | ||||
|         $defaults = array(); | ||||
|         $allow = array(); | ||||
|         $route = null; | ||||
|  | ||||
|         foreach ($this->denormalizeValues($_SERVER) as $key => $value) { | ||||
|             $name = $key; | ||||
|  | ||||
|             // skip non-routing variables | ||||
|             // this improves performance when $_SERVER contains many usual | ||||
|             // variables like HTTP_*, DOCUMENT_ROOT, REQUEST_URI, ... | ||||
|             if (false === strpos($name, '_ROUTING_')) { | ||||
|                 continue; | ||||
|             } | ||||
|  | ||||
|             while (0 === strpos($name, 'REDIRECT_')) { | ||||
|                 $name = substr($name, 9); | ||||
|             } | ||||
|  | ||||
|             // expect _ROUTING_<type>_<name> | ||||
|             // or _ROUTING_<type> | ||||
|  | ||||
|             if (0 !== strpos($name, '_ROUTING_')) { | ||||
|                 continue; | ||||
|             } | ||||
|             if (false !== $pos = strpos($name, '_', 9)) { | ||||
|                 $type = substr($name, 9, $pos-9); | ||||
|                 $name = substr($name, $pos+1); | ||||
|             } else { | ||||
|                 $type = substr($name, 9); | ||||
|             } | ||||
|  | ||||
|             if ('param' === $type) { | ||||
|                 if ('' !== $value) { | ||||
|                     $parameters[$name] = $value; | ||||
|                 } | ||||
|             } elseif ('default' === $type) { | ||||
|                 $defaults[$name] = $value; | ||||
|             } elseif ('route' === $type) { | ||||
|                 $route = $value; | ||||
|             } elseif ('allow' === $type) { | ||||
|                 $allow[] = $name; | ||||
|             } | ||||
|  | ||||
|             unset($_SERVER[$key]); | ||||
|         } | ||||
|  | ||||
|         if (null !== $route) { | ||||
|             $parameters['_route'] = $route; | ||||
|  | ||||
|             return $this->mergeDefaults($parameters, $defaults); | ||||
|         } elseif (0 < count($allow)) { | ||||
|             throw new MethodNotAllowedException($allow); | ||||
|         } else { | ||||
|             return parent::match($pathinfo); | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Denormalizes an array of values. | ||||
|      * | ||||
|      * @param string[] $values | ||||
|      * | ||||
|      * @return array | ||||
|      */ | ||||
|     private function denormalizeValues(array $values) | ||||
|     { | ||||
|         $normalizedValues = array(); | ||||
|         foreach ($values as $key => $value) { | ||||
|             if (preg_match('~^(.*)\[(\d+)\]$~', $key, $matches)) { | ||||
|                 if (!isset($normalizedValues[$matches[1]])) { | ||||
|                     $normalizedValues[$matches[1]] = array(); | ||||
|                 } | ||||
|                 $normalizedValues[$matches[1]][(int) $matches[2]] = $value; | ||||
|             } else { | ||||
|                 $normalizedValues[$key] = $value; | ||||
|             } | ||||
|         } | ||||
|  | ||||
|         return $normalizedValues; | ||||
|     } | ||||
| } | ||||
							
								
								
									
										274
									
								
								vendor/symfony/routing/Symfony/Component/Routing/Matcher/Dumper/ApacheMatcherDumper.php
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										274
									
								
								vendor/symfony/routing/Symfony/Component/Routing/Matcher/Dumper/ApacheMatcherDumper.php
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,274 @@ | ||||
| <?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\Matcher\Dumper; | ||||
|  | ||||
| use Symfony\Component\Routing\Route; | ||||
|  | ||||
| /** | ||||
|  * Dumps a set of Apache mod_rewrite rules. | ||||
|  * | ||||
|  * @author Fabien Potencier <fabien@symfony.com> | ||||
|  * @author Kris Wallsmith <kris@symfony.com> | ||||
|  */ | ||||
| class ApacheMatcherDumper extends MatcherDumper | ||||
| { | ||||
|     /** | ||||
|      * Dumps a set of Apache mod_rewrite rules. | ||||
|      * | ||||
|      * Available options: | ||||
|      * | ||||
|      *  * script_name: The script name (app.php by default) | ||||
|      *  * base_uri:    The base URI ("" by default) | ||||
|      * | ||||
|      * @param array $options An array of options | ||||
|      * | ||||
|      * @return string A string to be used as Apache rewrite rules | ||||
|      * | ||||
|      * @throws \LogicException When the route regex is invalid | ||||
|      */ | ||||
|     public function dump(array $options = array()) | ||||
|     { | ||||
|         $options = array_merge(array( | ||||
|             'script_name' => 'app.php', | ||||
|             'base_uri'    => '', | ||||
|         ), $options); | ||||
|  | ||||
|         $options['script_name'] = self::escape($options['script_name'], ' ', '\\'); | ||||
|  | ||||
|         $rules = array("# skip \"real\" requests\nRewriteCond %{REQUEST_FILENAME} -f\nRewriteRule .* - [QSA,L]"); | ||||
|         $methodVars = array(); | ||||
|         $hostRegexUnique = 0; | ||||
|         $prevHostRegex = ''; | ||||
|  | ||||
|         foreach ($this->getRoutes()->all() as $name => $route) { | ||||
|  | ||||
|             $compiledRoute = $route->compile(); | ||||
|             $hostRegex = $compiledRoute->getHostRegex(); | ||||
|  | ||||
|             if (null !== $hostRegex && $prevHostRegex !== $hostRegex) { | ||||
|                 $prevHostRegex = $hostRegex; | ||||
|                 $hostRegexUnique++; | ||||
|  | ||||
|                 $rule = array(); | ||||
|  | ||||
|                 $regex = $this->regexToApacheRegex($hostRegex); | ||||
|                 $regex = self::escape($regex, ' ', '\\'); | ||||
|  | ||||
|                 $rule[] = sprintf('RewriteCond %%{HTTP:Host} %s', $regex); | ||||
|  | ||||
|                 $variables = array(); | ||||
|                 $variables[] = sprintf('E=__ROUTING_host_%s:1', $hostRegexUnique); | ||||
|  | ||||
|                 foreach ($compiledRoute->getHostVariables() as $i => $variable) { | ||||
|                     $variables[] = sprintf('E=__ROUTING_host_%s_%s:%%%d', $hostRegexUnique, $variable, $i+1); | ||||
|                 } | ||||
|  | ||||
|                 $variables = implode(',', $variables); | ||||
|  | ||||
|                 $rule[] = sprintf('RewriteRule .? - [%s]', $variables); | ||||
|  | ||||
|                 $rules[] = implode("\n", $rule); | ||||
|             } | ||||
|  | ||||
|             $rules[] = $this->dumpRoute($name, $route, $options, $hostRegexUnique); | ||||
|  | ||||
|             if ($req = $route->getRequirement('_method')) { | ||||
|                 $methods = explode('|', strtoupper($req)); | ||||
|                 $methodVars = array_merge($methodVars, $methods); | ||||
|             } | ||||
|         } | ||||
|         if (0 < count($methodVars)) { | ||||
|             $rule = array('# 405 Method Not Allowed'); | ||||
|             $methodVars = array_values(array_unique($methodVars)); | ||||
|             if (in_array('GET', $methodVars) && !in_array('HEAD', $methodVars)) { | ||||
|                 $methodVars[] = 'HEAD'; | ||||
|             } | ||||
|             foreach ($methodVars as $i => $methodVar) { | ||||
|                 $rule[] = sprintf('RewriteCond %%{ENV:_ROUTING__allow_%s} =1%s', $methodVar, isset($methodVars[$i + 1]) ? ' [OR]' : ''); | ||||
|             } | ||||
|             $rule[] = sprintf('RewriteRule .* %s [QSA,L]', $options['script_name']); | ||||
|  | ||||
|             $rules[] = implode("\n", $rule); | ||||
|         } | ||||
|  | ||||
|         return implode("\n\n", $rules)."\n"; | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Dumps a single route | ||||
|      * | ||||
|      * @param  string $name Route name | ||||
|      * @param  Route  $route The route | ||||
|      * @param  array  $options Options | ||||
|      * @param  bool   $hostRegexUnique Unique identifier for the host regex | ||||
|      * | ||||
|      * @return string The compiled route | ||||
|      */ | ||||
|     private function dumpRoute($name, $route, array $options, $hostRegexUnique) | ||||
|     { | ||||
|         $compiledRoute = $route->compile(); | ||||
|  | ||||
|         // prepare the apache regex | ||||
|         $regex = $this->regexToApacheRegex($compiledRoute->getRegex()); | ||||
|         $regex = '^'.self::escape(preg_quote($options['base_uri']).substr($regex, 1), ' ', '\\'); | ||||
|  | ||||
|         $methods = $this->getRouteMethods($route); | ||||
|  | ||||
|         $hasTrailingSlash = (!$methods || in_array('HEAD', $methods)) && '/$' === substr($regex, -2) && '^/$' !== $regex; | ||||
|  | ||||
|         $variables = array('E=_ROUTING_route:'.$name); | ||||
|         foreach ($compiledRoute->getHostVariables() as $variable) { | ||||
|             $variables[] = sprintf('E=_ROUTING_param_%s:%%{ENV:__ROUTING_host_%s_%s}', $variable, $hostRegexUnique, $variable); | ||||
|         } | ||||
|         foreach ($compiledRoute->getPathVariables() as $i => $variable) { | ||||
|             $variables[] = 'E=_ROUTING_param_'.$variable.':%'.($i + 1); | ||||
|         } | ||||
|         foreach ($this->normalizeValues($route->getDefaults()) as $key => $value) { | ||||
|             $variables[] = 'E=_ROUTING_default_'.$key.':'.strtr($value, array( | ||||
|                 ':'  => '\\:', | ||||
|                 '='  => '\\=', | ||||
|                 '\\' => '\\\\', | ||||
|                 ' '  => '\\ ', | ||||
|             )); | ||||
|         } | ||||
|         $variables = implode(',', $variables); | ||||
|  | ||||
|         $rule = array("# $name"); | ||||
|  | ||||
|         // method mismatch | ||||
|         if (0 < count($methods)) { | ||||
|             $allow = array(); | ||||
|             foreach ($methods as $method) { | ||||
|                 $allow[] = 'E=_ROUTING_allow_'.$method.':1'; | ||||
|             } | ||||
|  | ||||
|             if ($compiledRoute->getHostRegex()) { | ||||
|                 $rule[] = sprintf("RewriteCond %%{ENV:__ROUTING_host_%s} =1", $hostRegexUnique); | ||||
|             } | ||||
|  | ||||
|             $rule[] = "RewriteCond %{REQUEST_URI} $regex"; | ||||
|             $rule[] = sprintf("RewriteCond %%{REQUEST_METHOD} !^(%s)$ [NC]", implode('|', $methods)); | ||||
|             $rule[] = sprintf('RewriteRule .* - [S=%d,%s]', $hasTrailingSlash ? 2 : 1, implode(',', $allow)); | ||||
|         } | ||||
|  | ||||
|         // redirect with trailing slash appended | ||||
|         if ($hasTrailingSlash) { | ||||
|             if ($compiledRoute->getHostRegex()) { | ||||
|                 $rule[] = sprintf("RewriteCond %%{ENV:__ROUTING_host_%s} =1", $hostRegexUnique); | ||||
|             } | ||||
|  | ||||
|             $rule[] = 'RewriteCond %{REQUEST_URI} '.substr($regex, 0, -2).'$'; | ||||
|             $rule[] = 'RewriteRule .* $0/ [QSA,L,R=301]'; | ||||
|         } | ||||
|  | ||||
|         // the main rule | ||||
|  | ||||
|         if ($compiledRoute->getHostRegex()) { | ||||
|             $rule[] = sprintf("RewriteCond %%{ENV:__ROUTING_host_%s} =1", $hostRegexUnique); | ||||
|         } | ||||
|  | ||||
|         $rule[] = "RewriteCond %{REQUEST_URI} $regex"; | ||||
|         $rule[] = "RewriteRule .* {$options['script_name']} [QSA,L,$variables]"; | ||||
|  | ||||
|         return implode("\n", $rule); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Returns methods allowed for a route | ||||
|      * | ||||
|      * @param Route  $route The route | ||||
|      * | ||||
|      * @return array The methods | ||||
|      */ | ||||
|     private function getRouteMethods(Route $route) | ||||
|     { | ||||
|         $methods = array(); | ||||
|         if ($req = $route->getRequirement('_method')) { | ||||
|             $methods = explode('|', strtoupper($req)); | ||||
|             // GET and HEAD are equivalent | ||||
|             if (in_array('GET', $methods) && !in_array('HEAD', $methods)) { | ||||
|                 $methods[] = 'HEAD'; | ||||
|             } | ||||
|         } | ||||
|  | ||||
|         return $methods; | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Converts a regex to make it suitable for mod_rewrite | ||||
|      * | ||||
|      * @param string  $regex The regex | ||||
|      * | ||||
|      * @return string The converted regex | ||||
|      */ | ||||
|     private function regexToApacheRegex($regex) | ||||
|     { | ||||
|         $regexPatternEnd = strrpos($regex, $regex[0]); | ||||
|  | ||||
|         return preg_replace('/\?P<.+?>/', '', substr($regex, 1, $regexPatternEnd - 1)); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Escapes a string. | ||||
|      * | ||||
|      * @param string $string The string to be escaped | ||||
|      * @param string $char   The character to be escaped | ||||
|      * @param string $with   The character to be used for escaping | ||||
|      * | ||||
|      * @return string The escaped string | ||||
|      */ | ||||
|     private static function escape($string, $char, $with) | ||||
|     { | ||||
|         $escaped = false; | ||||
|         $output = ''; | ||||
|         foreach (str_split($string) as $symbol) { | ||||
|             if ($escaped) { | ||||
|                 $output .= $symbol; | ||||
|                 $escaped = false; | ||||
|                 continue; | ||||
|             } | ||||
|             if ($symbol === $char) { | ||||
|                 $output .= $with.$char; | ||||
|                 continue; | ||||
|             } | ||||
|             if ($symbol === $with) { | ||||
|                 $escaped = true; | ||||
|             } | ||||
|             $output .= $symbol; | ||||
|         } | ||||
|  | ||||
|         return $output; | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Normalizes an array of values. | ||||
|      * | ||||
|      * @param array $values | ||||
|      * | ||||
|      * @return string[] | ||||
|      */ | ||||
|     private function normalizeValues(array $values) | ||||
|     { | ||||
|         $normalizedValues = array(); | ||||
|         foreach ($values as $key => $value) { | ||||
|             if (is_array($value)) { | ||||
|                 foreach ($value as $index => $bit) { | ||||
|                     $normalizedValues[sprintf('%s[%s]', $key, $index)] = $bit; | ||||
|                 } | ||||
|             } else { | ||||
|                 $normalizedValues[$key] = (string) $value; | ||||
|             } | ||||
|         } | ||||
|  | ||||
|         return $normalizedValues; | ||||
|     } | ||||
| } | ||||
							
								
								
									
										159
									
								
								vendor/symfony/routing/Symfony/Component/Routing/Matcher/Dumper/DumperCollection.php
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										159
									
								
								vendor/symfony/routing/Symfony/Component/Routing/Matcher/Dumper/DumperCollection.php
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,159 @@ | ||||
| <?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\Matcher\Dumper; | ||||
|  | ||||
| /** | ||||
|  * Collection of routes. | ||||
|  * | ||||
|  * @author Arnaud Le Blanc <arnaud.lb@gmail.com> | ||||
|  */ | ||||
| class DumperCollection implements \IteratorAggregate | ||||
| { | ||||
|     /** | ||||
|      * @var DumperCollection|null | ||||
|      */ | ||||
|     private $parent; | ||||
|  | ||||
|     /** | ||||
|      * @var (DumperCollection|DumperRoute)[] | ||||
|      */ | ||||
|     private $children = array(); | ||||
|  | ||||
|     /** | ||||
|      * @var array | ||||
|      */ | ||||
|     private $attributes = array(); | ||||
|  | ||||
|     /** | ||||
|      * Returns the children routes and collections. | ||||
|      * | ||||
|      * @return (DumperCollection|DumperRoute)[] Array of DumperCollection|DumperRoute | ||||
|      */ | ||||
|     public function all() | ||||
|     { | ||||
|         return $this->children; | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Adds a route or collection | ||||
|      * | ||||
|      * @param DumperRoute|DumperCollection The route or collection | ||||
|      */ | ||||
|     public function add($child) | ||||
|     { | ||||
|         if ($child instanceof DumperCollection) { | ||||
|             $child->setParent($this); | ||||
|         } | ||||
|         $this->children[] = $child; | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Sets children. | ||||
|      * | ||||
|      * @param array $children The children | ||||
|      */ | ||||
|     public function setAll(array $children) | ||||
|     { | ||||
|         foreach ($children as $child) { | ||||
|             if ($child instanceof DumperCollection) { | ||||
|                 $child->setParent($this); | ||||
|             } | ||||
|         } | ||||
|         $this->children = $children; | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Returns an iterator over the children. | ||||
|      * | ||||
|      * @return \Iterator The iterator | ||||
|      */ | ||||
|     public function getIterator() | ||||
|     { | ||||
|         return new \ArrayIterator($this->children); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Returns the root of the collection. | ||||
|      * | ||||
|      * @return DumperCollection The root collection | ||||
|      */ | ||||
|     public function getRoot() | ||||
|     { | ||||
|         return (null !== $this->parent) ? $this->parent->getRoot() : $this; | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Returns the parent collection. | ||||
|      * | ||||
|      * @return DumperCollection|null The parent collection or null if the collection has no parent | ||||
|      */ | ||||
|     protected function getParent() | ||||
|     { | ||||
|         return $this->parent; | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Sets the parent collection. | ||||
|      * | ||||
|      * @param DumperCollection $parent The parent collection | ||||
|      */ | ||||
|     protected function setParent(DumperCollection $parent) | ||||
|     { | ||||
|         $this->parent = $parent; | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Returns true if the attribute is defined. | ||||
|      * | ||||
|      * @param string $name The attribute name | ||||
|      * | ||||
|      * @return Boolean true if the attribute is defined, false otherwise | ||||
|      */ | ||||
|     public function hasAttribute($name) | ||||
|     { | ||||
|         return array_key_exists($name, $this->attributes); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Returns an attribute by name. | ||||
|      * | ||||
|      * @param string $name    The attribute name | ||||
|      * @param mixed  $default Default value is the attribute doesn't exist | ||||
|      * | ||||
|      * @return mixed The attribute value | ||||
|      */ | ||||
|     public function getAttribute($name, $default = null) | ||||
|     { | ||||
|         return $this->hasAttribute($name) ? $this->attributes[$name] : $default; | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Sets an attribute by name. | ||||
|      * | ||||
|      * @param string $name  The attribute name | ||||
|      * @param mixed  $value The attribute value | ||||
|      */ | ||||
|     public function setAttribute($name, $value) | ||||
|     { | ||||
|         $this->attributes[$name] = $value; | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Sets multiple attributes. | ||||
|      * | ||||
|      * @param array $attributes The attributes | ||||
|      */ | ||||
|     public function setAttributes($attributes) | ||||
|     { | ||||
|         $this->attributes = $attributes; | ||||
|     } | ||||
| } | ||||
							
								
								
									
										108
									
								
								vendor/symfony/routing/Symfony/Component/Routing/Matcher/Dumper/DumperPrefixCollection.php
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										108
									
								
								vendor/symfony/routing/Symfony/Component/Routing/Matcher/Dumper/DumperPrefixCollection.php
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,108 @@ | ||||
| <?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\Matcher\Dumper; | ||||
|  | ||||
| /** | ||||
|  * Prefix tree of routes preserving routes order. | ||||
|  * | ||||
|  * @author Arnaud Le Blanc <arnaud.lb@gmail.com> | ||||
|  */ | ||||
| class DumperPrefixCollection extends DumperCollection | ||||
| { | ||||
|     /** | ||||
|      * @var string | ||||
|      */ | ||||
|     private $prefix = ''; | ||||
|  | ||||
|     /** | ||||
|      * Returns the prefix. | ||||
|      * | ||||
|      * @return string The prefix | ||||
|      */ | ||||
|     public function getPrefix() | ||||
|     { | ||||
|         return $this->prefix; | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Sets the prefix. | ||||
|      * | ||||
|      * @param string $prefix The prefix | ||||
|      */ | ||||
|     public function setPrefix($prefix) | ||||
|     { | ||||
|         $this->prefix = $prefix; | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Adds a route in the tree. | ||||
|      * | ||||
|      * @param DumperRoute $route The route | ||||
|      * | ||||
|      * @return DumperPrefixCollection The node the route was added to | ||||
|      * | ||||
|      * @throws \LogicException | ||||
|      */ | ||||
|     public function addPrefixRoute(DumperRoute $route) | ||||
|     { | ||||
|         $prefix = $route->getRoute()->compile()->getStaticPrefix(); | ||||
|  | ||||
|         // Same prefix, add to current leave | ||||
|         if ($this->prefix === $prefix) { | ||||
|             $this->add($route); | ||||
|  | ||||
|             return $this; | ||||
|         } | ||||
|  | ||||
|         // Prefix starts with route's prefix | ||||
|         if ('' === $this->prefix || 0 === strpos($prefix, $this->prefix)) { | ||||
|             $collection = new DumperPrefixCollection(); | ||||
|             $collection->setPrefix(substr($prefix, 0, strlen($this->prefix)+1)); | ||||
|             $this->add($collection); | ||||
|  | ||||
|             return $collection->addPrefixRoute($route); | ||||
|         } | ||||
|  | ||||
|         // No match, fallback to parent (recursively) | ||||
|  | ||||
|         if (null === $parent = $this->getParent()) { | ||||
|             throw new \LogicException("The collection root must not have a prefix"); | ||||
|         } | ||||
|  | ||||
|         return $parent->addPrefixRoute($route); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Merges nodes whose prefix ends with a slash | ||||
|      * | ||||
|      * Children of a node whose prefix ends with a slash are moved to the parent node | ||||
|      */ | ||||
|     public function mergeSlashNodes() | ||||
|     { | ||||
|         $children = array(); | ||||
|  | ||||
|         foreach ($this as $child) { | ||||
|             if ($child instanceof self) { | ||||
|                 $child->mergeSlashNodes(); | ||||
|                 if ('/' === substr($child->prefix, -1)) { | ||||
|                     $children = array_merge($children, $child->all()); | ||||
|                 } else { | ||||
|                     $children[] = $child; | ||||
|                 } | ||||
|             } else { | ||||
|                 $children[] = $child; | ||||
|             } | ||||
|         } | ||||
|  | ||||
|         $this->setAll($children); | ||||
|     } | ||||
| } | ||||
							
								
								
									
										64
									
								
								vendor/symfony/routing/Symfony/Component/Routing/Matcher/Dumper/DumperRoute.php
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										64
									
								
								vendor/symfony/routing/Symfony/Component/Routing/Matcher/Dumper/DumperRoute.php
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,64 @@ | ||||
| <?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\Matcher\Dumper; | ||||
|  | ||||
| use Symfony\Component\Routing\Route; | ||||
|  | ||||
| /** | ||||
|  * Container for a Route. | ||||
|  * | ||||
|  * @author Arnaud Le Blanc <arnaud.lb@gmail.com> | ||||
|  */ | ||||
| class DumperRoute | ||||
| { | ||||
|     /** | ||||
|      * @var string | ||||
|      */ | ||||
|     private $name; | ||||
|  | ||||
|     /** | ||||
|      * @var Route | ||||
|      */ | ||||
|     private $route; | ||||
|  | ||||
|     /** | ||||
|      * Constructor. | ||||
|      * | ||||
|      * @param string $name  The route name | ||||
|      * @param Route  $route The route | ||||
|      */ | ||||
|     public function __construct($name, Route $route) | ||||
|     { | ||||
|         $this->name = $name; | ||||
|         $this->route = $route; | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Returns the route name. | ||||
|      * | ||||
|      * @return string The route name | ||||
|      */ | ||||
|     public function getName() | ||||
|     { | ||||
|         return $this->name; | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Returns the route. | ||||
|      * | ||||
|      * @return Route The route | ||||
|      */ | ||||
|     public function getRoute() | ||||
|     { | ||||
|         return $this->route; | ||||
|     } | ||||
| } | ||||
							
								
								
									
										45
									
								
								vendor/symfony/routing/Symfony/Component/Routing/Matcher/Dumper/MatcherDumper.php
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										45
									
								
								vendor/symfony/routing/Symfony/Component/Routing/Matcher/Dumper/MatcherDumper.php
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,45 @@ | ||||
| <?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\Matcher\Dumper; | ||||
|  | ||||
| use Symfony\Component\Routing\RouteCollection; | ||||
|  | ||||
| /** | ||||
|  * MatcherDumper is the abstract class for all built-in matcher dumpers. | ||||
|  * | ||||
|  * @author Fabien Potencier <fabien@symfony.com> | ||||
|  */ | ||||
| abstract class MatcherDumper implements MatcherDumperInterface | ||||
| { | ||||
|     /** | ||||
|      * @var RouteCollection | ||||
|      */ | ||||
|     private $routes; | ||||
|  | ||||
|     /** | ||||
|      * Constructor. | ||||
|      * | ||||
|      * @param RouteCollection $routes The RouteCollection to dump | ||||
|      */ | ||||
|     public function __construct(RouteCollection $routes) | ||||
|     { | ||||
|         $this->routes = $routes; | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * {@inheritdoc} | ||||
|      */ | ||||
|     public function getRoutes() | ||||
|     { | ||||
|         return $this->routes; | ||||
|     } | ||||
| } | ||||
							
								
								
									
										37
									
								
								vendor/symfony/routing/Symfony/Component/Routing/Matcher/Dumper/MatcherDumperInterface.php
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										37
									
								
								vendor/symfony/routing/Symfony/Component/Routing/Matcher/Dumper/MatcherDumperInterface.php
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,37 @@ | ||||
| <?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\Matcher\Dumper; | ||||
|  | ||||
| /** | ||||
|  * MatcherDumperInterface is the interface that all matcher dumper classes must implement. | ||||
|  * | ||||
|  * @author Fabien Potencier <fabien@symfony.com> | ||||
|  */ | ||||
| interface MatcherDumperInterface | ||||
| { | ||||
|     /** | ||||
|      * Dumps a set of routes to a string representation of executable code | ||||
|      * that can then be used to match a request against these routes. | ||||
|      * | ||||
|      * @param array $options An array of options | ||||
|      * | ||||
|      * @return string Executable code | ||||
|      */ | ||||
|     public function dump(array $options = array()); | ||||
|  | ||||
|     /** | ||||
|      * Gets the routes to dump. | ||||
|      * | ||||
|      * @return RouteCollection A RouteCollection instance | ||||
|      */ | ||||
|     public function getRoutes(); | ||||
| } | ||||
							
								
								
									
										378
									
								
								vendor/symfony/routing/Symfony/Component/Routing/Matcher/Dumper/PhpMatcherDumper.php
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										378
									
								
								vendor/symfony/routing/Symfony/Component/Routing/Matcher/Dumper/PhpMatcherDumper.php
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,378 @@ | ||||
| <?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\Matcher\Dumper; | ||||
|  | ||||
| use Symfony\Component\Routing\Route; | ||||
| use Symfony\Component\Routing\RouteCollection; | ||||
|  | ||||
| /** | ||||
|  * PhpMatcherDumper creates a PHP class able to match URLs for a given set of routes. | ||||
|  * | ||||
|  * @author Fabien Potencier <fabien@symfony.com> | ||||
|  * @author Tobias Schultze <http://tobion.de> | ||||
|  * @author Arnaud Le Blanc <arnaud.lb@gmail.com> | ||||
|  */ | ||||
| class PhpMatcherDumper extends MatcherDumper | ||||
| { | ||||
|     /** | ||||
|      * Dumps a set of routes to a PHP class. | ||||
|      * | ||||
|      * Available options: | ||||
|      * | ||||
|      *  * class:      The class name | ||||
|      *  * base_class: The base class name | ||||
|      * | ||||
|      * @param array $options An array of options | ||||
|      * | ||||
|      * @return string A PHP class representing the matcher class | ||||
|      */ | ||||
|     public function dump(array $options = array()) | ||||
|     { | ||||
|         $options = array_replace(array( | ||||
|             'class'      => 'ProjectUrlMatcher', | ||||
|             'base_class' => 'Symfony\\Component\\Routing\\Matcher\\UrlMatcher', | ||||
|         ), $options); | ||||
|  | ||||
|         // trailing slash support is only enabled if we know how to redirect the user | ||||
|         $interfaces = class_implements($options['base_class']); | ||||
|         $supportsRedirections = isset($interfaces['Symfony\\Component\\Routing\\Matcher\\RedirectableUrlMatcherInterface']); | ||||
|  | ||||
|         return <<<EOF | ||||
| <?php | ||||
|  | ||||
| use Symfony\Component\Routing\Exception\MethodNotAllowedException; | ||||
| use Symfony\Component\Routing\Exception\ResourceNotFoundException; | ||||
| use Symfony\Component\Routing\RequestContext; | ||||
|  | ||||
| /** | ||||
|  * {$options['class']} | ||||
|  * | ||||
|  * This class has been auto-generated | ||||
|  * by the Symfony Routing Component. | ||||
|  */ | ||||
| class {$options['class']} extends {$options['base_class']} | ||||
| { | ||||
|     /** | ||||
|      * Constructor. | ||||
|      */ | ||||
|     public function __construct(RequestContext \$context) | ||||
|     { | ||||
|         \$this->context = \$context; | ||||
|     } | ||||
|  | ||||
| {$this->generateMatchMethod($supportsRedirections)} | ||||
| } | ||||
|  | ||||
| EOF; | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Generates the code for the match method implementing UrlMatcherInterface. | ||||
|      * | ||||
|      * @param Boolean $supportsRedirections Whether redirections are supported by the base class | ||||
|      * | ||||
|      * @return string Match method as PHP code | ||||
|      */ | ||||
|     private function generateMatchMethod($supportsRedirections) | ||||
|     { | ||||
|         $code = rtrim($this->compileRoutes($this->getRoutes(), $supportsRedirections), "\n"); | ||||
|  | ||||
|         return <<<EOF | ||||
|     public function match(\$pathinfo) | ||||
|     { | ||||
|         \$allow = array(); | ||||
|         \$pathinfo = rawurldecode(\$pathinfo); | ||||
|  | ||||
| $code | ||||
|  | ||||
|         throw 0 < count(\$allow) ? new MethodNotAllowedException(array_unique(\$allow)) : new ResourceNotFoundException(); | ||||
|     } | ||||
| EOF; | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Generates PHP code to match a RouteCollection with all its routes. | ||||
|      * | ||||
|      * @param RouteCollection $routes               A RouteCollection instance | ||||
|      * @param Boolean         $supportsRedirections Whether redirections are supported by the base class | ||||
|      * | ||||
|      * @return string PHP code | ||||
|      */ | ||||
|     private function compileRoutes(RouteCollection $routes, $supportsRedirections) | ||||
|     { | ||||
|         $fetchedHost = false; | ||||
|  | ||||
|         $groups = $this->groupRoutesByHostRegex($routes); | ||||
|         $code = ''; | ||||
|  | ||||
|         foreach ($groups as $collection) { | ||||
|             if (null !== $regex = $collection->getAttribute('host_regex')) { | ||||
|                 if (!$fetchedHost) { | ||||
|                     $code .= "        \$host = \$this->context->getHost();\n\n"; | ||||
|                     $fetchedHost = true; | ||||
|                 } | ||||
|  | ||||
|                 $code .= sprintf("        if (preg_match(%s, \$host, \$hostMatches)) {\n", var_export($regex, true)); | ||||
|             } | ||||
|  | ||||
|             $tree = $this->buildPrefixTree($collection); | ||||
|             $groupCode = $this->compilePrefixRoutes($tree, $supportsRedirections); | ||||
|  | ||||
|             if (null !== $regex) { | ||||
|                 // apply extra indention at each line (except empty ones) | ||||
|                 $groupCode = preg_replace('/^.{2,}$/m', '    $0', $groupCode); | ||||
|                 $code .= $groupCode; | ||||
|                 $code .= "        }\n\n"; | ||||
|             } else { | ||||
|                 $code .= $groupCode; | ||||
|             } | ||||
|         } | ||||
|  | ||||
|         return $code; | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Generates PHP code recursively to match a tree of routes | ||||
|      * | ||||
|      * @param DumperPrefixCollection $collection           A DumperPrefixCollection instance | ||||
|      * @param Boolean                $supportsRedirections Whether redirections are supported by the base class | ||||
|      * @param string                 $parentPrefix         Prefix of the parent collection | ||||
|      * | ||||
|      * @return string PHP code | ||||
|      */ | ||||
|     private function compilePrefixRoutes(DumperPrefixCollection $collection, $supportsRedirections, $parentPrefix = '') | ||||
|     { | ||||
|         $code = ''; | ||||
|         $prefix = $collection->getPrefix(); | ||||
|         $optimizable = 1 < strlen($prefix) && 1 < count($collection->all()); | ||||
|         $optimizedPrefix = $parentPrefix; | ||||
|  | ||||
|         if ($optimizable) { | ||||
|             $optimizedPrefix = $prefix; | ||||
|  | ||||
|             $code .= sprintf("    if (0 === strpos(\$pathinfo, %s)) {\n", var_export($prefix, true)); | ||||
|         } | ||||
|  | ||||
|         foreach ($collection as $route) { | ||||
|             if ($route instanceof DumperCollection) { | ||||
|                 $code .= $this->compilePrefixRoutes($route, $supportsRedirections, $optimizedPrefix); | ||||
|             } else { | ||||
|                 $code .= $this->compileRoute($route->getRoute(), $route->getName(), $supportsRedirections, $optimizedPrefix)."\n"; | ||||
|             } | ||||
|         } | ||||
|  | ||||
|         if ($optimizable) { | ||||
|             $code .= "    }\n\n"; | ||||
|             // apply extra indention at each line (except empty ones) | ||||
|             $code = preg_replace('/^.{2,}$/m', '    $0', $code); | ||||
|         } | ||||
|  | ||||
|         return $code; | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Compiles a single Route to PHP code used to match it against the path info. | ||||
|      * | ||||
|      * @param Route       $route                A Route instance | ||||
|      * @param string      $name                 The name of the Route | ||||
|      * @param Boolean     $supportsRedirections Whether redirections are supported by the base class | ||||
|      * @param string|null $parentPrefix         The prefix of the parent collection used to optimize the code | ||||
|      * | ||||
|      * @return string PHP code | ||||
|      * | ||||
|      * @throws \LogicException | ||||
|      */ | ||||
|     private function compileRoute(Route $route, $name, $supportsRedirections, $parentPrefix = null) | ||||
|     { | ||||
|         $code = ''; | ||||
|         $compiledRoute = $route->compile(); | ||||
|         $conditions = array(); | ||||
|         $hasTrailingSlash = false; | ||||
|         $matches = false; | ||||
|         $hostMatches = false; | ||||
|         $methods = array(); | ||||
|  | ||||
|         if ($req = $route->getRequirement('_method')) { | ||||
|             $methods = explode('|', strtoupper($req)); | ||||
|             // GET and HEAD are equivalent | ||||
|             if (in_array('GET', $methods) && !in_array('HEAD', $methods)) { | ||||
|                 $methods[] = 'HEAD'; | ||||
|             } | ||||
|         } | ||||
|  | ||||
|         $supportsTrailingSlash = $supportsRedirections && (!$methods || in_array('HEAD', $methods)); | ||||
|  | ||||
|         if (!count($compiledRoute->getPathVariables()) && false !== preg_match('#^(.)\^(?P<url>.*?)\$\1#', $compiledRoute->getRegex(), $m)) { | ||||
|             if ($supportsTrailingSlash && substr($m['url'], -1) === '/') { | ||||
|                 $conditions[] = sprintf("rtrim(\$pathinfo, '/') === %s", var_export(rtrim(str_replace('\\', '', $m['url']), '/'), true)); | ||||
|                 $hasTrailingSlash = true; | ||||
|             } else { | ||||
|                 $conditions[] = sprintf("\$pathinfo === %s", var_export(str_replace('\\', '', $m['url']), true)); | ||||
|             } | ||||
|         } else { | ||||
|             if ($compiledRoute->getStaticPrefix() && $compiledRoute->getStaticPrefix() !== $parentPrefix) { | ||||
|                 $conditions[] = sprintf("0 === strpos(\$pathinfo, %s)", var_export($compiledRoute->getStaticPrefix(), true)); | ||||
|             } | ||||
|  | ||||
|             $regex = $compiledRoute->getRegex(); | ||||
|             if ($supportsTrailingSlash && $pos = strpos($regex, '/$')) { | ||||
|                 $regex = substr($regex, 0, $pos).'/?$'.substr($regex, $pos + 2); | ||||
|                 $hasTrailingSlash = true; | ||||
|             } | ||||
|             $conditions[] = sprintf("preg_match(%s, \$pathinfo, \$matches)", var_export($regex, true)); | ||||
|  | ||||
|             $matches = true; | ||||
|         } | ||||
|  | ||||
|         if ($compiledRoute->getHostVariables()) { | ||||
|             $hostMatches = true; | ||||
|         } | ||||
|  | ||||
|         $conditions = implode(' && ', $conditions); | ||||
|  | ||||
|         $code .= <<<EOF | ||||
|         // $name | ||||
|         if ($conditions) { | ||||
|  | ||||
| EOF; | ||||
|  | ||||
|         if ($methods) { | ||||
|             $gotoname = 'not_'.preg_replace('/[^A-Za-z0-9_]/', '', $name); | ||||
|  | ||||
|             if (1 === count($methods)) { | ||||
|                 $code .= <<<EOF | ||||
|             if (\$this->context->getMethod() != '$methods[0]') { | ||||
|                 \$allow[] = '$methods[0]'; | ||||
|                 goto $gotoname; | ||||
|             } | ||||
|  | ||||
|  | ||||
| EOF; | ||||
|             } else { | ||||
|                 $methods = implode("', '", $methods); | ||||
|                 $code .= <<<EOF | ||||
|             if (!in_array(\$this->context->getMethod(), array('$methods'))) { | ||||
|                 \$allow = array_merge(\$allow, array('$methods')); | ||||
|                 goto $gotoname; | ||||
|             } | ||||
|  | ||||
|  | ||||
| EOF; | ||||
|             } | ||||
|         } | ||||
|  | ||||
|         if ($hasTrailingSlash) { | ||||
|             $code .= <<<EOF | ||||
|             if (substr(\$pathinfo, -1) !== '/') { | ||||
|                 return \$this->redirect(\$pathinfo.'/', '$name'); | ||||
|             } | ||||
|  | ||||
|  | ||||
| EOF; | ||||
|         } | ||||
|  | ||||
|         if ($scheme = $route->getRequirement('_scheme')) { | ||||
|             if (!$supportsRedirections) { | ||||
|                 throw new \LogicException('The "_scheme" requirement is only supported for URL matchers that implement RedirectableUrlMatcherInterface.'); | ||||
|             } | ||||
|  | ||||
|             $code .= <<<EOF | ||||
|             if (\$this->context->getScheme() !== '$scheme') { | ||||
|                 return \$this->redirect(\$pathinfo, '$name', '$scheme'); | ||||
|             } | ||||
|  | ||||
|  | ||||
| EOF; | ||||
|         } | ||||
|  | ||||
|         // optimize parameters array | ||||
|         if ($matches || $hostMatches) { | ||||
|             $vars = array(); | ||||
|             if ($hostMatches) { | ||||
|                 $vars[] = '$hostMatches'; | ||||
|             } | ||||
|             if ($matches) { | ||||
|                 $vars[] = '$matches'; | ||||
|             } | ||||
|             $vars[] = "array('_route' => '$name')"; | ||||
|  | ||||
|             $code .= sprintf("            return \$this->mergeDefaults(array_replace(%s), %s);\n" | ||||
|                 , implode(', ', $vars), str_replace("\n", '', var_export($route->getDefaults(), true))); | ||||
|  | ||||
|         } elseif ($route->getDefaults()) { | ||||
|             $code .= sprintf("            return %s;\n", str_replace("\n", '', var_export(array_replace($route->getDefaults(), array('_route' => $name)), true))); | ||||
|         } else { | ||||
|             $code .= sprintf("            return array('_route' => '%s');\n", $name); | ||||
|         } | ||||
|         $code .= "        }\n"; | ||||
|  | ||||
|         if ($methods) { | ||||
|             $code .= "        $gotoname:\n"; | ||||
|         } | ||||
|  | ||||
|         return $code; | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Groups consecutive routes having the same host regex. | ||||
|      * | ||||
|      * The result is a collection of collections of routes having the same host regex. | ||||
|      * | ||||
|      * @param RouteCollection $routes A flat RouteCollection | ||||
|      * | ||||
|      * @return DumperCollection A collection with routes grouped by host regex in sub-collections | ||||
|      */ | ||||
|     private function groupRoutesByHostRegex(RouteCollection $routes) | ||||
|     { | ||||
|         $groups = new DumperCollection(); | ||||
|  | ||||
|         $currentGroup = new DumperCollection(); | ||||
|         $currentGroup->setAttribute('host_regex', null); | ||||
|         $groups->add($currentGroup); | ||||
|  | ||||
|         foreach ($routes as $name => $route) { | ||||
|             $hostRegex = $route->compile()->getHostRegex(); | ||||
|             if ($currentGroup->getAttribute('host_regex') !== $hostRegex) { | ||||
|                 $currentGroup = new DumperCollection(); | ||||
|                 $currentGroup->setAttribute('host_regex', $hostRegex); | ||||
|                 $groups->add($currentGroup); | ||||
|             } | ||||
|             $currentGroup->add(new DumperRoute($name, $route)); | ||||
|         } | ||||
|  | ||||
|         return $groups; | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Organizes the routes into a prefix tree. | ||||
|      * | ||||
|      * Routes order is preserved such that traversing the tree will traverse the | ||||
|      * routes in the origin order. | ||||
|      * | ||||
|      * @param DumperCollection $collection A collection of routes | ||||
|      * | ||||
|      * @return DumperPrefixCollection | ||||
|      */ | ||||
|     private function buildPrefixTree(DumperCollection $collection) | ||||
|     { | ||||
|         $tree = new DumperPrefixCollection(); | ||||
|         $current = $tree; | ||||
|  | ||||
|         foreach ($collection as $route) { | ||||
|             $current = $current->addPrefixRoute($route); | ||||
|         } | ||||
|  | ||||
|         $tree->mergeSlashNodes(); | ||||
|  | ||||
|         return $tree; | ||||
|     } | ||||
| } | ||||
							
								
								
									
										61
									
								
								vendor/symfony/routing/Symfony/Component/Routing/Matcher/RedirectableUrlMatcher.php
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										61
									
								
								vendor/symfony/routing/Symfony/Component/Routing/Matcher/RedirectableUrlMatcher.php
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,61 @@ | ||||
| <?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\Matcher; | ||||
|  | ||||
| use Symfony\Component\Routing\Exception\ResourceNotFoundException; | ||||
| use Symfony\Component\Routing\Route; | ||||
|  | ||||
| /** | ||||
|  * @author Fabien Potencier <fabien@symfony.com> | ||||
|  * | ||||
|  * @api | ||||
|  */ | ||||
| abstract class RedirectableUrlMatcher extends UrlMatcher implements RedirectableUrlMatcherInterface | ||||
| { | ||||
|     /** | ||||
|      * {@inheritdoc} | ||||
|      */ | ||||
|     public function match($pathinfo) | ||||
|     { | ||||
|         try { | ||||
|             $parameters = parent::match($pathinfo); | ||||
|         } catch (ResourceNotFoundException $e) { | ||||
|             if ('/' === substr($pathinfo, -1) || !in_array($this->context->getMethod(), array('HEAD', 'GET'))) { | ||||
|                 throw $e; | ||||
|             } | ||||
|  | ||||
|             try { | ||||
|                 parent::match($pathinfo.'/'); | ||||
|  | ||||
|                 return $this->redirect($pathinfo.'/', null); | ||||
|             } catch (ResourceNotFoundException $e2) { | ||||
|                 throw $e; | ||||
|             } | ||||
|         } | ||||
|  | ||||
|         return $parameters; | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * {@inheritDoc} | ||||
|      */ | ||||
|     protected function handleRouteRequirements($pathinfo, $name, Route $route) | ||||
|     { | ||||
|         // check HTTP scheme requirement | ||||
|         $scheme = $route->getRequirement('_scheme'); | ||||
|         if ($scheme && $this->context->getScheme() !== $scheme) { | ||||
|             return array(self::ROUTE_MATCH, $this->redirect($pathinfo, $name, $scheme)); | ||||
|         } | ||||
|  | ||||
|         return array(self::REQUIREMENT_MATCH, null); | ||||
|     } | ||||
| } | ||||
							
								
								
									
										35
									
								
								vendor/symfony/routing/Symfony/Component/Routing/Matcher/RedirectableUrlMatcherInterface.php
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										35
									
								
								vendor/symfony/routing/Symfony/Component/Routing/Matcher/RedirectableUrlMatcherInterface.php
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,35 @@ | ||||
| <?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\Matcher; | ||||
|  | ||||
| /** | ||||
|  * RedirectableUrlMatcherInterface knows how to redirect the user. | ||||
|  * | ||||
|  * @author Fabien Potencier <fabien@symfony.com> | ||||
|  * | ||||
|  * @api | ||||
|  */ | ||||
| interface RedirectableUrlMatcherInterface | ||||
| { | ||||
|     /** | ||||
|      * Redirects the user to another URL. | ||||
|      * | ||||
|      * @param string      $path   The path info to redirect to. | ||||
|      * @param string      $route  The route name that matched | ||||
|      * @param string|null $scheme The URL scheme (null to keep the current one) | ||||
|      * | ||||
|      * @return array An array of parameters | ||||
|      * | ||||
|      * @api | ||||
|      */ | ||||
|     public function redirect($path, $route, $scheme = null); | ||||
| } | ||||
							
								
								
									
										39
									
								
								vendor/symfony/routing/Symfony/Component/Routing/Matcher/RequestMatcherInterface.php
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										39
									
								
								vendor/symfony/routing/Symfony/Component/Routing/Matcher/RequestMatcherInterface.php
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,39 @@ | ||||
| <?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\Matcher; | ||||
|  | ||||
| use Symfony\Component\HttpFoundation\Request; | ||||
| use Symfony\Component\Routing\Exception\ResourceNotFoundException; | ||||
| use Symfony\Component\Routing\Exception\MethodNotAllowedException; | ||||
|  | ||||
| /** | ||||
|  * RequestMatcherInterface is the interface that all request matcher classes must implement. | ||||
|  * | ||||
|  * @author Fabien Potencier <fabien@symfony.com> | ||||
|  */ | ||||
| interface RequestMatcherInterface | ||||
| { | ||||
|     /** | ||||
|      * Tries to match a request with a set of routes. | ||||
|      * | ||||
|      * If the matcher can not find information, it must throw one of the exceptions documented | ||||
|      * below. | ||||
|      * | ||||
|      * @param Request $request The request to match | ||||
|      * | ||||
|      * @return array An array of parameters | ||||
|      * | ||||
|      * @throws ResourceNotFoundException If no matching resource could be found | ||||
|      * @throws MethodNotAllowedException If a matching resource was found but the request method is not allowed | ||||
|      */ | ||||
|     public function matchRequest(Request $request); | ||||
| } | ||||
							
								
								
									
										121
									
								
								vendor/symfony/routing/Symfony/Component/Routing/Matcher/TraceableUrlMatcher.php
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										121
									
								
								vendor/symfony/routing/Symfony/Component/Routing/Matcher/TraceableUrlMatcher.php
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,121 @@ | ||||
| <?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\Matcher; | ||||
|  | ||||
| use Symfony\Component\Routing\Exception\ExceptionInterface; | ||||
| use Symfony\Component\Routing\Route; | ||||
| use Symfony\Component\Routing\RouteCollection; | ||||
| use Symfony\Component\Routing\Matcher\UrlMatcher; | ||||
|  | ||||
| /** | ||||
|  * TraceableUrlMatcher helps debug path info matching by tracing the match. | ||||
|  * | ||||
|  * @author Fabien Potencier <fabien@symfony.com> | ||||
|  */ | ||||
| class TraceableUrlMatcher extends UrlMatcher | ||||
| { | ||||
|     const ROUTE_DOES_NOT_MATCH = 0; | ||||
|     const ROUTE_ALMOST_MATCHES = 1; | ||||
|     const ROUTE_MATCHES        = 2; | ||||
|  | ||||
|     protected $traces; | ||||
|  | ||||
|     public function getTraces($pathinfo) | ||||
|     { | ||||
|         $this->traces = array(); | ||||
|  | ||||
|         try { | ||||
|             $this->match($pathinfo); | ||||
|         } catch (ExceptionInterface $e) { | ||||
|         } | ||||
|  | ||||
|         return $this->traces; | ||||
|     } | ||||
|  | ||||
|     protected function matchCollection($pathinfo, RouteCollection $routes) | ||||
|     { | ||||
|         foreach ($routes as $name => $route) { | ||||
|             $compiledRoute = $route->compile(); | ||||
|  | ||||
|             if (!preg_match($compiledRoute->getRegex(), $pathinfo, $matches)) { | ||||
|                 // does it match without any requirements? | ||||
|                 $r = new Route($route->getPath(), $route->getDefaults(), array(), $route->getOptions()); | ||||
|                 $cr = $r->compile(); | ||||
|                 if (!preg_match($cr->getRegex(), $pathinfo)) { | ||||
|                     $this->addTrace(sprintf('Path "%s" does not match', $route->getPath()), self::ROUTE_DOES_NOT_MATCH, $name, $route); | ||||
|  | ||||
|                     continue; | ||||
|                 } | ||||
|  | ||||
|                 foreach ($route->getRequirements() as $n => $regex) { | ||||
|                     $r = new Route($route->getPath(), $route->getDefaults(), array($n => $regex), $route->getOptions()); | ||||
|                     $cr = $r->compile(); | ||||
|  | ||||
|                     if (in_array($n, $cr->getVariables()) && !preg_match($cr->getRegex(), $pathinfo)) { | ||||
|                         $this->addTrace(sprintf('Requirement for "%s" does not match (%s)', $n, $regex), self::ROUTE_ALMOST_MATCHES, $name, $route); | ||||
|  | ||||
|                         continue 2; | ||||
|                     } | ||||
|                 } | ||||
|  | ||||
|                 continue; | ||||
|             } | ||||
|  | ||||
|             // check host requirement | ||||
|             $hostMatches = array(); | ||||
|             if ($compiledRoute->getHostRegex() && !preg_match($compiledRoute->getHostRegex(), $this->context->getHost(), $hostMatches)) { | ||||
|                 $this->addTrace(sprintf('Host "%s" does not match the requirement ("%s")', $this->context->getHost(), $route->getHost()), self::ROUTE_ALMOST_MATCHES, $name, $route); | ||||
|  | ||||
|                 return true; | ||||
|             } | ||||
|  | ||||
|             // check HTTP method requirement | ||||
|             if ($req = $route->getRequirement('_method')) { | ||||
|                 // HEAD and GET are equivalent as per RFC | ||||
|                 if ('HEAD' === $method = $this->context->getMethod()) { | ||||
|                     $method = 'GET'; | ||||
|                 } | ||||
|  | ||||
|                 if (!in_array($method, $req = explode('|', strtoupper($req)))) { | ||||
|                     $this->allow = array_merge($this->allow, $req); | ||||
|  | ||||
|                     $this->addTrace(sprintf('Method "%s" does not match the requirement ("%s")', $this->context->getMethod(), implode(', ', $req)), self::ROUTE_ALMOST_MATCHES, $name, $route); | ||||
|  | ||||
|                     continue; | ||||
|                 } | ||||
|             } | ||||
|  | ||||
|             // check HTTP scheme requirement | ||||
|             if ($scheme = $route->getRequirement('_scheme')) { | ||||
|                 if ($this->context->getScheme() !== $scheme) { | ||||
|                     $this->addTrace(sprintf('Scheme "%s" does not match the requirement ("%s"); the user will be redirected', $this->context->getScheme(), $scheme), self::ROUTE_ALMOST_MATCHES, $name, $route); | ||||
|  | ||||
|                     return true; | ||||
|                 } | ||||
|             } | ||||
|  | ||||
|             $this->addTrace('Route matches!', self::ROUTE_MATCHES, $name, $route); | ||||
|  | ||||
|             return true; | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     private function addTrace($log, $level = self::ROUTE_DOES_NOT_MATCH, $name = null, $route = null) | ||||
|     { | ||||
|         $this->traces[] = array( | ||||
|             'log'   => $log, | ||||
|             'name'  => $name, | ||||
|             'level' => $level, | ||||
|             'path'  => null !== $route ? $route->getPath() : null, | ||||
|         ); | ||||
|     } | ||||
| } | ||||
							
								
								
									
										208
									
								
								vendor/symfony/routing/Symfony/Component/Routing/Matcher/UrlMatcher.php
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										208
									
								
								vendor/symfony/routing/Symfony/Component/Routing/Matcher/UrlMatcher.php
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,208 @@ | ||||
| <?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\Matcher; | ||||
|  | ||||
| use Symfony\Component\Routing\Exception\MethodNotAllowedException; | ||||
| use Symfony\Component\Routing\Exception\ResourceNotFoundException; | ||||
| use Symfony\Component\Routing\RouteCollection; | ||||
| use Symfony\Component\Routing\RequestContext; | ||||
| use Symfony\Component\Routing\Route; | ||||
|  | ||||
| /** | ||||
|  * UrlMatcher matches URL based on a set of routes. | ||||
|  * | ||||
|  * @author Fabien Potencier <fabien@symfony.com> | ||||
|  * | ||||
|  * @api | ||||
|  */ | ||||
| class UrlMatcher implements UrlMatcherInterface | ||||
| { | ||||
|     const REQUIREMENT_MATCH     = 0; | ||||
|     const REQUIREMENT_MISMATCH  = 1; | ||||
|     const ROUTE_MATCH           = 2; | ||||
|  | ||||
|     /** | ||||
|      * @var RequestContext | ||||
|      */ | ||||
|     protected $context; | ||||
|  | ||||
|     /** | ||||
|      * @var array | ||||
|      */ | ||||
|     protected $allow = array(); | ||||
|  | ||||
|     /** | ||||
|      * @var RouteCollection | ||||
|      */ | ||||
|     protected $routes; | ||||
|  | ||||
|     /** | ||||
|      * Constructor. | ||||
|      * | ||||
|      * @param RouteCollection $routes  A RouteCollection instance | ||||
|      * @param RequestContext  $context The context | ||||
|      * | ||||
|      * @api | ||||
|      */ | ||||
|     public function __construct(RouteCollection $routes, RequestContext $context) | ||||
|     { | ||||
|         $this->routes = $routes; | ||||
|         $this->context = $context; | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * {@inheritdoc} | ||||
|      */ | ||||
|     public function setContext(RequestContext $context) | ||||
|     { | ||||
|         $this->context = $context; | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * {@inheritdoc} | ||||
|      */ | ||||
|     public function getContext() | ||||
|     { | ||||
|         return $this->context; | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * {@inheritdoc} | ||||
|      */ | ||||
|     public function match($pathinfo) | ||||
|     { | ||||
|         $this->allow = array(); | ||||
|  | ||||
|         if ($ret = $this->matchCollection(rawurldecode($pathinfo), $this->routes)) { | ||||
|             return $ret; | ||||
|         } | ||||
|  | ||||
|         throw 0 < count($this->allow) | ||||
|             ? new MethodNotAllowedException(array_unique(array_map('strtoupper', $this->allow))) | ||||
|             : new ResourceNotFoundException(); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Tries to match a URL with a set of routes. | ||||
|      * | ||||
|      * @param string          $pathinfo The path info to be parsed | ||||
|      * @param RouteCollection $routes   The set of routes | ||||
|      * | ||||
|      * @return array An array of parameters | ||||
|      * | ||||
|      * @throws ResourceNotFoundException If the resource could not be found | ||||
|      * @throws MethodNotAllowedException If the resource was found but the request method is not allowed | ||||
|      */ | ||||
|     protected function matchCollection($pathinfo, RouteCollection $routes) | ||||
|     { | ||||
|         foreach ($routes as $name => $route) { | ||||
|             $compiledRoute = $route->compile(); | ||||
|  | ||||
|             // check the static prefix of the URL first. Only use the more expensive preg_match when it matches | ||||
|             if ('' !== $compiledRoute->getStaticPrefix() && 0 !== strpos($pathinfo, $compiledRoute->getStaticPrefix())) { | ||||
|                 continue; | ||||
|             } | ||||
|  | ||||
|             if (!preg_match($compiledRoute->getRegex(), $pathinfo, $matches)) { | ||||
|                 continue; | ||||
|             } | ||||
|  | ||||
|             $hostMatches = array(); | ||||
|             if ($compiledRoute->getHostRegex() && !preg_match($compiledRoute->getHostRegex(), $this->context->getHost(), $hostMatches)) { | ||||
|                 continue; | ||||
|             } | ||||
|  | ||||
|             // check HTTP method requirement | ||||
|             if ($req = $route->getRequirement('_method')) { | ||||
|                 // HEAD and GET are equivalent as per RFC | ||||
|                 if ('HEAD' === $method = $this->context->getMethod()) { | ||||
|                     $method = 'GET'; | ||||
|                 } | ||||
|  | ||||
|                 if (!in_array($method, $req = explode('|', strtoupper($req)))) { | ||||
|                     $this->allow = array_merge($this->allow, $req); | ||||
|  | ||||
|                     continue; | ||||
|                 } | ||||
|             } | ||||
|  | ||||
|             $status = $this->handleRouteRequirements($pathinfo, $name, $route); | ||||
|  | ||||
|             if (self::ROUTE_MATCH === $status[0]) { | ||||
|                 return $status[1]; | ||||
|             } | ||||
|  | ||||
|             if (self::REQUIREMENT_MISMATCH === $status[0]) { | ||||
|                 continue; | ||||
|             } | ||||
|  | ||||
|             return $this->getAttributes($route, $name, array_replace($matches, $hostMatches)); | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Returns an array of values to use as request attributes. | ||||
|      * | ||||
|      * As this method requires the Route object, it is not available | ||||
|      * in matchers that do not have access to the matched Route instance | ||||
|      * (like the PHP and Apache matcher dumpers). | ||||
|      * | ||||
|      * @param Route  $route      The route we are matching against | ||||
|      * @param string $name       The name of the route | ||||
|      * @param array  $attributes An array of attributes from the matcher | ||||
|      * | ||||
|      * @return array An array of parameters | ||||
|      */ | ||||
|     protected function getAttributes(Route $route, $name, array $attributes) | ||||
|     { | ||||
|         $attributes['_route'] = $name; | ||||
|  | ||||
|         return $this->mergeDefaults($attributes, $route->getDefaults()); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Handles specific route requirements. | ||||
|      * | ||||
|      * @param string $pathinfo The path | ||||
|      * @param string $name     The route name | ||||
|      * @param Route  $route    The route | ||||
|      * | ||||
|      * @return array The first element represents the status, the second contains additional information | ||||
|      */ | ||||
|     protected function handleRouteRequirements($pathinfo, $name, Route $route) | ||||
|     { | ||||
|         // check HTTP scheme requirement | ||||
|         $scheme = $route->getRequirement('_scheme'); | ||||
|         $status = $scheme && $scheme !== $this->context->getScheme() ? self::REQUIREMENT_MISMATCH : self::REQUIREMENT_MATCH; | ||||
|  | ||||
|         return array($status, null); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Get merged default parameters. | ||||
|      * | ||||
|      * @param array $params   The parameters | ||||
|      * @param array $defaults The defaults | ||||
|      * | ||||
|      * @return array Merged default parameters | ||||
|      */ | ||||
|     protected function mergeDefaults($params, $defaults) | ||||
|     { | ||||
|         foreach ($params as $key => $value) { | ||||
|             if (!is_int($key)) { | ||||
|                 $defaults[$key] = $value; | ||||
|             } | ||||
|         } | ||||
|  | ||||
|         return $defaults; | ||||
|     } | ||||
| } | ||||
							
								
								
									
										43
									
								
								vendor/symfony/routing/Symfony/Component/Routing/Matcher/UrlMatcherInterface.php
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										43
									
								
								vendor/symfony/routing/Symfony/Component/Routing/Matcher/UrlMatcherInterface.php
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,43 @@ | ||||
| <?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\Matcher; | ||||
|  | ||||
| use Symfony\Component\Routing\RequestContextAwareInterface; | ||||
| use Symfony\Component\Routing\Exception\ResourceNotFoundException; | ||||
| use Symfony\Component\Routing\Exception\MethodNotAllowedException; | ||||
|  | ||||
| /** | ||||
|  * UrlMatcherInterface is the interface that all URL matcher classes must implement. | ||||
|  * | ||||
|  * @author Fabien Potencier <fabien@symfony.com> | ||||
|  * | ||||
|  * @api | ||||
|  */ | ||||
| interface UrlMatcherInterface extends RequestContextAwareInterface | ||||
| { | ||||
|     /** | ||||
|      * Tries to match a URL path with a set of routes. | ||||
|      * | ||||
|      * If the matcher can not find information, it must throw one of the exceptions documented | ||||
|      * below. | ||||
|      * | ||||
|      * @param string $pathinfo The path info to be parsed (raw format, i.e. not urldecoded) | ||||
|      * | ||||
|      * @return array An array of parameters | ||||
|      * | ||||
|      * @throws ResourceNotFoundException If the resource could not be found | ||||
|      * @throws MethodNotAllowedException If the resource was found but the request method is not allowed | ||||
|      * | ||||
|      * @api | ||||
|      */ | ||||
|     public function match($pathinfo); | ||||
| } | ||||
		Reference in New Issue
	
	Block a user