Skip to content

Commit

Permalink
Merge branch 'master' into support_composite_types
Browse files Browse the repository at this point in the history
  • Loading branch information
Tigrov committed Nov 14, 2023
2 parents 12cd867 + 9782f39 commit f39f761
Show file tree
Hide file tree
Showing 9 changed files with 68 additions and 117 deletions.
23 changes: 23 additions & 0 deletions .github/workflows/bc.yml
Original file line number Diff line number Diff line change
@@ -0,0 +1,23 @@
on:
pull_request:
paths:
- 'src/**'
- '.github/workflows/bc.yml'
- 'composer.json'
push:
branches: ['master']
paths:
- 'src/**'
- '.github/workflows/bc.yml'
- 'composer.json'

name: backwards compatibility

jobs:
roave_bc_check:
uses: yiisoft/actions/.github/workflows/bc.yml@master
with:
os: >-
['ubuntu-latest']
php: >-
['8.1']
15 changes: 0 additions & 15 deletions .github/workflows/bc.yml_

This file was deleted.

11 changes: 9 additions & 2 deletions CHANGELOG.md
Original file line number Diff line number Diff line change
@@ -1,13 +1,20 @@
# PostgreSQL driver for Yii Database Change Log

## 1.1.1 under development
## 1.2.1 under development

- Enh #301: Refactor `JsonExpressionBuilder` (@Tigrov)
- no changes in this release.

## 1.2.0 November 12, 2023

- Chg #319: Remove use of abstract type `SchemaInterface::TYPE_JSONB` (@Tigrov)
- Enh #300: Refactor `ArrayExpressionBuilder` (@Tigrov)
- Enh #301: Refactor `JsonExpressionBuilder` (@Tigrov)
- Enh #302: Refactor `ColumnSchema` (@Tigrov)
- Enh #303: Support Composite types (@Tigrov)
- Enh #321: Move methods from `Command` to `AbstractPdoCommand` class (@Tigrov)
- Bug #302: Fix incorrect convert string value for BIT type (@Tigrov)
- Bug #309: Fix retrieving sequence name from default value (@Tigrov)
- Bug #313: Refactor `DMLQueryBuilder`, related with yiisoft/db#746 (@Tigrov)

## 1.1.0 July 24, 2023

Expand Down
5 changes: 2 additions & 3 deletions composer.json
Original file line number Diff line number Diff line change
Expand Up @@ -25,7 +25,7 @@
"ext-json": "*",
"ext-pdo": "*",
"ext-pdo_pgsql": "*",
"yiisoft/db": "^1.0",
"yiisoft/db": "^1.2",
"yiisoft/json": "^1.0"
},
"require-dev": {
Expand All @@ -36,8 +36,7 @@
"spatie/phpunit-watcher": "^1.23",
"vimeo/psalm": "^4.3|^5.6",
"yiisoft/aliases": "^2.0",
"yiisoft/cache": "^2.0",
"yiisoft/cache-file": "^2.0",
"yiisoft/cache-file": "^3.1",
"yiisoft/var-dumper": "^1.5"
},
"autoload": {
Expand Down
3 changes: 1 addition & 2 deletions src/Builder/ArrayExpressionBuilder.php
Original file line number Diff line number Diff line change
Expand Up @@ -14,7 +14,6 @@
use Yiisoft\Db\Expression\JsonExpression;
use Yiisoft\Db\Query\QueryInterface;
use Yiisoft\Db\QueryBuilder\QueryBuilderInterface;
use Yiisoft\Db\Schema\SchemaInterface;

use function implode;
use function in_array;
Expand Down Expand Up @@ -157,7 +156,7 @@ private function typecastValue(
return $value;
}

if (in_array($expression->getType(), [SchemaInterface::TYPE_JSON, SchemaInterface::TYPE_JSONB], true)) {
if (in_array($expression->getType(), ['json', 'jsonb'], true)) {
return new JsonExpression($value);
}

Expand Down
47 changes: 0 additions & 47 deletions src/Command.php
Original file line number Diff line number Diff line change
Expand Up @@ -4,11 +4,7 @@

namespace Yiisoft\Db\Pgsql;

use Exception;
use Throwable;
use Yiisoft\Db\Driver\Pdo\AbstractPdoCommand;
use Yiisoft\Db\Exception\ConvertException;
use Yiisoft\Db\QueryBuilder\QueryBuilderInterface;

/**
* Implements a database command that can be executed with a PDO (PHP Data Object) database connection for PostgreSQL
Expand All @@ -24,47 +20,4 @@ public function showDatabases(): array

return $this->setSql($sql)->queryColumn();
}

protected function getQueryBuilder(): QueryBuilderInterface
{
return $this->db->getQueryBuilder();
}

/**
* @psalm-suppress UnusedClosureParam
*
* @throws \Yiisoft\Db\Exception\Exception
* @throws Throwable
*/
protected function internalExecute(string|null $rawSql): void
{
$attempt = 0;

while (true) {
try {
if (
++$attempt === 1
&& $this->isolationLevel !== null
&& $this->db->getTransaction() === null
) {
$this->db->transaction(
function () use ($rawSql): void {
$this->internalExecute($rawSql);
},
$this->isolationLevel
);
} else {
$this->pdoStatement?->execute();
}
break;
} catch (Exception $e) {
$rawSql = $rawSql ?: $this->getRawSql();
$e = (new ConvertException($e, $rawSql))->run();

if ($this->retryHandler === null || !($this->retryHandler)($e, $attempt)) {
throw $e;
}
}
}
}
}
73 changes: 27 additions & 46 deletions src/DMLQueryBuilder.php
Original file line number Diff line number Diff line change
Expand Up @@ -9,8 +9,8 @@
use Yiisoft\Db\Query\QueryInterface;
use Yiisoft\Db\QueryBuilder\AbstractDMLQueryBuilder;

use function array_map;
use function implode;
use function reset;

/**
* Implements a DML (Data Manipulation Language) SQL statements for PostgreSQL Server.
Expand All @@ -20,20 +20,15 @@ final class DMLQueryBuilder extends AbstractDMLQueryBuilder
public function insertWithReturningPks(string $table, QueryInterface|array $columns, array &$params = []): string
{
$sql = $this->insert($table, $columns, $params);

$tableSchema = $this->schema->getTableSchema($table);

$returnColumns = [];
if ($tableSchema !== null) {
$returnColumns = $tableSchema->getPrimaryKey();
}
$returnColumns = $this->schema->getTableSchema($table)?->getPrimaryKey();

if (!empty($returnColumns)) {
$returning = [];
foreach ($returnColumns as $name) {
$returning[] = $this->quoter->quoteColumnName($name);
}
$sql .= ' RETURNING ' . implode(', ', $returning);
$returnColumns = array_map(
[$this->quoter, 'quoteColumnName'],
$returnColumns,
);

$sql .= ' RETURNING ' . implode(', ', $returnColumns);
}

return $sql;
Expand All @@ -43,73 +38,59 @@ public function resetSequence(string $table, int|string $value = null): string
{
$tableSchema = $this->schema->getTableSchema($table);

if ($tableSchema !== null && ($sequence = $tableSchema->getSequenceName()) !== null) {
/**
* @link https://www.postgresql.org/docs/8.1/static/functions-sequence.html
*/
$sequence = $this->quoter->quoteTableName($sequence);
$table = $this->quoter->quoteTableName($table);
if ($tableSchema === null) {
throw new InvalidArgumentException("Table not found: '$table'.");
}

if ($value === null) {
$pk = $tableSchema->getPrimaryKey();
$key = $this->quoter->quoteColumnName(reset($pk));
$value = "(SELECT COALESCE(MAX($key),0) FROM $table)+1";
}
$sequence = $tableSchema->getSequenceName();

return "SELECT SETVAL('$sequence',$value,false)";
if ($sequence === null) {
throw new InvalidArgumentException("There is not sequence associated with table '$table'.");
}

if ($tableSchema === null) {
throw new InvalidArgumentException("Table not found: '$table'.");
/** @link https://www.postgresql.org/docs/8.1/static/functions-sequence.html */
$sequence = $this->quoter->quoteTableName($sequence);

if ($value === null) {
$table = $this->quoter->quoteTableName($table);
$key = $tableSchema->getPrimaryKey()[0];
$key = $this->quoter->quoteColumnName($key);
$value = "(SELECT COALESCE(MAX($key),0) FROM $table)+1";
}

throw new InvalidArgumentException("There is not sequence associated with table '$table'.");
return "SELECT SETVAL('$sequence',$value,false)";
}

public function upsert(
string $table,
QueryInterface|array $insertColumns,
$updateColumns,
bool|array $updateColumns,
array &$params = []
): string {
$insertSql = $this->insert($table, $insertColumns, $params);

/** @psalm-var array $uniqueNames */
[$uniqueNames, , $updateNames] = $this->prepareUpsertColumns(
$table,
$insertColumns,
$updateColumns,
);
[$uniqueNames, , $updateNames] = $this->prepareUpsertColumns($table, $insertColumns, $updateColumns);

if (empty($uniqueNames)) {
return $insertSql;
}

if ($updateNames === []) {
if ($updateColumns === false || $updateNames === []) {
/** there are no columns to update */
$updateColumns = false;
}

if ($updateColumns === false) {
return "$insertSql ON CONFLICT DO NOTHING";
}

if ($updateColumns === true) {
$updateColumns = [];

/** @psalm-var string $name */
/** @psalm-var string[] $updateNames */
foreach ($updateNames as $name) {
$updateColumns[$name] = new Expression(
'EXCLUDED.' . $this->quoter->quoteColumnName($name)
);
}
}

/**
* @psalm-var array $updateColumns
* @psalm-var string[] $uniqueNames
* @psalm-var string[] $updates
*/
[$updates, $params] = $this->prepareUpdateSets($table, $updateColumns, $params);

return $insertSql
Expand Down
5 changes: 5 additions & 0 deletions tests/Provider/QueryBuilderProvider.php
Original file line number Diff line number Diff line change
Expand Up @@ -429,6 +429,11 @@ public static function upsert(): array
'VALUES (:qp0, :qp1, :qp2, :qp3) ON CONFLICT ("email") ' .
'DO UPDATE SET "address"=EXCLUDED."address", "status"=EXCLUDED."status", "profile_id"=EXCLUDED."profile_id"',
],
'regular values with unique at not the first position' => [
3 => 'INSERT INTO "T_upsert" ("address", "email", "status", "profile_id") ' .
'VALUES (:qp0, :qp1, :qp2, :qp3) ON CONFLICT ("email") ' .
'DO UPDATE SET "address"=EXCLUDED."address", "status"=EXCLUDED."status", "profile_id"=EXCLUDED."profile_id"',
],
'regular values with update part' => [
2 => ['address' => 'foo {{city}}', 'status' => 2, 'orders' => new Expression('"T_upsert"."orders" + 1')],
3 => 'INSERT INTO "T_upsert" ("email", "address", "status", "profile_id") ' .
Expand Down
3 changes: 1 addition & 2 deletions tests/QueryBuilderTest.php
Original file line number Diff line number Diff line change
Expand Up @@ -4,7 +4,6 @@

namespace Yiisoft\Db\Pgsql\Tests;

use Generator;
use Throwable;
use Yiisoft\Db\Driver\Pdo\PdoConnectionInterface;
use Yiisoft\Db\Exception\Exception;
Expand Down Expand Up @@ -253,7 +252,7 @@ public function testAddUnique(string $name, string $table, array|string $columns
/**
* @dataProvider \Yiisoft\Db\Pgsql\Tests\Provider\QueryBuilderProvider::batchInsert
*/
public function testBatchInsert(string $table, array $columns, iterable|Generator $rows, string $expected): void
public function testBatchInsert(string $table, array $columns, iterable $rows, string $expected): void
{
parent::testBatchInsert($table, $columns, $rows, $expected);
}
Expand Down

0 comments on commit f39f761

Please sign in to comment.