From d686df4aa58baf2ed64c42a1b23408154b8ad6f5 Mon Sep 17 00:00:00 2001 From: Lee Rowlands <lee.rowlands@previousnext.com.au> Date: Mon, 9 Jan 2023 09:27:50 +1000 Subject: [PATCH] Issue #1984588 by dpi, larowlan, AaronMcHale, smustgrave, xjm, acbramley: Add Block Content revision UI --- core/core.link_relation_types.yml | 6 + .../Controller/VersionHistoryController.php | 9 +- .../block_content/block_content.install | 24 ++ .../src/BlockContentAccessControlHandler.php | 46 ++- .../src/BlockContentPermissions.php | 36 +- .../block_content/src/Entity/BlockContent.php | 10 +- .../BlockContentRevisionDeleteTest.php | 85 +++++ .../BlockContentRevisionRevertTest.php | 87 +++++ ...BlockContentRevisionVersionHistoryTest.php | 94 +++++ .../Kernel/BlockContentAccessHandlerTest.php | 341 ++++++++++++++---- 10 files changed, 650 insertions(+), 88 deletions(-) create mode 100644 core/modules/block_content/tests/src/Functional/BlockContentRevisionDeleteTest.php create mode 100644 core/modules/block_content/tests/src/Functional/BlockContentRevisionRevertTest.php create mode 100644 core/modules/block_content/tests/src/Functional/BlockContentRevisionVersionHistoryTest.php diff --git a/core/core.link_relation_types.yml b/core/core.link_relation_types.yml index 50da734474bb..c974fbe25cb4 100644 --- a/core/core.link_relation_types.yml +++ b/core/core.link_relation_types.yml @@ -15,6 +15,12 @@ delete-multiple-form: revision: uri: https://drupal.org/link-relations/revision description: A particular version of this resource. +revision-revert-form: + uri: https://drupal.org/link-relations/revision-revert-form + description: A form where a particular version of this resource can be reverted. +revision-delete-form: + uri: https://drupal.org/link-relations/revision-delete-form + description: A form where a particular version of this resource can be deleted. create: uri: https://drupal.org/link-relations/create description: A REST resource URL where a resource of this type can be created. diff --git a/core/lib/Drupal/Core/Entity/Controller/VersionHistoryController.php b/core/lib/Drupal/Core/Entity/Controller/VersionHistoryController.php index def6fe1cde7b..8d069e5aeabc 100644 --- a/core/lib/Drupal/Core/Entity/Controller/VersionHistoryController.php +++ b/core/lib/Drupal/Core/Entity/Controller/VersionHistoryController.php @@ -11,6 +11,7 @@ use Drupal\Core\Entity\RevisionableStorageInterface; use Drupal\Core\Language\LanguageInterface; use Drupal\Core\Language\LanguageManagerInterface; +use Drupal\Core\Link; use Drupal\Core\Render\RendererInterface; use Drupal\Core\Routing\RouteMatchInterface; use Drupal\Core\Entity\RevisionLogInterface; @@ -155,10 +156,10 @@ protected function getRevisionDescription(RevisionableInterface $revision): arra $linkText = $revision->access('view label') ? $revision->label() : $this->t('- Restricted access -'); } - $revisionViewLink = $revision->toLink($linkText, 'revision'); - $context['revision'] = $revisionViewLink->getUrl()->access() - ? $revisionViewLink->toString() - : (string) $revisionViewLink->getText(); + $url = $revision->hasLinkTemplate('revision') ? $revision->toUrl('revision') : NULL; + $context['revision'] = $url && $url->access() + ? Link::fromTextAndUrl($linkText, $url)->toString() + : (string) $linkText; $context['message'] = $revision instanceof RevisionLogInterface ? [ '#markup' => $revision->getRevisionLogMessage(), '#allowed_tags' => Xss::getHtmlTagList(), diff --git a/core/modules/block_content/block_content.install b/core/modules/block_content/block_content.install index af22b6252740..3a9fb83b7b56 100644 --- a/core/modules/block_content/block_content.install +++ b/core/modules/block_content/block_content.install @@ -5,9 +5,33 @@ * Install, update and uninstall functions for the block_content module. */ +use Drupal\Core\Entity\Form\RevisionDeleteForm; +use Drupal\Core\Entity\Form\RevisionRevertForm; +use Drupal\Core\Entity\Routing\RevisionHtmlRouteProvider; +use Drupal\Core\StringTranslation\TranslatableMarkup; + /** * Implements hook_update_last_removed(). */ function block_content_update_last_removed() { return 8600; } + +/** + * Update entity definition to handle revision routes. + */ +function block_content_update_10100(&$sandbox = NULL): TranslatableMarkup { + $entityDefinitionUpdateManager = \Drupal::entityDefinitionUpdateManager(); + $definition = $entityDefinitionUpdateManager->getEntityType('block_content'); + $routeProviders = $definition->get('route_provider'); + $routeProviders['revision'] = RevisionHtmlRouteProvider::class; + $definition + ->setFormClass('revision-delete', RevisionDeleteForm::class) + ->setFormClass('revision-revert', RevisionRevertForm::class) + ->set('route_provider', $routeProviders) + ->setLinkTemplate('revision-delete-form', '/block/{block_content}/revision/{block_content_revision}/delete') + ->setLinkTemplate('revision-revert-form', '/block/{block_content}/revision/{block_content_revision}/revert') + ->setLinkTemplate('version-history', '/block/{block_content}/revisions'); + $entityDefinitionUpdateManager->updateEntityType($definition); + return \t('Added revision routes to Custom block entity type.'); +} diff --git a/core/modules/block_content/src/BlockContentAccessControlHandler.php b/core/modules/block_content/src/BlockContentAccessControlHandler.php index 1a6a31abd08e..0f09a45689d5 100644 --- a/core/modules/block_content/src/BlockContentAccessControlHandler.php +++ b/core/modules/block_content/src/BlockContentAccessControlHandler.php @@ -5,6 +5,7 @@ use Drupal\block_content\Access\DependentAccessInterface; use Drupal\block_content\Event\BlockContentGetDependencyEvent; use Drupal\Core\Access\AccessResult; +use Drupal\Core\Access\AccessResultInterface; use Drupal\Core\Entity\EntityHandlerInterface; use Drupal\Core\Entity\EntityInterface; use Drupal\Core\Entity\EntityAccessControlHandler; @@ -54,27 +55,40 @@ public static function createInstance(ContainerInterface $container, EntityTypeI * {@inheritdoc} */ protected function checkAccess(EntityInterface $entity, $operation, AccountInterface $account) { - // Allow view and update access to user with the 'edit any (type) block - // content' permission or the 'administer blocks' permission. - $edit_any_permission = 'edit any ' . $entity->bundle() . ' block content'; - if ($operation === 'view') { - $access = AccessResult::allowedIf($entity->isPublished()) + assert($entity instanceof BlockContentInterface); + $bundle = $entity->bundle(); + $forbidIfNotDefaultAndLatest = fn (): AccessResultInterface => AccessResult::forbiddenIf($entity->isDefaultRevision() && $entity->isLatestRevision()); + $forbidIfNotReusable = fn (): AccessResultInterface => AccessResult::forbiddenIf($entity->isReusable() === FALSE, sprintf('Block content must be reusable to use `%s` operation', $operation)); + $access = match ($operation) { + // Allow view and update access to user with the 'edit any (type) block + // content' permission or the 'administer blocks' permission. + 'view' => AccessResult::allowedIf($entity->isPublished()) ->orIf(AccessResult::allowedIfHasPermission($account, 'administer blocks')) - ->orIf(AccessResult::allowedIfHasPermission($account, $edit_any_permission)); - } - elseif ($operation === 'update') { - $access = AccessResult::allowedIfHasPermission($account, 'administer blocks') - ->orIf(AccessResult::allowedIfHasPermission($account, $edit_any_permission)); - } - else { - $access = parent::checkAccess($entity, $operation, $account); - } + ->orIf(AccessResult::allowedIfHasPermission($account, 'edit any ' . $bundle . ' block content')), + 'update' => AccessResult::allowedIfHasPermission($account, 'administer blocks') + ->orIf(AccessResult::allowedIfHasPermission($account, 'edit any ' . $bundle . ' block content')), + + // Revisions. + 'view all revisions' => AccessResult::allowedIfHasPermissions($account, [ + 'administer blocks', + 'view any ' . $bundle . ' block content history', + ], 'OR'), + 'revert' => AccessResult::allowedIfHasPermissions($account, [ + 'administer blocks', + 'revert any ' . $bundle . ' block content revisions', + ], 'OR')->orIf($forbidIfNotDefaultAndLatest())->orIf($forbidIfNotReusable()), + 'delete revision' => AccessResult::allowedIfHasPermissions($account, [ + 'administer blocks', + 'delete any ' . $bundle . ' block content revisions', + ], 'OR')->orIf($forbidIfNotDefaultAndLatest())->orIf($forbidIfNotReusable()), + + default => parent::checkAccess($entity, $operation, $account), + }; // Add the entity as a cacheable dependency because access will at least be // determined by whether the block is reusable. $access->addCacheableDependency($entity); - /** @var \Drupal\block_content\BlockContentInterface $entity */ - if ($entity->isReusable() === FALSE) { + if ($entity->isReusable() === FALSE && $access->isForbidden() !== TRUE) { if (!$entity instanceof DependentAccessInterface) { throw new \LogicException("Non-reusable block entities must implement \Drupal\block_content\Access\DependentAccessInterface for access control."); } diff --git a/core/modules/block_content/src/BlockContentPermissions.php b/core/modules/block_content/src/BlockContentPermissions.php index e6be17d0aad4..762379379e35 100644 --- a/core/modules/block_content/src/BlockContentPermissions.php +++ b/core/modules/block_content/src/BlockContentPermissions.php @@ -3,17 +3,40 @@ namespace Drupal\block_content; use Drupal\block_content\Entity\BlockContentType; +use Drupal\Core\DependencyInjection\ContainerInjectionInterface; use Drupal\Core\Entity\BundlePermissionHandlerTrait; +use Drupal\Core\Entity\EntityTypeManagerInterface; use Drupal\Core\StringTranslation\StringTranslationTrait; +use Symfony\Component\DependencyInjection\ContainerInterface; /** * Provide dynamic permissions for blocks of different types. */ -class BlockContentPermissions { +class BlockContentPermissions implements ContainerInjectionInterface { use StringTranslationTrait; use BundlePermissionHandlerTrait; + /** + * Constructs a BlockContentPermissions instance. + * + * @param \Drupal\Core\Entity\EntityTypeManagerInterface $entityTypeManager + * Entity type manager. + */ + public function __construct( + protected EntityTypeManagerInterface $entityTypeManager, + ) { + } + + /** + * {@inheritdoc} + */ + public static function create(ContainerInterface $container) { + return new static( + $container->get('entity_type.manager'), + ); + } + /** * Build permissions for each block type. * @@ -21,7 +44,7 @@ class BlockContentPermissions { * The block type permissions. */ public function blockTypePermissions() { - return $this->generatePermissions(BlockContentType::loadMultiple(), [$this, 'buildPermissions']); + return $this->generatePermissions($this->entityTypeManager->getStorage('block_content_type')->loadMultiple(), [$this, 'buildPermissions']); } /** @@ -40,6 +63,15 @@ protected function buildPermissions(BlockContentType $type) { "edit any $type_id block content" => [ 'title' => $this->t('%type_name: Edit any block content', $type_params), ], + "view any $type_id block content history" => [ + 'title' => $this->t('%type_name: View any block content history pages', $type_params), + ], + "revert any $type_id block content revisions" => [ + 'title' => $this->t('%type_name: Revert any block content revisions', $type_params), + ], + "delete any $type_id block content revisions" => [ + 'title' => $this->t('%type_name: Delete any block content revisions', $type_params), + ], ]; } diff --git a/core/modules/block_content/src/Entity/BlockContent.php b/core/modules/block_content/src/Entity/BlockContent.php index b12536d95d8f..dcbe2005b311 100644 --- a/core/modules/block_content/src/Entity/BlockContent.php +++ b/core/modules/block_content/src/Entity/BlockContent.php @@ -34,7 +34,12 @@ * "add" = "Drupal\block_content\BlockContentForm", * "edit" = "Drupal\block_content\BlockContentForm", * "delete" = "Drupal\block_content\Form\BlockContentDeleteForm", - * "default" = "Drupal\block_content\BlockContentForm" + * "default" = "Drupal\block_content\BlockContentForm", + * "revision-delete" = \Drupal\Core\Entity\Form\RevisionDeleteForm::class, + * "revision-revert" = \Drupal\Core\Entity\Form\RevisionRevertForm::class, + * }, + * "route_provider" = { + * "revision" = \Drupal\Core\Entity\Routing\RevisionHtmlRouteProvider::class, * }, * "translation" = "Drupal\block_content\BlockContentTranslationHandler" * }, @@ -50,6 +55,9 @@ * "edit-form" = "/block/{block_content}", * "collection" = "/admin/structure/block/block-content", * "create" = "/block", + * "revision-delete-form" = "/block/{block_content}/revision/{block_content_revision}/delete", + * "revision-revert-form" = "/block/{block_content}/revision/{block_content_revision}/revert", + * "version-history" = "/block/{block_content}/revisions", * }, * translatable = TRUE, * entity_keys = { diff --git a/core/modules/block_content/tests/src/Functional/BlockContentRevisionDeleteTest.php b/core/modules/block_content/tests/src/Functional/BlockContentRevisionDeleteTest.php new file mode 100644 index 000000000000..fb96554072e9 --- /dev/null +++ b/core/modules/block_content/tests/src/Functional/BlockContentRevisionDeleteTest.php @@ -0,0 +1,85 @@ +<?php + +declare(strict_types = 1); + +namespace Drupal\Tests\block_content\Functional; + +/** + * Block content revision delete form test. + * + * @group block_content + * @coversDefaultClass \Drupal\Core\Entity\Form\RevisionDeleteForm + */ +class BlockContentRevisionDeleteTest extends BlockContentTestBase { + + /** + * {@inheritdoc} + */ + protected $defaultTheme = 'stark'; + + /** + * {@inheritdoc} + */ + protected $permissions = [ + 'view any basic block content history', + 'delete any basic block content revisions', + ]; + + /** + * {@inheritdoc} + */ + protected function setUp(): void { + parent::setUp(); + $this->drupalLogin($this->adminUser); + $this->drupalPlaceBlock('page_title_block'); + } + + /** + * Tests revision delete. + */ + public function testDeleteForm(): void { + $entity = $this->createBlockContent(save: FALSE) + ->setRevisionCreationTime((new \DateTimeImmutable('11 January 2009 4pm'))->getTimestamp()) + ->setRevisionTranslationAffected(TRUE); + $entity->setNewRevision(); + $entity->save(); + $revisionId = $entity->getRevisionId(); + + // Cannot delete latest revision. + $this->drupalGet($entity->toUrl('revision-delete-form')); + $this->assertSession()->statusCodeEquals(403); + + // Create a new latest revision. + $entity + ->setRevisionCreationTime((new \DateTimeImmutable('11 January 2009 5pm'))->getTimestamp()) + ->setRevisionTranslationAffected(TRUE) + ->setNewRevision(); + $entity->save(); + + // Reload the entity. + $revision = \Drupal::entityTypeManager()->getStorage('block_content') + ->loadRevision($revisionId); + $this->drupalGet($revision->toUrl('revision-delete-form')); + $this->assertSession()->pageTextContains('Are you sure you want to delete the revision from Sun, 01/11/2009 - 16:00?'); + $this->assertSession()->buttonExists('Delete'); + $this->assertSession()->linkExists('Cancel'); + + $countRevisions = static function (): int { + return (int) \Drupal::entityTypeManager()->getStorage('block_content') + ->getQuery() + ->accessCheck(FALSE) + ->allRevisions() + ->count() + ->execute(); + }; + + $count = $countRevisions(); + $this->submitForm([], 'Delete'); + $this->assertEquals($count - 1, $countRevisions()); + $this->assertSession()->statusCodeEquals(200); + $this->assertSession()->addressEquals(sprintf('block/%s/revisions', $entity->id())); + $this->assertSession()->pageTextContains(sprintf('Revision from Sun, 01/11/2009 - 16:00 of basic %s has been deleted.', $entity->label())); + $this->assertSession()->elementsCount('css', 'table tbody tr', 1); + } + +} diff --git a/core/modules/block_content/tests/src/Functional/BlockContentRevisionRevertTest.php b/core/modules/block_content/tests/src/Functional/BlockContentRevisionRevertTest.php new file mode 100644 index 000000000000..48ab25c989cb --- /dev/null +++ b/core/modules/block_content/tests/src/Functional/BlockContentRevisionRevertTest.php @@ -0,0 +1,87 @@ +<?php + +declare(strict_types = 1); + +namespace Drupal\Tests\block_content\Functional; + +/** + * Block content revision form test. + * + * @group block_content + * @coversDefaultClass \Drupal\Core\Entity\Form\RevisionRevertForm + */ +class BlockContentRevisionRevertTest extends BlockContentTestBase { + + /** + * {@inheritdoc} + */ + protected $defaultTheme = 'stark'; + + /** + * {@inheritdoc} + */ + protected $permissions = [ + 'view any basic block content history', + 'revert any basic block content revisions', + ]; + + /** + * {@inheritdoc} + */ + protected function setUp(): void { + parent::setUp(); + $this->drupalLogin($this->adminUser); + $this->drupalPlaceBlock('page_title_block'); + } + + /** + * Tests revision revert. + */ + public function testRevertForm(): void { + $entity = $this->createBlockContent(save: FALSE) + ->setRevisionCreationTime((new \DateTimeImmutable('11 January 2009 4pm'))->getTimestamp()) + ->setRevisionTranslationAffected(TRUE); + $entity->setNewRevision(); + $entity->save(); + $revisionId = $entity->getRevisionId(); + + // Cannot revert latest revision. + $this->drupalGet($entity->toUrl('revision-revert-form')); + $this->assertSession()->statusCodeEquals(403); + + // Create a new latest revision. + $entity + ->setRevisionCreationTime((new \DateTimeImmutable('11 January 2009 5pm'))->getTimestamp()) + ->setRevisionTranslationAffected(TRUE) + ->setNewRevision(); + $entity->save(); + + // Reload the entity. + $revision = \Drupal::entityTypeManager()->getStorage('block_content') + ->loadRevision($revisionId); + $this->drupalGet($revision->toUrl('revision-revert-form')); + $this->assertSession()->pageTextContains('Are you sure you want to revert to the revision from Sun, 01/11/2009 - 16:00?'); + $this->assertSession()->buttonExists('Revert'); + $this->assertSession()->linkExists('Cancel'); + + $countRevisions = static function (): int { + return (int) \Drupal::entityTypeManager()->getStorage('block_content') + ->getQuery() + ->accessCheck(FALSE) + ->allRevisions() + ->count() + ->execute(); + }; + + $count = $countRevisions(); + $this->submitForm([], 'Revert'); + $this->assertEquals($count + 1, $countRevisions()); + $this->assertSession()->statusCodeEquals(200); + $this->assertSession()->addressEquals(sprintf('block/%s/revisions', $entity->id())); + $this->assertSession()->pageTextContains(sprintf('basic %s has been reverted to the revision from Sun, 01/11/2009 - 16:00.', $entity->label())); + // Three rows, from the top: the newly reverted revision, the revision from + // 5pm, and the revision from 4pm. + $this->assertSession()->elementsCount('css', 'table tbody tr', 3); + } + +} diff --git a/core/modules/block_content/tests/src/Functional/BlockContentRevisionVersionHistoryTest.php b/core/modules/block_content/tests/src/Functional/BlockContentRevisionVersionHistoryTest.php new file mode 100644 index 000000000000..dd3fad4aac5d --- /dev/null +++ b/core/modules/block_content/tests/src/Functional/BlockContentRevisionVersionHistoryTest.php @@ -0,0 +1,94 @@ +<?php + +declare(strict_types = 1); + +namespace Drupal\Tests\block_content\Functional; + +/** + * Block content version history test. + * + * @group block_content + * @coversDefaultClass \Drupal\Core\Entity\Controller\VersionHistoryController + */ +class BlockContentRevisionVersionHistoryTest extends BlockContentTestBase { + + /** + * {@inheritdoc} + */ + protected $defaultTheme = 'stark'; + + /** + * {@inheritdoc} + */ + protected $permissions = [ + 'view any basic block content history', + 'revert any basic block content revisions', + 'delete any basic block content revisions', + ]; + + /** + * {@inheritdoc} + */ + protected function setUp(): void { + parent::setUp(); + $this->drupalLogin($this->adminUser); + } + + /** + * Tests version history page. + */ + public function testVersionHistory(): void { + $entity = $this->createBlockContent(save: FALSE); + + $entity + ->setInfo('first revision') + ->setRevisionCreationTime((new \DateTimeImmutable('1st June 2020 7am'))->getTimestamp()) + ->setRevisionLogMessage('first revision log') + ->setRevisionUser($this->drupalCreateUser(name: 'first author')) + ->setNewRevision(); + $entity->save(); + + $entity + ->setInfo('second revision') + ->setRevisionCreationTime((new \DateTimeImmutable('2nd June 2020 8am'))->getTimestamp()) + ->setRevisionLogMessage('second revision log') + ->setRevisionUser($this->drupalCreateUser(name: 'second author')) + ->setNewRevision(); + $entity->save(); + + $entity + ->setInfo('third revision') + ->setRevisionCreationTime((new \DateTimeImmutable('3rd June 2020 9am'))->getTimestamp()) + ->setRevisionLogMessage('third revision log') + ->setRevisionUser($this->drupalCreateUser(name: 'third author')) + ->setNewRevision(); + $entity->save(); + + $this->drupalGet($entity->toUrl('version-history')); + $this->assertSession()->elementsCount('css', 'table tbody tr', 3); + + // Order is newest to oldest revision by creation order. + $row1 = $this->assertSession()->elementExists('css', 'table tbody tr:nth-child(1)'); + // Latest revision does not have revert or delete revision operation. + $this->assertSession()->elementNotExists('named', ['link', 'Revert'], $row1); + $this->assertSession()->elementNotExists('named', ['link', 'Delete'], $row1); + $this->assertSession()->elementTextContains('css', 'table tbody tr:nth-child(1)', 'Current revision'); + $this->assertSession()->elementTextContains('css', 'table tbody tr:nth-child(1)', 'third revision log'); + $this->assertSession()->elementTextContains('css', 'table tbody tr:nth-child(1)', '06/03/2020 - 09:00 by third author'); + + $row2 = $this->assertSession()->elementExists('css', 'table tbody tr:nth-child(2)'); + $this->assertSession()->elementExists('named', ['link', 'Revert'], $row2); + $this->assertSession()->elementExists('named', ['link', 'Delete'], $row2); + $this->assertSession()->elementTextNotContains('css', 'table tbody tr:nth-child(2)', 'Current revision'); + $this->assertSession()->elementTextContains('css', 'table tbody tr:nth-child(2)', 'second revision log'); + $this->assertSession()->elementTextContains('css', 'table tbody tr:nth-child(2)', '06/02/2020 - 08:00 by second author'); + + $row3 = $this->assertSession()->elementExists('css', 'table tbody tr:nth-child(3)'); + $this->assertSession()->elementExists('named', ['link', 'Revert'], $row3); + $this->assertSession()->elementExists('named', ['link', 'Delete'], $row3); + $this->assertSession()->elementTextNotContains('css', 'table tbody tr:nth-child(2)', 'Current revision'); + $this->assertSession()->elementTextContains('css', 'table tbody tr:nth-child(3)', 'first revision log'); + $this->assertSession()->elementTextContains('css', 'table tbody tr:nth-child(3)', '06/01/2020 - 07:00 by first author'); + } + +} diff --git a/core/modules/block_content/tests/src/Kernel/BlockContentAccessHandlerTest.php b/core/modules/block_content/tests/src/Kernel/BlockContentAccessHandlerTest.php index 62f63b5f7ad4..1db8a0e220a2 100644 --- a/core/modules/block_content/tests/src/Kernel/BlockContentAccessHandlerTest.php +++ b/core/modules/block_content/tests/src/Kernel/BlockContentAccessHandlerTest.php @@ -6,7 +6,10 @@ use Drupal\block_content\Entity\BlockContent; use Drupal\block_content\Entity\BlockContentType; use Drupal\Core\Access\AccessibleInterface; -use Drupal\Core\Access\AccessResult; +use Drupal\Core\Access\AccessResultAllowed; +use Drupal\Core\Access\AccessResultForbidden; +use Drupal\Core\Access\AccessResultNeutral; +use Drupal\Core\Access\AccessResultReasonInterface; use Drupal\KernelTests\KernelTestBase; use Drupal\user\Entity\Role; use Drupal\user\Entity\User; @@ -97,11 +100,50 @@ protected function setUp(): void { } /** + * Test block content entity access. + * + * @param string $operation + * The entity operation to test. + * @param bool $published + * Whether the latest revision should be published. + * @param bool $reusable + * Whether the block content should be reusable. Non-reusable blocks are + * typically used in Layout Builder. + * @param array $permissions + * Permissions to grant to the test user. + * @param bool $isLatest + * Whether the block content should be the latest revision when checking + * access. If FALSE, multiple revisions will be created, and an older + * revision will be loaded before checking access. + * @param string|null $parent_access + * Whether the test user has access to the parent entity, valid values are + * class names of classes implementing AccessResultInterface. Set to NULL to + * assert parent will not be called. + * @param string $expected_access + * The expected access for the user and block content. Valid values are + * class names of classes implementing AccessResultInterface + * @param string|null $expected_access_message + * The expected access message. + * * @covers ::checkAccess * * @dataProvider providerTestAccess + * + * @phpstan-param class-string<\Drupal\Core\Access\AccessResultInterface>|null $parent_access + * @phpstan-param class-string<\Drupal\Core\Access\AccessResultInterface> $expected_access */ - public function testAccess($operation, $published, $reusable, $permissions, $parent_access, $expected_access) { + public function testAccess(string $operation, bool $published, bool $reusable, array $permissions, bool $isLatest, ?string $parent_access, string $expected_access, ?string $expected_access_message = NULL) { + /** @var \Drupal\Core\Entity\RevisionableStorageInterface $entityStorage */ + $entityStorage = \Drupal::entityTypeManager()->getStorage('block_content'); + + $loadRevisionId = NULL; + if (!$isLatest) { + // Save a historical revision, then setup for a new revision to be saved. + $this->blockEntity->save(); + $loadRevisionId = $this->blockEntity->getRevisionId(); + $this->blockEntity = $entityStorage->createRevision($this->blockEntity); + } + $published ? $this->blockEntity->setPublished() : $this->blockEntity->setUnpublished(); $reusable ? $this->blockEntity->setReusable() : $this->blockEntity->setNonReusable(); @@ -119,22 +161,9 @@ public function testAccess($operation, $published, $reusable, $permissions, $par $user->addRole($this->role->id()); $user->save(); - if ($parent_access) { + if ($parent_access !== NULL) { $parent_entity = $this->prophesize(AccessibleInterface::class); - $expected_parent_result = NULL; - switch ($parent_access) { - case 'allowed': - $expected_parent_result = AccessResult::allowed(); - break; - - case 'neutral': - $expected_parent_result = AccessResult::neutral(); - break; - - case 'forbidden': - $expected_parent_result = AccessResult::forbidden(); - break; - } + $expected_parent_result = new ($parent_access)(); $parent_entity->access($operation, $user, TRUE) ->willReturn($expected_parent_result) ->shouldBeCalled(); @@ -144,125 +173,131 @@ public function testAccess($operation, $published, $reusable, $permissions, $par } $this->blockEntity->save(); - $result = $this->accessControlHandler->access($this->blockEntity, $operation, $user, TRUE); - switch ($expected_access) { - case 'allowed': - $this->assertTrue($result->isAllowed()); - break; - - case 'forbidden': - $this->assertTrue($result->isForbidden()); - break; - - case 'neutral': - $this->assertTrue($result->isNeutral()); - break; + // Reload a previous revision. + if ($loadRevisionId !== NULL) { + $this->blockEntity = $entityStorage->loadRevision($loadRevisionId); + } - default: - $this->fail('Unexpected access type'); + $result = $this->accessControlHandler->access($this->blockEntity, $operation, $user, TRUE); + $this->assertInstanceOf($expected_access, $result); + if ($expected_access_message !== NULL) { + $this->assertInstanceOf(AccessResultReasonInterface::class, $result); + $this->assertEquals($expected_access_message, $result->getReason()); } } /** * Data provider for testAccess(). */ - public function providerTestAccess() { + public function providerTestAccess(): array { $cases = [ 'view:published:reusable' => [ 'view', TRUE, TRUE, [], + TRUE, NULL, - 'allowed', + AccessResultAllowed::class, ], 'view:unpublished:reusable' => [ 'view', FALSE, TRUE, [], + TRUE, NULL, - 'neutral', + AccessResultNeutral::class, ], 'view:unpublished:reusable:admin' => [ 'view', FALSE, TRUE, ['administer blocks'], + TRUE, NULL, - 'allowed', + AccessResultAllowed::class, ], 'view:unpublished:reusable:per-block-editor:basic' => [ 'view', FALSE, TRUE, ['edit any basic block content'], + TRUE, NULL, - 'neutral', + AccessResultNeutral::class, ], 'view:unpublished:reusable:per-block-editor:square' => [ 'view', FALSE, TRUE, ['edit any square block content'], + TRUE, NULL, - 'allowed', + AccessResultAllowed::class, ], 'view:published:reusable:admin' => [ 'view', TRUE, TRUE, ['administer blocks'], + TRUE, NULL, - 'allowed', + AccessResultAllowed::class, ], 'view:published:reusable:per-block-editor:basic' => [ 'view', TRUE, TRUE, ['edit any basic block content'], + TRUE, NULL, - 'allowed', + AccessResultAllowed::class, ], 'view:published:reusable:per-block-editor:square' => [ 'view', TRUE, TRUE, ['edit any square block content'], + TRUE, NULL, - 'allowed', + AccessResultAllowed::class, ], 'view:published:non_reusable' => [ 'view', TRUE, FALSE, [], + TRUE, NULL, - 'forbidden', + AccessResultForbidden::class, ], 'view:published:non_reusable:parent_allowed' => [ 'view', TRUE, FALSE, [], - 'allowed', - 'allowed', + TRUE, + AccessResultAllowed::class, + AccessResultAllowed::class, ], 'view:published:non_reusable:parent_neutral' => [ 'view', TRUE, FALSE, [], - 'neutral', - 'neutral', + TRUE, + AccessResultNeutral::class, + AccessResultNeutral::class, ], 'view:published:non_reusable:parent_forbidden' => [ 'view', TRUE, FALSE, [], - 'forbidden', - 'forbidden', + TRUE, + AccessResultForbidden::class, + AccessResultForbidden::class, ], ]; foreach (['update', 'delete'] as $operation) { @@ -272,80 +307,90 @@ public function providerTestAccess() { TRUE, TRUE, [], + TRUE, NULL, - 'neutral', + AccessResultNeutral::class, ], $operation . ':unpublished:reusable' => [ $operation, FALSE, TRUE, [], + TRUE, NULL, - 'neutral', + AccessResultNeutral::class, ], $operation . ':unpublished:reusable:admin' => [ $operation, FALSE, TRUE, ['administer blocks'], + TRUE, NULL, - 'allowed', + AccessResultAllowed::class, ], $operation . ':published:reusable:admin' => [ $operation, TRUE, TRUE, ['administer blocks'], + TRUE, NULL, - 'allowed', + AccessResultAllowed::class, ], $operation . ':published:non_reusable' => [ $operation, TRUE, FALSE, [], + TRUE, NULL, - 'forbidden', + AccessResultForbidden::class, ], $operation . ':published:non_reusable:parent_allowed' => [ $operation, TRUE, FALSE, [], - 'allowed', - 'neutral', + TRUE, + AccessResultAllowed::class, + AccessResultNeutral::class, ], $operation . ':published:non_reusable:parent_neutral' => [ $operation, TRUE, FALSE, [], - 'neutral', - 'neutral', + TRUE, + AccessResultNeutral::class, + AccessResultNeutral::class, ], $operation . ':published:non_reusable:parent_forbidden' => [ $operation, TRUE, FALSE, [], - 'forbidden', - 'forbidden', + TRUE, + AccessResultForbidden::class, + AccessResultForbidden::class, ], $operation . ':unpublished:reusable:per-block-editor:basic' => [ $operation, FALSE, TRUE, ['edit any basic block content'], + TRUE, NULL, - 'neutral', + AccessResultNeutral::class, ], $operation . ':published:reusable:per-block-editor:basic' => [ $operation, TRUE, TRUE, ['edit any basic block content'], + TRUE, NULL, - 'neutral', + AccessResultNeutral::class, ], ]; } @@ -356,16 +401,18 @@ public function providerTestAccess() { FALSE, TRUE, ['edit any square block content'], + TRUE, NULL, - 'allowed', + AccessResultAllowed::class, ], 'update:published:reusable:per-block-editor:square' => [ 'update', TRUE, TRUE, ['edit any square block content'], + TRUE, NULL, - 'allowed', + AccessResultAllowed::class, ], ]; @@ -375,18 +422,182 @@ public function providerTestAccess() { FALSE, TRUE, ['edit any square block content'], + TRUE, NULL, - 'neutral', + AccessResultNeutral::class, ], 'delete:published:reusable:per-block-editor:square' => [ 'delete', TRUE, TRUE, ['edit any square block content'], + TRUE, NULL, - 'neutral', + AccessResultNeutral::class, ], ]; + + // View all revisions: + $cases['view all revisions:none'] = [ + 'view all revisions', + TRUE, + TRUE, + [], + TRUE, + NULL, + AccessResultNeutral::class, + ]; + $cases['view all revisions:administer blocks'] = [ + 'view all revisions', + TRUE, + TRUE, + ['administer blocks'], + TRUE, + NULL, + AccessResultAllowed::class, + ]; + $cases['view all revisions:view bundle'] = [ + 'view all revisions', + TRUE, + TRUE, + ['view any square block content history'], + TRUE, + NULL, + AccessResultAllowed::class, + ]; + + // Revert revisions: + $cases['revert:none:latest'] = [ + 'revert', + TRUE, + TRUE, + [], + TRUE, + NULL, + AccessResultForbidden::class, + ]; + $cases['revert:none:historical'] = [ + 'revert', + TRUE, + TRUE, + [], + FALSE, + NULL, + AccessResultNeutral::class, + ]; + $cases['revert:administer blocks:latest'] = [ + 'revert', + TRUE, + TRUE, + ['administer blocks'], + TRUE, + NULL, + AccessResultForbidden::class, + ]; + $cases['revert:administer blocks:historical'] = [ + 'revert', + TRUE, + TRUE, + ['administer blocks'], + FALSE, + NULL, + AccessResultAllowed::class, + ]; + $cases['revert:revert bundle:latest'] = [ + 'revert', + TRUE, + TRUE, + ['administer blocks'], + TRUE, + NULL, + AccessResultForbidden::class, + ]; + $cases['revert:revert bundle:historical'] = [ + 'revert', + TRUE, + TRUE, + ['revert any square block content revisions'], + FALSE, + NULL, + AccessResultAllowed::class, + ]; + $cases['revert:revert bundle:historical:non reusable'] = [ + 'revert', + TRUE, + FALSE, + ['revert any square block content revisions'], + FALSE, + NULL, + AccessResultForbidden::class, + 'Block content must be reusable to use `revert` operation', + ]; + + // Delete revisions: + $cases['delete revision:none:latest'] = [ + 'delete revision', + TRUE, + TRUE, + [], + TRUE, + NULL, + AccessResultForbidden::class, + ]; + $cases['delete revision:none:historical'] = [ + 'delete revision', + TRUE, + TRUE, + [], + FALSE, + NULL, + AccessResultNeutral::class, + ]; + $cases['delete revision:administer blocks:latest'] = [ + 'delete revision', + TRUE, + TRUE, + ['administer blocks'], + TRUE, + NULL, + AccessResultForbidden::class, + ]; + $cases['delete revision:administer blocks:historical'] = [ + 'delete revision', + TRUE, + TRUE, + ['administer blocks'], + FALSE, + NULL, + AccessResultAllowed::class, + ]; + $cases['delete revision:delete bundle:latest'] = [ + 'delete revision', + TRUE, + TRUE, + ['administer blocks'], + TRUE, + NULL, + AccessResultForbidden::class, + ]; + $cases['delete revision:delete bundle:historical'] = [ + 'delete revision', + TRUE, + TRUE, + ['delete any square block content revisions'], + FALSE, + NULL, + AccessResultAllowed::class, + ]; + $cases['delete revision:delete bundle:historical:non reusable'] = [ + 'delete revision', + TRUE, + FALSE, + ['delete any square block content revisions'], + FALSE, + NULL, + AccessResultForbidden::class, + 'Block content must be reusable to use `delete revision` operation', + ]; + return $cases; } -- GitLab