Blame view

sources/3rdparty/symfony/routing/Symfony/Component/Routing/RouteCompiler.php 4.2 KB
03e52840d   Kload   Init
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
  <?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;
  
  /**
   * RouteCompiler compiles Route instances to CompiledRoute instances.
   *
   * @author Fabien Potencier <fabien@symfony.com>
   */
  class RouteCompiler implements RouteCompilerInterface
  {
      /**
       * Compiles the current route instance.
       *
       * @param Route $route A Route instance
       *
       * @return CompiledRoute A CompiledRoute instance
       */
      public function compile(Route $route)
      {
          $pattern = $route->getPattern();
          $len = strlen($pattern);
          $tokens = array();
          $variables = array();
          $pos = 0;
          preg_match_all('#.\{([\w\d_]+)\}#', $pattern, $matches, PREG_OFFSET_CAPTURE | PREG_SET_ORDER);
          foreach ($matches as $match) {
              if ($text = substr($pattern, $pos, $match[0][1] - $pos)) {
                  $tokens[] = array('text', $text);
              }
              $seps = array($pattern[$pos]);
              $pos = $match[0][1] + strlen($match[0][0]);
              $var = $match[1][0];
  
              if ($req = $route->getRequirement($var)) {
                  $regexp = $req;
              } else {
                  if ($pos !== $len) {
                      $seps[] = $pattern[$pos];
                  }
                  $regexp = sprintf('[^%s]+?', preg_quote(implode('', array_unique($seps)), '#'));
              }
  
              $tokens[] = array('variable', $match[0][0][0], $regexp, $var);
  
              if (in_array($var, $variables)) {
                  throw new \LogicException(sprintf('Route pattern "%s" cannot reference variable name "%s" more than once.', $route->getPattern(), $var));
              }
  
              $variables[] = $var;
          }
  
          if ($pos < $len) {
              $tokens[] = array('text', substr($pattern, $pos));
          }
  
          // find the first optional token
          $firstOptional = PHP_INT_MAX;
          for ($i = count($tokens) - 1; $i >= 0; $i--) {
              $token = $tokens[$i];
              if ('variable' === $token[0] && $route->hasDefault($token[3])) {
                  $firstOptional = $i;
              } else {
                  break;
              }
          }
  
          // compute the matching regexp
          $regexp = '';
          for ($i = 0, $nbToken = count($tokens); $i < $nbToken; $i++) {
              $regexp .= $this->computeRegexp($tokens, $i, $firstOptional);
          }
  
          return new CompiledRoute(
              $route,
              'text' === $tokens[0][0] ? $tokens[0][1] : '',
              sprintf("#^%s$#s", $regexp),
              array_reverse($tokens),
              $variables
          );
      }
  
      /**
       * Computes the regexp used to match the token.
       *
       * @param array   $tokens        The route tokens
       * @param integer $index         The index of the current token
       * @param integer $firstOptional The index of the first optional token
       *
       * @return string The regexp
       */
      private function computeRegexp(array $tokens, $index, $firstOptional)
      {
          $token = $tokens[$index];
          if ('text' === $token[0]) {
              // Text tokens
              return preg_quote($token[1], '#');
          } else {
              // Variable tokens
              if (0 === $index && 0 === $firstOptional && 1 == count($tokens)) {
                  // When the only token is an optional variable token, the separator is required
                  return sprintf('%s(?P<%s>%s)?', preg_quote($token[1], '#'), $token[3], $token[2]);
              } else {
                  $nbTokens = count($tokens);
                  $regexp = sprintf('%s(?P<%s>%s)', preg_quote($token[1], '#'), $token[3], $token[2]);
                  if ($index >= $firstOptional) {
                      // Enclose each optional tokens in a subpattern to make it optional
                      $regexp = "(?:$regexp";
                      if ($nbTokens - 1 == $index) {
                          // Close the optional subpatterns
                          $regexp .= str_repeat(")?", $nbTokens - $firstOptional);
                      }
                  }
  
                  return $regexp;
              }
          }
      }
  }