2024-06-20 14:10:42 +00:00
< ? php
/*
* This file is part of Twig .
*
* ( c ) Fabien Potencier
*
* For the full copyright and license information , please view the LICENSE
* file that was distributed with this source code .
*/
namespace Twig\Node\Expression ;
use Twig\Compiler ;
use Twig\Error\SyntaxError ;
use Twig\Extension\ExtensionInterface ;
use Twig\Node\Node ;
2025-01-13 09:56:01 +00:00
use Twig\TwigCallableInterface ;
use Twig\TwigFilter ;
use Twig\TwigFunction ;
use Twig\TwigTest ;
use Twig\Util\CallableArgumentsExtractor ;
2024-09-05 17:51:48 +00:00
use Twig\Util\ReflectionCallable ;
2024-06-20 14:10:42 +00:00
abstract class CallExpression extends AbstractExpression
{
2024-09-05 17:51:48 +00:00
private $reflector = null ;
2024-06-20 14:10:42 +00:00
protected function compileCallable ( Compiler $compiler )
{
2025-01-13 09:56:01 +00:00
$twigCallable = $this -> getTwigCallable ();
$callable = $twigCallable -> getCallable ();
2024-06-20 14:10:42 +00:00
if ( \is_string ( $callable ) && ! str_contains ( $callable , '::' )) {
$compiler -> raw ( $callable );
} else {
2025-01-13 09:56:01 +00:00
$rc = $this -> reflectCallable ( $twigCallable );
2024-09-05 17:51:48 +00:00
$r = $rc -> getReflector ();
$callable = $rc -> getCallable ();
2024-06-20 14:10:42 +00:00
if ( \is_string ( $callable )) {
$compiler -> raw ( $callable );
} elseif ( \is_array ( $callable ) && \is_string ( $callable [ 0 ])) {
if ( ! $r instanceof \ReflectionMethod || $r -> isStatic ()) {
2024-09-05 17:51:48 +00:00
$compiler -> raw ( \sprintf ( '%s::%s' , $callable [ 0 ], $callable [ 1 ]));
2024-06-20 14:10:42 +00:00
} else {
2024-09-05 17:51:48 +00:00
$compiler -> raw ( \sprintf ( '$this->env->getRuntime(\'%s\')->%s' , $callable [ 0 ], $callable [ 1 ]));
2024-06-20 14:10:42 +00:00
}
} elseif ( \is_array ( $callable ) && $callable [ 0 ] instanceof ExtensionInterface ) {
$class = \get_class ( $callable [ 0 ]);
if ( ! $compiler -> getEnvironment () -> hasExtension ( $class )) {
// Compile a non-optimized call to trigger a \Twig\Error\RuntimeError, which cannot be a compile-time error
2024-09-05 17:51:48 +00:00
$compiler -> raw ( \sprintf ( '$this->env->getExtension(\'%s\')' , $class ));
2024-06-20 14:10:42 +00:00
} else {
2024-09-05 17:51:48 +00:00
$compiler -> raw ( \sprintf ( '$this->extensions[\'%s\']' , ltrim ( $class , '\\' )));
2024-06-20 14:10:42 +00:00
}
2024-09-05 17:51:48 +00:00
$compiler -> raw ( \sprintf ( '->%s' , $callable [ 1 ]));
2024-06-20 14:10:42 +00:00
} else {
2025-01-13 09:56:01 +00:00
$compiler -> raw ( \sprintf ( '$this->env->get%s(\'%s\')->getCallable()' , ucfirst ( $this -> getAttribute ( 'type' )), $twigCallable -> getDynamicName ()));
2024-06-20 14:10:42 +00:00
}
}
$this -> compileArguments ( $compiler );
}
protected function compileArguments ( Compiler $compiler , $isArray = false ) : void
{
2024-09-05 17:51:48 +00:00
if ( \func_num_args () >= 2 ) {
trigger_deprecation ( 'twig/twig' , '3.11' , 'Passing a second argument to "%s()" is deprecated.' , __METHOD__ );
}
2024-06-20 14:10:42 +00:00
$compiler -> raw ( $isArray ? '[' : '(' );
$first = true ;
2025-01-13 09:56:01 +00:00
$twigCallable = $this -> getAttribute ( 'twig_callable' );
if ( $twigCallable -> needsCharset ()) {
2024-06-20 14:10:42 +00:00
$compiler -> raw ( '$this->env->getCharset()' );
$first = false ;
}
2025-01-13 09:56:01 +00:00
if ( $twigCallable -> needsEnvironment ()) {
2024-06-20 14:10:42 +00:00
if ( ! $first ) {
$compiler -> raw ( ', ' );
}
$compiler -> raw ( '$this->env' );
$first = false ;
}
2025-01-13 09:56:01 +00:00
if ( $twigCallable -> needsContext ()) {
2024-06-20 14:10:42 +00:00
if ( ! $first ) {
$compiler -> raw ( ', ' );
}
$compiler -> raw ( '$context' );
$first = false ;
}
2025-01-13 09:56:01 +00:00
foreach ( $twigCallable -> getArguments () as $argument ) {
if ( ! $first ) {
$compiler -> raw ( ', ' );
2024-06-20 14:10:42 +00:00
}
2025-01-13 09:56:01 +00:00
$compiler -> string ( $argument );
$first = false ;
2024-06-20 14:10:42 +00:00
}
if ( $this -> hasNode ( 'node' )) {
if ( ! $first ) {
$compiler -> raw ( ', ' );
}
$compiler -> subcompile ( $this -> getNode ( 'node' ));
$first = false ;
}
if ( $this -> hasNode ( 'arguments' )) {
2025-01-13 09:56:01 +00:00
$arguments = ( new CallableArgumentsExtractor ( $this , $this -> getTwigCallable ())) -> extractArguments ( $this -> getNode ( 'arguments' ));
2024-06-20 14:10:42 +00:00
foreach ( $arguments as $node ) {
if ( ! $first ) {
$compiler -> raw ( ', ' );
}
$compiler -> subcompile ( $node );
$first = false ;
}
}
$compiler -> raw ( $isArray ? ']' : ')' );
}
2025-01-13 09:56:01 +00:00
/**
* @ deprecated since Twig 3.12 , use Twig\Util\CallableArgumentsExtractor :: getArguments () instead
*/
2024-06-20 14:10:42 +00:00
protected function getArguments ( $callable , $arguments )
{
2025-01-13 09:56:01 +00:00
trigger_deprecation ( 'twig/twig' , '3.12' , 'The "%s()" method is deprecated, use Twig\Util\CallableArgumentsExtractor::getArguments() instead.' , __METHOD__ );
2024-06-20 14:10:42 +00:00
$callType = $this -> getAttribute ( 'type' );
$callName = $this -> getAttribute ( 'name' );
$parameters = [];
$named = false ;
foreach ( $arguments as $name => $node ) {
if ( ! \is_int ( $name )) {
$named = true ;
$name = $this -> normalizeName ( $name );
} elseif ( $named ) {
2024-09-05 17:51:48 +00:00
throw new SyntaxError ( \sprintf ( 'Positional arguments cannot be used after named arguments for %s "%s".' , $callType , $callName ), $this -> getTemplateLine (), $this -> getSourceContext ());
2024-06-20 14:10:42 +00:00
}
$parameters [ $name ] = $node ;
}
2025-01-13 09:56:01 +00:00
$isVariadic = $this -> getAttribute ( 'twig_callable' ) -> isVariadic ();
2024-06-20 14:10:42 +00:00
if ( ! $named && ! $isVariadic ) {
return $parameters ;
}
if ( ! $callable ) {
if ( $named ) {
2024-09-05 17:51:48 +00:00
$message = \sprintf ( 'Named arguments are not supported for %s "%s".' , $callType , $callName );
2024-06-20 14:10:42 +00:00
} else {
2024-09-05 17:51:48 +00:00
$message = \sprintf ( 'Arbitrary positional arguments are not supported for %s "%s".' , $callType , $callName );
2024-06-20 14:10:42 +00:00
}
throw new \LogicException ( $message );
}
[ $callableParameters , $isPhpVariadic ] = $this -> getCallableParameters ( $callable , $isVariadic );
$arguments = [];
$names = [];
$missingArguments = [];
$optionalArguments = [];
$pos = 0 ;
foreach ( $callableParameters as $callableParameter ) {
$name = $this -> normalizeName ( $callableParameter -> name );
if ( \PHP_VERSION_ID >= 80000 && 'range' === $callable ) {
if ( 'start' === $name ) {
$name = 'low' ;
} elseif ( 'end' === $name ) {
$name = 'high' ;
}
}
$names [] = $name ;
if ( \array_key_exists ( $name , $parameters )) {
if ( \array_key_exists ( $pos , $parameters )) {
2024-09-05 17:51:48 +00:00
throw new SyntaxError ( \sprintf ( 'Argument "%s" is defined twice for %s "%s".' , $name , $callType , $callName ), $this -> getTemplateLine (), $this -> getSourceContext ());
2024-06-20 14:10:42 +00:00
}
if ( \count ( $missingArguments )) {
2024-09-05 17:51:48 +00:00
throw new SyntaxError ( \sprintf (
2024-06-20 14:10:42 +00:00
'Argument "%s" could not be assigned for %s "%s(%s)" because it is mapped to an internal PHP function which cannot determine default value for optional argument%s "%s".' ,
$name , $callType , $callName , implode ( ', ' , $names ), \count ( $missingArguments ) > 1 ? 's' : '' , implode ( '", "' , $missingArguments )
), $this -> getTemplateLine (), $this -> getSourceContext ());
}
$arguments = array_merge ( $arguments , $optionalArguments );
$arguments [] = $parameters [ $name ];
unset ( $parameters [ $name ]);
$optionalArguments = [];
} elseif ( \array_key_exists ( $pos , $parameters )) {
$arguments = array_merge ( $arguments , $optionalArguments );
$arguments [] = $parameters [ $pos ];
unset ( $parameters [ $pos ]);
$optionalArguments = [];
++ $pos ;
} elseif ( $callableParameter -> isDefaultValueAvailable ()) {
$optionalArguments [] = new ConstantExpression ( $callableParameter -> getDefaultValue (), - 1 );
} elseif ( $callableParameter -> isOptional ()) {
2025-01-13 09:56:01 +00:00
if ( ! $parameters ) {
2024-06-20 14:10:42 +00:00
break ;
} else {
$missingArguments [] = $name ;
}
} else {
2024-09-05 17:51:48 +00:00
throw new SyntaxError ( \sprintf ( 'Value for argument "%s" is required for %s "%s".' , $name , $callType , $callName ), $this -> getTemplateLine (), $this -> getSourceContext ());
2024-06-20 14:10:42 +00:00
}
}
if ( $isVariadic ) {
$arbitraryArguments = $isPhpVariadic ? new VariadicExpression ([], - 1 ) : new ArrayExpression ([], - 1 );
foreach ( $parameters as $key => $value ) {
if ( \is_int ( $key )) {
$arbitraryArguments -> addElement ( $value );
} else {
$arbitraryArguments -> addElement ( $value , new ConstantExpression ( $key , - 1 ));
}
unset ( $parameters [ $key ]);
}
if ( $arbitraryArguments -> count ()) {
$arguments = array_merge ( $arguments , $optionalArguments );
$arguments [] = $arbitraryArguments ;
}
}
2025-01-13 09:56:01 +00:00
if ( $parameters ) {
2024-06-20 14:10:42 +00:00
$unknownParameter = null ;
foreach ( $parameters as $parameter ) {
if ( $parameter instanceof Node ) {
$unknownParameter = $parameter ;
break ;
}
}
throw new SyntaxError (
2024-09-05 17:51:48 +00:00
\sprintf (
2024-06-20 14:10:42 +00:00
'Unknown argument%s "%s" for %s "%s(%s)".' ,
\count ( $parameters ) > 1 ? 's' : '' , implode ( '", "' , array_keys ( $parameters )), $callType , $callName , implode ( ', ' , $names )
),
$unknownParameter ? $unknownParameter -> getTemplateLine () : $this -> getTemplateLine (),
$unknownParameter ? $unknownParameter -> getSourceContext () : $this -> getSourceContext ()
);
}
return $arguments ;
}
2025-01-13 09:56:01 +00:00
/**
* @ deprecated since Twig 3.12
*/
2024-06-20 14:10:42 +00:00
protected function normalizeName ( string $name ) : string
{
2025-01-13 09:56:01 +00:00
trigger_deprecation ( 'twig/twig' , '3.12' , 'The "%s()" method is deprecated.' , __METHOD__ );
2024-06-20 14:10:42 +00:00
return strtolower ( preg_replace ([ '/([A-Z]+)([A-Z][a-z])/' , '/([a-z\d])([A-Z])/' ], [ '\\1_\\2' , '\\1_\\2' ], $name ));
}
2025-01-13 09:56:01 +00:00
// To be removed in 4.0
2024-06-20 14:10:42 +00:00
private function getCallableParameters ( $callable , bool $isVariadic ) : array
{
2025-01-13 09:56:01 +00:00
$twigCallable = $this -> getAttribute ( 'twig_callable' );
$rc = $this -> reflectCallable ( $twigCallable );
2024-09-05 17:51:48 +00:00
$r = $rc -> getReflector ();
$callableName = $rc -> getName ();
2024-06-20 14:10:42 +00:00
$parameters = $r -> getParameters ();
if ( $this -> hasNode ( 'node' )) {
array_shift ( $parameters );
}
2025-01-13 09:56:01 +00:00
if ( $twigCallable -> needsCharset ()) {
2024-06-20 14:10:42 +00:00
array_shift ( $parameters );
}
2025-01-13 09:56:01 +00:00
if ( $twigCallable -> needsEnvironment ()) {
2024-06-20 14:10:42 +00:00
array_shift ( $parameters );
}
2025-01-13 09:56:01 +00:00
if ( $twigCallable -> needsContext ()) {
2024-06-20 14:10:42 +00:00
array_shift ( $parameters );
}
2025-01-13 09:56:01 +00:00
foreach ( $twigCallable -> getArguments () as $argument ) {
array_shift ( $parameters );
2024-06-20 14:10:42 +00:00
}
2025-01-13 09:56:01 +00:00
2024-06-20 14:10:42 +00:00
$isPhpVariadic = false ;
if ( $isVariadic ) {
$argument = end ( $parameters );
2024-09-05 17:51:48 +00:00
$isArray = $argument && $argument -> hasType () && $argument -> getType () instanceof \ReflectionNamedType && 'array' === $argument -> getType () -> getName ();
2024-06-20 14:10:42 +00:00
if ( $isArray && $argument -> isDefaultValueAvailable () && [] === $argument -> getDefaultValue ()) {
array_pop ( $parameters );
} elseif ( $argument && $argument -> isVariadic ()) {
array_pop ( $parameters );
$isPhpVariadic = true ;
} else {
2025-01-13 09:56:01 +00:00
throw new \LogicException ( \sprintf ( 'The last parameter of "%s" for %s "%s" must be an array with default value, eg. "array $arg = []".' , $callableName , $this -> getAttribute ( 'type' ), $twigCallable -> getName ()));
2024-06-20 14:10:42 +00:00
}
}
return [ $parameters , $isPhpVariadic ];
}
2025-01-13 09:56:01 +00:00
private function reflectCallable ( TwigCallableInterface $callable ) : ReflectionCallable
2024-06-20 14:10:42 +00:00
{
2024-09-05 17:51:48 +00:00
if ( ! $this -> reflector ) {
2025-01-13 09:56:01 +00:00
$this -> reflector = new ReflectionCallable ( $callable );
2024-06-20 14:10:42 +00:00
}
2024-09-05 17:51:48 +00:00
return $this -> reflector ;
2024-06-20 14:10:42 +00:00
}
2025-01-13 09:56:01 +00:00
/**
* Overrides the Twig callable based on attributes ( as potentially , attributes changed between the creation and the compilation of the node ) .
*
* To be removed in 4.0 and replace by $this -> getAttribute ( 'twig_callable' ) .
*/
private function getTwigCallable () : TwigCallableInterface
{
$current = $this -> getAttribute ( 'twig_callable' );
$this -> setAttribute ( 'twig_callable' , match ( $this -> getAttribute ( 'type' )) {
'test' => ( new TwigTest (
$this -> getAttribute ( 'name' ),
$this -> hasAttribute ( 'callable' ) ? $this -> getAttribute ( 'callable' ) : $current -> getCallable (),
[
'is_variadic' => $this -> hasAttribute ( 'is_variadic' ) ? $this -> getAttribute ( 'is_variadic' ) : $current -> isVariadic (),
],
)) -> withDynamicArguments ( $this -> getAttribute ( 'name' ), $this -> hasAttribute ( 'dynamic_name' ) ? $this -> getAttribute ( 'dynamic_name' ) : $current -> getDynamicName (), $this -> hasAttribute ( 'arguments' ) ? $this -> getAttribute ( 'arguments' ) : $current -> getArguments ()),
'function' => ( new TwigFunction (
$this -> hasAttribute ( 'name' ) ? $this -> getAttribute ( 'name' ) : $current -> getName (),
$this -> hasAttribute ( 'callable' ) ? $this -> getAttribute ( 'callable' ) : $current -> getCallable (),
[
'needs_environment' => $this -> hasAttribute ( 'needs_environment' ) ? $this -> getAttribute ( 'needs_environment' ) : $current -> needsEnvironment (),
'needs_context' => $this -> hasAttribute ( 'needs_context' ) ? $this -> getAttribute ( 'needs_context' ) : $current -> needsContext (),
'needs_charset' => $this -> hasAttribute ( 'needs_charset' ) ? $this -> getAttribute ( 'needs_charset' ) : $current -> needsCharset (),
'is_variadic' => $this -> hasAttribute ( 'is_variadic' ) ? $this -> getAttribute ( 'is_variadic' ) : $current -> isVariadic (),
],
)) -> withDynamicArguments ( $this -> getAttribute ( 'name' ), $this -> hasAttribute ( 'dynamic_name' ) ? $this -> getAttribute ( 'dynamic_name' ) : $current -> getDynamicName (), $this -> hasAttribute ( 'arguments' ) ? $this -> getAttribute ( 'arguments' ) : $current -> getArguments ()),
'filter' => ( new TwigFilter (
$this -> getAttribute ( 'name' ),
$this -> hasAttribute ( 'callable' ) ? $this -> getAttribute ( 'callable' ) : $current -> getCallable (),
[
'needs_environment' => $this -> hasAttribute ( 'needs_environment' ) ? $this -> getAttribute ( 'needs_environment' ) : $current -> needsEnvironment (),
'needs_context' => $this -> hasAttribute ( 'needs_context' ) ? $this -> getAttribute ( 'needs_context' ) : $current -> needsContext (),
'needs_charset' => $this -> hasAttribute ( 'needs_charset' ) ? $this -> getAttribute ( 'needs_charset' ) : $current -> needsCharset (),
'is_variadic' => $this -> hasAttribute ( 'is_variadic' ) ? $this -> getAttribute ( 'is_variadic' ) : $current -> isVariadic (),
],
)) -> withDynamicArguments ( $this -> getAttribute ( 'name' ), $this -> hasAttribute ( 'dynamic_name' ) ? $this -> getAttribute ( 'dynamic_name' ) : $current -> getDynamicName (), $this -> hasAttribute ( 'arguments' ) ? $this -> getAttribute ( 'arguments' ) : $current -> getArguments ()),
});
return $this -> getAttribute ( 'twig_callable' );
}
2024-06-20 14:10:42 +00:00
}