1
|
|
|
<?php |
2
|
|
|
|
3
|
|
|
declare(strict_types=1); |
4
|
|
|
|
5
|
|
|
namespace ApiSkeletons\Doctrine\ORM\GraphQL\Hydrator; |
6
|
|
|
|
7
|
|
|
use ApiSkeletons\Doctrine\ORM\GraphQL\Container; |
8
|
|
|
use ApiSkeletons\Doctrine\ORM\GraphQL\Hydrator\Filter\Password; |
9
|
|
|
use ApiSkeletons\Doctrine\ORM\GraphQL\Type\Entity\EntityTypeContainer; |
10
|
|
|
use Doctrine\Laminas\Hydrator\DoctrineObject; |
11
|
|
|
use Doctrine\ORM\EntityManager; |
12
|
|
|
use GraphQL\Error\Error; |
13
|
|
|
use Laminas\Hydrator\Filter; |
14
|
|
|
use Laminas\Hydrator\NamingStrategy\MapNamingStrategy; |
15
|
|
|
use Laminas\Hydrator\NamingStrategy\NamingStrategyEnabledInterface; |
16
|
|
|
use Laminas\Hydrator\NamingStrategy\NamingStrategyInterface; |
17
|
|
|
use Laminas\Hydrator\Strategy\StrategyInterface; |
18
|
|
|
|
19
|
|
|
use function assert; |
20
|
|
|
use function class_implements; |
21
|
|
|
use function in_array; |
22
|
|
|
|
23
|
|
|
/** |
24
|
|
|
* This factory is used in the Metadata\Entity class to create a hydrator |
25
|
|
|
* for the current entity |
26
|
|
|
*/ |
27
|
|
|
class HydratorContainer extends Container |
28
|
|
|
{ |
29
|
|
|
public function __construct( |
30
|
|
|
protected EntityManager $entityManager, |
31
|
|
|
protected EntityTypeContainer $entityTypeContainer, |
32
|
|
|
) { |
33
|
|
|
// Register default strategies |
34
|
|
|
$this |
35
|
|
|
->set(Strategy\AssociationDefault::class, new Strategy\AssociationDefault()) |
36
|
|
|
->set(Strategy\FieldDefault::class, new Strategy\FieldDefault()) |
37
|
|
|
->set(Strategy\ToBoolean::class, new Strategy\ToBoolean()) |
38
|
|
|
->set(Strategy\ToFloat::class, new Strategy\ToFloat()) |
39
|
|
|
->set(Strategy\ToInteger::class, new Strategy\ToInteger()) |
40
|
|
|
->set(Password::class, new Password()); |
41
|
|
|
} |
42
|
|
|
|
43
|
|
|
/** @throws Error */ |
44
|
|
|
public function get(string $id): mixed |
45
|
|
|
{ |
46
|
|
|
// Custom hydrators should already be registered |
47
|
|
|
if ($this->has($id)) { |
48
|
|
|
return parent::get($id); |
49
|
|
|
} |
50
|
|
|
|
51
|
|
|
$entity = $this->entityTypeContainer->get($id); |
52
|
|
|
$config = $entity->getMetadata(); |
53
|
|
|
$hydrator = new DoctrineObject($this->entityManager, $config['byValue']); |
54
|
|
|
|
55
|
|
|
// Create field strategy and assign to hydrator |
56
|
|
|
foreach ($config['fields'] as $fieldName => $fieldMetadata) { |
57
|
|
|
assert( |
58
|
|
|
in_array(StrategyInterface::class, class_implements($fieldMetadata['hydratorStrategy'])), |
59
|
|
|
'Strategy must implement ' . StrategyInterface::class, |
60
|
|
|
); |
61
|
|
|
|
62
|
|
|
$hydrator->addStrategy($fieldName, $this->get($fieldMetadata['hydratorStrategy'])); |
63
|
|
|
} |
64
|
|
|
|
65
|
|
|
// Create filters and assign to hydrator |
66
|
|
|
foreach ($config['hydratorFilters'] as $name => $filterConfig) { |
67
|
|
|
// Default filters to AND |
68
|
|
|
$condition = $filterConfig['condition'] ?? Filter\FilterComposite::CONDITION_AND; |
69
|
|
|
$filterClass = $filterConfig['filter']; |
70
|
|
|
assert( |
71
|
|
|
in_array(Filter\FilterInterface::class, class_implements($filterClass)), |
72
|
|
|
'Filter must implement ' . StrategyInterface::class, |
73
|
|
|
); |
74
|
|
|
|
75
|
|
|
$hydrator->addFilter($name, $this->get($filterClass), $condition); |
76
|
|
|
} |
77
|
|
|
|
78
|
|
|
// Create naming strategy for aliases and assign to hydrator |
79
|
|
|
if ($entity->getAliasMap()) { |
80
|
|
|
$hydrator->setNamingStrategy(MapNamingStrategy::createFromExtractionMap($entity->getAliasMap())); |
81
|
|
|
} |
82
|
|
|
|
83
|
|
|
$this->set($id, $hydrator); |
84
|
|
|
|
85
|
|
|
return $hydrator; |
86
|
|
|
} |
87
|
|
|
} |
88
|
|
|
|