From 1d3b224ef6b57dd7040caca3f502bc2495b7257c Mon Sep 17 00:00:00 2001 From: Austin Collier Date: Sat, 16 Mar 2024 18:08:09 -0600 Subject: [PATCH 1/2] changed it so PdoWrapper returns collections --- flight/database/PdoWrapper.php | 27 ++++++++++++++++++--------- tests/PdoWrapperTest.php | 7 +++++++ 2 files changed, 25 insertions(+), 9 deletions(-) diff --git a/flight/database/PdoWrapper.php b/flight/database/PdoWrapper.php index 842038c..4821cb9 100644 --- a/flight/database/PdoWrapper.php +++ b/flight/database/PdoWrapper.php @@ -4,6 +4,7 @@ declare(strict_types=1); namespace flight\database; +use flight\util\Collection; use PDO; use PDOStatement; @@ -47,8 +48,9 @@ class PdoWrapper extends PDO */ public function fetchField(string $sql, array $params = []) { - $data = $this->fetchRow($sql, $params); - return reset($data); + $collection_data = $this->fetchRow($sql, $params); + $array_data = $collection_data->getData(); + return reset($array_data); } /** @@ -59,13 +61,13 @@ class PdoWrapper extends PDO * @param string $sql - Ex: "SELECT * FROM table WHERE something = ?" * @param array $params - Ex: [ $something ] * - * @return array + * @return Collection */ - public function fetchRow(string $sql, array $params = []): array + public function fetchRow(string $sql, array $params = []): Collection { $sql .= stripos($sql, 'LIMIT') === false ? ' LIMIT 1' : ''; $result = $this->fetchAll($sql, $params); - return count($result) > 0 ? $result[0] : []; + return count($result) > 0 ? $result[0] : new Collection(); } /** @@ -79,17 +81,24 @@ class PdoWrapper extends PDO * @param string $sql - Ex: "SELECT * FROM table WHERE something = ?" * @param array $params - Ex: [ $something ] * - * @return array> + * @return array */ - public function fetchAll(string $sql, array $params = []): array + public function fetchAll(string $sql, array $params = []) { $processed_sql_data = $this->processInStatementSql($sql, $params); $sql = $processed_sql_data['sql']; $params = $processed_sql_data['params']; $statement = $this->prepare($sql); $statement->execute($params); - $result = $statement->fetchAll(); - return is_array($result) ? $result : []; + $results = $statement->fetchAll(); + if (is_array($results) === true && count($results) > 0) { + foreach ($results as &$result) { + $result = new Collection($result); + } + } else { + $results = []; + } + return $results; } /** diff --git a/tests/PdoWrapperTest.php b/tests/PdoWrapperTest.php index 324c47b..0f41a92 100644 --- a/tests/PdoWrapperTest.php +++ b/tests/PdoWrapperTest.php @@ -88,6 +88,13 @@ class PdoWrapperTest extends TestCase $this->assertEquals('three', $rows[2]['name']); } + public function testFetchAllNoRows() + { + $rows = $this->pdo_wrapper->fetchAll('SELECT * FROM test WHERE 1 = 2'); + $this->assertCount(0, $rows); + $this->assertSame([], $rows); + } + public function testFetchAllWithNamedParams() { $rows = $this->pdo_wrapper->fetchAll('SELECT * FROM test WHERE name = :name', [ 'name' => 'two']); From 42a3d84d8acbb820443fea4a971f79c93ebba84e Mon Sep 17 00:00:00 2001 From: Austin Collier Date: Sun, 17 Mar 2024 08:22:06 -0600 Subject: [PATCH 2/2] Changed Collection to allow for reset --- flight/Engine.php | 2 ++ flight/database/PdoWrapper.php | 5 ++--- flight/util/Collection.php | 28 ++++++++++++++++++++++++++++ tests/CollectionTest.php | 23 +++++++++++++++++++++++ 4 files changed, 55 insertions(+), 3 deletions(-) diff --git a/flight/Engine.php b/flight/Engine.php index b0ae09d..fdeabb2 100644 --- a/flight/Engine.php +++ b/flight/Engine.php @@ -63,6 +63,8 @@ use flight\net\Route; * # HTTP caching * @method void etag(string $id, ('strong'|'weak') $type = 'strong') Handles ETag HTTP caching. * @method void lastModified(int $time) Handles last modified HTTP caching. + * + * phpcs:disable PSR2.Methods.MethodDeclaration.Underscore */ class Engine { diff --git a/flight/database/PdoWrapper.php b/flight/database/PdoWrapper.php index 4821cb9..bdbabb8 100644 --- a/flight/database/PdoWrapper.php +++ b/flight/database/PdoWrapper.php @@ -48,9 +48,8 @@ class PdoWrapper extends PDO */ public function fetchField(string $sql, array $params = []) { - $collection_data = $this->fetchRow($sql, $params); - $array_data = $collection_data->getData(); - return reset($array_data); + $result = $this->fetchRow($sql, $params); + return reset($result); } /** diff --git a/flight/util/Collection.php b/flight/util/Collection.php index 6ffe0b5..29147d1 100644 --- a/flight/util/Collection.php +++ b/flight/util/Collection.php @@ -20,6 +20,18 @@ use JsonSerializable; */ class Collection implements ArrayAccess, Iterator, Countable, JsonSerializable { + /** + * This is to allow for reset() to work properly. + * + * WARNING! This MUST be the first variable in this class!!! + * + * PHPStan is ignoring this because we don't need actually to read the property + * + * @var mixed + * @phpstan-ignore-next-line + */ + private $first_property = null; + /** * Collection data. * @@ -35,6 +47,7 @@ class Collection implements ArrayAccess, Iterator, Countable, JsonSerializable public function __construct(array $data = []) { $this->data = $data; + $this->handleReset(); } /** @@ -55,6 +68,7 @@ class Collection implements ArrayAccess, Iterator, Countable, JsonSerializable public function __set(string $key, $value): void { $this->data[$key] = $value; + $this->handleReset(); } /** @@ -71,6 +85,7 @@ class Collection implements ArrayAccess, Iterator, Countable, JsonSerializable public function __unset(string $key): void { unset($this->data[$key]); + $this->handleReset(); } /** @@ -100,6 +115,7 @@ class Collection implements ArrayAccess, Iterator, Countable, JsonSerializable } else { $this->data[$offset] = $value; } + $this->handleReset(); } /** @@ -120,6 +136,17 @@ class Collection implements ArrayAccess, Iterator, Countable, JsonSerializable public function offsetUnset($offset): void { unset($this->data[$offset]); + $this->handleReset(); + } + + /** + * This is to allow for reset() of a Collection to work properly. + * + * @return void + */ + public function handleReset() + { + $this->first_property = reset($this->data); } /** @@ -207,6 +234,7 @@ class Collection implements ArrayAccess, Iterator, Countable, JsonSerializable public function setData(array $data): void { $this->data = $data; + $this->handleReset(); } #[\ReturnTypeWillChange] diff --git a/tests/CollectionTest.php b/tests/CollectionTest.php index b6eac46..4f742fe 100644 --- a/tests/CollectionTest.php +++ b/tests/CollectionTest.php @@ -99,4 +99,27 @@ class CollectionTest extends TestCase $this->collection->clear(); $this->assertEquals(0, $this->collection->count()); } + + public function testResetByProperty() + { + $this->collection->a = 11; + $this->collection->b = 22; + $result = reset($this->collection); + $this->assertEquals(11, $result); + } + + public function testResetBySetData() + { + $this->collection->setData(['a' => 11, 'b' => 22]); + $result = reset($this->collection); + $this->assertEquals(11, $result); + } + + public function testResetByArraySet() + { + $this->collection['a'] = 11; + $this->collection['b'] = 22; + $result = reset($this->collection); + $this->assertEquals(11, $result); + } }