InsertElementMutation::processElementValue()   A
last analyzed

Complexity

Conditions 1
Paths 1

Size

Total Lines 12
Code Lines 9

Duplication

Lines 0
Ratio 0 %

Code Coverage

Tests 0
CRAP Score 2

Importance

Changes 1
Bugs 0 Features 0
Metric Value
cc 1
eloc 9
c 1
b 0
f 0
nc 1
nop 2
dl 0
loc 12
ccs 0
cts 8
cp 0
crap 2
rs 9.9666
1
<?php
2
3
declare(strict_types=1);
4
5
namespace Remorhaz\JSON\Pointer\Processor\Mutator;
6
7
use ArrayIterator;
8
use Generator;
9
use Iterator;
10
use Remorhaz\JSON\Data\Event\AfterElementEvent;
11
use Remorhaz\JSON\Data\Event\AfterElementEventInterface;
12
use Remorhaz\JSON\Data\Event\BeforeElementEvent;
13
use Remorhaz\JSON\Data\Event\BeforeElementEventInterface;
14
use Remorhaz\JSON\Data\Event\EventInterface;
15
use Remorhaz\JSON\Data\Export\EventDecoder;
16
use Remorhaz\JSON\Data\Path\PathInterface;
17
use Remorhaz\JSON\Data\Value\NodeValueInterface;
18
use Remorhaz\JSON\Data\Walker\MutationInterface;
19
use Remorhaz\JSON\Data\Walker\ValueWalkerInterface;
20
21
final class InsertElementMutation implements MutationInterface
22
{
23
24
    private $value;
25
26
    private $path;
27
28
    private $elementIndex;
29
30
    private $elementCounter = 0;
31
32
    private $eventDecoder;
33
34
    public function __construct(NodeValueInterface $value, PathInterface $path, int $elementIndex)
35
    {
36
        $this->value = $value;
37
        $this->path = $path;
38
        $this->elementIndex = $elementIndex;
39
        $this->eventDecoder = new EventDecoder();
40
    }
41
42
    public function __invoke(EventInterface $event, ValueWalkerInterface $valueWalker): Iterator
43
    {
44
        if (!$this->parentPathMatches($event)) {
45
            yield $event;
46
47
            return;
48
        }
49
50
        switch (true) {
51
            case $event instanceof BeforeElementEventInterface:
52
                yield from $this->processBeforeElementEvent($event, $valueWalker);
53
                break;
54
            case $event instanceof AfterElementEventInterface:
55
                yield from $this->processAfterElementEvent($event);
56
                break;
57
58
            default:
59
                yield from $this->processElementValue($event, $valueWalker);
60
        }
61
    }
62
63
    public function reset(): void
64
    {
65
        $this->elementCounter = 0;
66
    }
67
68
    private function parentPathMatches(EventInterface $event): bool
69
    {
70
        $pathElements = $event->getPath()->getElements();
71
        if (empty($pathElements)) {
72
            return false;
73
        }
74
75
        return $event
76
            ->getPath()
77
            ->copyParent()
78
            ->equals($this->path);
79
    }
80
81
    private function processBeforeElementEvent(
82
        BeforeElementEventInterface $event,
83
        ValueWalkerInterface $valueWalker
84
    ): Generator {
85
        if ($event->getIndex() < $this->elementIndex) {
86
            yield $event;
87
88
            return;
89
        }
90
91
        if ($event->getIndex() == $this->elementIndex) {
92
            $elementPath = $this->path->copyWithElement($this->elementIndex);
93
            yield new BeforeElementEvent($this->elementIndex, $elementPath);
94
            yield from $valueWalker->createEventIterator($this->value, $elementPath);
95
            yield new AfterElementEvent($this->elementIndex, $elementPath);
96
        }
97
        $shiftedIndex = $event->getIndex() + 1;
98
        $path = $event
99
            ->getPath()
100
            ->copyParent()
101
            ->copyWithElement($shiftedIndex);
102
        yield new BeforeElementEvent($shiftedIndex, $path);
103
    }
104
105
    private function processAfterElementEvent(AfterElementEventInterface $event): Generator
106
    {
107
        $this->elementCounter++;
108
        if ($event->getIndex() < $this->elementIndex) {
109
            yield $event;
110
111
            return;
112
        }
113
114
        $shiftedIndex = $event->getIndex() + 1;
115
        $path = $event
116
            ->getPath()
117
            ->copyParent()
118
            ->copyWithElement($shiftedIndex);
119
        yield new AfterElementEvent($shiftedIndex, $path);
120
    }
121
122
    private function processElementValue(EventInterface $event, ValueWalkerInterface $valueWalker): Generator
123
    {
124
        $path = $event
125
            ->getPath()
126
            ->copyParent()
127
            ->copyWithElement($this->elementCounter + 1);
128
129
        yield from $valueWalker->createEventIterator(
130
            $this
131
                ->eventDecoder
132
                ->exportExistingEvents(new ArrayIterator([$event])),
133
            $path,
134
        );
135
    }
136
}
137