From fab2c91beb95631b4c7abc852f662bf39dab20ab Mon Sep 17 00:00:00 2001 From: Johannes Meyer Date: Fri, 20 Oct 2023 13:26:12 +0200 Subject: [PATCH] Hydrator: Fix that columns can only be mapped once Given the two relations `a.b` and `a.b.c` and `a.b.c_id` == `a.b.c.id`, both paths are qualified to `a_b_c_id`, resulting in only one column being returned by the datasource. Such a column must be kept and shared until the last reference is fulfilled. --- src/Hydrator.php | 37 ++++++++++++++++++++++++++++++++++--- 1 file changed, 34 insertions(+), 3 deletions(-) diff --git a/src/Hydrator.php b/src/Hydrator.php index e3cd23d..f1ff4b9 100644 --- a/src/Hydrator.php +++ b/src/Hydrator.php @@ -13,6 +13,9 @@ class Hydrator /** @var array Additional hydration rules for the model's relations */ protected $hydrators = []; + /** @var array> Map of columns to referencing paths */ + protected $columnToTargetMap = []; + /** @var Query The query the hydration rules are for */ protected $query; @@ -70,11 +73,31 @@ public function add($path) } } + $this->updateColumnToTargetMap($path, $columnToPropertyMap); $this->hydrators[$path] = [$target, $relation, $columnToPropertyMap, $defaults]; return $this; } + /** + * Update which columns the given path is referencing + * + * @param string $path + * @param array $columnToPropertyMap + * + * @return void + */ + protected function updateColumnToTargetMap(string $path, array $columnToPropertyMap): void + { + foreach ($columnToPropertyMap as $qualifiedColumnPath => $_) { + if (isset($this->columnToTargetMap[$qualifiedColumnPath])) { + $this->columnToTargetMap[$qualifiedColumnPath][$path] = true; + } else { + $this->columnToTargetMap[$qualifiedColumnPath] = [$path => true]; + } + } + } + /** * Hydrate the given raw database rows into the specified model * @@ -120,7 +143,7 @@ public function hydrate(array $data, Model $model) } } - $subject->setProperties($this->extractAndMap($data, $columnToPropertyMap)); + $subject->setProperties($this->extractAndMap($data, $columnToPropertyMap, $path)); $this->query->getResolver()->getBehaviors($target)->retrieve($subject); $defaultsToApply[] = [$subject, $defaults]; } @@ -181,15 +204,23 @@ public function hydrate(array $data, Model $model) * * @param array $data * @param array $columnToPropertyMap + * @param string $path * * @return array */ - protected function extractAndMap(array &$data, array $columnToPropertyMap) + protected function extractAndMap(array &$data, array $columnToPropertyMap, string $path) { $extracted = []; foreach (array_intersect_key($columnToPropertyMap, $data) as $column => $property) { $extracted[$property] = $data[$column]; - unset($data[$column]); + + if (isset($this->columnToTargetMap[$column][$path])) { + unset($this->columnToTargetMap[$column][$path]); + if (empty($this->columnToTargetMap[$column])) { + // Only unset a column once it's really not required anymore + unset($data[$column], $this->columnToTargetMap[$column]); + } + } } return $extracted;