Code Coverage
 
Lines
Functions and Methods
Classes and Traits
Total
76.92% covered (warning)
76.92%
40 / 52
66.67% covered (warning)
66.67%
6 / 9
CRAP
0.00% covered (danger)
0.00%
0 / 1
DomainToStorageContext
76.92% covered (warning)
76.92%
40 / 52
66.67% covered (warning)
66.67%
6 / 9
34.31
0.00% covered (danger)
0.00%
0 / 1
 __construct
100.00% covered (success)
100.00%
1 / 1
100.00% covered (success)
100.00%
1 / 1
1
 createFromContext
100.00% covered (success)
100.00%
14 / 14
100.00% covered (success)
100.00%
1 / 1
2
 withStorageProperty
100.00% covered (success)
100.00%
2 / 2
100.00% covered (success)
100.00%
1 / 1
1
 withDomainObject
0.00% covered (danger)
0.00%
0 / 1
0.00% covered (danger)
0.00%
0 / 1
2
 withArrayKey
100.00% covered (success)
100.00%
1 / 1
100.00% covered (success)
100.00%
1 / 1
1
 setStoragePropertyValue
100.00% covered (success)
100.00%
1 / 1
100.00% covered (success)
100.00%
1 / 1
1
 getStoragePropertyValue
66.67% covered (warning)
66.67%
2 / 3
0.00% covered (danger)
0.00%
0 / 1
3.33
 dynamicCast
54.55% covered (warning)
54.55%
12 / 22
0.00% covered (danger)
0.00%
0 / 1
28.87
 clone
100.00% covered (success)
100.00%
7 / 7
100.00% covered (success)
100.00%
1 / 1
3
1<?php
2namespace Apie\StorageMetadata\Mediators;
3
4use Apie\Core\FileStorage\FileStorageInterface;
5use Apie\Core\TypeUtils;
6use Apie\Core\Utils\ConverterUtils;
7use Apie\StorageMetadata\DomainToStorageConverter;
8use Apie\StorageMetadata\Exceptions\CouldNotCastPropertyException;
9use Apie\StorageMetadata\Interfaces\StorageDtoInterface;
10use Apie\StorageMetadataBuilder\Interfaces\MixedStorageInterface;
11use Apie\TypeConverter\TypeConverter;
12use ReflectionClass;
13use ReflectionNamedType;
14use ReflectionProperty;
15use ReflectionType;
16use Throwable;
17
18final class DomainToStorageContext
19{
20    /**
21     * @var ReflectionClass<object>
22     */
23    public readonly ?ReflectionClass $domainClass;
24
25    public readonly ReflectionProperty $storageProperty;
26
27    public readonly int|string $arrayKey;
28
29    public readonly ?DomainToStorageContext $parentContext;
30
31    /**
32     * @template T of object
33     * @param T $domainObject
34     * @param ReflectionClass<T>|null $domainClass
35     */
36    public function __construct(
37        public readonly DomainToStorageConverter $domainToStorageConverter,
38        public readonly TypeConverter $typeConverter,
39        public readonly StorageDtoInterface $storageObject,
40        public readonly object $domainObject,
41        public readonly FileStorageInterface $fileStorage,
42        ?ReflectionClass $domainClass = null
43    ) {
44        $this->domainClass = $domainClass ?? new ReflectionClass($domainObject);
45    }
46
47    /**
48     * @template T of object
49     * @param T $domainObject
50     * @param ReflectionClass<T>|null $domainClass
51     */
52    public static function createFromContext(
53        DomainToStorageConverter $domainToStorageConverter,
54        TypeConverter $typeConverter,
55        StorageDtoInterface $storageObject,
56        object $domainObject,
57        FileStorageInterface $fileStorage,
58        ?ReflectionClass $domainClass = null,
59        ?DomainToStorageContext $context = null,
60    ): self {
61        $res = new self(
62            $domainToStorageConverter,
63            $typeConverter,
64            $storageObject,
65            $domainObject,
66            $fileStorage,
67            $domainClass
68        );
69        $fields = [
70            'parentContext' => $context,
71        ];
72
73        if (isset($context->arrayKey)) {
74            $fields['arrayKey'] = $context->arrayKey;
75        }
76
77        return $res->clone($fields);
78    }
79
80    public function withStorageProperty(ReflectionProperty $storageProperty): self
81    {
82        // we do this to throw an exception in case an incorrect property is entered here.
83        $storageProperty->isInitialized($this->storageObject);
84        return $this->clone(['storageProperty' => $storageProperty]);
85    }
86
87    public function withDomainObject(object $object): self
88    {
89        return $this->clone(['domainObject' => $object]);
90    }
91
92    public function withArrayKey(string|int $key): self
93    {
94        return $this->clone(['arrayKey' => $key]);
95    }
96
97    public function setStoragePropertyValue(mixed $value): void
98    {
99        $this->storageProperty->setValue($this->storageObject, $value);
100    }
101
102    public function getStoragePropertyValue(): mixed
103    {
104        if (!$this->storageProperty->isInitialized($this->storageObject)) {
105            return $this->storageProperty->hasDefaultValue() ? $this->storageProperty->getDefaultValue() : null;
106        }
107        return $this->storageProperty->getValue($this->storageObject);
108    }
109
110    public function dynamicCast(mixed $input, ?ReflectionType $wantedType): mixed
111    {
112        if ($input instanceof MixedStorageInterface) {
113            $input = $input->toOriginalObject();
114        }
115        if (!$wantedType || ('mixed' === (string) $wantedType)) {
116            return $input;
117        }
118        if ($input === null && $wantedType->allowsNull()) {
119            return null;
120        }
121        if (is_object($input)) {
122            $class = ConverterUtils::toReflectionClass($wantedType);
123            if ($class && $class->isInstance($input)) {
124                return $input;
125            }
126        } elseif ($wantedType instanceof ReflectionNamedType && $wantedType->getName() === get_debug_type($input)) {
127            return $input;
128        } elseif (TypeUtils::matchesType($wantedType, $input)) {
129            return $input;
130        }
131        try {
132            return $this->typeConverter->convertTo($input, $wantedType);
133        } catch (Throwable $error) {
134            throw new CouldNotCastPropertyException(
135                $this->storageProperty ?? null,
136                $this->domainObject,
137                $wantedType,
138                $error
139            );
140        }
141    }
142
143    /**
144     * @param array<string, mixed> $altered
145     */
146    private function clone(array $altered): self
147    {
148        $properties = get_object_vars($this) + $altered;
149        if (!isset($properties['parentContext'])) {
150            $properties['parentContext'] = $this;
151        }
152        $clone = (new ReflectionClass($this))->newInstanceWithoutConstructor();
153        foreach ($properties as $propertyName => $propertyValue) {
154            $clone->$propertyName = $propertyValue;
155        }
156
157        return $clone;
158    }
159}