From 4f1dbb95c9551b4359cdb2b188d0094bd70aa138 Mon Sep 17 00:00:00 2001 From: Sergei Tigrov Date: Sat, 14 Oct 2023 14:53:59 +0700 Subject: [PATCH] Fix CTE query expressions (#761) * Fix cte * Fix tests * Add `ExpressionInterface` as alias type and update tests * Update psalm * Update tests * Add line to CHANGELOG.md --- CHANGELOG.md | 1 + src/Query/Query.php | 2 +- src/Query/QueryPartsInterface.php | 5 ++- src/QueryBuilder/AbstractDQLQueryBuilder.php | 34 +++++++++++++- tests/AbstractQueryBuilderTest.php | 45 ++++++++++++++----- tests/Common/CommonQueryTest.php | 47 ++++++++++++++++++++ tests/Provider/QueryBuilderProvider.php | 11 +++++ 7 files changed, 130 insertions(+), 15 deletions(-) diff --git a/CHANGELOG.md b/CHANGELOG.md index 8cb849a49..63bf93448 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -8,6 +8,7 @@ - Bug #756: Fix `Quoter::quoteSql()` for SQL containing table with prefix (@Tigrov) - Bug #756: Fix `Quoter::getTableNameParts()` for cases when different quotes for tables and columns (@Tigrov) - Bug #756: Fix `Quoter::quoteTableName()` for sub-query with alias (@Tigrov) +- Bug #761: Quote aliases of CTE in `WITH` queries (@Tigrov) - Chg #765: Deprecate `SchemaInterface::TYPE_JSONB` (@Tigrov) ## 1.1.1 August 16, 2023 diff --git a/src/Query/Query.php b/src/Query/Query.php index 98e69b962..4ba681e7b 100644 --- a/src/Query/Query.php +++ b/src/Query/Query.php @@ -662,7 +662,7 @@ public function where(array|string|ExpressionInterface|null $condition, array $p return $this; } - public function withQuery(QueryInterface|string $query, string $alias, bool $recursive = false): static + public function withQuery(QueryInterface|string $query, ExpressionInterface|string $alias, bool $recursive = false): static { $this->withQueries[] = ['query' => $query, 'alias' => $alias, 'recursive' => $recursive]; return $this; diff --git a/src/Query/QueryPartsInterface.php b/src/Query/QueryPartsInterface.php index f6b9ca974..8c3ddfd84 100644 --- a/src/Query/QueryPartsInterface.php +++ b/src/Query/QueryPartsInterface.php @@ -645,10 +645,11 @@ public function where(array|string|ExpressionInterface|null $condition, array $p * Prepends an SQL statement using `WITH` syntax. * * @param QueryInterface|string $query The SQL statement to append using `UNION`. - * @param string $alias The query alias in `WITH` construction. + * @param ExpressionInterface|string $alias The query alias in `WITH` construction. + * To specify the alias in plain SQL, you may pass an instance of {@see ExpressionInterface}. * @param bool $recursive Its `true` if using `WITH RECURSIVE` and `false` if using `WITH`. */ - public function withQuery(QueryInterface|string $query, string $alias, bool $recursive = false): static; + public function withQuery(QueryInterface|string $query, ExpressionInterface|string $alias, bool $recursive = false): static; /** * Specifies the `WITH` query clause for the query. diff --git a/src/QueryBuilder/AbstractDQLQueryBuilder.php b/src/QueryBuilder/AbstractDQLQueryBuilder.php index 77418e5af..9f857328f 100644 --- a/src/QueryBuilder/AbstractDQLQueryBuilder.php +++ b/src/QueryBuilder/AbstractDQLQueryBuilder.php @@ -411,7 +411,7 @@ public function buildWithQueries(array $withs, array &$params): string $recursive = false; $result = []; - /** @psalm-var array $withs */ + /** @psalm-var array{query:string|Query, alias:ExpressionInterface|string, recursive:bool}[] $withs */ foreach ($withs as $with) { if ($with['recursive']) { $recursive = true; @@ -421,7 +421,9 @@ public function buildWithQueries(array $withs, array &$params): string [$with['query'], $params] = $this->build($with['query'], $params); } - $result[] = $with['alias'] . ' AS (' . $with['query'] . ')'; + $quotedAlias = $this->quoteCteAlias($with['alias']); + + $result[] = $quotedAlias . ' AS (' . $with['query'] . ')'; } return 'WITH ' . ($recursive ? 'RECURSIVE ' : '') . implode(', ', $result); @@ -610,4 +612,32 @@ private function quoteTableNames(array $tables, array &$params): array return $tables; } + + /** + * Quotes an alias of Common Table Expressions (CTE) + * + * @param ExpressionInterface|string $name The alias name with or without column names to quote. + * + * @return string The quoted alias. + */ + private function quoteCteAlias(ExpressionInterface|string $name): string + { + if ($name instanceof ExpressionInterface) { + return $this->buildExpression($name); + } + + if (!str_contains($name, '(')) { + return $this->quoter->quoteTableName($name); + } + + if (!str_ends_with($name, ')')) { + return $name; + } + + /** @psalm-suppress PossiblyUndefinedArrayOffset */ + [$name, $columns] = explode('(', substr($name, 0, -1), 2); + $name = trim($name); + + return $this->quoter->quoteTableName($name) . '(' . $this->buildColumns($columns) . ')'; + } } diff --git a/tests/AbstractQueryBuilderTest.php b/tests/AbstractQueryBuilderTest.php index 6a50636e9..31de64c71 100644 --- a/tests/AbstractQueryBuilderTest.php +++ b/tests/AbstractQueryBuilderTest.php @@ -668,7 +668,7 @@ public function testBuildWithQueries(): void $this->assertSame( DbHelper::replaceQuotes( <<getDriverName(), ), @@ -1131,7 +1131,7 @@ public function testBuildWithQuery(): void $this->assertSame( DbHelper::replaceQuotes( <<getDriverName(), ), @@ -1157,15 +1157,40 @@ public function testBuildWithQueryRecursive(): void [$sql, $params] = $qb->build($query); - $this->assertSame( - DbHelper::replaceQuotes( - <<getDriverName(), - ), - $sql, + $expected = DbHelper::replaceQuotes( + <<getDriverName(), ); + + if (in_array($db->getDriverName(), ['oci', 'sqlsrv'], true)) { + $expected = str_replace('WITH RECURSIVE ', 'WITH ', $expected); + } + + $this->assertSame($expected, $sql); + $this->assertSame([], $params); + } + + /** @dataProvider \Yiisoft\Db\Tests\Provider\QueryBuilderProvider::cteAliases */ + public function testBuildWithQueryAlias($alias, $expected) + { + $db = $this->getConnection(); + $qb = $db->getQueryBuilder(); + + $withQuery = (new Query($db))->from('t'); + $query = (new Query($db))->withQuery($withQuery, $alias)->from('t'); + + [$sql, $params] = $qb->build($query); + + $expectedSql = DbHelper::replaceQuotes( + <<getDriverName(), + ); + + $this->assertSame($expectedSql, $sql); $this->assertSame([], $params); } diff --git a/tests/Common/CommonQueryTest.php b/tests/Common/CommonQueryTest.php index 7b098207f..a1a6bb1d0 100644 --- a/tests/Common/CommonQueryTest.php +++ b/tests/Common/CommonQueryTest.php @@ -4,6 +4,7 @@ namespace Yiisoft\Db\Tests\Common; +use Yiisoft\Db\Expression\Expression; use Yiisoft\Db\Query\Query; use Yiisoft\Db\Tests\AbstractQueryTest; @@ -34,4 +35,50 @@ public function testColumnIndexByWithClosure() $db->close(); } + + public function testWithQuery() + { + $db = $this->getConnection(true); + + $with = (new Query($db)) + ->distinct() + ->select(['status']) + ->from('customer'); + + $query = (new Query($db)) + ->withQuery($with, 'statuses') + ->from('statuses'); + + $this->assertEquals(2, $query->count()); + + $db->close(); + } + + public function testWithQueryRecursive() + { + $db = $this->getConnection(); + $quoter = $db->getQuoter(); + $isOracle = $db->getDriverName() === 'oci'; + + /** Sum 1 to 10 equals 55 */ + $quotedName = $quoter->quoteColumnName('n'); + $union = (new Query($db)) + ->select(new Expression($quotedName . ' + 1')) + ->from('t') + ->where(['<', 'n', 10]); + + $with = (new Query($db)) + ->select(new Expression('1')) + ->from($isOracle ? new Expression('DUAL') : []) + ->union($union, true); + + $sum = (new Query($db)) + ->withQuery($with, 't(n)', true) + ->from('t') + ->sum($quotedName); + + $this->assertEquals(55, $sum); + + $db->close(); + } } diff --git a/tests/Provider/QueryBuilderProvider.php b/tests/Provider/QueryBuilderProvider.php index f86d147d9..0871a113b 100644 --- a/tests/Provider/QueryBuilderProvider.php +++ b/tests/Provider/QueryBuilderProvider.php @@ -1231,4 +1231,15 @@ public static function upsert(): array ], ]; } + + public static function cteAliases(): array + { + return [ + 'simple' => ['a', '[[a]]'], + 'with one column' => ['a(b)', '[[a]]([[b]])'], + 'with columns' => ['a(b,c,d)', '[[a]]([[b]], [[c]], [[d]])'], + 'with extra space' => ['a(b,c,d) ', 'a(b,c,d) '], + 'expression' => [new Expression('a(b,c,d)'), 'a(b,c,d)'], + ]; + } }