From 39b55c4956b51dc0af222d79f30da575a55cd124 Mon Sep 17 00:00:00 2001 From: Sergei Tigrov Date: Mon, 20 May 2024 17:12:00 +0700 Subject: [PATCH] Refactor `MagicRelationsTrait` (#325) --- src/Trait/MagicRelationsTrait.php | 71 ++++++++++++++++----------- tests/ActiveQueryTest.php | 3 +- tests/Stubs/ActiveRecord/Customer.php | 4 +- tests/Stubs/ActiveRecord/Order.php | 4 +- 4 files changed, 48 insertions(+), 34 deletions(-) diff --git a/src/Trait/MagicRelationsTrait.php b/src/Trait/MagicRelationsTrait.php index a2ca973dd..ae730f79a 100644 --- a/src/Trait/MagicRelationsTrait.php +++ b/src/Trait/MagicRelationsTrait.php @@ -4,13 +4,13 @@ namespace Yiisoft\ActiveRecord\Trait; -use Error; use ReflectionException; use ReflectionMethod; use Yiisoft\ActiveRecord\ActiveQueryInterface; use Yiisoft\ActiveRecord\ActiveRecordInterface; use Yiisoft\Db\Exception\InvalidArgumentException; +use function is_a; use function lcfirst; use function method_exists; use function substr; @@ -23,13 +23,21 @@ trait MagicRelationsTrait { /** - * Returns the relation object with the specified name. + * @inheritdoc * - * A relation is defined by a getter method which returns an {@see ActiveQueryInterface} object. + * A relation is defined by a getter method which has prefix `get` and suffix `Query` and returns an object + * implementing the {@see ActiveQueryInterface}. Normally this would be a relational {@see ActiveQuery} object. * - * It can be declared in either the Active Record class itself or one of its behaviors. + * For example, a relation named `orders` is defined using the following getter method: * - * @param string $name the relation name, e.g. `orders` for a relation defined via `getOrders()` method + * ```php + * public function getOrders(): ActiveQueryInterface + * { + * return $this->hasMany(Order::class, ['customer_id' => 'id']); + * } + * ``` + * + * @param string $name The relation name, for example `orders` for a relation defined via `getOrdersQuery()` method * (case-sensitive). * @param bool $throwException whether to throw exception if the relation does not exist. * @@ -43,42 +51,47 @@ public function relationQuery(string $name, bool $throwException = true): Active { $getter = 'get' . ucfirst($name); - try { - /** the relation could be defined in a behavior */ - $relation = $this->$getter(); - } catch (Error) { - if ($throwException) { - throw new InvalidArgumentException(static::class . ' has no relation named "' . $name . '".'); + if (!method_exists($this, $getter)) { + if (!$throwException) { + return null; } - return null; + throw new InvalidArgumentException(static::class . ' has no relation named "' . $name . '".'); } - if (!$relation instanceof ActiveQueryInterface) { - if ($throwException) { - throw new InvalidArgumentException(static::class . ' has no relation named "' . $name . '".'); + $method = new ReflectionMethod($this, $getter); + $type = $method->getReturnType(); + + if ( + $type === null + || !is_a('\\' . $type->getName(), ActiveQueryInterface::class, true) + ) { + if (!$throwException) { + return null; } - return null; - } + $typeName = $type === null ? 'mixed' : $type->getName(); - if (method_exists($this, $getter)) { - /** relation name is case sensitive, trying to validate it when the relation is defined within this class */ - $method = new ReflectionMethod($this, $getter); - $realName = lcfirst(substr($method->getName(), 3)); + throw new InvalidArgumentException( + 'Relation query method "' . static::class . '::' . $getter . '()" should return type "' + . ActiveQueryInterface::class . '", but returns "' . $typeName . '" type.' + ); + } - if ($realName !== $name) { - if ($throwException) { - throw new InvalidArgumentException( - 'Relation names are case sensitive. ' . static::class - . " has a relation named \"$realName\" instead of \"$name\"." - ); - } + /** relation name is case sensitive, trying to validate it when the relation is defined within this class */ + $realName = lcfirst(substr($method->getName(), 3)); + if ($realName !== $name) { + if (!$throwException) { return null; } + + throw new InvalidArgumentException( + 'Relation names are case sensitive. ' . static::class + . " has a relation named \"$realName\" instead of \"$name\"." + ); } - return $relation; + return $this->$getter(); } } diff --git a/tests/ActiveQueryTest.php b/tests/ActiveQueryTest.php index d3aacc1e5..52d05fbd5 100644 --- a/tests/ActiveQueryTest.php +++ b/tests/ActiveQueryTest.php @@ -2355,7 +2355,8 @@ public function testGetRelationInvalidArgumentExceptionHasNoRelationNamed(): voi $this->expectException(InvalidArgumentException::class); $this->expectExceptionMessage( - 'Yiisoft\ActiveRecord\Tests\Stubs\ActiveRecord\Customer has no relation named "item"' + 'Relation query method "Yiisoft\ActiveRecord\Tests\Stubs\ActiveRecord\Customer::getItem()" should return' + . ' type "Yiisoft\ActiveRecord\ActiveQueryInterface", but returns "void" type.' ); $query->relationQuery('item'); } diff --git a/tests/Stubs/ActiveRecord/Customer.php b/tests/Stubs/ActiveRecord/Customer.php index 02055d2a7..5895e3738 100644 --- a/tests/Stubs/ActiveRecord/Customer.php +++ b/tests/Stubs/ActiveRecord/Customer.php @@ -109,13 +109,13 @@ public function setOrdersReadOnly(): void { } - public function getOrderItems2() + public function getOrderItems2(): ActiveQuery { return $this->hasMany(OrderItem::class, ['order_id' => 'id']) ->via('ordersNoOrder'); } - public function getItems2() + public function getItems2(): ActiveQuery { return $this->hasMany(Item::class, ['id' => 'item_id']) ->via('orderItems2'); diff --git a/tests/Stubs/ActiveRecord/Order.php b/tests/Stubs/ActiveRecord/Order.php index 89152b557..177d7cbff 100644 --- a/tests/Stubs/ActiveRecord/Order.php +++ b/tests/Stubs/ActiveRecord/Order.php @@ -211,12 +211,12 @@ public function activeAttributes(): array ]; } - public function getOrderItemsFor8() + public function getOrderItemsFor8(): ActiveQuery { return $this->hasMany(OrderItemWithNullFK::class, ['order_id' => 'id'])->andOnCondition(['subtotal' => 8.0]); } - public function getItemsFor8() + public function getItemsFor8(): ActiveQuery { return $this->hasMany(Item::class, ['id' => 'item_id'])->via('orderItemsFor8'); }