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 f6684d2

Browse files
committedMay 2, 2024··
feat: introduce "in-memory" behavior
1 parent 71bbaa2 commit f6684d2

18 files changed

+501
-15
lines changed
 

‎composer.json

+6-1
Original file line numberDiff line numberDiff line change
@@ -44,7 +44,12 @@
4444
},
4545
"autoload": {
4646
"psr-4": { "Zenstruck\\Foundry\\": "src/" },
47-
"files": ["src/functions.php", "src/Persistence/functions.php", "src/phpunit_helper.php"]
47+
"files": [
48+
"src/functions.php",
49+
"src/Persistence/functions.php",
50+
"src/phpunit_helper.php",
51+
"src/InMemory/functions.php"
52+
]
4853
},
4954
"autoload-dev": {
5055
"psr-4": {

‎src/Configuration.php

+13-1
Original file line numberDiff line numberDiff line change
@@ -36,11 +36,13 @@ final class Configuration
3636
/** @var \Closure():self|self|null */
3737
private static \Closure|self|null $instance = null;
3838

39+
private bool $inMemory = false;
40+
3941
/**
4042
* @param InstantiatorCallable $instantiator
4143
*/
4244
public function __construct(
43-
public readonly FactoryRegistry $factories,
45+
public readonly FactoryRegistryInterface $factories,
4446
public readonly Faker\Generator $faker,
4547
callable $instantiator,
4648
public readonly StoryRegistry $stories,
@@ -90,4 +92,14 @@ public static function shutdown(): void
9092
StoryRegistry::reset();
9193
self::$instance = null;
9294
}
95+
96+
public function enableInMemory(): void
97+
{
98+
$this->inMemory = true;
99+
}
100+
101+
public function isInMemoryEnabled(): bool
102+
{
103+
return $this->inMemory;
104+
}
93105
}

‎src/Exception/CannotCreateFactory.php

+16
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,16 @@
1+
<?php
2+
3+
declare(strict_types=1);
4+
5+
namespace Zenstruck\Foundry\Exception;
6+
7+
/**
8+
* @author Nicolas PHILIPPE <nikophil@gmail.com>
9+
*/
10+
final class CannotCreateFactory extends \LogicException
11+
{
12+
public static function argumentCountError(\ArgumentCountError $e): static
13+
{
14+
return new self('Factories with dependencies (services) cannot be created before foundry is booted.', previous: $e);
15+
}
16+
}

‎src/Factory.php

+2-2
Original file line numberDiff line numberDiff line change
@@ -12,6 +12,7 @@
1212
namespace Zenstruck\Foundry;
1313

1414
use Faker;
15+
use Zenstruck\Foundry\Exception\CannotCreateFactory;
1516

1617
/**
1718
* @author Kevin Bond <kevinbond@gmail.com>
@@ -33,7 +34,6 @@ public function __construct()
3334
{
3435
}
3536

36-
3737
/**
3838
* @param Attributes $attributes
3939
*/
@@ -46,7 +46,7 @@ final public static function new(array|callable $attributes = []): static
4646
try {
4747
$factory ??= new static(); // @phpstan-ignore-line
4848
} catch (\ArgumentCountError $e) {
49-
throw new \LogicException('Factories with dependencies (services) cannot be created before foundry is booted.', previous: $e);
49+
throw CannotCreateFactory::argumentCountError($e);
5050
}
5151

5252
return $factory->initialize()->with($attributes);

‎src/FactoryRegistry.php

+9-10
Original file line numberDiff line numberDiff line change
@@ -11,12 +11,14 @@
1111

1212
namespace Zenstruck\Foundry;
1313

14+
use Zenstruck\Foundry\Exception\CannotCreateFactory;
15+
1416
/**
1517
* @author Kevin Bond <kevinbond@gmail.com>
1618
*
1719
* @internal
1820
*/
19-
final class FactoryRegistry
21+
final class FactoryRegistry implements FactoryRegistryInterface
2022
{
2123
/**
2224
* @param Factory<mixed>[] $factories
@@ -25,21 +27,18 @@ public function __construct(private iterable $factories)
2527
{
2628
}
2729

28-
/**
29-
* @template T of Factory
30-
*
31-
* @param class-string<T> $class
32-
*
33-
* @return T|null
34-
*/
35-
public function get(string $class): ?Factory
30+
public function get(string $class): Factory
3631
{
3732
foreach ($this->factories as $factory) {
3833
if ($class === $factory::class) {
3934
return $factory; // @phpstan-ignore-line
4035
}
4136
}
4237

43-
return null;
38+
try {
39+
return new $class();
40+
} catch (\ArgumentCountError $e) {
41+
throw CannotCreateFactory::argumentCountError($e);
42+
}
4443
}
4544
}

‎src/FactoryRegistryInterface.php

+29
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,29 @@
1+
<?php
2+
3+
/*
4+
* This file is part of the zenstruck/foundry package.
5+
*
6+
* (c) Kevin Bond <kevinbond@gmail.com>
7+
*
8+
* For the full copyright and license information, please view the LICENSE
9+
* file that was distributed with this source code.
10+
*/
11+
12+
namespace Zenstruck\Foundry;
13+
14+
/**
15+
* @author Nicolas PHILIPPE <nikophil@gmail.com>
16+
*
17+
* @internal
18+
*/
19+
interface FactoryRegistryInterface
20+
{
21+
/**
22+
* @template T of Factory
23+
*
24+
* @param class-string<T> $class
25+
*
26+
* @return T
27+
*/
28+
public function get(string $class): Factory;
29+
}

‎src/InMemory/AsInMemoryRepository.php

+22
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,22 @@
1+
<?php
2+
3+
declare(strict_types=1);
4+
5+
namespace Zenstruck\Foundry\InMemory;
6+
7+
// todo: remove this attribute in favor to interface?
8+
#[\Attribute(\Attribute::TARGET_CLASS)]
9+
/**
10+
* @author Nicolas PHILIPPE <nikophil@gmail.com>
11+
*/
12+
final class AsInMemoryRepository
13+
{
14+
public function __construct(
15+
public readonly string $class
16+
)
17+
{
18+
if (!class_exists($this->class)) {
19+
throw new \InvalidArgumentException("Wrong definition for \"AsInMemoryRepository\" attribute: class \"{$this->class}\" does not exist.");
20+
}
21+
}
22+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,52 @@
1+
<?php
2+
3+
declare(strict_types=1);
4+
5+
namespace Zenstruck\Foundry\InMemory\DependencyInjection;
6+
7+
use Symfony\Component\DependencyInjection\Compiler\CompilerPassInterface;
8+
use Symfony\Component\DependencyInjection\Compiler\ServiceLocatorTagPass;
9+
use Symfony\Component\DependencyInjection\ContainerBuilder;
10+
use Symfony\Component\DependencyInjection\Reference;
11+
use Zenstruck\Foundry\InMemory\InMemoryFactoryRegistry;
12+
13+
/**
14+
* @internal
15+
* @author Nicolas PHILIPPE <nikophil@gmail.com>
16+
*/
17+
final class InMemoryCompilerPass implements CompilerPassInterface
18+
{
19+
public function process(ContainerBuilder $container): void
20+
{
21+
// create a service locator with all "in memory" repositories, indexed by target class
22+
$inMemoryRepositoriesServices = $container->findTaggedServiceIds('foundry.in_memory.repository');
23+
$inMemoryRepositoriesLocator = ServiceLocatorTagPass::register(
24+
$container,
25+
array_combine(
26+
array_map(
27+
static function (array $tags) {
28+
if (\count($tags) !== 1) {
29+
throw new \LogicException('Cannot have multiple tags "foundry.in_memory.repository" on a service!');
30+
}
31+
32+
return $tags[0]['class'] ?? throw new \LogicException('Invalid tag definition of "foundry.in_memory.repository".');
33+
},
34+
array_values($inMemoryRepositoriesServices)
35+
),
36+
array_map(
37+
static fn(string $inMemoryRepositoryId) => new Reference($inMemoryRepositoryId),
38+
array_keys($inMemoryRepositoriesServices)
39+
),
40+
)
41+
);
42+
43+
// todo: should we check we only have a 1 repository per class?
44+
45+
$container->register('.zenstruck_foundry.in_memory.factory_registry')
46+
->setClass(InMemoryFactoryRegistry::class)
47+
->setDecoratedService('.zenstruck_foundry.factory_registry')
48+
->setArgument('$decorated', $container->getDefinition('.zenstruck_foundry.factory_registry'))
49+
->setArgument('$inMemoryRepositories', $inMemoryRepositoriesLocator)
50+
;
51+
}
52+
}
+43
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,43 @@
1+
<?php
2+
3+
declare(strict_types=1);
4+
5+
namespace Zenstruck\Foundry\InMemory;
6+
7+
/**
8+
* @template T of object
9+
* @implements InMemoryRepository<T>
10+
* @author Nicolas PHILIPPE <nikophil@gmail.com>
11+
*
12+
* This class will be used when a specific "in-memory" repository does not exist for a given class.
13+
*/
14+
final class GenericInMemoryRepository implements InMemoryRepository
15+
{
16+
/**
17+
* @var list<T>
18+
*/
19+
private array $elements = [];
20+
21+
/**
22+
* @param class-string<T> $class
23+
*/
24+
public function __construct(
25+
private readonly string $class
26+
)
27+
{
28+
}
29+
30+
/**
31+
* @param T $element
32+
*/
33+
public function _save(object $element): void
34+
{
35+
if (!$element instanceof $this->class) {
36+
throw new \InvalidArgumentException(sprintf('Given object of class "%s" is not an instance of expected "%s"', $element::class, $this->class));
37+
}
38+
39+
if (!in_array($element, $this->elements, true)) {
40+
$this->elements[] = $element;
41+
}
42+
}
43+
}
+73
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,73 @@
1+
<?php
2+
3+
declare(strict_types=1);
4+
5+
namespace Zenstruck\Foundry\InMemory;
6+
7+
use Symfony\Component\DependencyInjection\ServiceLocator;
8+
use Zenstruck\Foundry\Configuration;
9+
use Zenstruck\Foundry\Factory;
10+
use Zenstruck\Foundry\FactoryRegistryInterface;
11+
use Zenstruck\Foundry\Persistence\PersistentObjectFactory;
12+
13+
/**
14+
* @internal
15+
* @template T of object
16+
* @author Nicolas PHILIPPE <nikophil@gmail.com>
17+
*/
18+
final class InMemoryFactoryRegistry implements FactoryRegistryInterface
19+
{
20+
/**
21+
* @var array<class-string<T>, GenericInMemoryRepository<T>>
22+
*/
23+
private array $genericInMemoryRepositories = [];
24+
25+
public function __construct( // @phpstan-ignore-line
26+
private readonly FactoryRegistryInterface $decorated,
27+
/** @var ServiceLocator<InMemoryRepository> */
28+
private readonly ServiceLocator $inMemoryRepositories,
29+
) {
30+
}
31+
32+
/**
33+
* @template TFactory of Factory
34+
*
35+
* @param class-string<TFactory> $class
36+
*
37+
* @return TFactory
38+
*/
39+
public function get(string $class): Factory
40+
{
41+
$factory = $this->decorated->get($class);
42+
43+
if (!$factory instanceof PersistentObjectFactory) { // todo shall we support ObjectFactory as well?
44+
return $factory;
45+
}
46+
47+
$configuration = Configuration::instance();
48+
49+
if (!$configuration->isInMemoryEnabled()) {
50+
return $factory;
51+
}
52+
53+
return $factory->withoutPersisting()
54+
->afterInstantiate(
55+
fn(object $object) => $this->findInMemoryRepository($factory)->_save($object) // @phpstan-ignore-line
56+
);
57+
}
58+
59+
/**
60+
* @param PersistentObjectFactory<T> $factory
61+
*
62+
* @return InMemoryRepository<T>
63+
*/
64+
private function findInMemoryRepository(PersistentObjectFactory $factory): InMemoryRepository
65+
{
66+
$targetClass = $factory::class();
67+
if (!$this->inMemoryRepositories->has($targetClass)) {
68+
return $this->genericInMemoryRepositories[$targetClass] ??= new GenericInMemoryRepository($targetClass);
69+
}
70+
71+
return $this->inMemoryRepositories->get($targetClass);
72+
}
73+
}

‎src/InMemory/InMemoryRepository.php

+18
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,18 @@
1+
<?php
2+
3+
declare(strict_types=1);
4+
5+
namespace Zenstruck\Foundry\InMemory;
6+
7+
/**
8+
* @author Nicolas PHILIPPE <nikophil@gmail.com>
9+
*
10+
* @template T of object
11+
*/
12+
interface InMemoryRepository
13+
{
14+
/**
15+
* @param T $element
16+
*/
17+
public function _save(object $element): void;
18+
}

‎src/InMemory/functions.php

+22
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,22 @@
1+
<?php
2+
3+
/*
4+
* This file is part of the zenstruck/foundry package.
5+
*
6+
* (c) Kevin Bond <kevinbond@gmail.com>
7+
*
8+
* For the full copyright and license information, please view the LICENSE
9+
* file that was distributed with this source code.
10+
*/
11+
12+
namespace Zenstruck\Foundry\InMemory;
13+
14+
use Zenstruck\Foundry\Configuration;
15+
16+
/**
17+
* Enable "in memory" repositories globally.
18+
*/
19+
function enable_in_memory(): void
20+
{
21+
Configuration::instance()->enableInMemory();
22+
}

‎src/Persistence/PersistentObjectFactory.php

+4-1
Original file line numberDiff line numberDiff line change
@@ -274,7 +274,10 @@ protected function normalizeParameter(string $field, mixed $value): mixed
274274
$value->persist = $this->persist; // todo - breaks immutability
275275
}
276276

277-
if ($value instanceof self && Configuration::instance()->persistence()->relationshipMetadata(static::class(), $value::class(), $field)?->isCascadePersist) {
277+
if ($value instanceof self
278+
&& !Configuration::instance()->isInMemoryEnabled()
279+
&& Configuration::instance()->persistence()->relationshipMetadata(static::class(), $value::class(), $field)?->isCascadePersist
280+
) {
278281
$value->persist = false;
279282
}
280283

‎src/ZenstruckFoundryBundle.php

+19
Original file line numberDiff line numberDiff line change
@@ -12,11 +12,15 @@
1212
namespace Zenstruck\Foundry;
1313

1414
use Symfony\Component\Config\Definition\Configurator\DefinitionConfigurator;
15+
use Symfony\Component\DependencyInjection\ChildDefinition;
1516
use Symfony\Component\DependencyInjection\Compiler\CompilerPassInterface;
1617
use Symfony\Component\DependencyInjection\ContainerBuilder;
1718
use Symfony\Component\DependencyInjection\Loader\Configurator\ContainerConfigurator;
1819
use Symfony\Component\DependencyInjection\Reference;
1920
use Symfony\Component\HttpKernel\Bundle\AbstractBundle;
21+
use Zenstruck\Foundry\InMemory\AsInMemoryRepository;
22+
use Zenstruck\Foundry\InMemory\DependencyInjection\InMemoryCompilerPass;
23+
use Zenstruck\Foundry\InMemory\InMemoryRepository;
2024
use Zenstruck\Foundry\Object\Instantiator;
2125
use Zenstruck\Foundry\ORM\AbstractORMPersistenceStrategy;
2226

@@ -211,13 +215,28 @@ public function loadExtension(array $config, ContainerConfigurator $configurator
211215
->replaceArgument(1, $config['mongo'])
212216
;
213217
}
218+
219+
// tag with "foundry.in_memory.repository" all classes using attribute "AsInMemoryRepository"
220+
$container->registerAttributeForAutoconfiguration(
221+
AsInMemoryRepository::class,
222+
static function (ChildDefinition $definition, AsInMemoryRepository $attribute, \ReflectionClass $reflector) { // @phpstan-ignore-line
223+
if (!is_a($reflector->name, InMemoryRepository::class, true)) {
224+
throw new \LogicException(sprintf("Service \"%s\" with attribute \"AsInMemoryRepository\" must implement \"%s\".", $reflector->name, InMemoryRepository::class));
225+
}
226+
227+
$definition->addTag('foundry.in_memory.repository', ['class' => $attribute->class]);
228+
}
229+
);
214230
}
215231

216232
public function build(ContainerBuilder $container): void
217233
{
218234
parent::build($container);
219235

220236
$container->addCompilerPass($this);
237+
238+
// todo: should we find a way to decouple Foundry from its "plugins"?
239+
$container->addCompilerPass(new InMemoryCompilerPass());
221240
}
222241

223242
public function process(ContainerBuilder $container): void
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,36 @@
1+
<?php
2+
3+
declare(strict_types=1);
4+
5+
namespace Zenstruck\Foundry\Tests\Fixture\InMemory;
6+
7+
use Zenstruck\Foundry\InMemory\AsInMemoryRepository;
8+
use Zenstruck\Foundry\InMemory\InMemoryRepository;
9+
use Zenstruck\Foundry\Tests\Fixture\Entity\Address\StandardAddress;
10+
11+
/**
12+
* @implements InMemoryRepository<StandardAddress>
13+
*/
14+
#[AsInMemoryRepository(class: StandardAddress::class)]
15+
final class InMemoryStandardAddressRepository implements InMemoryRepository
16+
{
17+
/**
18+
* @var list<StandardAddress>
19+
*/
20+
private array $elements = [];
21+
22+
public function _save(object $element): void
23+
{
24+
if (!in_array($element, $this->elements, true)) {
25+
$this->elements[] = $element;
26+
}
27+
}
28+
29+
/**
30+
* @return list<StandardAddress>
31+
*/
32+
public function all(): array
33+
{
34+
return $this->elements;
35+
}
36+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,34 @@
1+
<?php
2+
3+
declare(strict_types=1);
4+
5+
namespace Zenstruck\Foundry\Tests\Fixture\InMemory;
6+
7+
use Zenstruck\Foundry\InMemory\AsInMemoryRepository;
8+
use Zenstruck\Foundry\InMemory\InMemoryRepository;
9+
use Zenstruck\Foundry\Tests\Fixture\Entity\Contact\StandardContact;
10+
11+
/**
12+
* @implements InMemoryRepository<StandardContact>
13+
*/
14+
#[AsInMemoryRepository(class: StandardContact::class)]
15+
final class InMemoryStandardContactRepository implements InMemoryRepository
16+
{
17+
/** @var list<StandardContact> */
18+
private array $elements = [];
19+
20+
public function _save(object $element): void
21+
{
22+
if (!in_array($element, $this->elements, true)) {
23+
$this->elements[] = $element;
24+
}
25+
}
26+
27+
/**
28+
* @return list<StandardContact>
29+
*/
30+
public function all(): array
31+
{
32+
return $this->elements;
33+
}
34+
}

‎tests/Fixture/TestKernel.php

+4
Original file line numberDiff line numberDiff line change
@@ -26,6 +26,8 @@
2626
use Zenstruck\Foundry\ORM\AbstractORMPersistenceStrategy;
2727
use Zenstruck\Foundry\Tests\Fixture\Factories\ArrayFactory;
2828
use Zenstruck\Foundry\Tests\Fixture\Factories\Object1Factory;
29+
use Zenstruck\Foundry\Tests\Fixture\InMemory\InMemoryStandardAddressRepository;
30+
use Zenstruck\Foundry\Tests\Fixture\InMemory\InMemoryStandardContactRepository;
2931
use Zenstruck\Foundry\Tests\Fixture\Stories\GlobalInvokableService;
3032
use Zenstruck\Foundry\Tests\Fixture\Stories\GlobalStory;
3133
use Zenstruck\Foundry\ZenstruckFoundryBundle;
@@ -145,6 +147,8 @@ protected function configureContainer(ContainerBuilder $c, LoaderInterface $load
145147
$c->register(GlobalInvokableService::class);
146148
$c->register(ArrayFactory::class)->setAutowired(true)->setAutoconfigured(true);
147149
$c->register(Object1Factory::class)->setAutowired(true)->setAutoconfigured(true);
150+
$c->register(InMemoryStandardAddressRepository::class)->setAutowired(true)->setAutoconfigured(true);
151+
$c->register(InMemoryStandardContactRepository::class)->setAutowired(true)->setAutoconfigured(true);
148152
}
149153

150154
protected function configureRoutes(RoutingConfigurator $routes): void
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,99 @@
1+
<?php
2+
3+
declare(strict_types=1);
4+
5+
namespace Zenstruck\Foundry\Tests\Integration\InMemory;
6+
7+
use Symfony\Bundle\FrameworkBundle\Test\KernelTestCase;
8+
use Zenstruck\Foundry\Test\Factories;
9+
use Zenstruck\Foundry\Test\ResetDatabase;
10+
use Zenstruck\Foundry\Tests\Fixture\Entity\Address\StandardAddress;
11+
use Zenstruck\Foundry\Tests\Fixture\Factories\Entity\Address\StandardAddressFactory;
12+
use Zenstruck\Foundry\Tests\Fixture\Factories\Entity\Category\StandardCategoryFactory;
13+
use Zenstruck\Foundry\Tests\Fixture\Factories\Entity\Contact\StandardContactFactory;
14+
use Zenstruck\Foundry\Tests\Fixture\InMemory\InMemoryStandardAddressRepository;
15+
use Zenstruck\Foundry\Tests\Fixture\InMemory\InMemoryStandardContactRepository;
16+
use function Zenstruck\Foundry\InMemory\enable_in_memory;
17+
18+
final class InMemoryTest extends KernelTestCase
19+
{
20+
use Factories;
21+
use ResetDatabase;
22+
23+
private InMemoryStandardAddressRepository $addressRepository;
24+
private InMemoryStandardContactRepository $contactRepository;
25+
26+
protected function setUp(): void
27+
{
28+
enable_in_memory();
29+
30+
$this->addressRepository = self::getContainer()->get(InMemoryStandardAddressRepository::class); // @phpstan-ignore-line
31+
$this->contactRepository = self::getContainer()->get(InMemoryStandardContactRepository::class); // @phpstan-ignore-line
32+
}
33+
34+
/**
35+
* @test
36+
*/
37+
public function create_one_does_not_persist_in_database(): void
38+
{
39+
$address = StandardAddressFactory::createOne();
40+
self::assertInstanceOf(StandardAddress::class, $address);
41+
42+
// todo!
43+
// StandardAddressFactory::assert()->count(0);
44+
45+
// id is autogenerated from the db, then it should be null
46+
self::assertNull($address->id);
47+
}
48+
49+
/**
50+
* @test
51+
*/
52+
public function create_many_does_not_persist_in_database(): void
53+
{
54+
$addresses = StandardAddressFactory::createMany(2);
55+
self::assertContainsOnlyInstancesOf(StandardAddress::class, $addresses);
56+
57+
// todo!
58+
// StandardAddressFactory::assert()->count(0);
59+
60+
foreach ($addresses as $address) {
61+
// id is autogenerated from the db, then it should be null
62+
self::assertNull($address->id);
63+
}
64+
}
65+
66+
/**
67+
* @test
68+
*/
69+
public function object_should_be_accessible_from_in_memory_repository(): void
70+
{
71+
$address = StandardAddressFactory::createOne();
72+
73+
self::assertSame([$address], $this->addressRepository->all());
74+
}
75+
76+
/**
77+
* @test
78+
*/
79+
public function nested_objects_should_be_accessible_from_their_respective_repository(): void
80+
{
81+
$contact = StandardContactFactory::createOne();
82+
83+
self::assertSame([$contact], $this->contactRepository->all());
84+
self::assertSame([$contact->getAddress()], $this->addressRepository->all());
85+
}
86+
87+
/**
88+
* @test
89+
*/
90+
public function can_use_generic_repository(): void
91+
{
92+
$category = StandardCategoryFactory::createOne();
93+
94+
// todo!
95+
// StandardCategoryFactory::assert()->count(0);
96+
97+
self::assertNull($category->id);
98+
}
99+
}

0 commit comments

Comments
 (0)
Please sign in to comment.