Issue #2472281 by pwolanin: 404/403 responses for non-existing nodes are cached in Page Cache/reverse proxy, are not invalidated when the node is created
parent
8f07297676
commit
afdbe0da9a
|
@ -196,6 +196,10 @@ services:
|
||||||
factory_method: get
|
factory_method: get
|
||||||
factory_service: cache_factory
|
factory_service: cache_factory
|
||||||
arguments: [discovery]
|
arguments: [discovery]
|
||||||
|
cache_router_rebuild_subscriber:
|
||||||
|
class: Drupal\Core\EventSubscriber\CacheRouterRebuildSubscriber
|
||||||
|
tags:
|
||||||
|
- { name: event_subscriber }
|
||||||
page_cache_request_policy:
|
page_cache_request_policy:
|
||||||
class: Drupal\Core\PageCache\DefaultRequestPolicy
|
class: Drupal\Core\PageCache\DefaultRequestPolicy
|
||||||
arguments: ['@session_configuration']
|
arguments: ['@session_configuration']
|
||||||
|
|
|
@ -504,6 +504,10 @@ abstract class Entity implements EntityInterface {
|
||||||
// listing's filtering requirements. A newly created entity may start to
|
// listing's filtering requirements. A newly created entity may start to
|
||||||
// appear in listings because it did not exist before.)
|
// appear in listings because it did not exist before.)
|
||||||
$tags = $this->getEntityType()->getListCacheTags();
|
$tags = $this->getEntityType()->getListCacheTags();
|
||||||
|
if ($this->hasLinkTemplate('canonical')) {
|
||||||
|
// Creating or updating an entity may change a cached 403 or 404 response.
|
||||||
|
$tags = Cache::mergeTags($tags, ['4xx-response']);
|
||||||
|
}
|
||||||
if ($update) {
|
if ($update) {
|
||||||
// An existing entity was updated, also invalidate its unique cache tag.
|
// An existing entity was updated, also invalidate its unique cache tag.
|
||||||
$tags = Cache::mergeTags($tags, $this->getCacheTags());
|
$tags = Cache::mergeTags($tags, $this->getCacheTags());
|
||||||
|
|
|
@ -0,0 +1,37 @@
|
||||||
|
<?php
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @file
|
||||||
|
* Contains \Drupal\Core\EventSubscriber\CacheRouterRebuildSubscriber.
|
||||||
|
*/
|
||||||
|
|
||||||
|
namespace Drupal\Core\EventSubscriber;
|
||||||
|
|
||||||
|
use Drupal\Core\Cache\Cache;
|
||||||
|
use Drupal\Core\Routing\RoutingEvents;
|
||||||
|
use Symfony\Component\EventDispatcher\EventSubscriberInterface;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Clear cache tags when the router is rebuilt.
|
||||||
|
*/
|
||||||
|
class CacheRouterRebuildSubscriber implements EventSubscriberInterface {
|
||||||
|
|
||||||
|
/**
|
||||||
|
* {@inheritdoc}
|
||||||
|
*/
|
||||||
|
public function onRouterFinished() {
|
||||||
|
// Requested URLs that formerly gave a 403/404 may now be valid.
|
||||||
|
Cache::invalidateTags(['4xx-response']);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* {@inheritdoc}
|
||||||
|
*/
|
||||||
|
public static function getSubscribedEvents() {
|
||||||
|
$events = [];
|
||||||
|
// Act only when the router rebuild is finished.
|
||||||
|
$events[RoutingEvents::FINISHED][] = ['onRouterFinished'];
|
||||||
|
return $events;
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
|
@ -138,6 +138,15 @@ class FinishResponseSubscriber implements EventSubscriberInterface {
|
||||||
if ($access_result instanceof CacheableDependencyInterface) {
|
if ($access_result instanceof CacheableDependencyInterface) {
|
||||||
$this->updateDrupalCacheHeaders($response, $access_result);
|
$this->updateDrupalCacheHeaders($response, $access_result);
|
||||||
}
|
}
|
||||||
|
// Add a cache tag to any 4xx response.
|
||||||
|
if ($response->isClientError()) {
|
||||||
|
$cache_tags = ['4xx-response'];
|
||||||
|
if ($response->headers->has('X-Drupal-Cache-Tags')) {
|
||||||
|
$existing_cache_tags = explode(' ', $response->headers->get('X-Drupal-Cache-Tags'));
|
||||||
|
$cache_tags = Cache::mergeTags($existing_cache_tags, $cache_tags);
|
||||||
|
}
|
||||||
|
$response->headers->set('X-Drupal-Cache-Tags', implode(' ', $cache_tags));
|
||||||
|
}
|
||||||
|
|
||||||
$is_cacheable = ($this->requestPolicy->check($request) === RequestPolicyInterface::ALLOW) && ($this->responsePolicy->check($response, $request) !== ResponsePolicyInterface::DENY);
|
$is_cacheable = ($this->requestPolicy->check($request) === RequestPolicyInterface::ALLOW) && ($this->responsePolicy->check($response, $request) !== ResponsePolicyInterface::DENY);
|
||||||
|
|
||||||
|
|
|
@ -10,6 +10,7 @@ namespace Drupal\system\Tests\Bootstrap;
|
||||||
use Drupal\Component\Datetime\DateTimePlus;
|
use Drupal\Component\Datetime\DateTimePlus;
|
||||||
use Drupal\Core\Routing\RequestContext;
|
use Drupal\Core\Routing\RequestContext;
|
||||||
use Drupal\Core\Url;
|
use Drupal\Core\Url;
|
||||||
|
use Drupal\entity_test\Entity\EntityTest;
|
||||||
use Drupal\simpletest\WebTestBase;
|
use Drupal\simpletest\WebTestBase;
|
||||||
use Drupal\Core\Cache\Cache;
|
use Drupal\Core\Cache\Cache;
|
||||||
use Drupal\user\Entity\Role;
|
use Drupal\user\Entity\Role;
|
||||||
|
@ -29,7 +30,7 @@ class PageCacheTest extends WebTestBase {
|
||||||
*
|
*
|
||||||
* @var array
|
* @var array
|
||||||
*/
|
*/
|
||||||
public static $modules = array('test_page_test', 'system_test');
|
public static $modules = array('test_page_test', 'system_test', 'entity_test');
|
||||||
|
|
||||||
protected function setUp() {
|
protected function setUp() {
|
||||||
parent::setUp();
|
parent::setUp();
|
||||||
|
@ -307,6 +308,52 @@ class PageCacheTest extends WebTestBase {
|
||||||
$this->assertNoCacheTag('config:user.role.authenticated');
|
$this->assertNoCacheTag('config:user.role.authenticated');
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Tests the 4xx-response cache tag is added and invalidated.
|
||||||
|
*/
|
||||||
|
function testPageCacheAnonymous403404() {
|
||||||
|
$admin_url = Url::fromRoute('system.admin');
|
||||||
|
$invalid_url = 'foo/' . $this->randomString();
|
||||||
|
$tests = [
|
||||||
|
403 => $admin_url,
|
||||||
|
404 => $invalid_url,
|
||||||
|
];
|
||||||
|
foreach ($tests as $code => $content_url) {
|
||||||
|
// Anonymous user, without permissions.
|
||||||
|
$this->drupalGet($content_url);
|
||||||
|
$this->assertResponse($code);
|
||||||
|
$this->assertEqual($this->drupalGetHeader('X-Drupal-Cache'), 'MISS');
|
||||||
|
$this->assertCacheTag('4xx-response');
|
||||||
|
$this->drupalGet($content_url);
|
||||||
|
$this->assertResponse($code);
|
||||||
|
$this->assertEqual($this->drupalGetHeader('X-Drupal-Cache'), 'HIT');
|
||||||
|
$entity_values = array(
|
||||||
|
'name' => $this->randomMachineName(),
|
||||||
|
'user_id' => 1,
|
||||||
|
'field_test_text' => array(
|
||||||
|
0 => array(
|
||||||
|
'value' => $this->randomString(),
|
||||||
|
'format' => 'plain_text',
|
||||||
|
)
|
||||||
|
),
|
||||||
|
);
|
||||||
|
$entity = EntityTest::create($entity_values);
|
||||||
|
$entity->save();
|
||||||
|
// Saving an entity clears 4xx cache tag.
|
||||||
|
$this->drupalGet($content_url);
|
||||||
|
$this->assertResponse($code);
|
||||||
|
$this->assertEqual($this->drupalGetHeader('X-Drupal-Cache'), 'MISS');
|
||||||
|
$this->drupalGet($content_url);
|
||||||
|
$this->assertResponse($code);
|
||||||
|
$this->assertEqual($this->drupalGetHeader('X-Drupal-Cache'), 'HIT');
|
||||||
|
// Rebuilding the router should invalidate the 4xx cache tag.
|
||||||
|
$this->container->get('router.builder')->rebuild();
|
||||||
|
$this->drupalGet($content_url);
|
||||||
|
$this->assertResponse($code);
|
||||||
|
$this->assertEqual($this->drupalGetHeader('X-Drupal-Cache'), 'MISS');
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Tests the omit_vary_cookie setting.
|
* Tests the omit_vary_cookie setting.
|
||||||
*/
|
*/
|
||||||
|
|
Loading…
Reference in New Issue