From bd93e686d3a00e51bb92dbad8800c015e8428bae Mon Sep 17 00:00:00 2001 From: SakiTakamachi Date: Sat, 7 Oct 2023 13:12:17 +0900 Subject: [PATCH 1/8] fix expected level in afterCommitCallbacksShouldBeExecuted --- src/Illuminate/Database/DatabaseTransactionsManager.php | 2 +- .../Foundation/Testing/DatabaseTransactionsManager.php | 2 +- 2 files changed, 2 insertions(+), 2 deletions(-) diff --git a/src/Illuminate/Database/DatabaseTransactionsManager.php b/src/Illuminate/Database/DatabaseTransactionsManager.php index a39924db761b..c759ac6a2840 100755 --- a/src/Illuminate/Database/DatabaseTransactionsManager.php +++ b/src/Illuminate/Database/DatabaseTransactionsManager.php @@ -127,7 +127,7 @@ public function callbackApplicableTransactions() */ public function afterCommitCallbacksShouldBeExecuted($level) { - return $level === 1; + return $level === 0; } /** diff --git a/src/Illuminate/Foundation/Testing/DatabaseTransactionsManager.php b/src/Illuminate/Foundation/Testing/DatabaseTransactionsManager.php index b2e2de142e99..08c1635443a6 100644 --- a/src/Illuminate/Foundation/Testing/DatabaseTransactionsManager.php +++ b/src/Illuminate/Foundation/Testing/DatabaseTransactionsManager.php @@ -42,6 +42,6 @@ public function callbackApplicableTransactions() */ public function afterCommitCallbacksShouldBeExecuted($level) { - return $level === 2; + return $level === 1; } } From b1aea1d6be67fa915544bfb98eab76e9a0d5d3fe Mon Sep 17 00:00:00 2001 From: SakiTakamachi Date: Sat, 7 Oct 2023 13:13:07 +0900 Subject: [PATCH 2/8] remove callbacksShouldIgnore --- .../Database/DatabaseTransactionsManager.php | 28 ------------------- 1 file changed, 28 deletions(-) diff --git a/src/Illuminate/Database/DatabaseTransactionsManager.php b/src/Illuminate/Database/DatabaseTransactionsManager.php index c759ac6a2840..e198f4f3f6d6 100755 --- a/src/Illuminate/Database/DatabaseTransactionsManager.php +++ b/src/Illuminate/Database/DatabaseTransactionsManager.php @@ -11,13 +11,6 @@ class DatabaseTransactionsManager */ protected $transactions; - /** - * The database transaction that should be ignored by callbacks. - * - * @var \Illuminate\Database\DatabaseTransactionRecord|null - */ - protected $callbacksShouldIgnore; - /** * Create a new database transactions manager instance. * @@ -54,10 +47,6 @@ public function rollback($connection, $level) $this->transactions = $this->transactions->reject( fn ($transaction) => $transaction->connection == $connection && $transaction->level > $level )->values(); - - if ($this->transactions->isEmpty()) { - $this->callbacksShouldIgnore = null; - } } /** @@ -75,10 +64,6 @@ public function commit($connection) $this->transactions = $forOtherConnections->values(); $forThisConnection->map->executeCallbacks(); - - if ($this->transactions->isEmpty()) { - $this->callbacksShouldIgnore = null; - } } /** @@ -96,19 +81,6 @@ public function addCallback($callback) $callback(); } - /** - * Specify that callbacks should ignore the given transaction when determining if they should be executed. - * - * @param \Illuminate\Database\DatabaseTransactionRecord $transaction - * @return $this - */ - public function callbacksShouldIgnore(DatabaseTransactionRecord $transaction) - { - $this->callbacksShouldIgnore = $transaction; - - return $this; - } - /** * Get the transactions that are applicable to callbacks. * From 491bedfa62d9faf916533ea23849e363a292ad3e Mon Sep 17 00:00:00 2001 From: SakiTakamachi Date: Sat, 7 Oct 2023 13:18:22 +0900 Subject: [PATCH 3/8] Fixed transaction level down timing --- .../Database/Concerns/ManagesTransactions.php | 11 +++++------ 1 file changed, 5 insertions(+), 6 deletions(-) diff --git a/src/Illuminate/Database/Concerns/ManagesTransactions.php b/src/Illuminate/Database/Concerns/ManagesTransactions.php index 80dac1c57b12..a690f7b5cb46 100644 --- a/src/Illuminate/Database/Concerns/ManagesTransactions.php +++ b/src/Illuminate/Database/Concerns/ManagesTransactions.php @@ -47,6 +47,8 @@ public function transaction(Closure $callback, $attempts = 1) $this->getPdo()->commit(); } + $this->transactions = max(0, $this->transactions - 1); + if ($this->afterCommitCallbacksShouldBeExecuted()) { $this->transactionsManager?->commit($this->getName()); } @@ -56,8 +58,6 @@ public function transaction(Closure $callback, $attempts = 1) ); continue; - } finally { - $this->transactions = max(0, $this->transactions - 1); } $this->fireConnectionEvent('committed'); @@ -194,12 +194,12 @@ public function commit() $this->getPdo()->commit(); } + $this->transactions = max(0, $this->transactions - 1); + if ($this->afterCommitCallbacksShouldBeExecuted()) { $this->transactionsManager?->commit($this->getName()); } - $this->transactions = max(0, $this->transactions - 1); - $this->fireConnectionEvent('committed'); } @@ -210,8 +210,7 @@ public function commit() */ protected function afterCommitCallbacksShouldBeExecuted() { - return $this->transactions == 0 || - $this->transactionsManager?->afterCommitCallbacksShouldBeExecuted($this->transactions); + return $this->transactionsManager?->afterCommitCallbacksShouldBeExecuted($this->transactions) || $this->transactions == 0; } /** From 1133ceb1934099d2d6f2714c27b966b67dea2090 Mon Sep 17 00:00:00 2001 From: SakiTakamachi Date: Sat, 7 Oct 2023 17:09:17 +0900 Subject: [PATCH 4/8] Fixed transaction level down timing From f360770d4fe321c60bf40df656b2cfddd966f46e Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?=E6=AD=A6=E7=94=B0=20=E6=86=B2=E5=A4=AA=E9=83=8E?= Date: Sat, 7 Oct 2023 21:45:17 +0900 Subject: [PATCH 5/8] Add test - after commit is executed on final commit --- tests/Database/DatabaseConnectionTest.php | 13 +++++++++++++ 1 file changed, 13 insertions(+) diff --git a/tests/Database/DatabaseConnectionTest.php b/tests/Database/DatabaseConnectionTest.php index 0df367eed634..eb6b86444900 100755 --- a/tests/Database/DatabaseConnectionTest.php +++ b/tests/Database/DatabaseConnectionTest.php @@ -7,6 +7,7 @@ use Exception; use Illuminate\Contracts\Events\Dispatcher; use Illuminate\Database\Connection; +use Illuminate\Database\DatabaseTransactionsManager; use Illuminate\Database\Events\QueryExecuted; use Illuminate\Database\Events\TransactionBeginning; use Illuminate\Database\Events\TransactionCommitted; @@ -289,6 +290,18 @@ public function testCommittingFiresEventsIfSet() $connection->commit(); } + public function testAfterCommitIsExecutedOnFinalCommit() + { + $pdo = $this->getMockBuilder(DatabaseConnectionTestMockPDO::class)->onlyMethods(['beginTransaction','commit'])->getMock(); + $transactionsManager = $this->getMockBuilder(DatabaseTransactionsManager::class)->onlyMethods(['afterCommitCallbacksShouldBeExecuted'])->getMock(); + $transactionsManager->expects($this->once())->method('afterCommitCallbacksShouldBeExecuted')->with(0)->willReturn(true); + + $connection = $this->getMockConnection([], $pdo); + $connection->setTransactionManager($transactionsManager); + + $connection->transaction(function(){}); + } + public function testRollBackedFiresEventsIfSet() { $pdo = $this->createMock(DatabaseConnectionTestMockPDO::class); From e131badc9d0f319d37ecb9a77a45f8b46ce8ea7f Mon Sep 17 00:00:00 2001 From: Saki Takamachi <34942839+SakiTakamachi@users.noreply.github.com> Date: Sun, 8 Oct 2023 02:18:31 +0900 Subject: [PATCH 6/8] style fix style fix --- tests/Database/DatabaseConnectionTest.php | 6 ++++-- 1 file changed, 4 insertions(+), 2 deletions(-) diff --git a/tests/Database/DatabaseConnectionTest.php b/tests/Database/DatabaseConnectionTest.php index eb6b86444900..9ad3819a0057 100755 --- a/tests/Database/DatabaseConnectionTest.php +++ b/tests/Database/DatabaseConnectionTest.php @@ -292,14 +292,16 @@ public function testCommittingFiresEventsIfSet() public function testAfterCommitIsExecutedOnFinalCommit() { - $pdo = $this->getMockBuilder(DatabaseConnectionTestMockPDO::class)->onlyMethods(['beginTransaction','commit'])->getMock(); + $pdo = $this->getMockBuilder(DatabaseConnectionTestMockPDO::class)->onlyMethods(['beginTransaction', 'commit'])->getMock(); $transactionsManager = $this->getMockBuilder(DatabaseTransactionsManager::class)->onlyMethods(['afterCommitCallbacksShouldBeExecuted'])->getMock(); $transactionsManager->expects($this->once())->method('afterCommitCallbacksShouldBeExecuted')->with(0)->willReturn(true); $connection = $this->getMockConnection([], $pdo); $connection->setTransactionManager($transactionsManager); - $connection->transaction(function(){}); + $connection->transaction(function () { + // do nothing + }); } public function testRollBackedFiresEventsIfSet() From 8339bfb3eefa12e05f9dd12e015af8d7ef34dfd0 Mon Sep 17 00:00:00 2001 From: Mior Muhammad Zaki Date: Mon, 9 Oct 2023 14:35:20 +0800 Subject: [PATCH 7/8] [10.x] Test Improvements Signed-off-by: Mior Muhammad Zaki --- ...loquentTransactionWithAfterCommitTests.php | 48 +++++++++++++++++++ 1 file changed, 48 insertions(+) diff --git a/tests/Integration/Database/EloquentTransactionWithAfterCommitTests.php b/tests/Integration/Database/EloquentTransactionWithAfterCommitTests.php index 41a2eed8a0da..d01d56401892 100644 --- a/tests/Integration/Database/EloquentTransactionWithAfterCommitTests.php +++ b/tests/Integration/Database/EloquentTransactionWithAfterCommitTests.php @@ -2,7 +2,11 @@ namespace Illuminate\Tests\Integration\Database; +use Illuminate\Bus\Queueable; +use Illuminate\Contracts\Queue\ShouldQueue; use Illuminate\Foundation\Auth\User; +use Illuminate\Foundation\Bus\Dispatchable; +use Illuminate\Queue\InteractsWithQueue; use Illuminate\Support\Facades\DB; use Orchestra\Testbench\Concerns\WithLaravelMigrations; use Orchestra\Testbench\Factories\UserFactory; @@ -41,6 +45,21 @@ public function testObserverCalledWithAfterCommitWhenInsideTransaction() $this->assertEquals(1, $observer::$calledTimes, 'Failed to assert the observer was called once.'); } + public function testObserverCalledWithAfterCommitWhenInsideTransactionWithDispatchSync() + { + User::observe($observer = EloquentTransactionWithAfterCommitTestsUserObserverUsingDispatchSync::resetting()); + + $user1 = DB::transaction(fn () => User::create(UserFactory::new()->raw())); + + $this->assertTrue($user1->exists); + $this->assertEquals(1, $observer::$calledTimes, 'Failed to assert the observer was called once.'); + + $this->assertDatabaseHas('password_reset_tokens', [ + 'email' => $user1->email, + 'token' => sha1($user1->email), + ]); + } + public function testObserverIsCalledOnTestsWithAfterCommitWhenUsingSavepoint() { User::observe($observer = EloquentTransactionWithAfterCommitTestsUserObserver::resetting()); @@ -102,3 +121,32 @@ public function created($user) static::$calledTimes++; } } + +class EloquentTransactionWithAfterCommitTestsUserObserverUsingDispatchSync extends EloquentTransactionWithAfterCommitTestsUserObserver +{ + public function created($user) + { + dispatch_sync(new EloquentTransactionWithAfterCommitTestsJob($user)); + + parent::created($user); + } +} + +class EloquentTransactionWithAfterCommitTestsJob implements ShouldQueue +{ + use Dispatchable, InteractsWithQueue, Queueable; + + public function __construct(public string $email) + { + // ... + } + + public function handle(): void + { + DB::transaction(function () { + DB::table('password_reset_tokens')->insert([ + ['email' => $this->email, 'token' => sha1($this->email), 'created_at' => now()], + ]); + }); + } +} From 364d3d72d341b52fa36591bf01fa28df3116671a Mon Sep 17 00:00:00 2001 From: Mior Muhammad Zaki Date: Mon, 9 Oct 2023 14:57:39 +0800 Subject: [PATCH 8/8] wip Signed-off-by: Mior Muhammad Zaki --- .../Database/EloquentTransactionWithAfterCommitTests.php | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/tests/Integration/Database/EloquentTransactionWithAfterCommitTests.php b/tests/Integration/Database/EloquentTransactionWithAfterCommitTests.php index d01d56401892..262ef43d9c5d 100644 --- a/tests/Integration/Database/EloquentTransactionWithAfterCommitTests.php +++ b/tests/Integration/Database/EloquentTransactionWithAfterCommitTests.php @@ -126,7 +126,7 @@ class EloquentTransactionWithAfterCommitTestsUserObserverUsingDispatchSync exten { public function created($user) { - dispatch_sync(new EloquentTransactionWithAfterCommitTestsJob($user)); + dispatch_sync(new EloquentTransactionWithAfterCommitTestsJob($user->email)); parent::created($user); }