vendor/twig/twig/src/Compiler.php line 102

Open in your IDE?
  1. <?php
  2. /*
  3.  * This file is part of Twig.
  4.  *
  5.  * (c) Fabien Potencier
  6.  * (c) Armin Ronacher
  7.  *
  8.  * For the full copyright and license information, please view the LICENSE
  9.  * file that was distributed with this source code.
  10.  */
  11. namespace Twig;
  12. use Twig\Node\Node;
  13. /**
  14.  * @author Fabien Potencier <fabien@symfony.com>
  15.  */
  16. class Compiler
  17. {
  18.     private $lastLine;
  19.     private $source;
  20.     private $indentation;
  21.     private $debugInfo = [];
  22.     private $sourceOffset;
  23.     private $sourceLine;
  24.     private $varNameSalt 0;
  25.     private $didUseEcho false;
  26.     private $didUseEchoStack = [];
  27.     public function __construct(
  28.         private Environment $env,
  29.     ) {
  30.     }
  31.     public function getEnvironment(): Environment
  32.     {
  33.         return $this->env;
  34.     }
  35.     public function getSource(): string
  36.     {
  37.         return $this->source;
  38.     }
  39.     /**
  40.      * @return $this
  41.      */
  42.     public function reset(int $indentation 0)
  43.     {
  44.         $this->lastLine null;
  45.         $this->source '';
  46.         $this->debugInfo = [];
  47.         $this->sourceOffset 0;
  48.         // source code starts at 1 (as we then increment it when we encounter new lines)
  49.         $this->sourceLine 1;
  50.         $this->indentation $indentation;
  51.         $this->varNameSalt 0;
  52.         return $this;
  53.     }
  54.     /**
  55.      * @return $this
  56.      */
  57.     public function compile(Node $nodeint $indentation 0)
  58.     {
  59.         $this->reset($indentation);
  60.         $this->didUseEchoStack[] = $this->didUseEcho;
  61.         try {
  62.             $this->didUseEcho false;
  63.             $node->compile($this);
  64.             if ($this->didUseEcho) {
  65.                 trigger_deprecation('twig/twig''3.9''Using "%s" is deprecated, use "yield" instead in "%s", then flag the class with #[\Twig\Attribute\YieldReady].'$this->didUseEcho\get_class($node));
  66.             }
  67.             return $this;
  68.         } finally {
  69.             $this->didUseEcho array_pop($this->didUseEchoStack);
  70.         }
  71.     }
  72.     /**
  73.      * @return $this
  74.      */
  75.     public function subcompile(Node $nodebool $raw true)
  76.     {
  77.         if (!$raw) {
  78.             $this->source .= str_repeat(' '$this->indentation 4);
  79.         }
  80.         $this->didUseEchoStack[] = $this->didUseEcho;
  81.         try {
  82.             $this->didUseEcho false;
  83.             $node->compile($this);
  84.             if ($this->didUseEcho) {
  85.                 trigger_deprecation('twig/twig''3.9''Using "%s" is deprecated, use "yield" instead in "%s", then flag the class with #[\Twig\Attribute\YieldReady].'$this->didUseEcho\get_class($node));
  86.             }
  87.             return $this;
  88.         } finally {
  89.             $this->didUseEcho array_pop($this->didUseEchoStack);
  90.         }
  91.     }
  92.     /**
  93.      * Adds a raw string to the compiled code.
  94.      *
  95.      * @return $this
  96.      */
  97.     public function raw(string $string)
  98.     {
  99.         $this->checkForEcho($string);
  100.         $this->source .= $string;
  101.         return $this;
  102.     }
  103.     /**
  104.      * Writes a string to the compiled code by adding indentation.
  105.      *
  106.      * @return $this
  107.      */
  108.     public function write(...$strings)
  109.     {
  110.         foreach ($strings as $string) {
  111.             $this->checkForEcho($string);
  112.             $this->source .= str_repeat(' '$this->indentation 4).$string;
  113.         }
  114.         return $this;
  115.     }
  116.     /**
  117.      * Adds a quoted string to the compiled code.
  118.      *
  119.      * @return $this
  120.      */
  121.     public function string(string $value)
  122.     {
  123.         $this->source .= \sprintf('"%s"'addcslashes($value"\0\t\"\$\\"));
  124.         return $this;
  125.     }
  126.     /**
  127.      * Returns a PHP representation of a given value.
  128.      *
  129.      * @return $this
  130.      */
  131.     public function repr($value)
  132.     {
  133.         if (\is_int($value) || \is_float($value)) {
  134.             if (false !== $locale setlocale(\LC_NUMERIC'0')) {
  135.                 setlocale(\LC_NUMERIC'C');
  136.             }
  137.             $this->raw(var_export($valuetrue));
  138.             if (false !== $locale) {
  139.                 setlocale(\LC_NUMERIC$locale);
  140.             }
  141.         } elseif (null === $value) {
  142.             $this->raw('null');
  143.         } elseif (\is_bool($value)) {
  144.             $this->raw($value 'true' 'false');
  145.         } elseif (\is_array($value)) {
  146.             $this->raw('[');
  147.             $first true;
  148.             foreach ($value as $key => $v) {
  149.                 if (!$first) {
  150.                     $this->raw(', ');
  151.                 }
  152.                 $first false;
  153.                 $this->repr($key);
  154.                 $this->raw(' => ');
  155.                 $this->repr($v);
  156.             }
  157.             $this->raw(']');
  158.         } else {
  159.             $this->string($value);
  160.         }
  161.         return $this;
  162.     }
  163.     /**
  164.      * @return $this
  165.      */
  166.     public function addDebugInfo(Node $node)
  167.     {
  168.         if ($node->getTemplateLine() != $this->lastLine) {
  169.             $this->write(\sprintf("// line %d\n"$node->getTemplateLine()));
  170.             $this->sourceLine += substr_count($this->source"\n"$this->sourceOffset);
  171.             $this->sourceOffset \strlen($this->source);
  172.             $this->debugInfo[$this->sourceLine] = $node->getTemplateLine();
  173.             $this->lastLine $node->getTemplateLine();
  174.         }
  175.         return $this;
  176.     }
  177.     public function getDebugInfo(): array
  178.     {
  179.         ksort($this->debugInfo);
  180.         return $this->debugInfo;
  181.     }
  182.     /**
  183.      * @return $this
  184.      */
  185.     public function indent(int $step 1)
  186.     {
  187.         $this->indentation += $step;
  188.         return $this;
  189.     }
  190.     /**
  191.      * @return $this
  192.      *
  193.      * @throws \LogicException When trying to outdent too much so the indentation would become negative
  194.      */
  195.     public function outdent(int $step 1)
  196.     {
  197.         // can't outdent by more steps than the current indentation level
  198.         if ($this->indentation $step) {
  199.             throw new \LogicException('Unable to call outdent() as the indentation would become negative.');
  200.         }
  201.         $this->indentation -= $step;
  202.         return $this;
  203.     }
  204.     public function getVarName(): string
  205.     {
  206.         return \sprintf('_v%d'$this->varNameSalt++);
  207.     }
  208.     private function checkForEcho(string $string): void
  209.     {
  210.         if ($this->didUseEcho) {
  211.             return;
  212.         }
  213.         $this->didUseEcho preg_match('/^\s*+(echo|print)\b/'$string$m) ? $m[1] : false;
  214.     }
  215. }