Passed
Pull Request — master (#91)
by Anton
02:38
created

Reacter::getReacterable()   A

Complexity

Conditions 2
Paths 2

Size

Total Lines 9
Code Lines 4

Duplication

Lines 0
Ratio 0 %

Importance

Changes 1
Bugs 0 Features 0
Metric Value
eloc 4
c 1
b 0
f 0
dl 0
loc 9
rs 10
cc 2
nc 2
nop 0
1
<?php
2
3
/*
4
 * This file is part of Laravel Love.
5
 *
6
 * (c) Anton Komarev <[email protected]>
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
declare(strict_types=1);
13
14
namespace Cog\Laravel\Love\Reacter\Models;
15
16
use Cog\Contracts\Love\Reactant\Exceptions\ReactantInvalid;
17
use Cog\Contracts\Love\Reactant\Models\Reactant as ReactantContract;
18
use Cog\Contracts\Love\Reacter\Exceptions\NotAssignedToReacterable;
19
use Cog\Contracts\Love\Reacter\Models\Reacter as ReacterContract;
20
use Cog\Contracts\Love\Reacterable\Models\Reacterable as ReacterableContract;
21
use Cog\Contracts\Love\Reaction\Exceptions\ReactionAlreadyExists;
22
use Cog\Contracts\Love\Reaction\Exceptions\ReactionNotExists;
23
use Cog\Contracts\Love\Reaction\Models\Reaction as ReactionContract;
24
use Cog\Contracts\Love\ReactionType\Models\ReactionType as ReactionTypeContract;
25
use Cog\Laravel\Love\Reaction\Models\Reaction;
26
use Cog\Laravel\Love\Support\Database\Eloquent\Model;
27
use Illuminate\Database\Eloquent\Relations\HasMany;
28
use Illuminate\Database\Eloquent\Relations\MorphTo;
29
30
final class Reacter extends Model implements
31
    ReacterContract
32
{
33
    protected $table = 'love_reacters';
34
35
    protected $fillable = [
36
        'type',
37
    ];
38
39
    protected $casts = [
40
        'id' => 'string',
41
    ];
42
43
    public function reacterable(): MorphTo
44
    {
45
        return $this->morphTo('reacterable', 'type', 'id', 'love_reacter_id');
46
    }
47
48
    public function reactions(): HasMany
49
    {
50
        return $this->hasMany(Reaction::class, 'reacter_id');
51
    }
52
53
    public function getId(): string
54
    {
55
        return $this->getAttributeValue('id');
0 ignored issues
show
Bug Best Practice introduced by
The expression return $this->getAttributeValue('id') could return the type boolean|null which is incompatible with the type-hinted return string. Consider adding an additional type-check to rule them out.
Loading history...
56
    }
57
58
    public function getReacterable(): ReacterableContract
59
    {
60
        $reacterable = $this->getAttribute('reacterable');
0 ignored issues
show
Bug introduced by
The attribute reacterable does not seem to exist on Cog\Laravel\Love\Reacter\Models\Reacter. Are you maybe missing a database migration?

If used attribute cannot be found there are two main reasons: 1. there is a creating database migration missing or 2. there is a major typo in either the declaration or the usage.

Loading history...
61
62
        if (is_null($reacterable)) {
63
            throw new NotAssignedToReacterable();
64
        }
65
66
        return $reacterable;
67
    }
68
69
    public function getReactions(): iterable
70
    {
71
        return $this->getAttribute('reactions');
72
    }
73
74
    public function reactTo(
75
        ReactantContract $reactant,
76
        ReactionTypeContract $reactionType,
77
        ?float $rate = null
78
    ): void {
79
        if ($reactant->isNull()) {
80
            throw ReactantInvalid::notExists();
81
        }
82
83
        $reaction = $this->findReaction($reactant, $reactionType);
84
85
        if (is_null($reaction)) {
86
            $this->reactions()->create([
87
                'reaction_type_id' => $reactionType->getId(),
88
                'reactant_id' => $reactant->getId(),
89
                'rate' => $rate,
90
            ]);
91
92
            return;
93
        }
94
95
        // TODO: Get or compare rate value using reaction contract
96
        if (is_null($rate) || $reaction->getAttributeValue('rate') === $rate) {
0 ignored issues
show
Bug introduced by
The method getAttributeValue() does not exist on Cog\Contracts\Love\Reaction\Models\Reaction. Since it exists in all sub-types, consider adding an abstract or default implementation to Cog\Contracts\Love\Reaction\Models\Reaction. ( Ignorable by Annotation )

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

96
        if (is_null($rate) || $reaction->/** @scrutinizer ignore-call */ getAttributeValue('rate') === $rate) {
Loading history...
97
            throw new ReactionAlreadyExists(
98
                sprintf('Reaction of type `%s` already exists.', $reactionType->getName())
99
            );
100
        }
101
102
        $reaction->update([
0 ignored issues
show
Bug introduced by
The method update() does not exist on Cog\Contracts\Love\Reaction\Models\Reaction. Since it exists in all sub-types, consider adding an abstract or default implementation to Cog\Contracts\Love\Reaction\Models\Reaction. ( Ignorable by Annotation )

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

102
        $reaction->/** @scrutinizer ignore-call */ 
103
                   update([
Loading history...
103
            'rate' => $rate,
104
        ]);
105
    }
106
107
    public function unreactTo(
108
        ReactantContract $reactant,
109
        ReactionTypeContract $reactionType
110
    ): void {
111
        if ($reactant->isNull()) {
112
            throw ReactantInvalid::notExists();
113
        }
114
115
        $reaction = $this->findReaction($reactant, $reactionType);
116
117
        if (is_null($reaction)) {
118
            throw new ReactionNotExists(
119
                sprintf('Reaction of type `%s` not exists.', $reactionType->getName())
120
            );
121
        }
122
123
        $reaction->delete();
0 ignored issues
show
Bug introduced by
The method delete() does not exist on Cog\Contracts\Love\Reaction\Models\Reaction. Since it exists in all sub-types, consider adding an abstract or default implementation to Cog\Contracts\Love\Reaction\Models\Reaction. ( Ignorable by Annotation )

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

123
        $reaction->/** @scrutinizer ignore-call */ 
124
                   delete();
Loading history...
124
    }
125
126
    public function hasReactedTo(
127
        ReactantContract $reactant,
128
        ?ReactionTypeContract $reactionType = null
129
    ): bool {
130
        if ($reactant->isNull()) {
131
            return false;
132
        }
133
134
        return $reactant->isReactedBy($this, $reactionType);
135
    }
136
137
    public function hasNotReactedTo(
138
        ReactantContract $reactant,
139
        ?ReactionTypeContract $reactionType = null
140
    ): bool {
141
        return $reactant->isNotReactedBy($this, $reactionType);
142
    }
143
144
    public function isEqualTo(
145
        ReacterContract $that
146
    ): bool {
147
        return $that->isNotNull()
148
            && $this->getId() === $that->getId();
149
    }
150
151
    public function isNotEqualTo(
152
        ReacterContract $that
153
    ): bool {
154
        return !$this->isEqualTo($that);
155
    }
156
157
    public function isNull(): bool
158
    {
159
        return !$this->exists;
160
    }
161
162
    public function isNotNull(): bool
163
    {
164
        return $this->exists;
165
    }
166
167
    /**
168
     * @param \Cog\Contracts\Love\Reactant\Models\Reactant $reactant
169
     * @param \Cog\Contracts\Love\ReactionType\Models\ReactionType $reactionType
170
     * @return null|\Cog\Contracts\Love\Reaction\Models\Reaction|\Illuminate\Database\Eloquent\Model
171
     */
172
    private function findReaction(
173
        ReactantContract $reactant,
174
        ReactionTypeContract $reactionType
175
    ): ?ReactionContract {
176
        return $this
177
            ->reactions()
178
            ->where('reactant_id', $reactant->getId())
179
            ->where('reaction_type_id', $reactionType->getId())
180
            ->first();
181
    }
182
}
183