Passed
Pull Request — master (#1200)
by Rene
02:34
created

ArrayCollectionHandler   A

Complexity

Total Complexity 20

Size/Duplication

Total Lines 140
Duplicated Lines 0 %

Test Coverage

Coverage 88.24%

Importance

Changes 0
Metric Value
eloc 63
c 0
b 0
f 0
dl 0
loc 140
ccs 30
cts 34
cp 0.8824
rs 10
wmc 20

4 Methods

Rating   Name   Duplication   Size   Complexity  
A getSubscribingMethods() 0 24 3
C deserializeCollection() 0 54 12
A __construct() 0 6 1
A serializeCollection() 0 21 4
1
<?php
2
3
declare(strict_types=1);
4
5
namespace JMS\Serializer\Handler;
6
7
use Doctrine\Common\Collections\ArrayCollection;
8
use Doctrine\Common\Collections\Collection;
9
use Doctrine\ODM\MongoDB\PersistentCollection as MongoPersistentCollection;
0 ignored issues
show
Bug introduced by
The type Doctrine\ODM\MongoDB\PersistentCollection was not found. Maybe you did not declare it correctly or list all dependencies?

The issue could also be caused by a filter entry in the build configuration. If the path has been excluded in your configuration, e.g. excluded_paths: ["lib/*"], you can move it to the dependency path list as follows:

filter:
    dependency_paths: ["lib/*"]

For further information see https://scrutinizer-ci.com/docs/tools/php/php-scrutinizer/#list-dependency-paths

Loading history...
10
use Doctrine\ODM\PHPCR\PersistentCollection as PhpcrPersistentCollection;
11
use Doctrine\ORM\PersistentCollection as OrmPersistentCollection;
12
use Doctrine\Persistence\ManagerRegistry;
13
use JMS\Serializer\DeserializationContext;
14
use JMS\Serializer\GraphNavigatorInterface;
15
use JMS\Serializer\Metadata\PropertyMetadata;
16
use JMS\Serializer\SerializationContext;
17
use JMS\Serializer\Visitor\DeserializationVisitorInterface;
18
use JMS\Serializer\Visitor\SerializationVisitorInterface;
19
20
final class ArrayCollectionHandler implements SubscribingHandlerInterface
21
{
22 329
    public const COLLECTION_TYPES = [
23
        'ArrayCollection',
24 329
        ArrayCollection::class,
25 329
        OrmPersistentCollection::class,
26
        MongoPersistentCollection::class,
27 329
        PhpcrPersistentCollection::class,
28
    ];
29 329
30 329
    /**
31
     * @var bool
32 329
     */
33
    private $initializeExcluded;
34
35
    /**
36
     * @var ManagerRegistry|null
37
     */
38
    private $managerRegistry;
39 329
40 329
    public function __construct(
41 329
        bool $initializeExcluded = true,
42 329
        ?ManagerRegistry $managerRegistry = null
43 329
    ) {
44 329
        $this->initializeExcluded = $initializeExcluded;
45 329
        $this->managerRegistry = $managerRegistry;
46
    }
47
48 329
    /**
49 329
     * {@inheritdoc}
50 329
     */
51 329
    public static function getSubscribingMethods()
52 329
    {
53
        $methods = [];
54
        $formats = ['json', 'xml', 'yml'];
55
56
        foreach (self::COLLECTION_TYPES as $type) {
57 329
            foreach ($formats as $format) {
58
                $methods[] = [
59
                    'direction' => GraphNavigatorInterface::DIRECTION_SERIALIZATION,
60 9
                    'type' => $type,
61
                    'format' => $format,
62
                    'method' => 'serializeCollection',
63 9
                ];
64
65 9
                $methods[] = [
66
                    'direction' => GraphNavigatorInterface::DIRECTION_DESERIALIZATION,
67 9
                    'type' => $type,
68
                    'format' => $format,
69
                    'method' => 'deserializeCollection',
70
                ];
71
            }
72
        }
73
74
        return $methods;
75 9
    }
76
77 9
    /**
78 9
     * @return array|\ArrayObject
79
     */
80
    public function serializeCollection(SerializationVisitorInterface $visitor, Collection $collection, array $type, SerializationContext $context)
81 6
    {
82
        // We change the base type, and pass through possible parameters.
83
        $type['name'] = 'array';
84 6
85
        $context->stopVisiting($collection);
86 6
87
        if (false === $this->initializeExcluded) {
88
            $exclusionStrategy = $context->getExclusionStrategy();
89
            if (null !== $exclusionStrategy && $exclusionStrategy->shouldSkipClass($context->getMetadataFactory()->getMetadataForClass(\get_class($collection)), $context)) {
0 ignored issues
show
Bug introduced by
It seems like $context->getMetadataFac...get_class($collection)) can also be of type Metadata\ClassHierarchyMetadata; however, parameter $metadata of JMS\Serializer\Exclusion...tegy::shouldSkipClass() does only seem to accept JMS\Serializer\Metadata\ClassMetadata, maybe add an additional type check? ( Ignorable by Annotation )

If this is a false-positive, you can also ignore this issue in your code via the ignore-type  annotation

89
            if (null !== $exclusionStrategy && $exclusionStrategy->shouldSkipClass(/** @scrutinizer ignore-type */ $context->getMetadataFactory()->getMetadataForClass(\get_class($collection)), $context)) {
Loading history...
90
                $context->startVisiting($collection);
91
92
                return $visitor->visitArray([], $type);
93
            }
94
        }
95
96
        $result = $visitor->visitArray($collection->toArray(), $type);
97
98
        $context->startVisiting($collection);
99
100
        return $result;
101
    }
102
103
    /**
104
     * @param mixed $data
105
     */
106
    public function deserializeCollection(
107
        DeserializationVisitorInterface $visitor,
108
        $data,
109
        array $type,
110
        DeserializationContext $context
111
    ): Collection {
112
        // See above.
113
        $type['name'] = 'array';
114
115
        $elements = new ArrayCollection($visitor->visitArray($data, $type));
116
117
        if (null === $this->managerRegistry) {
118
            return $elements;
119
        }
120
121
        $currentMetadata = $context->getMetadataStack()->top();
122
        if (!$currentMetadata instanceof PropertyMetadata) {
123
            return $elements;
124
        }
125
126
127
        $objectManager = $this->managerRegistry->getManagerForClass($currentMetadata->class);
128
        if (null === $objectManager) {
129
            return $elements;
130
        }
131
132
        $classMetadata = $objectManager->getClassMetadata($currentMetadata->class);
133
        $currentObject = $visitor->getCurrentObject();
0 ignored issues
show
Bug introduced by
The method getCurrentObject() does not exist on JMS\Serializer\Visitor\D...izationVisitorInterface. Since it exists in all sub-types, consider adding an abstract or default implementation to JMS\Serializer\Visitor\D...izationVisitorInterface. ( Ignorable by Annotation )

If this is a false-positive, you can also ignore this issue in your code via the ignore-call  annotation

133
        /** @scrutinizer ignore-call */ 
134
        $currentObject = $visitor->getCurrentObject();
Loading history...
134
135
        if (
136
            is_array($currentMetadata->type)
137
            && in_array($currentMetadata->type['name'], ArrayCollectionHandler::COLLECTION_TYPES)
138
            && $classMetadata->isCollectionValuedAssociation($currentMetadata->name)
139
        ) {
140
            $collection = $classMetadata->getFieldValue($currentObject, $currentMetadata->name);
0 ignored issues
show
Bug introduced by
The method getFieldValue() does not exist on Doctrine\Persistence\Mapping\ClassMetadata. Did you maybe mean getFieldNames()? ( Ignorable by Annotation )

If this is a false-positive, you can also ignore this issue in your code via the ignore-call  annotation

140
            /** @scrutinizer ignore-call */ 
141
            $collection = $classMetadata->getFieldValue($currentObject, $currentMetadata->name);

This check looks for calls to methods that do not seem to exist on a given type. It looks for the method on the type itself as well as in inherited classes or implemented interfaces.

This is most likely a typographical error or the method has been renamed.

Loading history...
141
            if (!$collection instanceof OrmPersistentCollection) {
142
                return $elements;
143
            }
144
            foreach ($elements as $element) {
145
                if (!$collection->contains($element)) {
146
                    $collection->add($element);
147
                }
148
            }
149
150
            foreach ($collection as $collectionElement) {
151
                if (!$elements->contains($collectionElement)) {
152
                    $collection->removeElement($collectionElement);
153
                }
154
            }
155
156
            return $collection;
157
        }
158
159
        return $elements;
160
    }
161
}
162