From 213906e2867ddf6187bb140a3002db2c1144ab92 Mon Sep 17 00:00:00 2001 From: Alex Pott <alex.a.pott@googlemail.com> Date: Thu, 22 Feb 2024 13:05:24 +0000 Subject: [PATCH] Issue #3414287 by catch, longwave: Avoid reading session from the database multiple times during a big pipe request --- .../Session/ResponseKeepSessionOpenInterface.php | 16 ++++++++++++++++ core/lib/Drupal/Core/StackMiddleware/Session.php | 3 ++- core/modules/big_pipe/src/Render/BigPipe.php | 12 ------------ .../big_pipe/src/Render/BigPipeResponse.php | 3 ++- ...OpenTelemetryAuthenticatedPerformanceTest.php | 3 +-- .../StandardPerformanceTest.php | 4 ++-- 6 files changed, 23 insertions(+), 18 deletions(-) create mode 100644 core/lib/Drupal/Core/Session/ResponseKeepSessionOpenInterface.php diff --git a/core/lib/Drupal/Core/Session/ResponseKeepSessionOpenInterface.php b/core/lib/Drupal/Core/Session/ResponseKeepSessionOpenInterface.php new file mode 100644 index 000000000000..4f4cdcc30d0a --- /dev/null +++ b/core/lib/Drupal/Core/Session/ResponseKeepSessionOpenInterface.php @@ -0,0 +1,16 @@ +<?php + +namespace Drupal\Core\Session; + +/** + * Indicates that sessions for this response should be kept open after sending. + * + * By default, Drupal closes sessions as soon as the response is sent. If + * a response implements this interface, Drupal will skip this behavior and + * assume that the session will be closed manually later in the request. + * + * @see Drupal\Core\StackMiddleware\Session + * @see Drupal\big_pipe\src\Render\BigPipeResponse + * @internal + */ +interface ResponseKeepSessionOpenInterface {} diff --git a/core/lib/Drupal/Core/StackMiddleware/Session.php b/core/lib/Drupal/Core/StackMiddleware/Session.php index 94841e1508dd..6e8082b81e44 100644 --- a/core/lib/Drupal/Core/StackMiddleware/Session.php +++ b/core/lib/Drupal/Core/StackMiddleware/Session.php @@ -2,6 +2,7 @@ namespace Drupal\Core\StackMiddleware; +use Drupal\Core\Session\ResponseKeepSessionOpenInterface; use Symfony\Component\DependencyInjection\ContainerAwareTrait; use Symfony\Component\HttpFoundation\Request; use Symfony\Component\HttpFoundation\Response; @@ -58,7 +59,7 @@ public function handle(Request $request, $type = self::MAIN_REQUEST, $catch = TR $result = $this->httpKernel->handle($request, $type, $catch); - if ($type === self::MAIN_REQUEST && PHP_SAPI !== 'cli' && $request->hasSession()) { + if ($type === self::MAIN_REQUEST && !$result instanceof ResponseKeepSessionOpenInterface && PHP_SAPI !== 'cli' && $request->hasSession()) { $request->getSession()->save(); } diff --git a/core/modules/big_pipe/src/Render/BigPipe.php b/core/modules/big_pipe/src/Render/BigPipe.php index 1dab69c4e3c0..1eab778c0c21 100644 --- a/core/modules/big_pipe/src/Render/BigPipe.php +++ b/core/modules/big_pipe/src/Render/BigPipe.php @@ -220,16 +220,6 @@ public function __construct(RendererInterface $renderer, SessionInterface $sessi } } - /** - * Performs tasks before sending content (and rendering placeholders). - */ - protected function performPreSendTasks() { - // The content in the placeholders may depend on the session, and by the - // time the response is sent (see index.php), the session is already - // closed. Reopen it for the duration that we are rendering placeholders. - $this->session->start(); - } - /** * Performs tasks after sending content (and rendering placeholders). */ @@ -282,8 +272,6 @@ public function sendContent(BigPipeResponse $response) { $cumulative_assets = AttachedAssets::createFromRenderArray(['#attached' => $attachments]); $cumulative_assets->setAlreadyLoadedLibraries($attachments['library']); - $this->performPreSendTasks(); - // Find the closing </body> tag and get the strings before and after. But be // careful to use the latest occurrence of the string "</body>", to ensure // that strings in inline JavaScript or CDATA sections aren't used instead. diff --git a/core/modules/big_pipe/src/Render/BigPipeResponse.php b/core/modules/big_pipe/src/Render/BigPipeResponse.php index 573b7dcb5ef3..0af699617068 100644 --- a/core/modules/big_pipe/src/Render/BigPipeResponse.php +++ b/core/modules/big_pipe/src/Render/BigPipeResponse.php @@ -3,6 +3,7 @@ namespace Drupal\big_pipe\Render; use Drupal\Core\Render\HtmlResponse; +use Drupal\Core\Session\ResponseKeepSessionOpenInterface; /** * A response that is sent in chunks by the BigPipe service. @@ -18,7 +19,7 @@ * created in https://www.drupal.org/node/2577631. Only code internal to * BigPipe should instantiate or type hint to this class. */ -class BigPipeResponse extends HtmlResponse { +class BigPipeResponse extends HtmlResponse implements ResponseKeepSessionOpenInterface { /** * The BigPipe service. diff --git a/core/profiles/demo_umami/tests/src/FunctionalJavascript/OpenTelemetryAuthenticatedPerformanceTest.php b/core/profiles/demo_umami/tests/src/FunctionalJavascript/OpenTelemetryAuthenticatedPerformanceTest.php index 8e66e7df19ae..2612ecf5bac5 100644 --- a/core/profiles/demo_umami/tests/src/FunctionalJavascript/OpenTelemetryAuthenticatedPerformanceTest.php +++ b/core/profiles/demo_umami/tests/src/FunctionalJavascript/OpenTelemetryAuthenticatedPerformanceTest.php @@ -35,8 +35,7 @@ public function testFrontPageAuthenticatedWarmCache(): void { $performance_data = $this->collectPerformanceData(function () { $this->drupalGet('<front>'); }, 'authenticatedFrontPage'); - $this->assertGreaterThanOrEqual(10, $performance_data->getQueryCount()); - $this->assertLessThanOrEqual(12, $performance_data->getQueryCount()); + $this->assertCountBetween(9, 11, $performance_data->getQueryCount()); $this->assertSame(45, $performance_data->getCacheGetCount()); $this->assertSame(0, $performance_data->getCacheSetCount()); $this->assertSame(0, $performance_data->getCacheDeleteCount()); diff --git a/core/profiles/standard/tests/src/FunctionalJavascript/StandardPerformanceTest.php b/core/profiles/standard/tests/src/FunctionalJavascript/StandardPerformanceTest.php index c977aaf51d1c..5fece5679a74 100644 --- a/core/profiles/standard/tests/src/FunctionalJavascript/StandardPerformanceTest.php +++ b/core/profiles/standard/tests/src/FunctionalJavascript/StandardPerformanceTest.php @@ -113,7 +113,7 @@ public function testLogin(): void { $this->submitLoginForm($account); }); - $this->assertCountBetween(26, 31, $performance_data->getQueryCount()); + $this->assertCountBetween(25, 30, $performance_data->getQueryCount()); $this->assertSame(64, $performance_data->getCacheGetCount()); $this->assertSame(1, $performance_data->getCacheSetCount()); $this->assertSame(1, $performance_data->getCacheDeleteCount()); @@ -146,7 +146,7 @@ public function testLoginBlock(): void { $performance_data = $this->collectPerformanceData(function () use ($account) { $this->submitLoginForm($account); }); - $this->assertCountBetween(31, 34, $performance_data->getQueryCount()); + $this->assertCountBetween(30, 33, $performance_data->getQueryCount()); $this->assertSame(85, $performance_data->getCacheGetCount()); $this->assertSame(1, $performance_data->getCacheSetCount()); $this->assertSame(1, $performance_data->getCacheDeleteCount()); -- GitLab