From e68ca9358fa3e56ef9ae0746474bc89ffa236d38 Mon Sep 17 00:00:00 2001 From: catch <catch@35733.no-reply.drupal.org> Date: Fri, 17 Jan 2020 11:48:31 +0000 Subject: [PATCH] Issue #2145751 by jibran, yongt9412, DuaelFr, Berdir, pnagornyak, arpad.rozsa, mbovan, Wim Leers, Nixou, effulgentsia, Fabianx, kristiaanvandeneynde, anavarre, dawehner: Introduce ENTITY_TYPE_list:BUNDLE cache tag and add it to single bundle listing --- .../Core/Config/Entity/ConfigEntityBase.php | 8 +- core/lib/Drupal/Core/Entity/EntityBase.php | 19 ++++- .../modules/cache_test/cache_test.routing.yml | 7 ++ .../src/Controller/CacheTestController.php | 26 ++++++ .../Entity/EntityBundleListCacheTest.php | 80 +++++++++++++++++++ .../Tests/Core/Entity/EntityUnitTest.php | 61 ++++++++++++++ 6 files changed, 196 insertions(+), 5 deletions(-) create mode 100644 core/tests/Drupal/FunctionalTests/Entity/EntityBundleListCacheTest.php diff --git a/core/lib/Drupal/Core/Config/Entity/ConfigEntityBase.php b/core/lib/Drupal/Core/Config/Entity/ConfigEntityBase.php index 38809c28c35c..04e94f83abbc 100644 --- a/core/lib/Drupal/Core/Config/Entity/ConfigEntityBase.php +++ b/core/lib/Drupal/Core/Config/Entity/ConfigEntityBase.php @@ -490,7 +490,7 @@ public function onDependencyRemoval(array $dependencies) { * already invalidates it. */ protected function invalidateTagsOnSave($update) { - Cache::invalidateTags($this->getEntityType()->getListCacheTags()); + Cache::invalidateTags($this->getListCacheTagsToInvalidate()); } /** @@ -500,7 +500,11 @@ protected function invalidateTagsOnSave($update) { * config system already invalidates them. */ protected static function invalidateTagsOnDelete(EntityTypeInterface $entity_type, array $entities) { - Cache::invalidateTags($entity_type->getListCacheTags()); + $tags = $entity_type->getListCacheTags(); + foreach ($entities as $entity) { + $tags = Cache::mergeTags($tags, $entity->getListCacheTagsToInvalidate()); + } + Cache::invalidateTags($tags); } /** diff --git a/core/lib/Drupal/Core/Entity/EntityBase.php b/core/lib/Drupal/Core/Entity/EntityBase.php index a20ebd4386d4..8532f6298ef7 100644 --- a/core/lib/Drupal/Core/Entity/EntityBase.php +++ b/core/lib/Drupal/Core/Entity/EntityBase.php @@ -477,12 +477,24 @@ public function getCacheContexts() { return $this->cacheContexts; } + /** + * The list cache tags to invalidate for this entity. + * + * @return string[] + * Set of list cache tags. + */ + protected function getListCacheTagsToInvalidate() { + $tags = $this->getEntityType()->getListCacheTags(); + if ($this->getEntityType()->hasKey('bundle')) { + $tags[] = $this->getEntityTypeId() . '_list:' . $this->bundle(); + } + return $tags; + } + /** * {@inheritdoc} */ public function getCacheTagsToInvalidate() { - // @todo Add bundle-specific listing cache tag? - // https://www.drupal.org/node/2145751 if ($this->isNew()) { return []; } @@ -547,7 +559,7 @@ protected function invalidateTagsOnSave($update) { // updated entity may start to appear in a listing because it now meets that // listing's filtering requirements. A newly created entity may start to // appear in listings because it did not exist before.) - $tags = $this->getEntityType()->getListCacheTags(); + $tags = $this->getListCacheTagsToInvalidate(); if ($this->hasLinkTemplate('canonical')) { // Creating or updating an entity may change a cached 403 or 404 response. $tags = Cache::mergeTags($tags, ['4xx-response']); @@ -576,6 +588,7 @@ protected static function invalidateTagsOnDelete(EntityTypeInterface $entity_typ // cache tag, but subsequent list pages would not be invalidated, hence we // must invalidate its list cache tags as well.) $tags = Cache::mergeTags($tags, $entity->getCacheTagsToInvalidate()); + $tags = Cache::mergeTags($tags, $entity->getListCacheTagsToInvalidate()); } Cache::invalidateTags($tags); } diff --git a/core/modules/system/tests/modules/cache_test/cache_test.routing.yml b/core/modules/system/tests/modules/cache_test/cache_test.routing.yml index fb87d3ded0e8..c785e19c9ed7 100644 --- a/core/modules/system/tests/modules/cache_test/cache_test.routing.yml +++ b/core/modules/system/tests/modules/cache_test/cache_test.routing.yml @@ -4,3 +4,10 @@ cache_test.url_bubbling: _controller: '\Drupal\cache_test\Controller\CacheTestController::urlBubbling' requirements: _access: 'TRUE' + +cache_test_list.bundle_tags: + path: '/cache-test-list/{entity_type_id}/{bundle}' + defaults: + _controller: '\Drupal\cache_test\Controller\CacheTestController::bundleTags' + requirements: + _access: 'TRUE' diff --git a/core/modules/system/tests/modules/cache_test/src/Controller/CacheTestController.php b/core/modules/system/tests/modules/cache_test/src/Controller/CacheTestController.php index de2e8fa82a3b..b773d5bb5ae0 100644 --- a/core/modules/system/tests/modules/cache_test/src/Controller/CacheTestController.php +++ b/core/modules/system/tests/modules/cache_test/src/Controller/CacheTestController.php @@ -19,4 +19,30 @@ public function urlBubbling() { ]; } + /** + * Bundle listing tags invalidation. + * + * @param string $entity_type_id + * The entity type ID. + * @param string $bundle + * The bundle. + * + * @return array + * Renderable array. + */ + public function bundleTags($entity_type_id, $bundle) { + $storage = \Drupal::entityTypeManager()->getStorage($entity_type_id); + $entity_ids = $storage->getQuery()->condition('type', $bundle)->execute(); + $page = []; + + $entities = $storage->loadMultiple($entity_ids); + foreach ($entities as $entity) { + $page[$entity->id()] = [ + '#markup' => $entity->label(), + ]; + } + $page['#cache']['tags'] = [$entity_type_id . '_list:' . $bundle]; + return $page; + } + } diff --git a/core/tests/Drupal/FunctionalTests/Entity/EntityBundleListCacheTest.php b/core/tests/Drupal/FunctionalTests/Entity/EntityBundleListCacheTest.php new file mode 100644 index 000000000000..207c0b74b7de --- /dev/null +++ b/core/tests/Drupal/FunctionalTests/Entity/EntityBundleListCacheTest.php @@ -0,0 +1,80 @@ +<?php + +namespace Drupal\FunctionalTests\Entity; + +use Drupal\Core\Url; +use Drupal\entity_test\Entity\EntityTestBundle; +use Drupal\entity_test\Entity\EntityTestWithBundle; +use Drupal\Tests\BrowserTestBase; +use Drupal\Tests\system\Functional\Cache\AssertPageCacheContextsAndTagsTrait; + +/** + * Tests that bundle tags are invalidated when entities change. + * + * @group Entity + */ +class EntityBundleListCacheTest extends BrowserTestBase { + + use AssertPageCacheContextsAndTagsTrait; + + /** + * Modules to enable. + * + * @var array + */ + public static $modules = ['cache_test', 'entity_test']; + + /** + * {@inheritdoc} + */ + protected $defaultTheme = 'classy'; + + /** + * {@inheritdoc} + */ + protected function setUp() { + parent::setUp(); + EntityTestBundle::create([ + 'id' => 'bundle_a', + 'label' => 'Bundle A', + ])->save(); + EntityTestBundle::create([ + 'id' => 'bundle_b', + 'label' => 'Bundle B', + ])->save(); + } + + /** + * Tests that tags are invalidated when an entity with that bundle changes. + */ + public function testBundleListingCache() { + // Access to lists of test entities with each bundle. + $bundle_a_url = Url::fromRoute('cache_test_list.bundle_tags', ['entity_type_id' => 'entity_test_with_bundle', 'bundle' => 'bundle_a']); + $bundle_b_url = Url::fromRoute('cache_test_list.bundle_tags', ['entity_type_id' => 'entity_test_with_bundle', 'bundle' => 'bundle_b']); + $this->drupalGet($bundle_a_url); + $this->assertSession()->responseHeaderEquals('X-Drupal-Cache', 'MISS'); + $this->assertCacheTags(['rendered', 'entity_test_with_bundle_list:bundle_a']); + + $this->drupalGet($bundle_a_url); + $this->assertSession()->responseHeaderEquals('X-Drupal-Cache', 'HIT'); + $this->assertCacheTags(['rendered', 'entity_test_with_bundle_list:bundle_a']); + $this->drupalGet($bundle_b_url); + $this->assertSession()->responseHeaderEquals('X-Drupal-Cache', 'MISS'); + $this->assertCacheTags(['rendered', 'entity_test_with_bundle_list:bundle_b']); + $this->drupalGet($bundle_b_url); + $this->assertSession()->responseHeaderEquals('X-Drupal-Cache', 'HIT'); + $entity1 = EntityTestWithBundle::create(['type' => 'bundle_a', 'name' => 'entity1']); + $entity1->save(); + // Check that tags are invalidated after creating an entity of the current + // bundle. + $this->drupalGet($bundle_a_url); + $this->assertSession()->responseHeaderEquals('X-Drupal-Cache', 'MISS'); + $this->drupalGet($bundle_a_url); + $this->assertSession()->responseHeaderEquals('X-Drupal-Cache', 'HIT'); + // Check that tags are not invalidated after creating an entity of a + // different bundle than the current in the request. + $this->drupalGet($bundle_b_url); + $this->assertSession()->responseHeaderEquals('X-Drupal-Cache', 'HIT'); + } + +} diff --git a/core/tests/Drupal/Tests/Core/Entity/EntityUnitTest.php b/core/tests/Drupal/Tests/Core/Entity/EntityUnitTest.php index 1e001cc99256..7933611bf2eb 100644 --- a/core/tests/Drupal/Tests/Core/Entity/EntityUnitTest.php +++ b/core/tests/Drupal/Tests/Core/Entity/EntityUnitTest.php @@ -441,6 +441,43 @@ public function testPostSave() { $this->entity->postSave($storage, TRUE); } + /** + * @covers ::postSave + */ + public function testPostSaveBundle() { + $this->cacheTagsInvalidator->expects($this->at(0)) + ->method('invalidateTags') + ->with([ + // List cache tag. + $this->entityTypeId . '_list', + $this->entityTypeId . '_list:' . $this->entity->bundle(), + ]); + $this->cacheTagsInvalidator->expects($this->at(1)) + ->method('invalidateTags') + ->with([ + // Own cache tag. + $this->entityTypeId . ':' . $this->values['id'], + // List cache tag. + $this->entityTypeId . '_list', + $this->entityTypeId . '_list:' . $this->entity->bundle(), + ]); + + $this->entityType->expects($this->atLeastOnce()) + ->method('hasKey') + ->with('bundle') + ->willReturn(TRUE); + + // This method is internal, so check for errors on calling it only. + $storage = $this->createMock('\Drupal\Core\Entity\EntityStorageInterface'); + + // A creation should trigger the invalidation of the global list cache tag + // and the one for the bundle. + $this->entity->postSave($storage, FALSE); + // An update should trigger the invalidation of the "list", bundle list and + // the "own" cache tags. + $this->entity->postSave($storage, TRUE); + } + /** * @covers ::preCreate */ @@ -491,6 +528,30 @@ public function testPostDelete() { $this->entity->postDelete($storage, $entities); } + /** + * @covers ::postDelete + */ + public function testPostDeleteBundle() { + $this->cacheTagsInvalidator->expects($this->once()) + ->method('invalidateTags') + ->with([ + $this->entityTypeId . ':' . $this->values['id'], + $this->entityTypeId . '_list', + $this->entityTypeId . '_list:' . $this->entity->bundle(), + ]); + $this->entityType->expects($this->atLeastOnce()) + ->method('hasKey') + ->with('bundle') + ->willReturn(TRUE); + $storage = $this->createMock('\Drupal\Core\Entity\EntityStorageInterface'); + $storage->expects($this->once()) + ->method('getEntityType') + ->willReturn($this->entityType); + + $entities = [$this->values['id'] => $this->entity]; + $this->entity->postDelete($storage, $entities); + } + /** * @covers ::postLoad */ -- GitLab