-
-
Notifications
You must be signed in to change notification settings - Fork 46
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
Lazy services #232
base: master
Are you sure you want to change the base?
Lazy services #232
Changes from all commits
a0fdad8
61f2bff
1b4cfae
abb2e83
15f8a66
dbed5b3
4bdd4bc
592c239
e4bd46f
2f8c8b5
8212a39
5136d8d
47dc088
40ad592
125231d
51789d0
1d59fbd
95cdd9e
0b0b784
20db77b
d1b0c55
feff3b7
5bf1e81
1d04fbc
bc7f70e
fedff5e
51fefa2
File filter
Filter by extension
Conversations
Jump to
Diff view
Diff view
There are no files selected for viewing
Original file line number | Diff line number | Diff line change |
---|---|---|
|
@@ -10,11 +10,13 @@ | |
use Psr\Container\NotFoundExceptionInterface; | ||
use Throwable; | ||
use Yiisoft\Definitions\ArrayDefinition; | ||
use Yiisoft\Definitions\Contract\DefinitionInterface; | ||
use Yiisoft\Definitions\DefinitionStorage; | ||
use Yiisoft\Definitions\Exception\CircularReferenceException; | ||
use Yiisoft\Definitions\Exception\InvalidConfigException; | ||
use Yiisoft\Definitions\Exception\NotInstantiableException; | ||
use Yiisoft\Definitions\Helpers\DefinitionValidator; | ||
use Yiisoft\Definitions\LazyDefinition; | ||
use Yiisoft\Di\Helpers\DefinitionNormalizer; | ||
use Yiisoft\Di\Helpers\DefinitionParser; | ||
use Yiisoft\Di\Reference\TagReference; | ||
|
@@ -30,12 +32,15 @@ | |
|
||
/** | ||
* Container implements a [dependency injection](https://en.wikipedia.org/wiki/Dependency_injection) container. | ||
* | ||
* @psalm-import-type MethodOrPropertyItem from ArrayDefinition | ||
*/ | ||
final class Container implements ContainerInterface | ||
{ | ||
private const META_TAGS = 'tags'; | ||
private const META_RESET = 'reset'; | ||
private const ALLOWED_META = [self::META_TAGS, self::META_RESET]; | ||
private const META_LAZY = 'lazy'; | ||
private const ALLOWED_META = [self::META_TAGS, self::META_RESET, self::META_LAZY]; | ||
|
||
/** | ||
* @var DefinitionStorage Storage of object definitions. | ||
|
@@ -199,14 +204,13 @@ | |
*/ | ||
private function addDefinition(string $id, mixed $definition): void | ||
{ | ||
/** @var mixed $definition */ | ||
[$definition, $meta] = DefinitionParser::parse($definition); | ||
if ($this->validate) { | ||
$this->validateDefinition($definition, $id); | ||
$this->validateMeta($meta); | ||
} | ||
/** | ||
* @psalm-var array{reset?:Closure,tags?:string[]} $meta | ||
* @psalm-var array{reset?:Closure,lazy?:bool,tags?:string[]} $meta | ||
*/ | ||
|
||
if (isset($meta[self::META_TAGS])) { | ||
|
@@ -215,6 +219,9 @@ | |
if (isset($meta[self::META_RESET])) { | ||
$this->setDefinitionResetter($id, $meta[self::META_RESET]); | ||
} | ||
if (isset($meta[self::META_LAZY]) && $meta[self::META_LAZY] === true) { | ||
$definition = $this->decorateLazy($id, $definition); | ||
} | ||
|
||
unset($this->instances[$id]); | ||
$this->addDefinitionToStorage($id, $definition); | ||
|
@@ -276,7 +283,7 @@ | |
|
||
$this->delegates->attach($delegate); | ||
} | ||
$this->definitions->setDelegateContainer($this->delegates); | ||
Check warning on line 286 in src/Container.php
|
||
} | ||
|
||
/** | ||
|
@@ -303,7 +310,7 @@ | |
|
||
$definition = array_merge( | ||
$class === null ? [] : [ArrayDefinition::CLASS_NAME => $class], | ||
[ArrayDefinition::CONSTRUCTOR => $constructorArguments], | ||
Check warning on line 313 in src/Container.php
|
||
// extract only value from parsed definition method | ||
array_map(fn (array $data): mixed => $data[2], $methodsAndProperties), | ||
); | ||
|
@@ -489,7 +496,7 @@ | |
); | ||
} | ||
|
||
$this->building[$id] = 1; | ||
Check warning on line 499 in src/Container.php
|
||
try { | ||
/** @var mixed $object */ | ||
$object = $this->buildInternal($id); | ||
|
@@ -621,4 +628,52 @@ | |
|
||
return $providerInstance; | ||
} | ||
|
||
private function decorateLazy(string $id, mixed $definition): DefinitionInterface | ||
{ | ||
$class = class_exists($id) || interface_exists($id) ? $id : null; | ||
|
||
if (is_array($definition) && isset($definition[DefinitionParser::IS_PREPARED_ARRAY_DEFINITION_DATA])) { | ||
/** | ||
* @psalm-var array{ | ||
* class: class-string|null, | ||
* '__construct()': array, | ||
* methodsAndProperties: array<string, MethodOrPropertyItem> | ||
* } $definition | ||
*/ | ||
if (empty($class)) { | ||
$class = $definition[ArrayDefinition::CLASS_NAME]; | ||
} | ||
$this->checkClassOnNullForLazyService($class, $id, $definition); | ||
$preparedDefinition = ArrayDefinition::fromPreparedData( | ||
$definition[ArrayDefinition::CLASS_NAME] ?? $class, | ||
$definition[ArrayDefinition::CONSTRUCTOR], | ||
$definition['methodsAndProperties'], | ||
); | ||
} else { | ||
$this->checkClassOnNullForLazyService($class, $id, $definition); | ||
$preparedDefinition = $definition; | ||
} | ||
|
||
|
||
|
||
return new LazyDefinition($preparedDefinition, $class); | ||
} | ||
|
||
/** | ||
* @psalm-param class-string|null $class | ||
* @psalm-assert class-string $class | ||
*/ | ||
private function checkClassOnNullForLazyService(?string $class, string $id, mixed $definition): void | ||
{ | ||
if (empty($class)) { | ||
throw new InvalidConfigException( | ||
sprintf( | ||
'Invalid definition: lazy services are only available with array definitions or references. Got type "%s" for definition ID: "%s"', | ||
get_debug_type($definition), | ||
$id, | ||
) | ||
); | ||
} | ||
} | ||
} |
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,120 @@ | ||
<?php | ||
|
||
declare(strict_types=1); | ||
|
||
namespace Yiisoft\Di\Tests\Unit; | ||
|
||
use PHPUnit\Framework\TestCase; | ||
use ProxyManager\Factory\LazyLoadingValueHolderFactory; | ||
use ProxyManager\Proxy\LazyLoadingInterface; | ||
use Yiisoft\Di\Container; | ||
use Yiisoft\Di\ContainerConfig; | ||
use Yiisoft\Di\Tests\Support\EngineMarkOne; | ||
|
||
class LazyServiceContainerTest extends TestCase | ||
{ | ||
protected function setUp(): void | ||
{ | ||
if (!class_exists(LazyLoadingValueHolderFactory::class)) { | ||
$this->markTestSkipped( | ||
'You should install `friendsofphp/proxy-manager-lts` if you want to use lazy services.' | ||
); | ||
} | ||
} | ||
|
||
public function testIsTheSameObject(): void | ||
{ | ||
$class = EngineMarkOne::class; | ||
$number = 55; | ||
|
||
$config = ContainerConfig::create() | ||
->withDefinitions([ | ||
EngineMarkOne::class => [ | ||
'class' => $class, | ||
'setNumber()' => [$number], | ||
'lazy' => true, | ||
], | ||
]); | ||
$container = new Container($config); | ||
|
||
/* @var EngineMarkOne $object */ | ||
$object = $container->get($class); | ||
|
||
self::assertInstanceOf(LazyLoadingInterface::class, $object); | ||
self::assertInstanceOf(EngineMarkOne::class, $object); | ||
self::assertFalse($object->isProxyInitialized()); | ||
self::assertEquals($number, $object->getNumber()); | ||
self::assertTrue($object->isProxyInitialized()); | ||
|
||
/* @var EngineMarkOne $object */ | ||
$object = $container->get($class); | ||
|
||
self::assertInstanceOf(LazyLoadingInterface::class, $object); | ||
self::assertInstanceOf(EngineMarkOne::class, $object); | ||
self::assertTrue($object->isProxyInitialized()); | ||
} | ||
|
||
/** | ||
* @dataProvider lazyDefinitionDataProvider | ||
*/ | ||
public function testLazy(array $definitions, string $id): void | ||
{ | ||
$config = ContainerConfig::create() | ||
->withDefinitions($definitions); | ||
$container = new Container($config); | ||
|
||
$object = $container->get($id); | ||
|
||
self::assertInstanceOf(LazyLoadingInterface::class, $object); | ||
} | ||
|
||
public function lazyDefinitionDataProvider(): array | ||
{ | ||
return [ | ||
'class as definition name' => [ | ||
[ | ||
EngineMarkOne::class => [ | ||
'lazy' => true, | ||
], | ||
], | ||
EngineMarkOne::class, | ||
], | ||
'class as key' => [ | ||
[ | ||
EngineMarkOne::class => [ | ||
'class' => EngineMarkOne::class, | ||
'lazy' => true, | ||
], | ||
], | ||
EngineMarkOne::class, | ||
], | ||
'alias as key' => [ | ||
[ | ||
'mark_one' => [ | ||
'class' => EngineMarkOne::class, | ||
'lazy' => true, | ||
], | ||
], | ||
'mark_one', | ||
], | ||
'dedicated array definition' => [ | ||
[ | ||
EngineMarkOne::class => [ | ||
'definition' => ['class' => EngineMarkOne::class], | ||
'lazy' => true, | ||
], | ||
], | ||
EngineMarkOne::class, | ||
], | ||
'dedicated callback definition' => [ | ||
[ | ||
EngineMarkOne::class => [ | ||
'definition' => fn () => new EngineMarkOne(), | ||
'lazy' => true, | ||
], | ||
], | ||
EngineMarkOne::class, | ||
], | ||
]; | ||
} | ||
} |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
It's always
false
in tests