diff --git a/core/core.services.yml b/core/core.services.yml index 2b81197e13f643370c3b84e8d6b0db09636c9a8f..6681e7daa7108c53d929c0ca322ed8787ce7d5f1 100644 --- a/core/core.services.yml +++ b/core/core.services.yml @@ -140,6 +140,11 @@ services: class: Drupal\Core\PageCache\ResponsePolicy\KillSwitch tags: - { name: page_cache_response_policy } + page_cache_no_server_error: + class: Drupal\Core\PageCache\ResponsePolicy\NoServerError + public: false + tags: + - { name: page_cache_response_policy } config.manager: class: Drupal\Core\Config\ConfigManager arguments: ['@entity.manager', '@config.factory', '@config.typed', '@string_translation', '@config.storage', '@event_dispatcher'] diff --git a/core/lib/Drupal/Core/PageCache/ResponsePolicy/NoServerError.php b/core/lib/Drupal/Core/PageCache/ResponsePolicy/NoServerError.php new file mode 100644 index 0000000000000000000000000000000000000000..f3cd90e6cf3fa1ca27e2508dbf528232ccadc50c --- /dev/null +++ b/core/lib/Drupal/Core/PageCache/ResponsePolicy/NoServerError.php @@ -0,0 +1,28 @@ +<?php + +/** + * @file + * Contains \Drupal\Core\PageCache\ResponsePolicy\NoServerError. + */ + +namespace Drupal\Core\PageCache\ResponsePolicy; + +use Drupal\Core\PageCache\ResponsePolicyInterface; +use Symfony\Component\HttpFoundation\Request; +use Symfony\Component\HttpFoundation\Response; + +/** + * A policy denying caching of a server error (HTTP 5xx) responses. + */ +class NoServerError implements ResponsePolicyInterface { + + /** + * {@inheritdoc} + */ + public function check(Response $response, Request $request) { + if ($response->isServerError()) { + return static::DENY; + } + } + +} diff --git a/core/modules/system/src/Tests/System/ErrorHandlerTest.php b/core/modules/system/src/Tests/System/ErrorHandlerTest.php index 597a374820bee194dea0153f9a9e53274d71dc05..6d353e9498633baaabab5de4d2414105cba6da2c 100644 --- a/core/modules/system/src/Tests/System/ErrorHandlerTest.php +++ b/core/modules/system/src/Tests/System/ErrorHandlerTest.php @@ -130,6 +130,22 @@ function testExceptionHandler() { $this->assertTrue(strpos($this->drupalGetHeader(':status'), '500 Service unavailable (with message)'), 'Received expected HTTP status line.'); $this->assertErrorMessage($error_renderer_exception); + // Enable the page cache and disable error reporting, ensure that 5xx + // responses are not cached. + $config = $this->config('system.performance'); + $config->set('cache.page.use_internal', 1); + $config->set('cache.page.max_age', 300); + $config->save(); + $this->config('system.logging') + ->set('error_level', ERROR_REPORTING_HIDE) + ->save(); + + $this->drupalGet('error-test/trigger-exception'); + $this->assertFalse($this->drupalGetHeader('X-Drupal-Cache')); + $this->assertIdentical(strpos($this->drupalGetHeader('Cache-Control'), 'public'), FALSE, 'Received expected HTTP status line.'); + $this->assertTrue(strpos($this->drupalGetHeader(':status'), '500 Service unavailable (with message)'), 'Received expected HTTP status line.'); + $this->assertNoErrorMessage($error_exception); + // The exceptions are expected. Do not interpret them as a test failure. // Not using File API; a potential error must trigger a PHP warning. unlink(\Drupal::root() . '/' . $this->siteDirectory . '/error.log');