vendor/twig/twig/src/ExpressionParser.php line 169

  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\Attribute\FirstClassTwigCallableReady;
  13. use Twig\Error\SyntaxError;
  14. use Twig\Node\EmptyNode;
  15. use Twig\Node\Expression\AbstractExpression;
  16. use Twig\Node\Expression\ArrayExpression;
  17. use Twig\Node\Expression\ArrowFunctionExpression;
  18. use Twig\Node\Expression\Binary\AbstractBinary;
  19. use Twig\Node\Expression\Binary\ConcatBinary;
  20. use Twig\Node\Expression\ConstantExpression;
  21. use Twig\Node\Expression\GetAttrExpression;
  22. use Twig\Node\Expression\MacroReferenceExpression;
  23. use Twig\Node\Expression\Ternary\ConditionalTernary;
  24. use Twig\Node\Expression\TestExpression;
  25. use Twig\Node\Expression\Unary\AbstractUnary;
  26. use Twig\Node\Expression\Unary\NegUnary;
  27. use Twig\Node\Expression\Unary\NotUnary;
  28. use Twig\Node\Expression\Unary\PosUnary;
  29. use Twig\Node\Expression\Unary\SpreadUnary;
  30. use Twig\Node\Expression\Variable\AssignContextVariable;
  31. use Twig\Node\Expression\Variable\ContextVariable;
  32. use Twig\Node\Expression\Variable\LocalVariable;
  33. use Twig\Node\Expression\Variable\TemplateVariable;
  34. use Twig\Node\Node;
  35. use Twig\Node\Nodes;
  36. /**
  37.  * Parses expressions.
  38.  *
  39.  * This parser implements a "Precedence climbing" algorithm.
  40.  *
  41.  * @see https://www.engr.mun.ca/~theo/Misc/exp_parsing.htm
  42.  * @see https://en.wikipedia.org/wiki/Operator-precedence_parser
  43.  *
  44.  * @author Fabien Potencier <fabien@symfony.com>
  45.  */
  46. class ExpressionParser
  47. {
  48.     public const OPERATOR_LEFT 1;
  49.     public const OPERATOR_RIGHT 2;
  50.     /** @var array<string, array{precedence: int, precedence_change?: OperatorPrecedenceChange, class: class-string<AbstractUnary>}> */
  51.     private $unaryOperators;
  52.     /** @var array<string, array{precedence: int, precedence_change?: OperatorPrecedenceChange, class: class-string<AbstractBinary>, associativity: self::OPERATOR_*}> */
  53.     private $binaryOperators;
  54.     private $readyNodes = [];
  55.     private array $precedenceChanges = [];
  56.     private bool $deprecationCheck true;
  57.     public function __construct(
  58.         private Parser $parser,
  59.         private Environment $env,
  60.     ) {
  61.         $this->unaryOperators $env->getUnaryOperators();
  62.         $this->binaryOperators $env->getBinaryOperators();
  63.         $ops = [];
  64.         foreach ($this->unaryOperators as $n => $c) {
  65.             $ops[] = $c + ['name' => $n'type' => 'unary'];
  66.         }
  67.         foreach ($this->binaryOperators as $n => $c) {
  68.             $ops[] = $c + ['name' => $n'type' => 'binary'];
  69.         }
  70.         foreach ($ops as $config) {
  71.             if (!isset($config['precedence_change'])) {
  72.                 continue;
  73.             }
  74.             $name $config['type'].'_'.$config['name'];
  75.             $min min($config['precedence_change']->getNewPrecedence(), $config['precedence']);
  76.             $max max($config['precedence_change']->getNewPrecedence(), $config['precedence']);
  77.             foreach ($ops as $c) {
  78.                 if ($c['precedence'] > $min && $c['precedence'] < $max) {
  79.                     $this->precedenceChanges[$c['type'].'_'.$c['name']][] = $name;
  80.                 }
  81.             }
  82.         }
  83.     }
  84.     public function parseExpression($precedence 0)
  85.     {
  86.         if (\func_num_args() > 1) {
  87.             trigger_deprecation('twig/twig''3.15''Passing a second argument ($allowArrow) to "%s()" is deprecated.'__METHOD__);
  88.         }
  89.         if ($arrow $this->parseArrow()) {
  90.             return $arrow;
  91.         }
  92.         $expr $this->getPrimary();
  93.         $token $this->parser->getCurrentToken();
  94.         while ($this->isBinary($token) && $this->binaryOperators[$token->getValue()]['precedence'] >= $precedence) {
  95.             $op $this->binaryOperators[$token->getValue()];
  96.             $this->parser->getStream()->next();
  97.             if ('is not' === $token->getValue()) {
  98.                 $expr $this->parseNotTestExpression($expr);
  99.             } elseif ('is' === $token->getValue()) {
  100.                 $expr $this->parseTestExpression($expr);
  101.             } elseif (isset($op['callable'])) {
  102.                 $expr $op['callable']($this->parser$expr);
  103.             } else {
  104.                 $previous $this->setDeprecationCheck(true);
  105.                 try {
  106.                     $expr1 $this->parseExpression(self::OPERATOR_LEFT === $op['associativity'] ? $op['precedence'] + $op['precedence']);
  107.                 } finally {
  108.                     $this->setDeprecationCheck($previous);
  109.                 }
  110.                 $class $op['class'];
  111.                 $expr = new $class($expr$expr1$token->getLine());
  112.             }
  113.             $expr->setAttribute('operator''binary_'.$token->getValue());
  114.             $this->triggerPrecedenceDeprecations($expr);
  115.             $token $this->parser->getCurrentToken();
  116.         }
  117.         if (=== $precedence) {
  118.             return $this->parseConditionalExpression($expr);
  119.         }
  120.         return $expr;
  121.     }
  122.     private function triggerPrecedenceDeprecations(AbstractExpression $expr): void
  123.     {
  124.         // Check that the all nodes that are between the 2 precedences have explicit parentheses
  125.         if (!$expr->hasAttribute('operator') || !isset($this->precedenceChanges[$expr->getAttribute('operator')])) {
  126.             return;
  127.         }
  128.         if (str_starts_with($unaryOp $expr->getAttribute('operator'), 'unary')) {
  129.             if ($expr->hasExplicitParentheses()) {
  130.                 return;
  131.             }
  132.             $target explode('_'$unaryOp)[1];
  133.             /** @var AbstractExpression $node */
  134.             $node $expr->getNode('node');
  135.             foreach ($this->precedenceChanges as $operatorName => $changes) {
  136.                 if (!\in_array($unaryOp$changes)) {
  137.                     continue;
  138.                 }
  139.                 if ($node->hasAttribute('operator') && $operatorName === $node->getAttribute('operator')) {
  140.                     $change $this->unaryOperators[$target]['precedence_change'];
  141.                     trigger_deprecation($change->getPackage(), $change->getVersion(), \sprintf('Add explicit parentheses around the "%s" unary operator to avoid behavior change in the next major version as its precedence will change in "%s" at line %d.'$target$this->parser->getStream()->getSourceContext()->getName(), $node->getTemplateLine()));
  142.                 }
  143.             }
  144.         } else {
  145.             foreach ($this->precedenceChanges[$expr->getAttribute('operator')] as $operatorName) {
  146.                 foreach ($expr as $node) {
  147.                     /** @var AbstractExpression $node */
  148.                     if ($node->hasAttribute('operator') && $operatorName === $node->getAttribute('operator') && !$node->hasExplicitParentheses()) {
  149.                         $op explode('_'$operatorName)[1];
  150.                         $change $this->binaryOperators[$op]['precedence_change'];
  151.                         trigger_deprecation($change->getPackage(), $change->getVersion(), \sprintf('Add explicit parentheses around the "%s" binary operator to avoid behavior change in the next major version as its precedence will change in "%s" at line %d.'$op$this->parser->getStream()->getSourceContext()->getName(), $node->getTemplateLine()));
  152.                     }
  153.                 }
  154.             }
  155.         }
  156.     }
  157.     /**
  158.      * @return ArrowFunctionExpression|null
  159.      */
  160.     private function parseArrow()
  161.     {
  162.         $stream $this->parser->getStream();
  163.         // short array syntax (one argument, no parentheses)?
  164.         if ($stream->look(1)->test(Token::ARROW_TYPE)) {
  165.             $line $stream->getCurrent()->getLine();
  166.             $token $stream->expect(Token::NAME_TYPE);
  167.             $names = [new AssignContextVariable($token->getValue(), $token->getLine())];
  168.             $stream->expect(Token::ARROW_TYPE);
  169.             return new ArrowFunctionExpression($this->parseExpression(), new Nodes($names), $line);
  170.         }
  171.         // first, determine if we are parsing an arrow function by finding => (long form)
  172.         $i 0;
  173.         if (!$stream->look($i)->test(Token::PUNCTUATION_TYPE'(')) {
  174.             return null;
  175.         }
  176.         ++$i;
  177.         while (true) {
  178.             // variable name
  179.             ++$i;
  180.             if (!$stream->look($i)->test(Token::PUNCTUATION_TYPE',')) {
  181.                 break;
  182.             }
  183.             ++$i;
  184.         }
  185.         if (!$stream->look($i)->test(Token::PUNCTUATION_TYPE')')) {
  186.             return null;
  187.         }
  188.         ++$i;
  189.         if (!$stream->look($i)->test(Token::ARROW_TYPE)) {
  190.             return null;
  191.         }
  192.         // yes, let's parse it properly
  193.         $token $stream->expect(Token::PUNCTUATION_TYPE'(');
  194.         $line $token->getLine();
  195.         $names = [];
  196.         while (true) {
  197.             $token $stream->expect(Token::NAME_TYPE);
  198.             $names[] = new AssignContextVariable($token->getValue(), $token->getLine());
  199.             if (!$stream->nextIf(Token::PUNCTUATION_TYPE',')) {
  200.                 break;
  201.             }
  202.         }
  203.         $stream->expect(Token::PUNCTUATION_TYPE')');
  204.         $stream->expect(Token::ARROW_TYPE);
  205.         return new ArrowFunctionExpression($this->parseExpression(), new Nodes($names), $line);
  206.     }
  207.     private function getPrimary(): AbstractExpression
  208.     {
  209.         $token $this->parser->getCurrentToken();
  210.         if ($this->isUnary($token)) {
  211.             $operator $this->unaryOperators[$token->getValue()];
  212.             $this->parser->getStream()->next();
  213.             $expr $this->parseExpression($operator['precedence']);
  214.             $class $operator['class'];
  215.             $expr = new $class($expr$token->getLine());
  216.             $expr->setAttribute('operator''unary_'.$token->getValue());
  217.             if ($this->deprecationCheck) {
  218.                 $this->triggerPrecedenceDeprecations($expr);
  219.             }
  220.             return $this->parsePostfixExpression($expr);
  221.         } elseif ($token->test(Token::PUNCTUATION_TYPE'(')) {
  222.             $this->parser->getStream()->next();
  223.             $previous $this->setDeprecationCheck(false);
  224.             try {
  225.                 $expr $this->parseExpression()->setExplicitParentheses();
  226.             } finally {
  227.                 $this->setDeprecationCheck($previous);
  228.             }
  229.             $this->parser->getStream()->expect(Token::PUNCTUATION_TYPE')''An opened parenthesis is not properly closed');
  230.             return $this->parsePostfixExpression($expr);
  231.         }
  232.         return $this->parsePrimaryExpression();
  233.     }
  234.     private function parseConditionalExpression($expr): AbstractExpression
  235.     {
  236.         while ($this->parser->getStream()->nextIf(Token::PUNCTUATION_TYPE'?')) {
  237.             $expr2 $this->parseExpression();
  238.             if ($this->parser->getStream()->nextIf(Token::PUNCTUATION_TYPE':')) {
  239.                 // Ternary operator (expr ? expr2 : expr3)
  240.                 $expr3 $this->parseExpression();
  241.             } else {
  242.                 // Ternary without else (expr ? expr2)
  243.                 $expr3 = new ConstantExpression(''$this->parser->getCurrentToken()->getLine());
  244.             }
  245.             $expr = new ConditionalTernary($expr$expr2$expr3$this->parser->getCurrentToken()->getLine());
  246.         }
  247.         return $expr;
  248.     }
  249.     private function isUnary(Token $token): bool
  250.     {
  251.         return $token->test(Token::OPERATOR_TYPE) && isset($this->unaryOperators[$token->getValue()]);
  252.     }
  253.     private function isBinary(Token $token): bool
  254.     {
  255.         return $token->test(Token::OPERATOR_TYPE) && isset($this->binaryOperators[$token->getValue()]);
  256.     }
  257.     public function parsePrimaryExpression()
  258.     {
  259.         $token $this->parser->getCurrentToken();
  260.         switch (true) {
  261.             case $token->test(Token::NAME_TYPE):
  262.                 $this->parser->getStream()->next();
  263.                 switch ($token->getValue()) {
  264.                     case 'true':
  265.                     case 'TRUE':
  266.                         $node = new ConstantExpression(true$token->getLine());
  267.                         break;
  268.                     case 'false':
  269.                     case 'FALSE':
  270.                         $node = new ConstantExpression(false$token->getLine());
  271.                         break;
  272.                     case 'none':
  273.                     case 'NONE':
  274.                     case 'null':
  275.                     case 'NULL':
  276.                         $node = new ConstantExpression(null$token->getLine());
  277.                         break;
  278.                     default:
  279.                         if ('(' === $this->parser->getCurrentToken()->getValue()) {
  280.                             $node $this->getFunctionNode($token->getValue(), $token->getLine());
  281.                         } else {
  282.                             $node = new ContextVariable($token->getValue(), $token->getLine());
  283.                         }
  284.                 }
  285.                 break;
  286.             case $token->test(Token::NUMBER_TYPE):
  287.                 $this->parser->getStream()->next();
  288.                 $node = new ConstantExpression($token->getValue(), $token->getLine());
  289.                 break;
  290.             case $token->test(Token::STRING_TYPE):
  291.             case $token->test(Token::INTERPOLATION_START_TYPE):
  292.                 $node $this->parseStringExpression();
  293.                 break;
  294.             case $token->test(Token::PUNCTUATION_TYPE):
  295.                 $node = match ($token->getValue()) {
  296.                     '[' => $this->parseSequenceExpression(),
  297.                     '{' => $this->parseMappingExpression(),
  298.                     default => throw new SyntaxError(\sprintf('Unexpected token "%s" of value "%s".'$token->toEnglish(), $token->getValue()), $token->getLine(), $this->parser->getStream()->getSourceContext()),
  299.                 };
  300.                 break;
  301.             case $token->test(Token::OPERATOR_TYPE):
  302.                 if (preg_match(Lexer::REGEX_NAME$token->getValue(), $matches) && $matches[0] == $token->getValue()) {
  303.                     // in this context, string operators are variable names
  304.                     $this->parser->getStream()->next();
  305.                     $node = new ContextVariable($token->getValue(), $token->getLine());
  306.                     break;
  307.                 }
  308.                 if ('=' === $token->getValue() && ('==' === $this->parser->getStream()->look(-1)->getValue() || '!=' === $this->parser->getStream()->look(-1)->getValue())) {
  309.                     throw new SyntaxError(\sprintf('Unexpected operator of value "%s". Did you try to use "===" or "!==" for strict comparison? Use "is same as(value)" instead.'$token->getValue()), $token->getLine(), $this->parser->getStream()->getSourceContext());
  310.                 }
  311.                 // no break
  312.             default:
  313.                 throw new SyntaxError(\sprintf('Unexpected token "%s" of value "%s".'$token->toEnglish(), $token->getValue()), $token->getLine(), $this->parser->getStream()->getSourceContext());
  314.         }
  315.         return $this->parsePostfixExpression($node);
  316.     }
  317.     public function parseStringExpression()
  318.     {
  319.         $stream $this->parser->getStream();
  320.         $nodes = [];
  321.         // a string cannot be followed by another string in a single expression
  322.         $nextCanBeString true;
  323.         while (true) {
  324.             if ($nextCanBeString && $token $stream->nextIf(Token::STRING_TYPE)) {
  325.                 $nodes[] = new ConstantExpression($token->getValue(), $token->getLine());
  326.                 $nextCanBeString false;
  327.             } elseif ($stream->nextIf(Token::INTERPOLATION_START_TYPE)) {
  328.                 $nodes[] = $this->parseExpression();
  329.                 $stream->expect(Token::INTERPOLATION_END_TYPE);
  330.                 $nextCanBeString true;
  331.             } else {
  332.                 break;
  333.             }
  334.         }
  335.         $expr array_shift($nodes);
  336.         foreach ($nodes as $node) {
  337.             $expr = new ConcatBinary($expr$node$node->getTemplateLine());
  338.         }
  339.         return $expr;
  340.     }
  341.     /**
  342.      * @deprecated since Twig 3.11, use parseSequenceExpression() instead
  343.      */
  344.     public function parseArrayExpression()
  345.     {
  346.         trigger_deprecation('twig/twig''3.11''Calling "%s()" is deprecated, use "parseSequenceExpression()" instead.'__METHOD__);
  347.         return $this->parseSequenceExpression();
  348.     }
  349.     public function parseSequenceExpression()
  350.     {
  351.         $stream $this->parser->getStream();
  352.         $stream->expect(Token::PUNCTUATION_TYPE'[''A sequence element was expected');
  353.         $node = new ArrayExpression([], $stream->getCurrent()->getLine());
  354.         $first true;
  355.         while (!$stream->test(Token::PUNCTUATION_TYPE']')) {
  356.             if (!$first) {
  357.                 $stream->expect(Token::PUNCTUATION_TYPE',''A sequence element must be followed by a comma');
  358.                 // trailing ,?
  359.                 if ($stream->test(Token::PUNCTUATION_TYPE']')) {
  360.                     break;
  361.                 }
  362.             }
  363.             $first false;
  364.             if ($stream->nextIf(Token::SPREAD_TYPE)) {
  365.                 $expr $this->parseExpression();
  366.                 $expr->setAttribute('spread'true);
  367.                 $node->addElement($expr);
  368.             } else {
  369.                 $node->addElement($this->parseExpression());
  370.             }
  371.         }
  372.         $stream->expect(Token::PUNCTUATION_TYPE']''An opened sequence is not properly closed');
  373.         return $node;
  374.     }
  375.     /**
  376.      * @deprecated since Twig 3.11, use parseMappingExpression() instead
  377.      */
  378.     public function parseHashExpression()
  379.     {
  380.         trigger_deprecation('twig/twig''3.11''Calling "%s()" is deprecated, use "parseMappingExpression()" instead.'__METHOD__);
  381.         return $this->parseMappingExpression();
  382.     }
  383.     public function parseMappingExpression()
  384.     {
  385.         $stream $this->parser->getStream();
  386.         $stream->expect(Token::PUNCTUATION_TYPE'{''A mapping element was expected');
  387.         $node = new ArrayExpression([], $stream->getCurrent()->getLine());
  388.         $first true;
  389.         while (!$stream->test(Token::PUNCTUATION_TYPE'}')) {
  390.             if (!$first) {
  391.                 $stream->expect(Token::PUNCTUATION_TYPE',''A mapping value must be followed by a comma');
  392.                 // trailing ,?
  393.                 if ($stream->test(Token::PUNCTUATION_TYPE'}')) {
  394.                     break;
  395.                 }
  396.             }
  397.             $first false;
  398.             if ($stream->nextIf(Token::SPREAD_TYPE)) {
  399.                 $value $this->parseExpression();
  400.                 $value->setAttribute('spread'true);
  401.                 $node->addElement($value);
  402.                 continue;
  403.             }
  404.             // a mapping key can be:
  405.             //
  406.             //  * a number -- 12
  407.             //  * a string -- 'a'
  408.             //  * a name, which is equivalent to a string -- a
  409.             //  * an expression, which must be enclosed in parentheses -- (1 + 2)
  410.             if ($token $stream->nextIf(Token::NAME_TYPE)) {
  411.                 $key = new ConstantExpression($token->getValue(), $token->getLine());
  412.                 // {a} is a shortcut for {a:a}
  413.                 if ($stream->test(Token::PUNCTUATION_TYPE, [',''}'])) {
  414.                     $value = new ContextVariable($key->getAttribute('value'), $key->getTemplateLine());
  415.                     $node->addElement($value$key);
  416.                     continue;
  417.                 }
  418.             } elseif (($token $stream->nextIf(Token::STRING_TYPE)) || $token $stream->nextIf(Token::NUMBER_TYPE)) {
  419.                 $key = new ConstantExpression($token->getValue(), $token->getLine());
  420.             } elseif ($stream->test(Token::PUNCTUATION_TYPE'(')) {
  421.                 $key $this->parseExpression();
  422.             } else {
  423.                 $current $stream->getCurrent();
  424.                 throw new SyntaxError(\sprintf('A mapping key must be a quoted string, a number, a name, or an expression enclosed in parentheses (unexpected token "%s" of value "%s".'$current->toEnglish(), $current->getValue()), $current->getLine(), $stream->getSourceContext());
  425.             }
  426.             $stream->expect(Token::PUNCTUATION_TYPE':''A mapping key must be followed by a colon (:)');
  427.             $value $this->parseExpression();
  428.             $node->addElement($value$key);
  429.         }
  430.         $stream->expect(Token::PUNCTUATION_TYPE'}''An opened mapping is not properly closed');
  431.         return $node;
  432.     }
  433.     public function parsePostfixExpression($node)
  434.     {
  435.         while (true) {
  436.             $token $this->parser->getCurrentToken();
  437.             if ($token->test(Token::PUNCTUATION_TYPE)) {
  438.                 if ('.' == $token->getValue() || '[' == $token->getValue()) {
  439.                     $node $this->parseSubscriptExpression($node);
  440.                 } elseif ('|' == $token->getValue()) {
  441.                     $node $this->parseFilterExpression($node);
  442.                 } else {
  443.                     break;
  444.                 }
  445.             } else {
  446.                 break;
  447.             }
  448.         }
  449.         return $node;
  450.     }
  451.     public function getFunctionNode($name$line)
  452.     {
  453.         if (null !== $alias $this->parser->getImportedSymbol('function'$name)) {
  454.             return new MacroReferenceExpression($alias['node']->getNode('var'), $alias['name'], $this->createArguments($line), $line);
  455.         }
  456.         $args $this->parseNamedArguments();
  457.         $function $this->getFunction($name$line);
  458.         if ($function->getParserCallable()) {
  459.             $fakeNode = new EmptyNode($line);
  460.             $fakeNode->setSourceContext($this->parser->getStream()->getSourceContext());
  461.             return ($function->getParserCallable())($this->parser$fakeNode$args$line);
  462.         }
  463.         if (!isset($this->readyNodes[$class $function->getNodeClass()])) {
  464.             $this->readyNodes[$class] = (bool) (new \ReflectionClass($class))->getConstructor()->getAttributes(FirstClassTwigCallableReady::class);
  465.         }
  466.         if (!$ready $this->readyNodes[$class]) {
  467.             trigger_deprecation('twig/twig''3.12''Twig node "%s" is not marked as ready for passing a "TwigFunction" in the constructor instead of its name; please update your code and then add #[FirstClassTwigCallableReady] attribute to the constructor.'$class);
  468.         }
  469.         return new $class($ready $function $function->getName(), $args$line);
  470.     }
  471.     public function parseSubscriptExpression($node)
  472.     {
  473.         if ('.' === $this->parser->getStream()->next()->getValue()) {
  474.             return $this->parseSubscriptExpressionDot($node);
  475.         }
  476.         return $this->parseSubscriptExpressionArray($node);
  477.     }
  478.     public function parseFilterExpression($node)
  479.     {
  480.         $this->parser->getStream()->next();
  481.         return $this->parseFilterExpressionRaw($node);
  482.     }
  483.     public function parseFilterExpressionRaw($node)
  484.     {
  485.         if (\func_num_args() > 1) {
  486.             trigger_deprecation('twig/twig''3.12''Passing a second argument to "%s()" is deprecated.'__METHOD__);
  487.         }
  488.         while (true) {
  489.             $token $this->parser->getStream()->expect(Token::NAME_TYPE);
  490.             if (!$this->parser->getStream()->test(Token::PUNCTUATION_TYPE'(')) {
  491.                 $arguments = new EmptyNode();
  492.             } else {
  493.                 $arguments $this->parseNamedArguments();
  494.             }
  495.             $filter $this->getFilter($token->getValue(), $token->getLine());
  496.             $ready true;
  497.             if (!isset($this->readyNodes[$class $filter->getNodeClass()])) {
  498.                 $this->readyNodes[$class] = (bool) (new \ReflectionClass($class))->getConstructor()->getAttributes(FirstClassTwigCallableReady::class);
  499.             }
  500.             if (!$ready $this->readyNodes[$class]) {
  501.                 trigger_deprecation('twig/twig''3.12''Twig node "%s" is not marked as ready for passing a "TwigFilter" in the constructor instead of its name; please update your code and then add #[FirstClassTwigCallableReady] attribute to the constructor.'$class);
  502.             }
  503.             $node = new $class($node$ready $filter : new ConstantExpression($filter->getName(), $token->getLine()), $arguments$token->getLine());
  504.             if (!$this->parser->getStream()->test(Token::PUNCTUATION_TYPE'|')) {
  505.                 break;
  506.             }
  507.             $this->parser->getStream()->next();
  508.         }
  509.         return $node;
  510.     }
  511.     /**
  512.      * Parses arguments.
  513.      *
  514.      * @return Node
  515.      *
  516.      * @throws SyntaxError
  517.      *
  518.      * @deprecated since Twig 3.19 Use parseNamedArguments() instead
  519.      */
  520.     public function parseArguments()
  521.     {
  522.         trigger_deprecation('twig/twig''3.19'\sprintf('The "%s()" method is deprecated, use "%s::parseNamedArguments()" instead.'__METHOD____CLASS__));
  523.         $namedArguments false;
  524.         $definition false;
  525.         if (\func_num_args() > 1) {
  526.             $definition func_get_arg(1);
  527.         }
  528.         if (\func_num_args() > 0) {
  529.             trigger_deprecation('twig/twig''3.15''Passing arguments to "%s()" is deprecated.'__METHOD__);
  530.             $namedArguments func_get_arg(0);
  531.         }
  532.         $args = [];
  533.         $stream $this->parser->getStream();
  534.         $stream->expect(Token::PUNCTUATION_TYPE'(''A list of arguments must begin with an opening parenthesis');
  535.         $hasSpread false;
  536.         while (!$stream->test(Token::PUNCTUATION_TYPE')')) {
  537.             if ($args) {
  538.                 $stream->expect(Token::PUNCTUATION_TYPE',''Arguments must be separated by a comma');
  539.                 // if the comma above was a trailing comma, early exit the argument parse loop
  540.                 if ($stream->test(Token::PUNCTUATION_TYPE')')) {
  541.                     break;
  542.                 }
  543.             }
  544.             if ($definition) {
  545.                 $token $stream->expect(Token::NAME_TYPEnull'An argument must be a name');
  546.                 $value = new ContextVariable($token->getValue(), $this->parser->getCurrentToken()->getLine());
  547.             } else {
  548.                 if ($stream->nextIf(Token::SPREAD_TYPE)) {
  549.                     $hasSpread true;
  550.                     $value = new SpreadUnary($this->parseExpression(), $stream->getCurrent()->getLine());
  551.                 } elseif ($hasSpread) {
  552.                     throw new SyntaxError('Normal arguments must be placed before argument unpacking.'$stream->getCurrent()->getLine(), $stream->getSourceContext());
  553.                 } else {
  554.                     $value $this->parseExpression();
  555.                 }
  556.             }
  557.             $name null;
  558.             if ($namedArguments && (($token $stream->nextIf(Token::OPERATOR_TYPE'=')) || (!$definition && $token $stream->nextIf(Token::PUNCTUATION_TYPE':')))) {
  559.                 if (!$value instanceof ContextVariable) {
  560.                     throw new SyntaxError(\sprintf('A parameter name must be a string, "%s" given.'\get_class($value)), $token->getLine(), $stream->getSourceContext());
  561.                 }
  562.                 $name $value->getAttribute('name');
  563.                 if ($definition) {
  564.                     $value $this->getPrimary();
  565.                     if (!$this->checkConstantExpression($value)) {
  566.                         throw new SyntaxError('A default value for an argument must be a constant (a boolean, a string, a number, a sequence, or a mapping).'$token->getLine(), $stream->getSourceContext());
  567.                     }
  568.                 } else {
  569.                     $value $this->parseExpression();
  570.                 }
  571.             }
  572.             if ($definition) {
  573.                 if (null === $name) {
  574.                     $name $value->getAttribute('name');
  575.                     $value = new ConstantExpression(null$this->parser->getCurrentToken()->getLine());
  576.                     $value->setAttribute('is_implicit'true);
  577.                 }
  578.                 $args[$name] = $value;
  579.             } else {
  580.                 if (null === $name) {
  581.                     $args[] = $value;
  582.                 } else {
  583.                     $args[$name] = $value;
  584.                 }
  585.             }
  586.         }
  587.         $stream->expect(Token::PUNCTUATION_TYPE')''A list of arguments must be closed by a parenthesis');
  588.         return new Nodes($args);
  589.     }
  590.     public function parseAssignmentExpression()
  591.     {
  592.         $stream $this->parser->getStream();
  593.         $targets = [];
  594.         while (true) {
  595.             $token $this->parser->getCurrentToken();
  596.             if ($stream->test(Token::OPERATOR_TYPE) && preg_match(Lexer::REGEX_NAME$token->getValue())) {
  597.                 // in this context, string operators are variable names
  598.                 $this->parser->getStream()->next();
  599.             } else {
  600.                 $stream->expect(Token::NAME_TYPEnull'Only variables can be assigned to');
  601.             }
  602.             $targets[] = new AssignContextVariable($token->getValue(), $token->getLine());
  603.             if (!$stream->nextIf(Token::PUNCTUATION_TYPE',')) {
  604.                 break;
  605.             }
  606.         }
  607.         return new Nodes($targets);
  608.     }
  609.     public function parseMultitargetExpression()
  610.     {
  611.         $targets = [];
  612.         while (true) {
  613.             $targets[] = $this->parseExpression();
  614.             if (!$this->parser->getStream()->nextIf(Token::PUNCTUATION_TYPE',')) {
  615.                 break;
  616.             }
  617.         }
  618.         return new Nodes($targets);
  619.     }
  620.     private function parseNotTestExpression(Node $node): NotUnary
  621.     {
  622.         return new NotUnary($this->parseTestExpression($node), $this->parser->getCurrentToken()->getLine());
  623.     }
  624.     private function parseTestExpression(Node $node): TestExpression
  625.     {
  626.         $stream $this->parser->getStream();
  627.         $test $this->getTest($node->getTemplateLine());
  628.         $arguments null;
  629.         if ($stream->test(Token::PUNCTUATION_TYPE'(')) {
  630.             $arguments $this->parseNamedArguments();
  631.         } elseif ($test->hasOneMandatoryArgument()) {
  632.             $arguments = new Nodes([=> $this->getPrimary()]);
  633.         }
  634.         if ('defined' === $test->getName() && $node instanceof ContextVariable && null !== $alias $this->parser->getImportedSymbol('function'$node->getAttribute('name'))) {
  635.             $node = new MacroReferenceExpression($alias['node']->getNode('var'), $alias['name'], new ArrayExpression([], $node->getTemplateLine()), $node->getTemplateLine());
  636.         }
  637.         $ready $test instanceof TwigTest;
  638.         if (!isset($this->readyNodes[$class $test->getNodeClass()])) {
  639.             $this->readyNodes[$class] = (bool) (new \ReflectionClass($class))->getConstructor()->getAttributes(FirstClassTwigCallableReady::class);
  640.         }
  641.         if (!$ready $this->readyNodes[$class]) {
  642.             trigger_deprecation('twig/twig''3.12''Twig node "%s" is not marked as ready for passing a "TwigTest" in the constructor instead of its name; please update your code and then add #[FirstClassTwigCallableReady] attribute to the constructor.'$class);
  643.         }
  644.         return new $class($node$ready $test $test->getName(), $arguments$this->parser->getCurrentToken()->getLine());
  645.     }
  646.     private function getTest(int $line): TwigTest
  647.     {
  648.         $stream $this->parser->getStream();
  649.         $name $stream->expect(Token::NAME_TYPE)->getValue();
  650.         if ($stream->test(Token::NAME_TYPE)) {
  651.             // try 2-words tests
  652.             $name $name.' '.$this->parser->getCurrentToken()->getValue();
  653.             if ($test $this->env->getTest($name)) {
  654.                 $stream->next();
  655.             }
  656.         } else {
  657.             $test $this->env->getTest($name);
  658.         }
  659.         if (!$test) {
  660.             if ($this->parser->shouldIgnoreUnknownTwigCallables()) {
  661.                 return new TwigTest($name, fn () => '');
  662.             }
  663.             $e = new SyntaxError(\sprintf('Unknown "%s" test.'$name), $line$stream->getSourceContext());
  664.             $e->addSuggestions($namearray_keys($this->env->getTests()));
  665.             throw $e;
  666.         }
  667.         if ($test->isDeprecated()) {
  668.             $stream $this->parser->getStream();
  669.             $src $stream->getSourceContext();
  670.             $test->triggerDeprecation($src->getPath() ?: $src->getName(), $stream->getCurrent()->getLine());
  671.         }
  672.         return $test;
  673.     }
  674.     private function getFunction(string $nameint $line): TwigFunction
  675.     {
  676.         try {
  677.             $function $this->env->getFunction($name);
  678.         } catch (SyntaxError $e) {
  679.             if (!$this->parser->shouldIgnoreUnknownTwigCallables()) {
  680.                 throw $e;
  681.             }
  682.             $function null;
  683.         }
  684.         if (!$function) {
  685.             if ($this->parser->shouldIgnoreUnknownTwigCallables()) {
  686.                 return new TwigFunction($name, fn () => '');
  687.             }
  688.             $e = new SyntaxError(\sprintf('Unknown "%s" function.'$name), $line$this->parser->getStream()->getSourceContext());
  689.             $e->addSuggestions($namearray_keys($this->env->getFunctions()));
  690.             throw $e;
  691.         }
  692.         if ($function->isDeprecated()) {
  693.             $src $this->parser->getStream()->getSourceContext();
  694.             $function->triggerDeprecation($src->getPath() ?: $src->getName(), $line);
  695.         }
  696.         return $function;
  697.     }
  698.     private function getFilter(string $nameint $line): TwigFilter
  699.     {
  700.         try {
  701.             $filter $this->env->getFilter($name);
  702.         } catch (SyntaxError $e) {
  703.             if (!$this->parser->shouldIgnoreUnknownTwigCallables()) {
  704.                 throw $e;
  705.             }
  706.             $filter null;
  707.         }
  708.         if (!$filter) {
  709.             if ($this->parser->shouldIgnoreUnknownTwigCallables()) {
  710.                 return new TwigFilter($name, fn () => '');
  711.             }
  712.             $e = new SyntaxError(\sprintf('Unknown "%s" filter.'$name), $line$this->parser->getStream()->getSourceContext());
  713.             $e->addSuggestions($namearray_keys($this->env->getFilters()));
  714.             throw $e;
  715.         }
  716.         if ($filter->isDeprecated()) {
  717.             $src $this->parser->getStream()->getSourceContext();
  718.             $filter->triggerDeprecation($src->getPath() ?: $src->getName(), $line);
  719.         }
  720.         return $filter;
  721.     }
  722.     // checks that the node only contains "constant" elements
  723.     // to be removed in 4.0
  724.     private function checkConstantExpression(Node $node): bool
  725.     {
  726.         if (!($node instanceof ConstantExpression || $node instanceof ArrayExpression
  727.             || $node instanceof NegUnary || $node instanceof PosUnary
  728.         )) {
  729.             return false;
  730.         }
  731.         foreach ($node as $n) {
  732.             if (!$this->checkConstantExpression($n)) {
  733.                 return false;
  734.             }
  735.         }
  736.         return true;
  737.     }
  738.     private function setDeprecationCheck(bool $deprecationCheck): bool
  739.     {
  740.         $current $this->deprecationCheck;
  741.         $this->deprecationCheck $deprecationCheck;
  742.         return $current;
  743.     }
  744.     private function createArguments(int $line): ArrayExpression
  745.     {
  746.         $arguments = new ArrayExpression([], $line);
  747.         foreach ($this->parseNamedArguments() as $k => $n) {
  748.             $arguments->addElement($n, new LocalVariable($k$line));
  749.         }
  750.         return $arguments;
  751.     }
  752.     /**
  753.      * @deprecated since Twig 3.19 Use parseNamedArguments() instead
  754.      */
  755.     public function parseOnlyArguments()
  756.     {
  757.         trigger_deprecation('twig/twig''3.19'\sprintf('The "%s()" method is deprecated, use "%s::parseNamedArguments()" instead.'__METHOD____CLASS__));
  758.         return $this->parseNamedArguments();
  759.     }
  760.     public function parseNamedArguments(): Nodes
  761.     {
  762.         $args = [];
  763.         $stream $this->parser->getStream();
  764.         $stream->expect(Token::PUNCTUATION_TYPE'(''A list of arguments must begin with an opening parenthesis');
  765.         $hasSpread false;
  766.         while (!$stream->test(Token::PUNCTUATION_TYPE')')) {
  767.             if ($args) {
  768.                 $stream->expect(Token::PUNCTUATION_TYPE',''Arguments must be separated by a comma');
  769.                 // if the comma above was a trailing comma, early exit the argument parse loop
  770.                 if ($stream->test(Token::PUNCTUATION_TYPE')')) {
  771.                     break;
  772.                 }
  773.             }
  774.             if ($stream->nextIf(Token::SPREAD_TYPE)) {
  775.                 $hasSpread true;
  776.                 $value = new SpreadUnary($this->parseExpression(), $stream->getCurrent()->getLine());
  777.             } elseif ($hasSpread) {
  778.                 throw new SyntaxError('Normal arguments must be placed before argument unpacking.'$stream->getCurrent()->getLine(), $stream->getSourceContext());
  779.             } else {
  780.                 $value $this->parseExpression();
  781.             }
  782.             $name null;
  783.             if (($token $stream->nextIf(Token::OPERATOR_TYPE'=')) || ($token $stream->nextIf(Token::PUNCTUATION_TYPE':'))) {
  784.                 if (!$value instanceof ContextVariable) {
  785.                     throw new SyntaxError(\sprintf('A parameter name must be a string, "%s" given.'\get_class($value)), $token->getLine(), $stream->getSourceContext());
  786.                 }
  787.                 $name $value->getAttribute('name');
  788.                 $value $this->parseExpression();
  789.             }
  790.             if (null === $name) {
  791.                 $args[] = $value;
  792.             } else {
  793.                 $args[$name] = $value;
  794.             }
  795.         }
  796.         $stream->expect(Token::PUNCTUATION_TYPE')''A list of arguments must be closed by a parenthesis');
  797.         return new Nodes($args);
  798.     }
  799.     private function parseSubscriptExpressionDot(Node $node): AbstractExpression
  800.     {
  801.         $stream $this->parser->getStream();
  802.         $token $stream->getCurrent();
  803.         $lineno $token->getLine();
  804.         $arguments = new ArrayExpression([], $lineno);
  805.         $type Template::ANY_CALL;
  806.         if ($stream->nextIf(Token::PUNCTUATION_TYPE'(')) {
  807.             $attribute $this->parseExpression();
  808.             $stream->expect(Token::PUNCTUATION_TYPE')');
  809.         } else {
  810.             $token $stream->next();
  811.             if (
  812.                 $token->test(Token::NAME_TYPE)
  813.                 || $token->test(Token::NUMBER_TYPE)
  814.                 || ($token->test(Token::OPERATOR_TYPE) && preg_match(Lexer::REGEX_NAME$token->getValue()))
  815.             ) {
  816.                 $attribute = new ConstantExpression($token->getValue(), $token->getLine());
  817.             } else {
  818.                 throw new SyntaxError(\sprintf('Expected name or number, got value "%s" of type %s.'$token->getValue(), $token->toEnglish()), $token->getLine(), $stream->getSourceContext());
  819.             }
  820.         }
  821.         if ($stream->test(Token::PUNCTUATION_TYPE'(')) {
  822.             $type Template::METHOD_CALL;
  823.             $arguments $this->createArguments($token->getLine());
  824.         }
  825.         if (
  826.             $node instanceof ContextVariable
  827.             && (
  828.                 null !== $this->parser->getImportedSymbol('template'$node->getAttribute('name'))
  829.                 || '_self' === $node->getAttribute('name') && $attribute instanceof ConstantExpression
  830.             )
  831.         ) {
  832.             return new MacroReferenceExpression(new TemplateVariable($node->getAttribute('name'), $node->getTemplateLine()), 'macro_'.$attribute->getAttribute('value'), $arguments$node->getTemplateLine());
  833.         }
  834.         return new GetAttrExpression($node$attribute$arguments$type$lineno);
  835.     }
  836.     private function parseSubscriptExpressionArray(Node $node): AbstractExpression
  837.     {
  838.         $stream $this->parser->getStream();
  839.         $token $stream->getCurrent();
  840.         $lineno $token->getLine();
  841.         $arguments = new ArrayExpression([], $lineno);
  842.         // slice?
  843.         $slice false;
  844.         if ($stream->test(Token::PUNCTUATION_TYPE':')) {
  845.             $slice true;
  846.             $attribute = new ConstantExpression(0$token->getLine());
  847.         } else {
  848.             $attribute $this->parseExpression();
  849.         }
  850.         if ($stream->nextIf(Token::PUNCTUATION_TYPE':')) {
  851.             $slice true;
  852.         }
  853.         if ($slice) {
  854.             if ($stream->test(Token::PUNCTUATION_TYPE']')) {
  855.                 $length = new ConstantExpression(null$token->getLine());
  856.             } else {
  857.                 $length $this->parseExpression();
  858.             }
  859.             $filter $this->getFilter('slice'$token->getLine());
  860.             $arguments = new Nodes([$attribute$length]);
  861.             $filter = new ($filter->getNodeClass())($node$filter$arguments$token->getLine());
  862.             $stream->expect(Token::PUNCTUATION_TYPE']');
  863.             return $filter;
  864.         }
  865.         $stream->expect(Token::PUNCTUATION_TYPE']');
  866.         return new GetAttrExpression($node$attribute$argumentsTemplate::ARRAY_CALL$lineno);
  867.     }
  868. }