staudenmeir /
eloquent-has-many-deep
| 1 | <?php |
||||||
| 2 | |||||||
| 3 | namespace Staudenmeir\EloquentHasManyDeep\Eloquent\Traits; |
||||||
| 4 | |||||||
| 5 | use Illuminate\Database\Eloquent\Relations\Relation; |
||||||
| 6 | use Illuminate\Database\Eloquent\SoftDeletingScope; |
||||||
| 7 | use Staudenmeir\EloquentHasManyDeep\Eloquent\Relations\ThirdParty\LaravelHasManyMerged\HasManyMerged; |
||||||
| 8 | use Staudenmeir\EloquentHasManyDeep\HasManyDeep; |
||||||
| 9 | use Staudenmeir\EloquentHasManyDeep\HasOneDeep; |
||||||
| 10 | use Staudenmeir\EloquentHasManyDeepContracts\Interfaces\ConcatenableRelation; |
||||||
| 11 | |||||||
| 12 | trait ConcatenatesRelationships |
||||||
| 13 | { |
||||||
| 14 | use ConcatenatesNativeRelationships; |
||||||
| 15 | |||||||
| 16 | /** |
||||||
| 17 | * Define a has-many-deep relationship from existing relationships. |
||||||
| 18 | * |
||||||
| 19 | * @param \Illuminate\Database\Eloquent\Relations\Relation|callable ...$relations |
||||||
| 20 | * @return \Staudenmeir\EloquentHasManyDeep\HasManyDeep |
||||||
| 21 | */ |
||||||
| 22 | public function hasManyDeepFromRelations(...$relations) |
||||||
| 23 | { |
||||||
| 24 | [ |
||||||
| 25 | $related, |
||||||
| 26 | $through, |
||||||
| 27 | $foreignKeys, |
||||||
| 28 | $localKeys, |
||||||
| 29 | $postGetCallbacks, |
||||||
| 30 | $customThroughKeyCallback, |
||||||
| 31 | $customEagerConstraintsCallback, |
||||||
| 32 | $customEagerMatchingCallback |
||||||
| 33 | ] = |
||||||
| 34 | $this->hasOneOrManyDeepFromRelations($relations); |
||||||
| 35 | |||||||
| 36 | $relation = $this->hasManyDeep($related, $through, $foreignKeys, $localKeys); |
||||||
|
0 ignored issues
–
show
|
|||||||
| 37 | |||||||
| 38 | $this->customizeHasOneOrManyDeepRelationship( |
||||||
| 39 | $relation, |
||||||
| 40 | $postGetCallbacks, |
||||||
| 41 | $customThroughKeyCallback, |
||||||
| 42 | $customEagerConstraintsCallback, |
||||||
| 43 | $customEagerMatchingCallback |
||||||
| 44 | ); |
||||||
| 45 | |||||||
| 46 | return $relation; |
||||||
| 47 | } |
||||||
| 48 | |||||||
| 49 | /** |
||||||
| 50 | * Define a has-one-deep relationship from existing relationships. |
||||||
| 51 | * |
||||||
| 52 | * @param \Illuminate\Database\Eloquent\Relations\Relation|callable ...$relations |
||||||
| 53 | * @return \Staudenmeir\EloquentHasManyDeep\HasOneDeep |
||||||
| 54 | */ |
||||||
| 55 | public function hasOneDeepFromRelations(...$relations) |
||||||
| 56 | { |
||||||
| 57 | [ |
||||||
| 58 | $related, |
||||||
| 59 | $through, |
||||||
| 60 | $foreignKeys, |
||||||
| 61 | $localKeys, |
||||||
| 62 | $postGetCallbacks, |
||||||
| 63 | $customThroughKeyCallback, |
||||||
| 64 | $customEagerConstraintsCallback, |
||||||
| 65 | $customEagerMatchingCallback |
||||||
| 66 | ] = $this->hasOneOrManyDeepFromRelations($relations); |
||||||
| 67 | |||||||
| 68 | $relation = $this->hasOneDeep($related, $through, $foreignKeys, $localKeys); |
||||||
|
0 ignored issues
–
show
The method
hasOneDeep() does not exist on Staudenmeir\EloquentHasM...ncatenatesRelationships. Did you maybe mean hasOneDeepFromRelationsWithConstraints()?
(
Ignorable by Annotation
)
If this is a false-positive, you can also ignore this issue in your code via the
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...
|
|||||||
| 69 | |||||||
| 70 | $this->customizeHasOneOrManyDeepRelationship( |
||||||
| 71 | $relation, |
||||||
| 72 | $postGetCallbacks, |
||||||
| 73 | $customThroughKeyCallback, |
||||||
| 74 | $customEagerConstraintsCallback, |
||||||
| 75 | $customEagerMatchingCallback |
||||||
| 76 | ); |
||||||
| 77 | |||||||
| 78 | return $relation; |
||||||
| 79 | } |
||||||
| 80 | |||||||
| 81 | /** |
||||||
| 82 | * Prepare a has-one-deep or has-many-deep relationship from existing relationships. |
||||||
| 83 | * |
||||||
| 84 | * @param \Illuminate\Database\Eloquent\Relations\Relation[]|callable[] $relations |
||||||
| 85 | * @return array |
||||||
| 86 | */ |
||||||
| 87 | protected function hasOneOrManyDeepFromRelations(array $relations) |
||||||
| 88 | { |
||||||
| 89 | $relations = $this->normalizeVariadicRelations($relations); |
||||||
| 90 | |||||||
| 91 | foreach ($relations as $i => $relation) { |
||||||
| 92 | if (is_callable($relation)) { |
||||||
| 93 | $relations[$i] = $relation(); |
||||||
| 94 | } |
||||||
| 95 | } |
||||||
| 96 | |||||||
| 97 | $related = null; |
||||||
| 98 | $through = []; |
||||||
| 99 | $foreignKeys = []; |
||||||
| 100 | $localKeys = []; |
||||||
| 101 | $postGetCallbacks = []; |
||||||
| 102 | $customThroughKeyCallback = null; |
||||||
| 103 | $customEagerConstraintsCallback = null; |
||||||
| 104 | $customEagerMatchingCallback = null; |
||||||
| 105 | |||||||
| 106 | foreach ($relations as $i => $relation) { |
||||||
| 107 | // https://github.com/korridor/laravel-has-many-merged |
||||||
| 108 | if (is_a($relation, 'Korridor\LaravelHasManyMerged\HasManyMerged', true)) { |
||||||
| 109 | $relation = HasManyMerged::fromBaseRelation($relation); |
||||||
| 110 | } |
||||||
| 111 | |||||||
| 112 | if ($relation instanceof ConcatenableRelation) { |
||||||
| 113 | [$through, $foreignKeys, $localKeys] = $relation->appendToDeepRelationship( |
||||||
| 114 | $through, |
||||||
| 115 | $foreignKeys, |
||||||
| 116 | $localKeys, |
||||||
| 117 | $i |
||||||
| 118 | ); |
||||||
| 119 | |||||||
| 120 | if (method_exists($relation, 'postGetCallback')) { |
||||||
| 121 | $postGetCallbacks[] = [$relation, 'postGetCallback']; |
||||||
| 122 | } |
||||||
| 123 | |||||||
| 124 | if ($i === 0) { |
||||||
| 125 | if (method_exists($relation, 'getThroughKeyForDeepRelationships')) { |
||||||
| 126 | $customThroughKeyCallback = [$relation, 'getThroughKeyForDeepRelationships']; |
||||||
| 127 | } |
||||||
| 128 | |||||||
| 129 | if (method_exists($relation, 'addEagerConstraintsToDeepRelationship')) { |
||||||
| 130 | $customEagerConstraintsCallback = [$relation, 'addEagerConstraintsToDeepRelationship']; |
||||||
| 131 | } |
||||||
| 132 | |||||||
| 133 | if (method_exists($relation, 'matchResultsForDeepRelationship')) { |
||||||
| 134 | $customEagerMatchingCallback = [$relation, 'matchResultsForDeepRelationship']; |
||||||
| 135 | } |
||||||
| 136 | } |
||||||
| 137 | } else { |
||||||
| 138 | $method = $this->hasOneOrManyDeepRelationMethod($relation); |
||||||
| 139 | |||||||
| 140 | [$through, $foreignKeys, $localKeys] = $this->$method($relation, $through, $foreignKeys, $localKeys); |
||||||
| 141 | } |
||||||
| 142 | |||||||
| 143 | if ($i === count($relations) - 1) { |
||||||
| 144 | $related = get_class($relation->getRelated()); |
||||||
| 145 | |||||||
| 146 | if ((new $related())->getTable() !== $relation->getRelated()->getTable()) { |
||||||
| 147 | $related .= ' from ' . $relation->getRelated()->getTable(); |
||||||
| 148 | } |
||||||
| 149 | } else { |
||||||
| 150 | $through[] = $this->hasOneOrManyThroughParent($relation, $relations[$i + 1]); |
||||||
| 151 | } |
||||||
| 152 | } |
||||||
| 153 | |||||||
| 154 | return [ |
||||||
| 155 | $related, |
||||||
| 156 | $through, |
||||||
| 157 | $foreignKeys, |
||||||
| 158 | $localKeys, |
||||||
| 159 | $postGetCallbacks, |
||||||
| 160 | $customThroughKeyCallback, |
||||||
| 161 | $customEagerConstraintsCallback, |
||||||
| 162 | $customEagerMatchingCallback |
||||||
| 163 | ]; |
||||||
| 164 | } |
||||||
| 165 | |||||||
| 166 | /** |
||||||
| 167 | * Prepare the through parent class from an existing relationship and its successor. |
||||||
| 168 | * |
||||||
| 169 | * @param \Illuminate\Database\Eloquent\Relations\Relation $relation |
||||||
| 170 | * @param \Illuminate\Database\Eloquent\Relations\Relation $successor |
||||||
| 171 | * @return string |
||||||
| 172 | */ |
||||||
| 173 | protected function hasOneOrManyThroughParent(Relation $relation, Relation $successor) |
||||||
| 174 | { |
||||||
| 175 | $through = get_class($relation->getRelated()); |
||||||
| 176 | |||||||
| 177 | if ($relation instanceof ConcatenableRelation && method_exists($relation, 'getTableForDeepRelationship')) { |
||||||
| 178 | return $through . ' from ' . $relation->getTableForDeepRelationship(); |
||||||
| 179 | } |
||||||
| 180 | |||||||
| 181 | if ((new $through())->getTable() !== $relation->getRelated()->getTable()) { |
||||||
| 182 | $through .= ' from ' . $relation->getRelated()->getTable(); |
||||||
| 183 | } |
||||||
| 184 | |||||||
| 185 | if (get_class($relation->getRelated()) === get_class($successor->getParent())) { |
||||||
| 186 | $table = $successor->getParent()->getTable(); |
||||||
| 187 | |||||||
| 188 | $segments = explode(' as ', $table); |
||||||
| 189 | |||||||
| 190 | if (isset($segments[1])) { |
||||||
| 191 | $through .= ' as ' . $segments[1]; |
||||||
| 192 | } |
||||||
| 193 | } |
||||||
| 194 | |||||||
| 195 | return $through; |
||||||
| 196 | } |
||||||
| 197 | |||||||
| 198 | /** |
||||||
| 199 | * Customize a has-one-deep or has-many-deep relationship. |
||||||
| 200 | * |
||||||
| 201 | * @param \Staudenmeir\EloquentHasManyDeep\HasManyDeep $relation |
||||||
| 202 | * @param callable[] $postGetCallbacks |
||||||
| 203 | * @param callable|null $customThroughKeyCallback |
||||||
| 204 | * @param callable|null $customEagerConstraintsCallback |
||||||
| 205 | * @param callable|null $customEagerMatchingCallback |
||||||
| 206 | * @return \Staudenmeir\EloquentHasManyDeep\HasManyDeep|\Staudenmeir\EloquentHasManyDeep\HasOneDeep |
||||||
| 207 | */ |
||||||
| 208 | protected function customizeHasOneOrManyDeepRelationship( |
||||||
| 209 | HasManyDeep $relation, |
||||||
| 210 | array $postGetCallbacks, |
||||||
| 211 | ?callable $customThroughKeyCallback, |
||||||
| 212 | ?callable $customEagerConstraintsCallback, |
||||||
| 213 | ?callable $customEagerMatchingCallback |
||||||
| 214 | ): HasManyDeep|HasOneDeep { |
||||||
| 215 | $relation->withPostGetCallbacks($postGetCallbacks); |
||||||
| 216 | |||||||
| 217 | if ($customThroughKeyCallback) { |
||||||
| 218 | $relation->withCustomThroughKeyCallback($customThroughKeyCallback); |
||||||
| 219 | } |
||||||
| 220 | |||||||
| 221 | if ($customEagerConstraintsCallback) { |
||||||
| 222 | $relation->withCustomEagerConstraintsCallback($customEagerConstraintsCallback); |
||||||
| 223 | } |
||||||
| 224 | |||||||
| 225 | if ($customEagerMatchingCallback) { |
||||||
| 226 | $relation->withCustomEagerMatchingCallback($customEagerMatchingCallback); |
||||||
| 227 | } |
||||||
| 228 | |||||||
| 229 | return $relation; |
||||||
| 230 | } |
||||||
| 231 | |||||||
| 232 | /** |
||||||
| 233 | * Define a has-many-deep relationship with constraints from existing relationships. |
||||||
| 234 | * |
||||||
| 235 | * @param callable ...$relations |
||||||
| 236 | * @return \Staudenmeir\EloquentHasManyDeep\HasManyDeep |
||||||
| 237 | */ |
||||||
| 238 | public function hasManyDeepFromRelationsWithConstraints(...$relations): HasManyDeep |
||||||
| 239 | { |
||||||
| 240 | $hasManyDeep = $this->hasManyDeepFromRelations(...$relations); |
||||||
| 241 | |||||||
| 242 | $this->addConstraintsToHasOneOrManyDeepRelationship($hasManyDeep, $relations); |
||||||
| 243 | |||||||
| 244 | return $hasManyDeep; |
||||||
| 245 | } |
||||||
| 246 | |||||||
| 247 | /** |
||||||
| 248 | * Define a has-one-deep relationship with constraints from existing relationships. |
||||||
| 249 | * |
||||||
| 250 | * @param callable ...$relations |
||||||
| 251 | * @return \Staudenmeir\EloquentHasManyDeep\HasOneDeep |
||||||
| 252 | */ |
||||||
| 253 | public function hasOneDeepFromRelationsWithConstraints(...$relations): HasOneDeep |
||||||
| 254 | { |
||||||
| 255 | $hasOneDeep = $this->hasOneDeepFromRelations(...$relations); |
||||||
| 256 | |||||||
| 257 | $this->addConstraintsToHasOneOrManyDeepRelationship($hasOneDeep, $relations); |
||||||
| 258 | |||||||
| 259 | return $hasOneDeep; |
||||||
| 260 | } |
||||||
| 261 | |||||||
| 262 | /** |
||||||
| 263 | * Add the constraints from existing relationships to a has-one-deep or has-many-deep relationship. |
||||||
| 264 | * |
||||||
| 265 | * @param \Staudenmeir\EloquentHasManyDeep\HasManyDeep $deepRelation |
||||||
| 266 | * @param callable[] $relations |
||||||
| 267 | * @return \Staudenmeir\EloquentHasManyDeep\HasManyDeep|\Staudenmeir\EloquentHasManyDeep\HasOneDeep |
||||||
| 268 | */ |
||||||
| 269 | protected function addConstraintsToHasOneOrManyDeepRelationship( |
||||||
| 270 | HasManyDeep $deepRelation, |
||||||
| 271 | array $relations |
||||||
| 272 | ): HasManyDeep|HasOneDeep { |
||||||
| 273 | $relations = $this->normalizeVariadicRelations($relations); |
||||||
| 274 | |||||||
| 275 | foreach ($relations as $i => $relation) { |
||||||
| 276 | $relationWithoutConstraints = Relation::noConstraints(function () use ($relation) { |
||||||
| 277 | return $relation(); |
||||||
| 278 | }); |
||||||
| 279 | |||||||
| 280 | $deepRelation->getQuery()->mergeWheres( |
||||||
| 281 | $relationWithoutConstraints->getQuery()->getQuery()->wheres, |
||||||
| 282 | $relationWithoutConstraints->getQuery()->getQuery()->getRawBindings()['where'] ?? [] |
||||||
| 283 | ); |
||||||
| 284 | |||||||
| 285 | $isLast = $i === count($relations) - 1; |
||||||
| 286 | |||||||
| 287 | $this->addRemovedScopesToHasOneOrManyDeepRelationship($deepRelation, $relationWithoutConstraints, $isLast); |
||||||
| 288 | } |
||||||
| 289 | |||||||
| 290 | return $deepRelation; |
||||||
| 291 | } |
||||||
| 292 | |||||||
| 293 | /** |
||||||
| 294 | * Add the removed scopes from an existing relationship to a has-one-deep or has-many-deep relationship. |
||||||
| 295 | * |
||||||
| 296 | * @param \Staudenmeir\EloquentHasManyDeep\HasManyDeep $deepRelation |
||||||
| 297 | * @param \Illuminate\Database\Eloquent\Relations\Relation $relation |
||||||
| 298 | * @param bool $isLastRelation |
||||||
| 299 | * @return void |
||||||
| 300 | */ |
||||||
| 301 | protected function addRemovedScopesToHasOneOrManyDeepRelationship( |
||||||
| 302 | HasManyDeep $deepRelation, |
||||||
| 303 | Relation $relation, |
||||||
| 304 | bool $isLastRelation |
||||||
| 305 | ): void { |
||||||
| 306 | $removedScopes = $relation->getQuery()->removedScopes(); |
||||||
| 307 | |||||||
| 308 | foreach ($removedScopes as $scope) { |
||||||
| 309 | if ($scope === SoftDeletingScope::class) { |
||||||
| 310 | if ($isLastRelation) { |
||||||
| 311 | /** @phpstan-ignore method.notFound */ |
||||||
| 312 | $deepRelation->withTrashed(); |
||||||
| 313 | } else { |
||||||
| 314 | /** @phpstan-ignore method.notFound */ |
||||||
| 315 | $deletedAtColumn = $relation->getRelated()->getQualifiedDeletedAtColumn(); |
||||||
| 316 | |||||||
| 317 | /** @phpstan-ignore method.notFound */ |
||||||
| 318 | $deepRelation->withTrashed($deletedAtColumn); |
||||||
| 319 | } |
||||||
| 320 | } |
||||||
| 321 | |||||||
| 322 | if ($scope === 'SoftDeletableHasManyThrough') { |
||||||
| 323 | /** @phpstan-ignore method.notFound */ |
||||||
| 324 | $deletedAtColumn = $relation->getParent()->getQualifiedDeletedAtColumn(); |
||||||
| 325 | |||||||
| 326 | /** @phpstan-ignore method.notFound */ |
||||||
| 327 | $deepRelation->withTrashed($deletedAtColumn); |
||||||
| 328 | } |
||||||
| 329 | |||||||
| 330 | if (str_starts_with($scope, HasManyDeep::class . ':')) { |
||||||
| 331 | $deletedAtColumn = explode(':', $scope)[1]; |
||||||
| 332 | |||||||
| 333 | /** @phpstan-ignore method.notFound */ |
||||||
| 334 | $deepRelation->withTrashed($deletedAtColumn); |
||||||
| 335 | } |
||||||
| 336 | } |
||||||
| 337 | } |
||||||
| 338 | |||||||
| 339 | /** |
||||||
| 340 | * Normalize the relations from a variadic parameter. |
||||||
| 341 | * |
||||||
| 342 | * @param array $relations |
||||||
| 343 | * @return array |
||||||
| 344 | */ |
||||||
| 345 | protected function normalizeVariadicRelations(array $relations): array |
||||||
| 346 | { |
||||||
| 347 | return is_array($relations[0]) && !is_callable($relations[0]) ? $relations[0] : $relations; |
||||||
| 348 | } |
||||||
| 349 | } |
||||||
| 350 |
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.