Skip to content
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.

Commit c3bedf9

Browse files
committedDec 2, 2024·
creating lazy services in PHP 8.4 WIP
1 parent d5b56b4 commit c3bedf9

File tree

5 files changed

+48
-9
lines changed

5 files changed

+48
-9
lines changed
 

‎src/DI/Definitions/ServiceDefinition.php

+27-9
Original file line numberDiff line numberDiff line change
@@ -11,6 +11,7 @@
1111

1212
use Nette;
1313
use Nette\DI\ServiceCreationException;
14+
use Nette\Utils\Strings;
1415

1516

1617
/**
@@ -24,6 +25,8 @@ final class ServiceDefinition extends Definition
2425
{
2526
use Nette\SmartObject;
2627

28+
public ?bool $lazy = null;
29+
2730
private Statement $creator;
2831

2932
/** @var Statement[] */
@@ -181,18 +184,33 @@ private function prependSelf(Statement $setup): Statement
181184

182185
public function generateCode(Nette\DI\PhpGenerator $generator): string
183186
{
184-
$code = $this->creator->generateCode($generator) . ";\n";
185-
if (!$this->setup) {
186-
return 'return ' . $code;
187+
$lines = [];
188+
foreach ([$this->creator, ...$this->setup] as $stmt) {
189+
$lines[] = $stmt->generateCode($generator) . ";\n";
187190
}
188191

189-
$code = '$service = ' . $code;
190-
foreach ($this->setup as $setup) {
191-
$code .= $setup->generateCode($generator) . ";\n";
192+
$class = $this->creator->getEntity();
193+
$lazy = PHP_VERSION_ID >= 80400
194+
&& $this->lazy
195+
&& is_string($class)
196+
&& ($ancestor = ($tmp = class_parents($class)) ? array_pop($tmp) : $class)
197+
&& !(new \ReflectionClass($ancestor))->isInternal()
198+
&& !preg_grep('#(?:func_get_arg|func_num_args)#i', $lines); // latteFactory workaround
199+
200+
if ($lazy) {
201+
$lines[0] = (new \ReflectionClass($class))->hasMethod('__construct')
202+
? $generator->formatPhp("\$service->__construct(...?:);\n", [$this->creator->arguments])
203+
: '';
204+
return "return new ReflectionClass($class::class)->newLazyGhost(function (\$service) {\n"
205+
. Strings::indent(implode('', $lines))
206+
. '});';
207+
208+
} elseif (count($lines) === 1) {
209+
return 'return ' . $lines[0];
210+
211+
} else {
212+
return '$service = ' . implode('', $lines) . 'return $service;';
192213
}
193-
194-
$code .= 'return $service;';
195-
return $code;
196214
}
197215

198216

‎src/DI/Extensions/DIExtension.php

+15
Original file line numberDiff line numberDiff line change
@@ -10,6 +10,7 @@
1010
namespace Nette\DI\Extensions;
1111

1212
use Nette;
13+
use Nette\DI\Definitions\ServiceDefinition;
1314
use Tracy;
1415

1516

@@ -36,6 +37,7 @@ public function __construct(bool $debugMode = false)
3637
public array $excluded = [];
3738
public ?string $parentClass = null;
3839
public object $export;
40+
public bool $lazy = false;
3941
};
4042
$this->config->export = new class {
4143
public bool $parameters = true;
@@ -56,6 +58,19 @@ public function loadConfiguration(): void
5658
}
5759

5860

61+
public function beforeCompile(): void
62+
{
63+
if ($this->config->lazy) {
64+
$builder = $this->getContainerBuilder();
65+
foreach ($builder->getDefinitions() as $def) {
66+
if ($def instanceof ServiceDefinition) {
67+
$def->lazy ??= true;
68+
}
69+
}
70+
}
71+
}
72+
73+
5974
public function afterCompile(Nette\PhpGenerator\ClassType $class): void
6075
{
6176
if ($this->config->parentClass) {

‎src/DI/Extensions/DefinitionSchema.php

+1
Original file line numberDiff line numberDiff line change
@@ -173,6 +173,7 @@ private static function getServiceSchema(): Schema
173173
'tags' => Expect::array(),
174174
'reset' => Expect::array(),
175175
'alteration' => Expect::bool(),
176+
'lazy' => Expect::bool(),
176177
]);
177178
}
178179

‎src/DI/Extensions/ServicesExtension.php

+4
Original file line numberDiff line numberDiff line change
@@ -113,6 +113,10 @@ private function updateServiceDefinition(Definitions\ServiceDefinition $definiti
113113
if (isset($config->inject)) {
114114
$definition->addTag(InjectExtension::TagInject, $config->inject);
115115
}
116+
117+
if (isset($config->lazy)) {
118+
$definition->lazy = $config->lazy;
119+
}
116120
}
117121

118122

‎tests/DI/Compiler.loadConfig.include.phpt

+1
Original file line numberDiff line numberDiff line change
@@ -49,6 +49,7 @@ Assert::equal([
4949
'tags' => [],
5050
'reset' => [],
5151
'alteration' => null,
52+
'lazy' => null,
5253
'defType' => Nette\DI\Definitions\ServiceDefinition::class,
5354
],
5455
],

0 commit comments

Comments
 (0)
Please sign in to comment.