Skip to content

Commit

Permalink
minor #4525 Optimize NameExpression compilation (fabpot)
Browse files Browse the repository at this point in the history
This PR was merged into the 3.x branch.

Discussion
----------

Optimize NameExpression compilation

Commits
-------

efd12ef Optimize NameExpression compilation
  • Loading branch information
fabpot committed Jan 2, 2025
2 parents 91c8c19 + efd12ef commit 4a871c1
Show file tree
Hide file tree
Showing 2 changed files with 39 additions and 12 deletions.
2 changes: 1 addition & 1 deletion src/Node/Expression/NameExpression.php
Original file line number Diff line number Diff line change
Expand Up @@ -39,7 +39,7 @@ public function compile(Compiler $compiler): void
$compiler->addDebugInfo($this);

if ($this->getAttribute('is_defined_test')) {
if (isset($this->specialVars[$name])) {
if (isset($this->specialVars[$name]) || $this->getAttribute('always_defined')) {
$compiler->repr(true);
} elseif (\PHP_VERSION_ID >= 70400) {
$compiler
Expand Down
49 changes: 38 additions & 11 deletions tests/Node/Expression/Variable/ContextVariableTest.php
Original file line number Diff line number Diff line change
Expand Up @@ -27,20 +27,47 @@ public function testConstructor()

public static function provideTests(): iterable
{
$node = new ContextVariable('foo', 1);
$self = new ContextVariable('_self', 1);
$context = new ContextVariable('_context', 1);
// special variables
foreach (['_self' => '$this->getTemplateName()', '_context' => '$context', '_charset' => '$this->env->getCharset()'] as $special => $compiled) {
$node = new ContextVariable($special, 1);
yield $special => [$node, "// line 1\n$compiled"];
$node = new ContextVariable($special, 1);
$node->setAttribute('is_defined_test', true);
yield $special.'_defined_test' => [$node, "// line 1\ntrue"];
}

$env = new Environment(new ArrayLoader(), ['strict_variables' => true]);
$env1 = new Environment(new ArrayLoader(), ['strict_variables' => false]);
$env = new Environment(new ArrayLoader(), ['strict_variables' => false]);
$envStrict = new Environment(new ArrayLoader(), ['strict_variables' => true]);

// regular
$node = new ContextVariable('foo', 1);
$output = '(isset($context["foo"]) || array_key_exists("foo", $context) ? $context["foo"] : (function () { throw new RuntimeError(\'Variable "foo" does not exist.\', 1, $this->source); })())';
yield 'strict' => [$node, "// line 1\n".$output, $envStrict];
yield 'non_strict' => [$node, self::createVariableGetter('foo', 1), $env];

return [
[$node, "// line 1\n".$output, $env],
[$node, self::createVariableGetter('foo', 1), $env1],
[$self, "// line 1\n\$this->getTemplateName()"],
[$context, "// line 1\n\$context"],
];
// ignore strict check
$node = new ContextVariable('foo', 1);
$node->setAttribute('ignore_strict_check', true);
yield 'ignore_strict_check_strict' => [$node, "// line 1\n(\$context[\"foo\"] ?? null)", $envStrict];
yield 'ignore_strict_check_non_strict' => [$node, "// line 1\n(\$context[\"foo\"] ?? null)", $env];

// always defined
$node = new ContextVariable('foo', 1);
$node->setAttribute('always_defined', true);
yield 'always_defined_strict' => [$node, "// line 1\n\$context[\"foo\"]", $envStrict];
yield 'always_defined_non_strict' => [$node, "// line 1\n\$context[\"foo\"]", $env];

// is defined test
$node = new ContextVariable('foo', 1);
$node->setAttribute('is_defined_test', true);
yield 'is_defined_test_strict' => [$node, "// line 1\narray_key_exists(\"foo\", \$context)", $envStrict];
yield 'is_defined_test_non_strict' => [$node, "// line 1\narray_key_exists(\"foo\", \$context)", $env];

// is defined test // always defined
$node = new ContextVariable('foo', 1);
$node->setAttribute('is_defined_test', true);
$node->setAttribute('always_defined', true);
yield 'is_defined_test_always_defined_strict' => [$node, "// line 1\ntrue", $envStrict];
yield 'is_defined_test_always_defined_non_strict' => [$node, "// line 1\ntrue", $env];
}
}

0 comments on commit 4a871c1

Please sign in to comment.