diff --git a/psalm.xml b/psalm.xml index e66ef57a1..19d34d11d 100644 --- a/psalm.xml +++ b/psalm.xml @@ -14,4 +14,7 @@ + + + diff --git a/src/ActiveRecord.php b/src/ActiveRecord.php index f636136a3..31ac97320 100644 --- a/src/ActiveRecord.php +++ b/src/ActiveRecord.php @@ -323,7 +323,7 @@ public function transactions(): array return []; } - public function update(array $attributeNames = null): false|int + public function update(array $attributeNames = null): int { if (!$this->isTransactional(self::OP_UPDATE)) { return $this->updateInternal($attributeNames); diff --git a/src/ActiveRecordInterface.php b/src/ActiveRecordInterface.php index f8a9952b8..6af7b3cd7 100644 --- a/src/ActiveRecordInterface.php +++ b/src/ActiveRecordInterface.php @@ -159,6 +159,12 @@ public function getIsNewRecord(): bool; * @return mixed The old primary key value. An array (column name => column value) is returned if the primary key * is composite or `$asArray` is true. A string is returned otherwise (`null` will be returned if the key value is * `null`). + * + * @psalm-return ( + * $asArray is true + * ? array + * : mixed|null + * ) */ public function getOldPrimaryKey(bool $asArray = false): mixed; @@ -172,6 +178,12 @@ public function getOldPrimaryKey(bool $asArray = false): mixed; * @return mixed The primary key value. An array (attribute name => attribute value) is returned if the primary key * is composite or `$asArray` is true. A string is returned otherwise (`null` will be returned if the key value is * `null`). + * + * @psalm-return ( + * $asArray is true + * ? array + * : mixed|null + * ) */ public function getPrimaryKey(bool $asArray = false): mixed; @@ -322,7 +334,7 @@ public function populateRelation(string $name, array|self|null $records): void; * @throws Exception * @throws InvalidConfigException * - * @psalm-return string[] The primary keys of the associated database table. + * @return string[] The primary keys of the associated database table. */ public function primaryKey(): array; @@ -377,7 +389,7 @@ public function setAttribute(string $name, mixed $value): void; * For this reason, you should use the following code to check if update() is successful or not: * * ```php - * if ($customer->update() !== false) { + * if ($customer->update() !== 0) { * // update successful * } else { * // update failed @@ -391,10 +403,9 @@ public function setAttribute(string $name, mixed $value): void; * outdated. * @throws Throwable In case update failed. * - * @return false|int The number of rows affected, or false if validation fails or {@seebeforeSave()} stops the - * updating process. + * @return int The number of rows affected. */ - public function update(array $attributeNames = null): false|int; + public function update(array $attributeNames = null): int; /** * Updates the whole table using the provided attribute values and conditions. @@ -410,10 +421,10 @@ public function update(array $attributeNames = null): false|int; * * ```php * $customerQuery = new ActiveQuery(Customer::class, $db); - * $aqClasses = $customerQuery->where('status = 2')->all(); - * foreach ($aqClasses as $aqClass) { - * $aqClass->status = 1; - * $aqClass->update(); + * $customers = $customerQuery->where('status = 2')->all(); + * foreach ($customers as $customer) { + * $customer->status = 1; + * $customer->update(); * } * ``` * diff --git a/src/BaseActiveRecord.php b/src/BaseActiveRecord.php index d6cb497e3..11b739de2 100644 --- a/src/BaseActiveRecord.php +++ b/src/BaseActiveRecord.php @@ -233,7 +233,7 @@ public function getOldAttributes(): array * @throws InvalidConfigException * @throws Exception */ - public function getOldPrimaryKey(bool $asArray = false): array|string|null + public function getOldPrimaryKey(bool $asArray = false): mixed { $keys = $this->primaryKey(); @@ -244,16 +244,13 @@ public function getOldPrimaryKey(bool $asArray = false): array|string|null ); } - if (!$asArray && count($keys) === 1) { - /** @psalm-var string|null */ + if ($asArray === false && count($keys) === 1) { return $this->oldAttributes[$keys[0]] ?? null; } $values = []; - /** @psalm-var list $keys */ foreach ($keys as $name) { - /** @psalm-var string|null */ $values[$name] = $this->oldAttributes[$name] ?? null; } @@ -270,9 +267,7 @@ public function getPrimaryKey(bool $asArray = false): mixed $values = []; - /** @psalm-var list $keys */ foreach ($keys as $name) { - /** @psalm-var string|null */ $values[$name] = $this->attributes[$name] ?? null; } @@ -655,7 +650,9 @@ public function save(array $attributeNames = null): bool return $this->insert($attributeNames); } - return $this->update($attributeNames) !== false; + $this->update($attributeNames); + + return true; } public function setAttribute(string $name, mixed $value): void @@ -734,7 +731,7 @@ public function setOldAttributes(array $values = null): void $this->oldAttributes = $values; } - public function update(array $attributeNames = null): false|int + public function update(array $attributeNames = null): int { return $this->updateInternal($attributeNames); } @@ -752,19 +749,11 @@ public function updateAttributes(array $attributes): int { $attrs = []; - $condition = $this->getOldPrimaryKey(true); - - if ($condition === null || $condition === []) { - return 0; - } - - /** @psalm-var mixed $value */ foreach ($attributes as $name => $value) { if (is_int($name)) { - /** @psalm-var mixed */ $attrs[] = $value; } else { - $this->$name = $value; + $this->setAttribute($name, $value); $attrs[] = $name; } } @@ -775,11 +764,10 @@ public function updateAttributes(array $attributes): int return 0; } - $rows = $this->updateAll($values, $this->getOldPrimaryKey(true) ?? []); + $rows = $this->updateAll($values, $this->getOldPrimaryKey(true)); - /** @psalm-var array $value */ foreach ($values as $name => $value) { - $this->oldAttributes[$name] = $this->attributes[$name]; + $this->oldAttributes[$name] = $value; } return $rows; @@ -843,6 +831,8 @@ public function updateAllCounters(array $counters, array|string $condition = '', * @param array $counters The counters to be updated (attribute name => increment value), use negative values if you * want to decrement the counters. * + * @psalm-param array $counters + * * @throws Exception * @throws NotSupportedException * @@ -852,18 +842,16 @@ public function updateAllCounters(array $counters, array|string $condition = '', */ public function updateCounters(array $counters): bool { - if ($this->updateAllCounters($counters, $this->getOldPrimaryKey(true) ?? '')) { - /** @psalm-var array $counters */ - foreach ($counters as $name => $value) { - $this->attributes[$name] = isset($this->attributes[$name]) - ? (int) $this->attributes[$name] + $value : $value; - $this->oldAttributes[$name] = $this->attributes[$name]; - } + if ($this->updateAllCounters($counters, $this->getOldPrimaryKey(true)) === 0) { + return false; + } - return true; + foreach ($counters as $name => $value) { + $this->attributes[$name] = (int)($this->attributes[$name] ?? 0) + $value; + $this->oldAttributes[$name] = $this->attributes[$name]; } - return false; + return true; } public function unlink(string $name, ActiveRecordInterface $arClass, bool $delete = false): void @@ -1173,41 +1161,27 @@ protected function updateInternal(array $attributes = null): int return 0; } - /** @psalm-var mixed $condition */ $condition = $this->getOldPrimaryKey(true); $lock = $this->optimisticLock(); - if ($lock !== null && is_array($condition)) { - $values[$lock] = (int) $this->$lock + 1; - /** @psalm-var array|string */ - $condition[$lock] = $this->$lock; - } + if ($lock !== null) { + $lockValue = (int)$this->getAttribute($lock); - /** - * We do not check the return value of updateAll() because it's possible that the UPDATE statement doesn't - * change anything and thus returns 0. - * - * @psalm-var array|string $condition - */ - $rows = $this->updateAll($values, $condition); + $condition[$lock] = $lockValue; + $values[$lock] = ++$lockValue; - if ($lock !== null && !$rows) { - throw new StaleObjectException('The object being updated is outdated.'); - } + $rows = $this->updateAll($values, $condition); - if (isset($values[$lock])) { - $this->$lock = $values[$lock]; - } + if ($rows === 0) { + throw new StaleObjectException('The object being updated is outdated.'); + } - $changedAttributes = []; + $this->setAttribute($lock, $lockValue); + } else { + $rows = $this->updateAll($values, $condition); + } - /** - * @psalm-var string $name - * @psalm-var mixed $value - */ foreach ($values as $name => $value) { - /** @psalm-var mixed */ - $changedAttributes[$name] = $this->oldAttributes[$name] ?? null; $this->oldAttributes[$name] = $value; } diff --git a/tests/ActiveRecordTest.php b/tests/ActiveRecordTest.php index 903ee53ce..8a9e6f084 100644 --- a/tests/ActiveRecordTest.php +++ b/tests/ActiveRecordTest.php @@ -836,4 +836,15 @@ public function testToArrayForArrayable(): void ]), ); } + + public function testSaveWithoutChanges() + { + $this->checkFixture($this->db, 'customer'); + + $customerQuery = new ActiveQuery(Customer::class, $this->db); + + $customer = $customerQuery->findOne(1); + + $this->assertTrue($customer->save()); + } }