Skip to content

Commit

Permalink
[GH-4687] Remove support for Connection::lastInsertId($name)
Browse files Browse the repository at this point in the history
morozov committed Jun 25, 2021

Verified

This commit was signed with the committer’s verified signature.
morozov Sergei Morozov
1 parent ea319d6 commit b2e0427
Showing 13 changed files with 38 additions and 231 deletions.
4 changes: 4 additions & 0 deletions UPGRADE.md
Original file line number Diff line number Diff line change
@@ -8,6 +8,10 @@ awareness about deprecated code.

# Upgrade to 4.0

## Removed support for `Connection::lastInsertId($name)`

The `Connection::lastInsertId()` method no longer accepts a sequence name.

## Removed defaults for MySQL table charset, collation and engine

The library no longer provides the default values for MySQL table charset, collation and engine.
20 changes: 4 additions & 16 deletions src/Connection.php
Original file line number Diff line number Diff line change
@@ -32,7 +32,6 @@
use Doctrine\DBAL\Schema\AbstractSchemaManager;
use Doctrine\DBAL\SQL\Parser;
use Doctrine\DBAL\Types\Type;
use Doctrine\Deprecations\Deprecation;
use Throwable;
use Traversable;

@@ -1104,31 +1103,20 @@ public function getTransactionNestingLevel(): int
}

/**
* Returns the ID of the last inserted row, or the last value from a sequence object,
* depending on the underlying driver.
* Returns the ID of the last inserted row.
*
* Note: This method may not return a meaningful or consistent result across different drivers,
* because the underlying database may not even support the notion of AUTO_INCREMENT/IDENTITY
* columns or sequences.
*
* @param string|null $name Name of the sequence object from which the ID should be returned.
* columns.
*
* @return string A string representation of the last inserted ID.
*
* @throws Exception
*/
public function lastInsertId(?string $name = null): string
public function lastInsertId(): string
{
if ($name !== null) {
Deprecation::trigger(
'doctrine/dbal',
'https://github.com/doctrine/dbal/issues/4687',
'The usage of Connection::lastInsertId() with a sequence name is deprecated.'
);
}

try {
return $this->getWrappedConnection()->lastInsertId($name);
return $this->getWrappedConnection()->lastInsertId();
} catch (Driver\Exception $e) {
throw $this->convertException($e);
}
4 changes: 2 additions & 2 deletions src/Driver/Connection.php
Original file line number Diff line number Diff line change
@@ -37,11 +37,11 @@ public function quote(string $value): string;
public function exec(string $sql): int;

/**
* Returns the ID of the last inserted row or sequence value.
* Returns the ID of the last inserted row.
*
* @throws Exception
*/
public function lastInsertId(?string $name = null): string;
public function lastInsertId(): string;

/**
* Initiates a transaction.
11 changes: 1 addition & 10 deletions src/Driver/IBMDB2/Connection.php
Original file line number Diff line number Diff line change
@@ -11,7 +11,6 @@
use Doctrine\DBAL\Driver\Result as ResultInterface;
use Doctrine\DBAL\Driver\ServerInfoAwareConnection;
use Doctrine\DBAL\Driver\Statement as DriverStatement;
use Doctrine\Deprecations\Deprecation;
use stdClass;

use function assert;
@@ -103,16 +102,8 @@ public function exec(string $sql): int
return db2_num_rows($stmt);
}

public function lastInsertId(?string $name = null): string
public function lastInsertId(): string
{
if ($name !== null) {
Deprecation::triggerIfCalledFromOutside(
'doctrine/dbal',
'https://github.com/doctrine/dbal/issues/4687',
'The usage of Connection::lastInsertId() with a sequence name is deprecated.'
);
}

return db2_last_insert_id($this->conn);
}

11 changes: 1 addition & 10 deletions src/Driver/Mysqli/Connection.php
Original file line number Diff line number Diff line change
@@ -10,7 +10,6 @@
use Doctrine\DBAL\Driver\Result as ResultInterface;
use Doctrine\DBAL\Driver\ServerInfoAwareConnection;
use Doctrine\DBAL\Driver\Statement as DriverStatement;
use Doctrine\Deprecations\Deprecation;
use mysqli;

use function assert;
@@ -126,16 +125,8 @@ public function exec(string $sql): int
return $this->conn->affected_rows;
}

public function lastInsertId(?string $name = null): string
public function lastInsertId(): string
{
if ($name !== null) {
Deprecation::triggerIfCalledFromOutside(
'doctrine/dbal',
'https://github.com/doctrine/dbal/issues/4687',
'The usage of Connection::lastInsertId() with a sequence name is deprecated.'
);
}

return (string) $this->conn->insert_id;
}

22 changes: 2 additions & 20 deletions src/Driver/OCI8/Connection.php
Original file line number Diff line number Diff line change
@@ -8,11 +8,9 @@
use Doctrine\DBAL\Driver\Exception\IdentityColumnsNotSupported;
use Doctrine\DBAL\Driver\OCI8\Exception\ConnectionFailed;
use Doctrine\DBAL\Driver\OCI8\Exception\Error;
use Doctrine\DBAL\Driver\OCI8\Exception\SequenceDoesNotExist;
use Doctrine\DBAL\Driver\Result as ResultInterface;
use Doctrine\DBAL\Driver\ServerInfoAwareConnection;
use Doctrine\DBAL\Driver\Statement as DriverStatement;
use Doctrine\Deprecations\Deprecation;

use function addcslashes;
use function assert;
@@ -97,25 +95,9 @@ public function exec(string $sql): int
return $this->prepare($sql)->execute()->rowCount();
}

public function lastInsertId(?string $name = null): string
public function lastInsertId(): string
{
if ($name === null) {
throw IdentityColumnsNotSupported::new();
}

Deprecation::triggerIfCalledFromOutside(
'doctrine/dbal',
'https://github.com/doctrine/dbal/issues/4687',
'The usage of Connection::lastInsertId() with a sequence name is deprecated.'
);

$result = $this->query('SELECT ' . $name . '.CURRVAL FROM DUAL')->fetchOne();

if ($result === false) {
throw SequenceDoesNotExist::new();
}

return $result;
throw IdentityColumnsNotSupported::new();
}

public function beginTransaction(): void
15 changes: 2 additions & 13 deletions src/Driver/PDO/Connection.php
Original file line number Diff line number Diff line change
@@ -8,7 +8,6 @@
use Doctrine\DBAL\Driver\Result as ResultInterface;
use Doctrine\DBAL\Driver\ServerInfoAwareConnection;
use Doctrine\DBAL\Driver\Statement as StatementInterface;
use Doctrine\Deprecations\Deprecation;
use PDO;
use PDOException;
use PDOStatement;
@@ -89,20 +88,10 @@ public function quote(string $value): string
return $this->connection->quote($value);
}

public function lastInsertId(?string $name = null): string
public function lastInsertId(): string
{
try {
if ($name === null) {
return $this->connection->lastInsertId();
}

Deprecation::triggerIfCalledFromOutside(
'doctrine/dbal',
'https://github.com/doctrine/dbal/issues/4687',
'The usage of Connection::lastInsertId() with a sequence name is deprecated.'
);

return $this->connection->lastInsertId($name);
return $this->connection->lastInsertId();
} catch (PDOException $exception) {
throw Exception::new($exception);
}
17 changes: 2 additions & 15 deletions src/Driver/PDO/SQLSrv/Connection.php
Original file line number Diff line number Diff line change
@@ -8,7 +8,6 @@
use Doctrine\DBAL\Driver\Result;
use Doctrine\DBAL\Driver\ServerInfoAwareConnection;
use Doctrine\DBAL\Driver\Statement as StatementInterface;
use Doctrine\Deprecations\Deprecation;
use PDO;

final class Connection implements ServerInfoAwareConnection
@@ -43,21 +42,9 @@ public function exec(string $sql): int
return $this->connection->exec($sql);
}

public function lastInsertId(?string $name = null): string
public function lastInsertId(): string
{
if ($name === null) {
return $this->connection->lastInsertId($name);
}

Deprecation::triggerIfCalledFromOutside(
'doctrine/dbal',
'https://github.com/doctrine/dbal/issues/4687',
'The usage of Connection::lastInsertId() with a sequence name is deprecated.'
);

return $this->prepare('SELECT CONVERT(VARCHAR(MAX), current_value) FROM sys.sequences WHERE name = ?')
->execute([$name])
->fetchOne();
return $this->connection->lastInsertId();
}

public function beginTransaction(): void
18 changes: 2 additions & 16 deletions src/Driver/SQLSrv/Connection.php
Original file line number Diff line number Diff line change
@@ -9,7 +9,6 @@
use Doctrine\DBAL\Driver\ServerInfoAwareConnection;
use Doctrine\DBAL\Driver\SQLSrv\Exception\Error;
use Doctrine\DBAL\Driver\Statement as DriverStatement;
use Doctrine\Deprecations\Deprecation;

use function sqlsrv_begin_transaction;
use function sqlsrv_commit;
@@ -87,22 +86,9 @@ public function exec(string $sql): int
return $rowsAffected;
}

public function lastInsertId(?string $name = null): string
public function lastInsertId(): string
{
if ($name !== null) {
Deprecation::triggerIfCalledFromOutside(
'doctrine/dbal',
'https://github.com/doctrine/dbal/issues/4687',
'The usage of Connection::lastInsertId() with a sequence name is deprecated.'
);

$result = $this->prepare('SELECT CONVERT(VARCHAR(MAX), current_value) FROM sys.sequences WHERE name = ?')
->execute([$name]);
} else {
$result = $this->query('SELECT @@IDENTITY');
}

return $result->fetchOne();
return $this->query('SELECT @@IDENTITY')->fetchOne();
}

public function beginTransaction(): void
13 changes: 2 additions & 11 deletions src/Portability/Connection.php
Original file line number Diff line number Diff line change
@@ -7,7 +7,6 @@
use Doctrine\DBAL\Driver\Connection as ConnectionInterface;
use Doctrine\DBAL\Driver\Result as DriverResult;
use Doctrine\DBAL\Driver\Statement as DriverStatement;
use Doctrine\Deprecations\Deprecation;

/**
* Portability wrapper for a Connection.
@@ -58,17 +57,9 @@ public function exec(string $sql): int
return $this->connection->exec($sql);
}

public function lastInsertId(?string $name = null): string
public function lastInsertId(): string
{
if ($name !== null) {
Deprecation::triggerIfCalledFromOutside(
'doctrine/dbal',
'https://github.com/doctrine/dbal/issues/4687',
'The usage of Connection::lastInsertId() with a sequence name is deprecated.'
);
}

return $this->connection->lastInsertId($name);
return $this->connection->lastInsertId();
}

public function beginTransaction(): void
45 changes: 0 additions & 45 deletions tests/Functional/Driver/OCI8/ConnectionTest.php

This file was deleted.

14 changes: 7 additions & 7 deletions tests/Functional/Ticket/DBAL630Test.php
Original file line number Diff line number Diff line change
@@ -25,8 +25,8 @@ protected function setUp(): void
}

try {
$this->connection->executeStatement('CREATE TABLE dbal630 (id SERIAL, bool_col BOOLEAN NOT NULL);');
$this->connection->executeStatement('CREATE TABLE dbal630_allow_nulls (id SERIAL, bool_col BOOLEAN);');
$this->connection->executeStatement('CREATE TABLE dbal630 (id SERIAL, bool_col BOOLEAN NOT NULL)');
$this->connection->executeStatement('CREATE TABLE dbal630_allow_nulls (id SERIAL, bool_col BOOLEAN)');
} catch (Exception $e) {
}

@@ -47,7 +47,7 @@ protected function tearDown(): void
public function testBooleanConversionSqlLiteral(): void
{
$this->connection->executeStatement('INSERT INTO dbal630 (bool_col) VALUES(false)');
$id = $this->connection->lastInsertId('dbal630_id_seq');
$id = $this->connection->lastInsertId();
self::assertNotEmpty($id);

$row = $this->connection->fetchAssociative('SELECT bool_col FROM dbal630 WHERE id = ?', [$id]);
@@ -63,7 +63,7 @@ public function testBooleanConversionBoolParamRealPrepares(): void
['false'],
[ParameterType::BOOLEAN]
);
$id = $this->connection->lastInsertId('dbal630_id_seq');
$id = $this->connection->lastInsertId();
self::assertNotEmpty($id);

$row = $this->connection->fetchAssociative('SELECT bool_col FROM dbal630 WHERE id = ?', [$id]);
@@ -84,7 +84,7 @@ public function testBooleanConversionBoolParamEmulatedPrepares(): void
$stmt->bindValue(1, $platform->convertBooleansToDatabaseValue('false'), ParameterType::BOOLEAN);
$stmt->execute();

$id = $this->connection->lastInsertId('dbal630_id_seq');
$id = $this->connection->lastInsertId();

self::assertNotEmpty($id);

@@ -111,7 +111,7 @@ public function testBooleanConversionNullParamEmulatedPrepares(
$stmt->bindValue(1, $platform->convertBooleansToDatabaseValue($statementValue));
$stmt->execute();

$id = $this->connection->lastInsertId('dbal630_allow_nulls_id_seq');
$id = $this->connection->lastInsertId();

self::assertNotEmpty($id);

@@ -142,7 +142,7 @@ public function testBooleanConversionNullParamEmulatedPreparesWithBooleanTypeInB
);
$stmt->execute();

$id = $this->connection->lastInsertId('dbal630_allow_nulls_id_seq');
$id = $this->connection->lastInsertId();

self::assertNotEmpty($id);

75 changes: 9 additions & 66 deletions tests/Functional/WriteTest.php
Original file line number Diff line number Diff line change
@@ -7,14 +7,10 @@
use DateTime;
use Doctrine\DBAL\Exception\DriverException;
use Doctrine\DBAL\ParameterType;
use Doctrine\DBAL\Schema\Sequence;
use Doctrine\DBAL\Schema\Table;
use Doctrine\DBAL\Tests\FunctionalTestCase;
use Throwable;

use function array_filter;
use function strtolower;

class WriteTest extends FunctionalTestCase
{
protected function setUp(): void
@@ -151,51 +147,21 @@ public function testLastInsertId(): void
}

self::assertEquals(1, $this->connection->insert('write_table', ['test_int' => 2, 'test_string' => 'bar']));
$num = $this->lastInsertId();
$num = $this->connection->lastInsertId();

self::assertGreaterThan(0, $num, 'LastInsertId() should be non-negative number.');
}

public function testLastInsertIdSequence(): void
public function testLastInsertIdNotSupported(): void
{
if (! $this->connection->getDatabasePlatform()->supportsSequences()) {
self::markTestSkipped('Test only works on platforms with sequences.');
}

$sequence = new Sequence('write_table_id_seq');
try {
$this->connection->createSchemaManager()->createSequence($sequence);
} catch (Throwable $e) {
}

$sequences = $this->connection->createSchemaManager()->listSequences();
self::assertCount(1, array_filter($sequences, static function ($sequence): bool {
return strtolower($sequence->getName()) === 'write_table_id_seq';
}));

$nextSequenceVal = $this->connection->fetchOne(
$this->connection->getDatabasePlatform()->getSequenceNextValSQL('write_table_id_seq')
);

$lastInsertId = $this->lastInsertId('write_table_id_seq');

self::assertGreaterThan(0, $lastInsertId);
self::assertEquals($nextSequenceVal, $lastInsertId);
}

public function testLastInsertIdNoSequenceGiven(): void
{
if (
! $this->connection->getDatabasePlatform()->supportsSequences()
|| $this->connection->getDatabasePlatform()->supportsIdentityColumns()
) {
if ($this->connection->getDatabasePlatform()->supportsIdentityColumns()) {
self::markTestSkipped(
"Test only works consistently on platforms that support sequences and don't support identity columns."
"Test only works consistently on platforms that don't support identity columns."
);
}

$this->expectException(DriverException::class);
$this->lastInsertId();
$this->connection->lastInsertId();
}

public function testInsertWithKeyValueTypes(): void
@@ -269,9 +235,9 @@ public function testEmptyIdentityInsert(): void
{
$platform = $this->connection->getDatabasePlatform();

if (! ($platform->supportsIdentityColumns() || $platform->usesSequenceEmulatedIdentityColumns())) {
if (! $platform->supportsIdentityColumns()) {
self::markTestSkipped(
'Test only works on platforms with identity columns or sequence emulated identity columns.'
'Test only works on platforms with identity columns.'
);
}

@@ -288,19 +254,15 @@ public function testEmptyIdentityInsert(): void
$this->connection->executeStatement($sql);
}

$seqName = $platform->usesSequenceEmulatedIdentityColumns()
? $platform->getIdentitySequenceName('test_empty_identity', 'id')
: null;

$sql = $platform->getEmptyIdentityInsertSQL('test_empty_identity', 'id');

$this->connection->executeStatement($sql);

$firstId = $this->lastInsertId($seqName);
$firstId = $this->connection->lastInsertId();

$this->connection->executeStatement($sql);

$secondId = $this->lastInsertId($seqName);
$secondId = $this->connection->lastInsertId();

self::assertGreaterThan($firstId, $secondId);
}
@@ -345,23 +307,4 @@ public function testDeleteWhereIsNull(): void

self::assertCount(0, $data);
}

/**
* Returns the ID of the last inserted row or skips the test if the currently used driver
* doesn't support this feature
*
* @throws DriverException
*/
private function lastInsertId(?string $name = null): string
{
try {
return $this->connection->lastInsertId($name);
} catch (DriverException $e) {
if ($e->getSQLState() === 'IM001') {
self::markTestSkipped($e->getMessage());
}

throw $e;
}
}
}

0 comments on commit b2e0427

Please sign in to comment.