From bd467519ab1dbd24d4945fa0bcf57ec70b4e9e60 Mon Sep 17 00:00:00 2001 From: David Grudl Date: Tue, 9 Jul 2024 12:50:36 +0200 Subject: [PATCH] {default} is evaluated lazily --- src/Latte/Essential/Nodes/VarNode.php | 29 ++++++--------- tests/tags/expected/inheritance.1.parent.php | 4 ++- tests/tags/var.default.phpt | 37 +++++++++++++++----- 3 files changed, 42 insertions(+), 28 deletions(-) diff --git a/src/Latte/Essential/Nodes/VarNode.php b/src/Latte/Essential/Nodes/VarNode.php index f7cb4c867..d3e67410b 100644 --- a/src/Latte/Essential/Nodes/VarNode.php +++ b/src/Latte/Essential/Nodes/VarNode.php @@ -11,7 +11,6 @@ use Latte\Compiler\Nodes\Php\Expression\AssignNode; use Latte\Compiler\Nodes\Php\Expression\VariableNode; -use Latte\Compiler\Nodes\Php\ExpressionNode; use Latte\Compiler\Nodes\Php\Scalar\NullNode; use Latte\Compiler\Nodes\StatementNode; use Latte\Compiler\PrintContext; @@ -67,26 +66,18 @@ private static function parseAssignments(Tag $tag, bool $default): array public function print(PrintContext $context): string { $res = []; - if ($this->default) { - foreach ($this->assignments as $assign) { + foreach ($this->assignments as $assign) { + if ($this->default) { assert($assign->var instanceof VariableNode); - if ($assign->var->name instanceof ExpressionNode) { - $var = $assign->var->name->print($context); - } else { - $var = $context->encodeString($assign->var->name); - } - $res[] = $var . ' => ' . $assign->expr->print($context); + $res[] = $context->format( + '%node ??= array_key_exists(%raw, get_defined_vars()) ? null : %node', + $assign->var, + $context->encodeString($assign->var->name), + $assign->expr, + ); + } else { + $res[] = $assign->print($context); } - - return $context->format( - 'extract([%raw], EXTR_SKIP) %line;', - implode(', ', $res), - $this->position, - ); - } - - foreach ($this->assignments as $assign) { - $res[] = $assign->print($context); } return $context->format( diff --git a/tests/tags/expected/inheritance.1.parent.php b/tests/tags/expected/inheritance.1.parent.php index cdad6e4fa..dd6c365bc 100644 --- a/tests/tags/expected/inheritance.1.parent.php +++ b/tests/tags/expected/inheritance.1.parent.php @@ -53,7 +53,9 @@ public function prepare(): array { extract($this->params); - extract(['class' => null, 'namespace' => null, 'top' => true], EXTR_SKIP) /* line %d% */; + $class ??= array_key_exists('class', get_defined_vars()) ? null : null; + $namespace ??= array_key_exists('namespace', get_defined_vars()) ? null : null; + $top ??= array_key_exists('top', get_defined_vars()) ? null : true /* line 1 */; return get_defined_vars(); } diff --git a/tests/tags/var.default.phpt b/tests/tags/var.default.phpt index e8e5bf843..5c70fe3f6 100644 --- a/tests/tags/var.default.phpt +++ b/tests/tags/var.default.phpt @@ -42,13 +42,37 @@ test('{var ...}', function () use ($latte) { test('{default ...}', function () use ($latte) { - Assert::contains("extract(['var' => null, 'var2' => null], EXTR_SKIP) /*", $latte->compile('{default $var, $var2}')); - Assert::contains("extract(['var' => 123], EXTR_SKIP) /*", $latte->compile('{default $var = 123}')); - Assert::contains("extract(['var1' => 123, 'var2' => 'nette framework'], EXTR_SKIP) /*", $latte->compile('{default $var1 = 123, $var2 = "nette framework"}')); + Assert::match(<<<'XX' + %A% + $var ??= array_key_exists('var', get_defined_vars()) ? null : null; + $var2 ??= array_key_exists('var2', get_defined_vars()) ? null : null /* line 1 */; + %A% + XX, $latte->compile('{default $var, $var2}')); + + Assert::contains( + '$var ??= array_key_exists(\'var\', get_defined_vars()) ? null : 123 /* line 1 */;', + $latte->compile('{default $var = 123}'), + ); + + Assert::match(<<<'XX' + %A% + $var1 ??= array_key_exists('var1', get_defined_vars()) ? null : 123; + $var2 ??= array_key_exists('var2', get_defined_vars()) ? null : 'nette framework' /* line 1 */; + %A% + XX, $latte->compile('{default $var1 = 123, $var2 = "nette framework"}')); // types - Assert::contains("extract(['var' => 123], EXTR_SKIP) /*", $latte->compile('{default null|int|string[] $var = 123}')); - Assert::contains("extract(['var1' => 123, 'var2' => 'nette framework'], EXTR_SKIP) /*", $latte->compile('{default int|string[] $var1 = 123, ?class $var2 = "nette framework"}')); + Assert::contains( + '$var ??= array_key_exists(\'var\', get_defined_vars()) ? null : 123 /* line 1 */;', + $latte->compile('{default null|int|string[] $var = 123}'), + ); + + Assert::match(<<<'XX' + %A% + $var1 ??= array_key_exists('var1', get_defined_vars()) ? null : 123; + $var2 ??= array_key_exists('var2', get_defined_vars()) ? null : 'nette framework' /* line 1 */; + %A% + XX, $latte->compile('{default int|string[] $var1 = 123, ?class $var2 = "nette framework"}')); // errors Assert::exception( @@ -62,7 +86,4 @@ test('{default ...}', function () use ($latte) { Latte\CompileException::class, 'Unexpected end in {default} (on line 1 at column 30)', ); - - // preprocess - Assert::contains("extract(['var1' => true ? 'a' : null], EXTR_SKIP) /*", $latte->compile('{default $var1 = true ? a}')); });