From e33858bc91ed83213ad938580b0b3b86d2fb29ff Mon Sep 17 00:00:00 2001 From: Sergei Tigrov Date: Sat, 25 May 2024 10:59:46 +0700 Subject: [PATCH 01/11] Refactor `ActiveQuery::removeDuplicatedModels()` to separate `ArrayAccess` implementation (#345) --- src/ActiveQuery.php | 75 ++++++++++++++++++++++++--------------------- 1 file changed, 40 insertions(+), 35 deletions(-) diff --git a/src/ActiveQuery.php b/src/ActiveQuery.php index dd866912b..c9a0394ec 100644 --- a/src/ActiveQuery.php +++ b/src/ActiveQuery.php @@ -23,6 +23,12 @@ use Yiisoft\Definitions\Exception\NotInstantiableException; use Yiisoft\Factory\NotFoundException; +use function array_column; +use function array_combine; +use function array_flip; +use function array_intersect_key; +use function array_key_first; +use function array_map; use function array_merge; use function array_values; use function count; @@ -289,53 +295,52 @@ public function populate(array $rows, Closure|string|null $indexBy = null): arra */ private function removeDuplicatedModels(array $models): array { - $hash = []; + $model = reset($models); - $pks = $this->getARInstance()->primaryKey(); + if ($this->asArray) { + $pks = $this->getARInstance()->primaryKey(); - if (count($pks) > 1) { - // Composite primary key. - foreach ($models as $i => $model) { - $key = []; - foreach ($pks as $pk) { - if (!isset($model[$pk])) { - // Don't continue if the primary key isn't part of the result set. - break 2; - } - $key[] = $model[$pk]; - } - - $key = serialize($key); - - if (isset($hash[$key])) { - unset($models[$i]); - } else { - $hash[$key] = true; - } + if (empty($pks)) { + throw new InvalidConfigException("Primary key of '$this->arClass' can not be empty."); } - } elseif (empty($pks)) { - throw new InvalidConfigException("Primary key of '$this->arClass' can not be empty."); - } else { - // Single column primary key. - $pk = reset($pks); - foreach ($models as $i => $model) { + foreach ($pks as $pk) { if (!isset($model[$pk])) { - // Don't continue if the primary key isn't part of the result set. - break; + return $models; } + } - $key = $model[$pk]; + if (count($pks) === 1) { + $hash = array_column($models, reset($pks)); + } else { + $flippedPks = array_flip($pks); + $hash = array_map( + static fn ($model): string => serialize(array_intersect_key($model, $flippedPks)), + $models + ); + } + } else { + $pks = $model->getPrimaryKey(true); - if (isset($hash[$key])) { - unset($models[$i]); - } else { - $hash[$key] = true; + if (empty($pks)) { + throw new InvalidConfigException("Primary key of '$this->arClass' can not be empty."); + } + + foreach ($pks as $pk) { + if ($pk === null) { + return $models; } } + + if (count($pks) === 1) { + $key = array_key_first($pks); + $hash = array_map(static fn ($model): string => (string) $model->getAttribute($key), $models); + } else { + $hash = array_map(static fn ($model): string => serialize($model->getPrimaryKey(true)), $models); + } } - return array_values($models); + return array_values(array_combine($hash, $models)); } /** From 1564264647d117ee089a833ded406a7e302d37ea Mon Sep 17 00:00:00 2001 From: Sergei Tigrov Date: Sat, 25 May 2024 15:30:56 +0700 Subject: [PATCH 02/11] Improve `getAttributes()` (#346) --- src/AbstractActiveRecord.php | 4 ++-- src/ActiveRecordInterface.php | 2 +- src/Trait/MagicPropertiesTrait.php | 4 ++-- 3 files changed, 5 insertions(+), 5 deletions(-) diff --git a/src/AbstractActiveRecord.php b/src/AbstractActiveRecord.php index 9be5bc6eb..9908244d1 100644 --- a/src/AbstractActiveRecord.php +++ b/src/AbstractActiveRecord.php @@ -115,11 +115,11 @@ public function getAttribute(string $name): mixed return $this->getObjectVars($this)[$name] ?? null; } - public function getAttributes(array $names = null, array $except = []): array + public function getAttributes(array|null $names = null, array $except = []): array { $names ??= $this->attributes(); - if ($except !== []) { + if (!empty($except)) { $names = array_diff($names, $except); } diff --git a/src/ActiveRecordInterface.php b/src/ActiveRecordInterface.php index cdcc7d25e..21d8a2327 100644 --- a/src/ActiveRecordInterface.php +++ b/src/ActiveRecordInterface.php @@ -132,7 +132,7 @@ public function getAttribute(string $name): mixed; * * @return array Attribute values (name => value). */ - public function getAttributes(array $names = null, array $except = []): array; + public function getAttributes(array|null $names = null, array $except = []): array; /** * Returns a value indicating whether the current record is new (not saved in the database). diff --git a/src/Trait/MagicPropertiesTrait.php b/src/Trait/MagicPropertiesTrait.php index 532c638c4..5a072062b 100644 --- a/src/Trait/MagicPropertiesTrait.php +++ b/src/Trait/MagicPropertiesTrait.php @@ -180,12 +180,12 @@ public function getAttribute(string $name): mixed return $this->attributes[$name] ?? null; } - public function getAttributes(array $names = null, array $except = []): array + public function getAttributes(array|null $names = null, array $except = []): array { $names ??= $this->attributes(); $attributes = array_merge($this->attributes, get_object_vars($this)); - if ($except !== []) { + if (!empty($except)) { $names = array_diff($names, $except); } From 4d6bcaf2abfc4a3837d59c76fa755fffeb413cc1 Mon Sep 17 00:00:00 2001 From: Sergei Tigrov Date: Sat, 25 May 2024 16:08:23 +0700 Subject: [PATCH 03/11] Refactor rename `getObjectVars()` to `getAttributesInternal()` (#347) --- src/AbstractActiveRecord.php | 22 ++++++-------- src/BaseActiveRecord.php | 4 +-- src/Trait/MagicPropertiesTrait.php | 49 +++++------------------------- 3 files changed, 18 insertions(+), 57 deletions(-) diff --git a/src/AbstractActiveRecord.php b/src/AbstractActiveRecord.php index 9908244d1..580e30150 100644 --- a/src/AbstractActiveRecord.php +++ b/src/AbstractActiveRecord.php @@ -56,17 +56,11 @@ public function __construct( /** * Returns the public and protected property values of an Active Record object. * - * This method is provided because a direct call of {@see get_object_vars()} within the {@see AbstractActiveRecord} - * class will return also private property values of {@see AbstractActiveRecord} class. - * - * @param ActiveRecordInterface $object - * * @return array - * @link https://www.php.net/manual/en/function.get-object-vars.php * * @psalm-return array */ - abstract protected function getObjectVars(ActiveRecordInterface $object): array; + abstract protected function getAttributesInternal(): array; /** * Inserts Active Record values into DB without considering transaction. @@ -112,7 +106,7 @@ public function equals(ActiveRecordInterface $record): bool public function getAttribute(string $name): mixed { - return $this->getObjectVars($this)[$name] ?? null; + return $this->getAttributesInternal()[$name] ?? null; } public function getAttributes(array|null $names = null, array $except = []): array @@ -123,7 +117,7 @@ public function getAttributes(array|null $names = null, array $except = []): arr $names = array_diff($names, $except); } - return array_intersect_key($this->getObjectVars($this), array_flip($names)); + return array_intersect_key($this->getAttributesInternal(), array_flip($names)); } public function getIsNewRecord(): bool @@ -345,11 +339,13 @@ public function instantiateQuery(string $arClass): ActiveQueryInterface */ public function isAttributeChanged(string $name, bool $identical = true): bool { - if (!isset($this->oldAttributes[$name])) { - return array_key_exists($name, $this->getObjectVars($this)); + $attributes = $this->getAttributesInternal(); + + if (empty($this->oldAttributes) || !array_key_exists($name, $this->oldAttributes)) { + return array_key_exists($name, $attributes); } - return $this->getAttribute($name) !== $this->oldAttributes[$name]; + return !array_key_exists($name, $attributes) || $attributes[$name] !== $this->oldAttributes[$name]; } public function isPrimaryKey(array $keys): bool @@ -674,7 +670,7 @@ public function setAttributes(array $values): void */ public function setIsNewRecord(bool $value): void { - $this->oldAttributes = $value ? null : $this->getObjectVars($this); + $this->oldAttributes = $value ? null : $this->getAttributesInternal(); } /** diff --git a/src/BaseActiveRecord.php b/src/BaseActiveRecord.php index 02c69ecd0..65652e026 100644 --- a/src/BaseActiveRecord.php +++ b/src/BaseActiveRecord.php @@ -230,9 +230,9 @@ protected function filterValidColumnNames(array $aliases): array return $columnNames; } - protected function getObjectVars(ActiveRecordInterface $object): array + protected function getAttributesInternal(): array { - return get_object_vars($object); + return get_object_vars($this); } protected function insertInternal(array $attributes = null): bool diff --git a/src/Trait/MagicPropertiesTrait.php b/src/Trait/MagicPropertiesTrait.php index 5a072062b..85825aed0 100644 --- a/src/Trait/MagicPropertiesTrait.php +++ b/src/Trait/MagicPropertiesTrait.php @@ -14,10 +14,6 @@ use Yiisoft\Db\Exception\InvalidConfigException; use Yiisoft\Db\Exception\UnknownPropertyException; -use function array_diff; -use function array_flip; -use function array_intersect_key; -use function array_key_exists; use function array_merge; use function get_object_vars; use function in_array; @@ -54,6 +50,7 @@ */ trait MagicPropertiesTrait { + /** @psalm-var array $attributes */ private array $attributes = []; /** @@ -171,49 +168,11 @@ public function __set(string $name, mixed $value): void throw new UnknownPropertyException('Setting unknown property: ' . static::class . '::' . $name); } - public function getAttribute(string $name): mixed - { - if ($name !== 'attributes' && property_exists($this, $name)) { - return get_object_vars($this)[$name] ?? null; - } - - return $this->attributes[$name] ?? null; - } - - public function getAttributes(array|null $names = null, array $except = []): array - { - $names ??= $this->attributes(); - $attributes = array_merge($this->attributes, get_object_vars($this)); - - if (!empty($except)) { - $names = array_diff($names, $except); - } - - return array_intersect_key($attributes, array_flip($names)); - } - public function hasAttribute(string $name): bool { return isset($this->attributes[$name]) || in_array($name, $this->attributes(), true); } - public function isAttributeChanged(string $name, bool $identical = true): bool - { - $hasOldAttribute = array_key_exists($name, $this->getOldAttributes()); - - if (!$hasOldAttribute) { - return property_exists($this, $name) && array_key_exists($name, get_object_vars($this)) - || array_key_exists($name, $this->attributes); - } - - if (property_exists($this, $name)) { - return $this->getOldAttribute($name) !== $this->$name; - } - - return !array_key_exists($name, $this->attributes) - || $this->getOldAttribute($name) !== $this->attributes[$name]; - } - public function setAttribute(string $name, mixed $value): void { if ($this->hasAttribute($name)) { @@ -264,6 +223,12 @@ public function canSetProperty(string $name, bool $checkVars = true): bool || $this->hasAttribute($name); } + /** @psalm-return array */ + protected function getAttributesInternal(): array + { + return array_merge($this->attributes, parent::getAttributesInternal()); + } + protected function populateAttribute(string $name, mixed $value): void { if ($name !== 'attributes' && property_exists($this, $name)) { From ad73d8f417e4e33b896d36558f28b682e7ab86d5 Mon Sep 17 00:00:00 2001 From: Sergei Tigrov Date: Sat, 25 May 2024 18:24:20 +0700 Subject: [PATCH 04/11] Rename `ArArrayHelper::populate()` to `ArArrayHelper::index()` (#348) --- src/ActiveQuery.php | 2 +- src/ArArrayHelper.php | 7 +++---- 2 files changed, 4 insertions(+), 5 deletions(-) diff --git a/src/ActiveQuery.php b/src/ActiveQuery.php index c9a0394ec..b68f9a808 100644 --- a/src/ActiveQuery.php +++ b/src/ActiveQuery.php @@ -275,7 +275,7 @@ public function populate(array $rows, Closure|string|null $indexBy = null): arra $this->addInverseRelations($models); } - return ArArrayHelper::populate($models, $indexBy); + return ArArrayHelper::index($models, $indexBy); } /** diff --git a/src/ArArrayHelper.php b/src/ArArrayHelper.php index 661623abb..391529ac3 100644 --- a/src/ArArrayHelper.php +++ b/src/ArArrayHelper.php @@ -96,8 +96,7 @@ public static function getValueByPath(ActiveRecordInterface|array $array, string } if (property_exists($array, $key)) { - $values = get_object_vars($array); - return array_key_exists($key, $values) ? $values[$key] : $default; + return array_key_exists($key, get_object_vars($array)) ? $array->$key : $default; } if ($array->isRelationPopulated($key)) { @@ -118,7 +117,7 @@ public static function getValueByPath(ActiveRecordInterface|array $array, string } /** - * Populates an array of rows with the specified column value as keys. + * Indexes an array of rows with the specified column value as keys. * * The input array should be multidimensional or an array of {@see ActiveRecordInterface} instances. * @@ -144,7 +143,7 @@ public static function getValueByPath(ActiveRecordInterface|array $array, string * * @return array[] */ - public static function populate(array $rows, Closure|string|null $indexBy = null): array + public static function index(array $rows, Closure|string|null $indexBy = null): array { if ($indexBy === null) { return $rows; From 804ca3fda599af5fa4f7f70ce6200abf2016a92f Mon Sep 17 00:00:00 2001 From: Sergei Tigrov Date: Sat, 25 May 2024 20:23:14 +0700 Subject: [PATCH 05/11] Refactor `indexBuckets()` remove `ArrayAccess` implementation (#349) --- src/ActiveRelationTrait.php | 21 ++++++++------------- 1 file changed, 8 insertions(+), 13 deletions(-) diff --git a/src/ActiveRelationTrait.php b/src/ActiveRelationTrait.php index da12861f5..e7b846b70 100644 --- a/src/ActiveRelationTrait.php +++ b/src/ActiveRelationTrait.php @@ -4,6 +4,7 @@ namespace Yiisoft\ActiveRecord; +use Closure; use ReflectionException; use Stringable; use Throwable; @@ -487,24 +488,18 @@ private function mapVia(array $map, array $viaMap): array } /** - * Indexes buckets by column name. + * Indexes buckets by a column name. * - * @param callable|string $indexBy the name of the column by which the query results should be indexed by. This can - * also be a callable(e.g. anonymous function) that returns the index value based on the given row data. + * @param Closure|string $indexBy the name of the column by which the query results should be indexed by. This can + * also be a {@see Closure} that returns the index value based on the given models data. */ - private function indexBuckets(array $buckets, callable|string $indexBy): array + private function indexBuckets(array $buckets, Closure|string $indexBy): array { - $result = []; - - foreach ($buckets as $key => $models) { - $result[$key] = []; - foreach ($models as $model) { - $index = is_string($indexBy) ? $model[$indexBy] : $indexBy($model); - $result[$key][$index] = $model; - } + foreach ($buckets as &$models) { + $models = ArArrayHelper::index($models, $indexBy); } - return $result; + return $buckets; } /** From 3beda5026f2908faf50826921bf573a1f89fa110 Mon Sep 17 00:00:00 2001 From: Sergei Tigrov Date: Mon, 27 May 2024 11:04:19 +0700 Subject: [PATCH 06/11] Refactor `populateRecord()` to split `ArrayAccess` implementation (#350) --- src/AbstractActiveRecord.php | 4 ++++ src/ArArrayHelper.php | 27 +++++++++++++++++++++++++++ src/BaseActiveRecord.php | 11 +++++------ 3 files changed, 36 insertions(+), 6 deletions(-) diff --git a/src/AbstractActiveRecord.php b/src/AbstractActiveRecord.php index 580e30150..21ef22c02 100644 --- a/src/AbstractActiveRecord.php +++ b/src/AbstractActiveRecord.php @@ -533,6 +533,10 @@ public function optimisticLock(): string|null */ public function populateRecord(array|object $row): void { + if ($row instanceof ActiveRecordInterface) { + $row = $row->getAttributes(); + } + foreach ($row as $name => $value) { $this->populateAttribute($name, $value); $this->oldAttributes[$name] = $value; diff --git a/src/ArArrayHelper.php b/src/ArArrayHelper.php index 391529ac3..d1b8246e8 100644 --- a/src/ArArrayHelper.php +++ b/src/ArArrayHelper.php @@ -5,11 +5,14 @@ namespace Yiisoft\ActiveRecord; use Closure; +use Traversable; use function array_combine; use function array_key_exists; use function array_map; use function get_object_vars; +use function is_array; +use function iterator_to_array; use function property_exists; use function strrpos; use function substr; @@ -162,4 +165,28 @@ public static function index(array $rows, Closure|string|null $indexBy = null): return $result; } + + /** + * Converts an object into an array. + * + * @param array|object $object The object to be converted into an array. + * + * @return array The array representation of the object. + */ + public static function toArray(array|object $object): array + { + if (is_array($object)) { + return $object; + } + + if ($object instanceof ActiveRecordInterface) { + return $object->getAttributes(); + } + + if ($object instanceof Traversable) { + return iterator_to_array($object); + } + + return get_object_vars($object); + } } diff --git a/src/BaseActiveRecord.php b/src/BaseActiveRecord.php index 65652e026..df4f6305e 100644 --- a/src/BaseActiveRecord.php +++ b/src/BaseActiveRecord.php @@ -162,16 +162,15 @@ public function loadDefaultValues(bool $skipIfSet = true): self public function populateRecord(array|object $row): void { + $row = ArArrayHelper::toArray($row); $columns = $this->getTableSchema()->getColumns(); + $rowColumns = array_intersect_key($row, $columns); - /** @psalm-var array[][] $row */ - foreach ($row as $name => $value) { - if (isset($columns[$name])) { - $row[$name] = $columns[$name]->phpTypecast($value); - } + foreach ($rowColumns as $name => &$value) { + $value = $columns[$name]->phpTypecast($value); } - parent::populateRecord($row); + parent::populateRecord($rowColumns + $row); } public function primaryKey(): array From 2c1010d7177dedcd511871fcc4c50d1096934df6 Mon Sep 17 00:00:00 2001 From: Sergei Tigrov Date: Mon, 27 May 2024 12:31:38 +0700 Subject: [PATCH 07/11] Refactor `populateRelation()` to separate `ArrayAccess` implementation (#351) --- src/ActiveRelationTrait.php | 34 +++++++++++++++++++++------------- 1 file changed, 21 insertions(+), 13 deletions(-) diff --git a/src/ActiveRelationTrait.php b/src/ActiveRelationTrait.php index e7b846b70..07b5df74d 100644 --- a/src/ActiveRelationTrait.php +++ b/src/ActiveRelationTrait.php @@ -283,13 +283,10 @@ public function populateRelation(string $name, array &$primaryModels): array $this->indexBy($indexBy); - $indexBy = $this->getIndexBy(); - if ($indexBy !== null && $this->multiple) { $buckets = $this->indexBuckets($buckets, $indexBy); } - $link = array_values($this->link); if (isset($viaQuery)) { $deepViaQuery = $viaQuery; @@ -297,30 +294,40 @@ public function populateRelation(string $name, array &$primaryModels): array $deepViaQuery = is_array($deepViaQuery->via) ? $deepViaQuery->via[1] : $deepViaQuery->via; } - $link = array_values($deepViaQuery->link); + $link = $deepViaQuery->link; + } else { + $link = $this->link; } foreach ($primaryModels as $i => $primaryModel) { $keys = null; + if ($this->multiple && count($link) === 1) { $primaryModelKey = reset($link); - $keys = $primaryModel[$primaryModelKey] ?? null; + + if ($primaryModel instanceof ActiveRecordInterface) { + $keys = $primaryModel->getAttribute($primaryModelKey); + } else { + $keys = $primaryModel[$primaryModelKey] ?? null; + } } + if (is_array($keys)) { $value = []; + foreach ($keys as $key) { $key = $this->normalizeModelKey($key); if (isset($buckets[$key])) { - if ($indexBy !== null) { - /** if indexBy is set, array_merge will cause renumbering of numeric array */ - foreach ($buckets[$key] as $bucketKey => $bucketValue) { - $value[$bucketKey] = $bucketValue; - } - } else { - $value = array_merge($value, $buckets[$key]); - } + $value[] = $buckets[$key]; } } + + if ($indexBy !== null) { + /** if indexBy is set, array_merge will cause renumbering of numeric array */ + $value = array_replace(...$value); + } else { + $value = array_merge(...$value); + } } else { $key = $this->getModelKey($primaryModel, $link); $value = $buckets[$key] ?? ($this->multiple ? [] : null); @@ -332,6 +339,7 @@ public function populateRelation(string $name, array &$primaryModels): array $primaryModels[$i][$name] = $value; } } + if ($this->inverseOf !== null) { $this->populateInverseRelation($primaryModels, $models, $name, $this->inverseOf); } From fe61e2d53cc2e2e91d53dc9df90dafef60eb72e2 Mon Sep 17 00:00:00 2001 From: Sergei Tigrov Date: Mon, 27 May 2024 13:12:58 +0700 Subject: [PATCH 08/11] Refactor `addInverseRelations()` (#352) --- src/ActiveRelationTrait.php | 31 ++++++++++++++----------------- 1 file changed, 14 insertions(+), 17 deletions(-) diff --git a/src/ActiveRelationTrait.php b/src/ActiveRelationTrait.php index 07b5df74d..5b30b74fb 100644 --- a/src/ActiveRelationTrait.php +++ b/src/ActiveRelationTrait.php @@ -194,24 +194,21 @@ private function addInverseRelations(array &$result): void return; } - foreach ($result as $i => $relatedModel) { - if ($relatedModel instanceof ActiveRecordInterface) { - if (!isset($inverseRelation)) { - /** @var ActiveQuery $inverseRelation */ - $inverseRelation = $relatedModel->relationQuery($this->inverseOf); - } - $relatedModel->populateRelation( - $this->inverseOf, - $inverseRelation->multiple ? [$this->primaryModel] : $this->primaryModel - ); - } else { - if (!isset($inverseRelation)) { - /** @var ActiveQuery $inverseRelation */ - $inverseRelation = $this->getARInstance()->relationQuery($this->inverseOf); - } + $relatedModel = reset($result); + + if ($relatedModel instanceof ActiveRecordInterface) { + $inverseRelation = $relatedModel->relationQuery($this->inverseOf); + $primaryModel = $inverseRelation->getMultiple() ? [$this->primaryModel] : $this->primaryModel; + + foreach ($result as $relatedModel) { + $relatedModel->populateRelation($this->inverseOf, $primaryModel); + } + } else { + $inverseRelation = $this->getARInstance()->relationQuery($this->inverseOf); + $primaryModel = $inverseRelation->getMultiple() ? [$this->primaryModel] : $this->primaryModel; - $result[$i][$this->inverseOf] = $inverseRelation->multiple - ? [$this->primaryModel] : $this->primaryModel; + foreach ($result as &$relatedModel) { + $relatedModel[$this->inverseOf] = $primaryModel; } } } From d8fa21dad9906123c6311ff80f1a5932804de870 Mon Sep 17 00:00:00 2001 From: Sergei Tigrov Date: Mon, 27 May 2024 17:06:16 +0700 Subject: [PATCH 09/11] Refactor `filterByModels()` to separate `ArrayAccess` implementation (#353) --- src/ActiveRelationTrait.php | 90 +++++++++++++++++++++---------------- 1 file changed, 52 insertions(+), 38 deletions(-) diff --git a/src/ActiveRelationTrait.php b/src/ActiveRelationTrait.php index 5b30b74fb..cd2e64d4d 100644 --- a/src/ActiveRelationTrait.php +++ b/src/ActiveRelationTrait.php @@ -12,9 +12,12 @@ use Yiisoft\Db\Exception\InvalidArgumentException; use Yiisoft\Db\Exception\InvalidConfigException; use Yiisoft\Db\Exception\NotSupportedException; -use Yiisoft\Db\Expression\ArrayExpression; use function array_combine; +use function array_diff_key; +use function array_fill_keys; +use function array_filter; +use function array_intersect_key; use function array_keys; use function array_merge; use function array_unique; @@ -542,63 +545,74 @@ private function prefixKeyColumns(array $attributes): array protected function filterByModels(array $models): void { $attributes = array_keys($this->link); - $attributes = $this->prefixKeyColumns($attributes); + $model = reset($models); $values = []; + if (count($attributes) === 1) { /** single key */ $attribute = reset($this->link); - foreach ($models as $model) { - $value = isset($model[$attribute]) || (is_object($model) && property_exists($model, $attribute)) ? $model[$attribute] : null; - if ($value !== null) { - if (is_array($value)) { - $values = array_merge($values, $value); - } elseif ($value instanceof ArrayExpression && $value->getDimension() === 1) { - $values = array_merge($values, $value->getValue()); - } else { - $values[] = $value; + + if ($model instanceof ActiveRecordInterface) { + foreach ($models as $model) { + $value = $model->getAttribute($attribute); + + if ($value !== null) { + if (is_array($value)) { + $values = [...$values, ...$value]; + } else { + $values[] = $value; + } + } + } + } else { + foreach ($models as $model) { + if (isset($model[$attribute])) { + $value = $model[$attribute]; + + if (is_array($value)) { + $values = [...$values, ...$value]; + } else { + $values[] = $value; + } } } } - if (empty($values)) { - $this->emulateExecution(); + if (!empty($values)) { + $scalarValues = array_filter($values, 'is_scalar'); + $nonScalarValues = array_diff_key($values, $scalarValues); + + $scalarValues = array_unique($scalarValues); + $values = [...$scalarValues, ...$nonScalarValues]; } } else { - /** - * composite keys ensure keys of $this->link are prefixed the same way as $attributes. - */ - $prefixedLink = array_combine($attributes, $this->link); + $nulls = array_fill_keys($this->link, null); - foreach ($models as $model) { - $v = []; + if ($model instanceof ActiveRecordInterface) { + foreach ($models as $model) { + $value = $model->getAttributes($this->link); - foreach ($prefixedLink as $attribute => $link) { - $v[$attribute] = $model[$link]; + if (!empty($value)) { + $values[] = array_combine($attributes, array_merge($nulls, $value)); + } } + } else { + foreach ($models as $model) { + $value = array_intersect_key($model, $nulls); - $values[] = $v; - - if (empty($v)) { - $this->emulateExecution(); + if (!empty($value)) { + $values[] = array_combine($attributes, array_merge($nulls, $value)); + } } } } - if (!empty($values)) { - $scalarValues = []; - $nonScalarValues = []; - foreach ($values as $value) { - if (is_scalar($value)) { - $scalarValues[] = $value; - } else { - $nonScalarValues[] = $value; - } - } - - $scalarValues = array_unique($scalarValues); - $values = [...$scalarValues, ...$nonScalarValues]; + if (empty($values)) { + $this->emulateExecution(); + $this->andWhere('1=0'); + return; } $this->andWhere(['in', $attributes, $values]); From b52db346f185288298829b5fb44d3bd108ebb3d0 Mon Sep 17 00:00:00 2001 From: Sergei Tigrov Date: Tue, 28 May 2024 10:18:18 +0700 Subject: [PATCH 10/11] Refactor `populateInverseRelation()` to separate `ArrayAccess` implementation (#354) --- src/ActiveRelationTrait.php | 103 +++++++++++++++++++++--------------- 1 file changed, 60 insertions(+), 43 deletions(-) diff --git a/src/ActiveRelationTrait.php b/src/ActiveRelationTrait.php index cd2e64d4d..5e615147e 100644 --- a/src/ActiveRelationTrait.php +++ b/src/ActiveRelationTrait.php @@ -21,7 +21,6 @@ use function array_keys; use function array_merge; use function array_unique; -use function array_values; use function count; use function is_array; use function is_object; @@ -276,9 +275,9 @@ public function populateRelation(string $name, array &$primaryModels): array $models = $this->all(); if (isset($viaModels, $viaQuery)) { - $buckets = $this->buildBuckets($models, $this->link, $viaModels, $viaQuery); + $buckets = $this->buildBuckets($models, $viaModels, $viaQuery); } else { - $buckets = $this->buildBuckets($models, $this->link); + $buckets = $this->buildBuckets($models); } $this->indexBy($indexBy); @@ -363,66 +362,84 @@ private function populateInverseRelation( $model = reset($models); if ($model instanceof ActiveRecordInterface) { - /** @var ActiveQuery $relation */ - $relation = $model->relationQuery($name); - } else { - /** @var ActiveQuery $relation */ - $relation = $this->getARInstance()->relationQuery($name); + $this->populateInverseRelationToModels($models, $primaryModels, $name); + return; } - if ($relation->getMultiple()) { - $buckets = $this->buildBuckets($primaryModels, $relation->getLink(), null, null, false); - if ($model instanceof ActiveRecordInterface) { - foreach ($models as $model) { - $key = $this->getModelKey($model, $relation->getLink()); - if ($model instanceof ActiveRecordInterface) { - $model->populateRelation($name, $buckets[$key] ?? []); + $primaryModel = reset($primaryModels); + + if ($primaryModel instanceof ActiveRecordInterface) { + if ($this->multiple) { + foreach ($primaryModels as $primaryModel) { + $models = $primaryModel->relation($primaryName); + if (!empty($models)) { + $this->populateInverseRelationToModels($models, $primaryModels, $name); + $primaryModel->populateRelation($primaryName, $models); } } } else { - foreach ($primaryModels as $i => $primaryModel) { - if ($this->multiple) { - foreach ($primaryModel as $j => $m) { - $key = $this->getModelKey($m, $relation->getLink()); - $primaryModels[$i][$j][$name] = $buckets[$key] ?? []; - } - } elseif (!empty($primaryModel[$primaryName])) { - $key = $this->getModelKey($primaryModel[$primaryName], $relation->getLink()); - $primaryModels[$i][$primaryName][$name] = $buckets[$key] ?? []; + foreach ($primaryModels as $primaryModel) { + $model = $primaryModel->relation($primaryName); + if (!empty($model)) { + $models = [$model]; + $this->populateInverseRelationToModels($models, $primaryModels, $name); + $primaryModel->populateRelation($primaryName, $models[0]); } } } - } elseif ($this->multiple) { - foreach ($primaryModels as $i => $primaryModel) { - foreach ($primaryModel[$primaryName] as $j => $m) { - if ($m instanceof ActiveRecordInterface) { - $m->populateRelation($name, $primaryModel); - } else { - $primaryModels[$i][$primaryName][$j][$name] = $primaryModel; + } else { + if ($this->multiple) { + foreach ($primaryModels as &$primaryModel) { + if (!empty($primaryModel[$primaryName])) { + $this->populateInverseRelationToModels($primaryModel[$primaryName], $primaryModels, $name); + } + } + } else { + foreach ($primaryModels as &$primaryModel) { + if (!empty($primaryModel[$primaryName])) { + $models = [$primaryModel[$primaryName]]; + $this->populateInverseRelationToModels($models, $primaryModels, $name); + $primaryModel[$primaryName] = $models[0]; } } } + } + } + + private function populateInverseRelationToModels(array &$models, array $primaryModels, string $name): void + { + $model = reset($models); + $isArray = is_array($model); + + /** @var ActiveQuery $relation */ + $relation = $isArray ? $this->getARInstance()->relationQuery($name) : $model->relationQuery($name); + $buckets = $relation->buildBuckets($primaryModels); + $link = $relation->getLink(); + $default = $relation->getMultiple() ? [] : null; + + if ($isArray) { + /** @var array $model */ + foreach ($models as &$model) { + $key = $this->getModelKey($model, $link); + $model[$name] = $buckets[$key] ?? $default; + } } else { - foreach ($primaryModels as $i => $primaryModel) { - if ($primaryModel[$primaryName] instanceof ActiveRecordInterface) { - $primaryModel[$primaryName]->populateRelation($name, $primaryModel); - } elseif (!empty($primaryModel[$primaryName])) { - $primaryModels[$i][$primaryName][$name] = $primaryModel; - } + /** @var ActiveRecordInterface $model */ + foreach ($models as $model) { + $key = $this->getModelKey($model, $link); + $model->populateRelation($name, $buckets[$key] ?? $default); } } } private function buildBuckets( array $models, - array $link, array $viaModels = null, - self $viaQuery = null, - bool $checkMultiple = true + self $viaQuery = null ): array { if ($viaModels !== null) { $map = []; - $linkValues = array_values($link); + $linkValues = $this->link; $viaLink = $viaQuery->link ?? []; $viaLinkKeys = array_keys($viaLink); $viaVia = null; @@ -452,7 +469,7 @@ private function buildBuckets( } $buckets = []; - $linkKeys = array_keys($link); + $linkKeys = array_keys($this->link); if (isset($map)) { foreach ($models as $model) { @@ -471,7 +488,7 @@ private function buildBuckets( } } - if ($checkMultiple && !$this->multiple) { + if (!$this->multiple) { foreach ($buckets as $i => $bucket) { $buckets[$i] = reset($bucket); } From bc8b6fbb04c7dbbed2200f2ef574384716989158 Mon Sep 17 00:00:00 2001 From: Sergei Tigrov Date: Tue, 28 May 2024 11:20:40 +0700 Subject: [PATCH 11/11] Improve performance (#355) --- src/AbstractActiveRecord.php | 10 +++----- src/ActiveRelationTrait.php | 49 +++++++++++------------------------- 2 files changed, 19 insertions(+), 40 deletions(-) diff --git a/src/AbstractActiveRecord.php b/src/AbstractActiveRecord.php index 21ef22c02..0be18c184 100644 --- a/src/AbstractActiveRecord.php +++ b/src/AbstractActiveRecord.php @@ -190,10 +190,8 @@ public function getOldPrimaryKey(bool $asArray = false): mixed ); } - if (count($keys) === 1) { - $key = $this->oldAttributes[$keys[0]] ?? null; - - return $asArray ? [$keys[0] => $key] : $key; + if ($asArray === false && count($keys) === 1) { + return $this->oldAttributes[$keys[0]] ?? null; } $values = []; @@ -209,8 +207,8 @@ public function getPrimaryKey(bool $asArray = false): mixed { $keys = $this->primaryKey(); - if (count($keys) === 1) { - return $asArray ? [$keys[0] => $this->getAttribute($keys[0])] : $this->getAttribute($keys[0]); + if ($asArray === false && count($keys) === 1) { + return $this->getAttribute($keys[0]); } $values = []; diff --git a/src/ActiveRelationTrait.php b/src/ActiveRelationTrait.php index 5e615147e..e15d8ddec 100644 --- a/src/ActiveRelationTrait.php +++ b/src/ActiveRelationTrait.php @@ -6,13 +6,13 @@ use Closure; use ReflectionException; -use Stringable; use Throwable; use Yiisoft\Db\Exception\Exception; use Yiisoft\Db\Exception\InvalidArgumentException; use Yiisoft\Db\Exception\InvalidConfigException; use Yiisoft\Db\Exception\NotSupportedException; +use function array_column; use function array_combine; use function array_diff_key; use function array_fill_keys; @@ -24,7 +24,6 @@ use function count; use function is_array; use function is_object; -use function is_scalar; use function is_string; use function key; use function reset; @@ -315,7 +314,8 @@ public function populateRelation(string $name, array &$primaryModels): array $value = []; foreach ($keys as $key) { - $key = $this->normalizeModelKey($key); + $key = (string) $key; + if (isset($buckets[$key])) { $value[] = $buckets[$key]; } @@ -489,9 +489,10 @@ private function buildBuckets( } if (!$this->multiple) { - foreach ($buckets as $i => $bucket) { - $buckets[$i] = reset($bucket); - } + return array_combine( + array_keys($buckets), + array_column($buckets, 0) + ); } return $buckets; @@ -635,14 +636,14 @@ protected function filterByModels(array $models): void $this->andWhere(['in', $attributes, $values]); } - private function getModelKey(ActiveRecordInterface|array $activeRecord, array $attributes): false|int|string + private function getModelKey(ActiveRecordInterface|array $activeRecord, array $attributes): string { $key = []; if (is_array($activeRecord)) { foreach ($attributes as $attribute) { if (isset($activeRecord[$attribute])) { - $key[] = $this->normalizeModelKey($activeRecord[$attribute]); + $key[] = (string) $activeRecord[$attribute]; } } } else { @@ -650,36 +651,16 @@ private function getModelKey(ActiveRecordInterface|array $activeRecord, array $a $value = $activeRecord->getAttribute($attribute); if ($value !== null) { - $key[] = $this->normalizeModelKey($value); + $key[] = (string) $value; } } } - if (count($key) > 1) { - return serialize($key); - } - - $key = reset($key); - - return is_scalar($key) ? $key : serialize($key); - } - - /** - * @param int|string|Stringable|null $value raw key value. - * - * @return int|string|null normalized key value. - */ - private function normalizeModelKey(int|string|Stringable|null $value): int|string|null - { - if ($value instanceof Stringable) { - /** - * ensure matching to special objects, which are convertible to string, for cross-DBMS relations, - * for example: `|MongoId` - */ - $value = (string) $value; - } - - return $value; + return match (count($key)) { + 0 => '', + 1 => $key[0], + default => serialize($key), + }; } /**