-
-
Notifications
You must be signed in to change notification settings - Fork 75
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
feat: introduce "in-memory" behavior
- Loading branch information
Showing
18 changed files
with
501 additions
and
15 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,16 @@ | ||
<?php | ||
|
||
declare(strict_types=1); | ||
|
||
namespace Zenstruck\Foundry\Exception; | ||
|
||
/** | ||
* @author Nicolas PHILIPPE <nikophil@gmail.com> | ||
*/ | ||
final class CannotCreateFactory extends \LogicException | ||
{ | ||
public static function argumentCountError(\ArgumentCountError $e): static | ||
{ | ||
return new self('Factories with dependencies (services) cannot be created before foundry is booted.', previous: $e); | ||
} | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,29 @@ | ||
<?php | ||
|
||
/* | ||
* This file is part of the zenstruck/foundry package. | ||
* | ||
* (c) Kevin Bond <kevinbond@gmail.com> | ||
* | ||
* For the full copyright and license information, please view the LICENSE | ||
* file that was distributed with this source code. | ||
*/ | ||
|
||
namespace Zenstruck\Foundry; | ||
|
||
/** | ||
* @author Nicolas PHILIPPE <nikophil@gmail.com> | ||
* | ||
* @internal | ||
*/ | ||
interface FactoryRegistryInterface | ||
{ | ||
/** | ||
* @template T of Factory | ||
* | ||
* @param class-string<T> $class | ||
* | ||
* @return T | ||
*/ | ||
public function get(string $class): Factory; | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,22 @@ | ||
<?php | ||
|
||
declare(strict_types=1); | ||
|
||
namespace Zenstruck\Foundry\InMemory; | ||
|
||
// todo: remove this attribute in favor to interface? | ||
#[\Attribute(\Attribute::TARGET_CLASS)] | ||
/** | ||
* @author Nicolas PHILIPPE <nikophil@gmail.com> | ||
*/ | ||
final class AsInMemoryRepository | ||
{ | ||
public function __construct( | ||
public readonly string $class | ||
) | ||
{ | ||
if (!class_exists($this->class)) { | ||
throw new \InvalidArgumentException("Wrong definition for \"AsInMemoryRepository\" attribute: class \"{$this->class}\" does not exist."); | ||
} | ||
} | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,52 @@ | ||
<?php | ||
|
||
declare(strict_types=1); | ||
|
||
namespace Zenstruck\Foundry\InMemory\DependencyInjection; | ||
|
||
use Symfony\Component\DependencyInjection\Compiler\CompilerPassInterface; | ||
use Symfony\Component\DependencyInjection\Compiler\ServiceLocatorTagPass; | ||
use Symfony\Component\DependencyInjection\ContainerBuilder; | ||
use Symfony\Component\DependencyInjection\Reference; | ||
use Zenstruck\Foundry\InMemory\InMemoryFactoryRegistry; | ||
|
||
/** | ||
* @internal | ||
* @author Nicolas PHILIPPE <nikophil@gmail.com> | ||
*/ | ||
final class InMemoryCompilerPass implements CompilerPassInterface | ||
{ | ||
public function process(ContainerBuilder $container): void | ||
{ | ||
// create a service locator with all "in memory" repositories, indexed by target class | ||
$inMemoryRepositoriesServices = $container->findTaggedServiceIds('foundry.in_memory.repository'); | ||
$inMemoryRepositoriesLocator = ServiceLocatorTagPass::register( | ||
$container, | ||
array_combine( | ||
array_map( | ||
static function (array $tags) { | ||
if (\count($tags) !== 1) { | ||
throw new \LogicException('Cannot have multiple tags "foundry.in_memory.repository" on a service!'); | ||
} | ||
|
||
return $tags[0]['class'] ?? throw new \LogicException('Invalid tag definition of "foundry.in_memory.repository".'); | ||
}, | ||
array_values($inMemoryRepositoriesServices) | ||
), | ||
array_map( | ||
static fn(string $inMemoryRepositoryId) => new Reference($inMemoryRepositoryId), | ||
array_keys($inMemoryRepositoriesServices) | ||
), | ||
) | ||
); | ||
|
||
// todo: should we check we only have a 1 repository per class? | ||
$container->register('.zenstruck_foundry.in_memory.factory_registry') | ||
->setClass(InMemoryFactoryRegistry::class) | ||
->setDecoratedService('.zenstruck_foundry.factory_registry') | ||
->setArgument('$decorated', $container->getDefinition('.zenstruck_foundry.factory_registry')) | ||
->setArgument('$inMemoryRepositories', $inMemoryRepositoriesLocator) | ||
; | ||
} | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,43 @@ | ||
<?php | ||
|
||
declare(strict_types=1); | ||
|
||
namespace Zenstruck\Foundry\InMemory; | ||
|
||
/** | ||
* @template T of object | ||
* @implements InMemoryRepository<T> | ||
* @author Nicolas PHILIPPE <nikophil@gmail.com> | ||
* | ||
* This class will be used when a specific "in-memory" repository does not exist for a given class. | ||
*/ | ||
final class GenericInMemoryRepository implements InMemoryRepository | ||
{ | ||
/** | ||
* @var list<T> | ||
*/ | ||
private array $elements = []; | ||
|
||
/** | ||
* @param class-string<T> $class | ||
*/ | ||
public function __construct( | ||
private readonly string $class | ||
) | ||
{ | ||
} | ||
|
||
/** | ||
* @param T $element | ||
*/ | ||
public function _save(object $element): void | ||
{ | ||
if (!$element instanceof $this->class) { | ||
throw new \InvalidArgumentException(sprintf('Given object of class "%s" is not an instance of expected "%s"', $element::class, $this->class)); | ||
} | ||
|
||
if (!in_array($element, $this->elements, true)) { | ||
$this->elements[] = $element; | ||
} | ||
} | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,73 @@ | ||
<?php | ||
|
||
declare(strict_types=1); | ||
|
||
namespace Zenstruck\Foundry\InMemory; | ||
|
||
use Symfony\Component\DependencyInjection\ServiceLocator; | ||
use Zenstruck\Foundry\Configuration; | ||
use Zenstruck\Foundry\Factory; | ||
use Zenstruck\Foundry\FactoryRegistryInterface; | ||
use Zenstruck\Foundry\Persistence\PersistentObjectFactory; | ||
|
||
/** | ||
* @internal | ||
* @template T of object | ||
* @author Nicolas PHILIPPE <nikophil@gmail.com> | ||
*/ | ||
final class InMemoryFactoryRegistry implements FactoryRegistryInterface | ||
{ | ||
/** | ||
* @var array<class-string<T>, GenericInMemoryRepository<T>> | ||
*/ | ||
private array $genericInMemoryRepositories = []; | ||
|
||
public function __construct( // @phpstan-ignore-line | ||
private readonly FactoryRegistryInterface $decorated, | ||
/** @var ServiceLocator<InMemoryRepository> */ | ||
private readonly ServiceLocator $inMemoryRepositories, | ||
) { | ||
} | ||
|
||
/** | ||
* @template TFactory of Factory | ||
* | ||
* @param class-string<TFactory> $class | ||
* | ||
* @return TFactory | ||
*/ | ||
public function get(string $class): Factory | ||
{ | ||
$factory = $this->decorated->get($class); | ||
|
||
if (!$factory instanceof PersistentObjectFactory) { // todo shall we support ObjectFactory as well? | ||
return $factory; | ||
} | ||
|
||
$configuration = Configuration::instance(); | ||
|
||
if (!$configuration->isInMemoryEnabled()) { | ||
return $factory; | ||
} | ||
|
||
return $factory->withoutPersisting() | ||
->afterInstantiate( | ||
fn(object $object) => $this->findInMemoryRepository($factory)->_save($object) // @phpstan-ignore-line | ||
); | ||
} | ||
|
||
/** | ||
* @param PersistentObjectFactory<T> $factory | ||
* | ||
* @return InMemoryRepository<T> | ||
*/ | ||
private function findInMemoryRepository(PersistentObjectFactory $factory): InMemoryRepository | ||
{ | ||
$targetClass = $factory::class(); | ||
if (!$this->inMemoryRepositories->has($targetClass)) { | ||
return $this->genericInMemoryRepositories[$targetClass] ??= new GenericInMemoryRepository($targetClass); | ||
} | ||
|
||
return $this->inMemoryRepositories->get($targetClass); | ||
} | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,18 @@ | ||
<?php | ||
|
||
declare(strict_types=1); | ||
|
||
namespace Zenstruck\Foundry\InMemory; | ||
|
||
/** | ||
* @author Nicolas PHILIPPE <nikophil@gmail.com> | ||
* | ||
* @template T of object | ||
*/ | ||
interface InMemoryRepository | ||
{ | ||
/** | ||
* @param T $element | ||
*/ | ||
public function _save(object $element): void; | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,22 @@ | ||
<?php | ||
|
||
/* | ||
* This file is part of the zenstruck/foundry package. | ||
* | ||
* (c) Kevin Bond <kevinbond@gmail.com> | ||
* | ||
* For the full copyright and license information, please view the LICENSE | ||
* file that was distributed with this source code. | ||
*/ | ||
|
||
namespace Zenstruck\Foundry\InMemory; | ||
|
||
use Zenstruck\Foundry\Configuration; | ||
|
||
/** | ||
* Enable "in memory" repositories globally. | ||
*/ | ||
function enable_in_memory(): void | ||
{ | ||
Configuration::instance()->enableInMemory(); | ||
} |
Oops, something went wrong.