From 4393eb8c94b8e8de8994bf9988126ccaa0155c86 Mon Sep 17 00:00:00 2001 From: Stefano Arlandini Date: Fri, 23 Apr 2021 00:58:11 +0200 Subject: [PATCH 1/3] Make TransactionHelper always respect the presence of the inTransaction() method on the driver connection class --- .../Migrations/Tools/TransactionHelper.php | 9 +- .../Tests/Tools/TransactionHelperTest.php | 93 +++++++++++++++++++ 2 files changed, 100 insertions(+), 2 deletions(-) create mode 100644 tests/Doctrine/Migrations/Tests/Tools/TransactionHelperTest.php diff --git a/lib/Doctrine/Migrations/Tools/TransactionHelper.php b/lib/Doctrine/Migrations/Tools/TransactionHelper.php index 866c591ec4..acfceeabea 100644 --- a/lib/Doctrine/Migrations/Tools/TransactionHelper.php +++ b/lib/Doctrine/Migrations/Tools/TransactionHelper.php @@ -5,7 +5,8 @@ namespace Doctrine\Migrations\Tools; use Doctrine\DBAL\Connection; -use PDO; + +use function method_exists; /** * @internal @@ -36,6 +37,10 @@ private static function inTransaction(Connection $connection): bool /* Attempt to commit or rollback while no transaction is running results in an exception since PHP 8 + pdo_mysql combination */ - return ! $wrappedConnection instanceof PDO || $wrappedConnection->inTransaction(); + if (method_exists($wrappedConnection, 'inTransaction')) { + return $wrappedConnection->inTransaction(); + } + + return true; } } diff --git a/tests/Doctrine/Migrations/Tests/Tools/TransactionHelperTest.php b/tests/Doctrine/Migrations/Tests/Tools/TransactionHelperTest.php new file mode 100644 index 0000000000..9279cc9790 --- /dev/null +++ b/tests/Doctrine/Migrations/Tests/Tools/TransactionHelperTest.php @@ -0,0 +1,93 @@ +createMock($driverConnectionFqcn); + $wrappedConnection->expects($this->once()) + ->method('inTransaction') + ->willReturn($commitExpected); + + $connection = $this->createMock(Connection::class); + $connection->expects($this->once()) + ->method('getWrappedConnection') + ->willReturn($wrappedConnection); + + $connection->expects($commitExpected ? $this->once() : $this->never()) + ->method('commit'); + + TransactionHelper::commitIfInTransaction($connection); + } + + public function testCommitIfInTransactionWithConnectionNotImplementingInTransactionMethod(): void + { + $connection = $this->createMock(Connection::class); + $connection->expects($this->once()) + ->method('getWrappedConnection') + ->willReturn($this->createMock(DriverConnection::class)); + + $connection->expects($this->once()) + ->method('commit'); + + TransactionHelper::commitIfInTransaction($connection); + } + + /** + * @dataProvider getDriverConnectionClassesImplementingInTransactionMethod + */ + public function testRollbackIfInTransactionWithConnectionImplementingInTransactionMethod(string $driverConnectionFqcn, bool $commitExpected): void + { + $wrappedConnection = $this->createMock($driverConnectionFqcn); + $wrappedConnection->expects($this->once()) + ->method('inTransaction') + ->willReturn($commitExpected); + + $connection = $this->createMock(Connection::class); + $connection->expects($this->once()) + ->method('getWrappedConnection') + ->willReturn($wrappedConnection); + + $connection->expects($commitExpected ? $this->once() : $this->never()) + ->method('rollback'); + + TransactionHelper::rollbackIfInTransaction($connection); + } + + public function testRollbackIfInTransactionWithConnectionNotImplementingInTransactionMethod(): void + { + $connection = $this->createMock(Connection::class); + $connection->expects($this->once()) + ->method('getWrappedConnection') + ->willReturn($this->createMock(DriverConnection::class)); + + $connection->expects($this->once()) + ->method('rollback'); + + TransactionHelper::rollbackIfInTransaction($connection); + } + + /** + * @return mixed[] + */ + public function getDriverConnectionClassesImplementingInTransactionMethod(): array + { + return [ + [PDOConnection::class, true], + [PDOConnection::class, false], + ]; + } +} From 687d85e5b3464bb51e41445427286ae1204d984d Mon Sep 17 00:00:00 2001 From: Stefano Arlandini Date: Fri, 23 Apr 2021 01:28:47 +0200 Subject: [PATCH 2/3] Fix PHPStan issues --- .../Tests/Tools/TransactionHelperTest.php | 24 +++++++++++-------- 1 file changed, 14 insertions(+), 10 deletions(-) diff --git a/tests/Doctrine/Migrations/Tests/Tools/TransactionHelperTest.php b/tests/Doctrine/Migrations/Tests/Tools/TransactionHelperTest.php index 9279cc9790..c767720cd6 100644 --- a/tests/Doctrine/Migrations/Tests/Tools/TransactionHelperTest.php +++ b/tests/Doctrine/Migrations/Tests/Tools/TransactionHelperTest.php @@ -13,21 +13,23 @@ final class TransactionHelperTest extends TestCase { /** + * @param class-string $driverConnectionFqcn + * * @dataProvider getDriverConnectionClassesImplementingInTransactionMethod */ public function testCommitIfInTransactionWithConnectionImplementingInTransactionMethod(string $driverConnectionFqcn, bool $commitExpected): void { $wrappedConnection = $this->createMock($driverConnectionFqcn); - $wrappedConnection->expects($this->once()) + $wrappedConnection->expects(self::once()) ->method('inTransaction') ->willReturn($commitExpected); $connection = $this->createMock(Connection::class); - $connection->expects($this->once()) + $connection->expects(self::once()) ->method('getWrappedConnection') ->willReturn($wrappedConnection); - $connection->expects($commitExpected ? $this->once() : $this->never()) + $connection->expects($commitExpected ? self::once() : self::never()) ->method('commit'); TransactionHelper::commitIfInTransaction($connection); @@ -36,32 +38,34 @@ public function testCommitIfInTransactionWithConnectionImplementingInTransaction public function testCommitIfInTransactionWithConnectionNotImplementingInTransactionMethod(): void { $connection = $this->createMock(Connection::class); - $connection->expects($this->once()) + $connection->expects(self::once()) ->method('getWrappedConnection') ->willReturn($this->createMock(DriverConnection::class)); - $connection->expects($this->once()) + $connection->expects(self::once()) ->method('commit'); TransactionHelper::commitIfInTransaction($connection); } /** + * @param class-string $driverConnectionFqcn + * * @dataProvider getDriverConnectionClassesImplementingInTransactionMethod */ public function testRollbackIfInTransactionWithConnectionImplementingInTransactionMethod(string $driverConnectionFqcn, bool $commitExpected): void { $wrappedConnection = $this->createMock($driverConnectionFqcn); - $wrappedConnection->expects($this->once()) + $wrappedConnection->expects(self::once()) ->method('inTransaction') ->willReturn($commitExpected); $connection = $this->createMock(Connection::class); - $connection->expects($this->once()) + $connection->expects(self::once()) ->method('getWrappedConnection') ->willReturn($wrappedConnection); - $connection->expects($commitExpected ? $this->once() : $this->never()) + $connection->expects($commitExpected ? self::once() : self::never()) ->method('rollback'); TransactionHelper::rollbackIfInTransaction($connection); @@ -70,11 +74,11 @@ public function testRollbackIfInTransactionWithConnectionImplementingInTransacti public function testRollbackIfInTransactionWithConnectionNotImplementingInTransactionMethod(): void { $connection = $this->createMock(Connection::class); - $connection->expects($this->once()) + $connection->expects(self::once()) ->method('getWrappedConnection') ->willReturn($this->createMock(DriverConnection::class)); - $connection->expects($this->once()) + $connection->expects(self::once()) ->method('rollback'); TransactionHelper::rollbackIfInTransaction($connection); From 22bfda85fbba835fd5381cc609f9a3635a9d52d6 Mon Sep 17 00:00:00 2001 From: Stefano Arlandini Date: Fri, 23 Apr 2021 10:51:39 +0200 Subject: [PATCH 3/3] Fix CR issues --- .../Event/Listeners/AutoCommitListenerTest.php | 6 ++++++ .../Tests/Tools/TransactionHelperTest.php | 2 +- .../Tests/Tracking/TableUpdaterTest.php | 9 +++++++++ .../Migrations/Tests/Version/ExecutorTest.php | 17 +++++++++++++++++ 4 files changed, 33 insertions(+), 1 deletion(-) diff --git a/tests/Doctrine/Migrations/Tests/Event/Listeners/AutoCommitListenerTest.php b/tests/Doctrine/Migrations/Tests/Event/Listeners/AutoCommitListenerTest.php index 8167315aee..7ce41503c0 100644 --- a/tests/Doctrine/Migrations/Tests/Event/Listeners/AutoCommitListenerTest.php +++ b/tests/Doctrine/Migrations/Tests/Event/Listeners/AutoCommitListenerTest.php @@ -5,6 +5,7 @@ namespace Doctrine\Migrations\Tests\Event\Listeners; use Doctrine\DBAL\Connection; +use Doctrine\DBAL\Driver\Connection as DriverConnection; use Doctrine\Migrations\Configuration\Configuration; use Doctrine\Migrations\Event\Listeners\AutoCommitListener; use Doctrine\Migrations\Event\MigrationsEventArgs; @@ -38,9 +39,14 @@ public function testListenerDoesNothingWhenConnecitonAutoCommitIsOn(): void public function testListenerDoesFinalCommitWhenAutoCommitIsOff(): void { + $this->conn->expects(self::once()) + ->method('getWrappedConnection') + ->willReturn($this->createMock(DriverConnection::class)); + $this->conn->expects(self::once()) ->method('isAutoCommit') ->willReturn(false); + $this->conn->expects(self::once()) ->method('commit'); diff --git a/tests/Doctrine/Migrations/Tests/Tools/TransactionHelperTest.php b/tests/Doctrine/Migrations/Tests/Tools/TransactionHelperTest.php index c767720cd6..716ca8e7d2 100644 --- a/tests/Doctrine/Migrations/Tests/Tools/TransactionHelperTest.php +++ b/tests/Doctrine/Migrations/Tests/Tools/TransactionHelperTest.php @@ -85,7 +85,7 @@ public function testRollbackIfInTransactionWithConnectionNotImplementingInTransa } /** - * @return mixed[] + * @return list, bool}> */ public function getDriverConnectionClassesImplementingInTransactionMethod(): array { diff --git a/tests/Doctrine/Migrations/Tests/Tracking/TableUpdaterTest.php b/tests/Doctrine/Migrations/Tests/Tracking/TableUpdaterTest.php index 1035be0e99..71c213bc01 100644 --- a/tests/Doctrine/Migrations/Tests/Tracking/TableUpdaterTest.php +++ b/tests/Doctrine/Migrations/Tests/Tracking/TableUpdaterTest.php @@ -5,6 +5,7 @@ namespace Doctrine\Migrations\Tests\Tracking; use Doctrine\DBAL\Connection; +use Doctrine\DBAL\Driver\Connection as DriverConnection; use Doctrine\DBAL\Platforms\AbstractPlatform; use Doctrine\DBAL\Schema\AbstractSchemaManager; use Doctrine\DBAL\Schema\Column; @@ -96,6 +97,10 @@ public function testUpdateMigrationTable(): void ->with($toSchema, $this->platform) ->willReturn(['ALTER TABLE table_name ADD COLUMN executed_at DATETIME DEFAULT NULL']); + $this->connection->expects(self::once()) + ->method('getWrappedConnection') + ->willReturn($this->createMock(DriverConnection::class)); + $this->connection->expects(self::once()) ->method('beginTransaction'); @@ -173,6 +178,10 @@ public function testUpdateMigrationTableRollback(): void ->with($toSchema, $this->platform) ->willReturn(['ALTER TABLE table_name ADD COLUMN executed_at DATETIME DEFAULT NULL']); + $this->connection->expects(self::once()) + ->method('getWrappedConnection') + ->willReturn($this->createMock(DriverConnection::class)); + $this->connection->expects(self::once()) ->method('beginTransaction'); diff --git a/tests/Doctrine/Migrations/Tests/Version/ExecutorTest.php b/tests/Doctrine/Migrations/Tests/Version/ExecutorTest.php index b8d11b6e13..2768240909 100644 --- a/tests/Doctrine/Migrations/Tests/Version/ExecutorTest.php +++ b/tests/Doctrine/Migrations/Tests/Version/ExecutorTest.php @@ -5,6 +5,7 @@ namespace Doctrine\Migrations\Tests\Version; use Doctrine\DBAL\Connection; +use Doctrine\DBAL\Driver\Connection as DriverConnection; use Doctrine\DBAL\Platforms\AbstractPlatform; use Doctrine\DBAL\Schema\Schema; use Doctrine\Migrations\AbstractMigration; @@ -63,6 +64,10 @@ public function testAddSql(): void public function testExecuteUp(): void { + $this->connection->expects(self::once()) + ->method('getWrappedConnection') + ->willReturn($this->createMock(DriverConnection::class)); + $this->outputWriter->expects(self::at(0)) ->method('write') ->with("\n ++ migrating 001\n"); @@ -113,6 +118,10 @@ public function testExecuteUp(): void */ public function executeUpShouldAppendDescriptionWhenItIsNotEmpty(): void { + $this->connection->expects(self::once()) + ->method('getWrappedConnection') + ->willReturn($this->createMock(DriverConnection::class)); + $this->outputWriter->expects(self::at(0)) ->method('write') ->with("\n ++ migrating 001 (testing)\n"); @@ -130,6 +139,10 @@ public function executeUpShouldAppendDescriptionWhenItIsNotEmpty(): void public function testExecuteDown(): void { + $this->connection->expects(self::once()) + ->method('getWrappedConnection') + ->willReturn($this->createMock(DriverConnection::class)); + $this->outputWriter->expects(self::at(0)) ->method('write') ->with("\n -- reverting 001\n"); @@ -180,6 +193,10 @@ public function testExecuteDown(): void */ public function executeDownShouldAppendDescriptionWhenItIsNotEmpty(): void { + $this->connection->expects(self::once()) + ->method('getWrappedConnection') + ->willReturn($this->createMock(DriverConnection::class)); + $this->outputWriter->expects(self::at(0)) ->method('write') ->with("\n -- reverting 001 (testing)\n");