Skip to content
Snippets Groups Projects
Commit 16567a01 authored by Bojan Zivanovic's avatar Bojan Zivanovic
Browse files

Merge pull request #15 from dawehner/entity-revision-view-controller

Add a revision view controller /node/{node}/revisions/{revision_id}
parents a9b5308a 04246cf0
No related branches found
No related tags found
No related merge requests found
...@@ -9,3 +9,9 @@ services: ...@@ -9,3 +9,9 @@ services:
class: Drupal\entity\RouteEnhancer\EntityRevisionRouteEnhancer class: Drupal\entity\RouteEnhancer\EntityRevisionRouteEnhancer
tags: tags:
- { name: route_enhancer, priority: 20 } - { name: route_enhancer, priority: 20 }
access_checker.entity_revision:
class: \Drupal\entity\Access\EntityRevisionRouteAccessChecker
arguments: ['@entity_type.manager']
tags:
- { name: access_check, applies_to: _entity_access_revision, needs_request: TRUE }
...@@ -14,6 +14,7 @@ use Drupal\Core\Entity\EntityTypeManagerInterface; ...@@ -14,6 +14,7 @@ use Drupal\Core\Entity\EntityTypeManagerInterface;
use Drupal\Core\Entity\RevisionableInterface; use Drupal\Core\Entity\RevisionableInterface;
use Drupal\Core\Routing\Access\AccessInterface; use Drupal\Core\Routing\Access\AccessInterface;
use Drupal\Core\Session\AccountInterface; use Drupal\Core\Session\AccountInterface;
use Symfony\Component\HttpFoundation\Request;
use Symfony\Component\Routing\Route; use Symfony\Component\Routing\Route;
/** /**
...@@ -46,7 +47,8 @@ class EntityRevisionRouteAccessChecker implements AccessInterface { ...@@ -46,7 +47,8 @@ class EntityRevisionRouteAccessChecker implements AccessInterface {
/** /**
* {@inheritdoc} * {@inheritdoc}
*/ */
public function access(Route $route, AccountInterface $account, RevisionableInterface $_entity_revision) { public function access(Route $route, AccountInterface $account, Request $request) {
$_entity_revision = $request->attributes->get('_entity_revision');
$operation = $route->getRequirement('_entity_access_revision'); $operation = $route->getRequirement('_entity_access_revision');
list(, $operation) = explode('.', $operation, 2); list(, $operation) = explode('.', $operation, 2);
return AccessResult::allowedIf($_entity_revision && $this->checkAccess($_entity_revision, $account, $operation))->cachePerPermissions(); return AccessResult::allowedIf($_entity_revision && $this->checkAccess($_entity_revision, $account, $operation))->cachePerPermissions();
......
<?php
/**
* @file
* Contains \Drupal\entity\Controller\RevisionController.
*/
namespace Drupal\entity\Controller;
use Drupal\Core\Entity\Controller\EntityViewController;
use Drupal\Core\Entity\EntityInterface;
/**
* Provides some controllers related with entity revisions.
*/
class RevisionController extends EntityViewController {
/**
* Provides a page to render a single entity revision.
*
* @param \Drupal\Core\Entity\EntityInterface $_entity_revision
* The Entity to be rendered. Note this variable is named $_entity_revision
* rather than $entity to prevent collisions with other named placeholders
* in the route.
* @param string $view_mode
* (optional) The view mode that should be used to display the entity.
* Defaults to 'full'.
*
* @return array
* A render array.
*/
public function view(EntityInterface $_entity_revision, $view_mode = 'full') {
return parent::view($_entity_revision, $view_mode);
}
}
<?php
/**
* @file
* Contains \Drupal\entity\Routing\RevisionRouteProvider.
*/
namespace Drupal\entity\Routing;
use Drupal\Core\Entity\EntityTypeInterface;
use Drupal\Core\Entity\Routing\EntityRouteProviderInterface;
use Symfony\Component\Routing\Route;
use Symfony\Component\Routing\RouteCollection;
/**
* Provides revision routes.
*/
class RevisionRouteProvider implements EntityRouteProviderInterface {
/**
* {@inheritdoc}
*/
public function getRoutes(EntityTypeInterface $entity_type) {
$collection = new RouteCollection();
$entity_type_id = $entity_type->id();
if ($view_route = $this->getRevisionViewRoute($entity_type)) {
$collection->add("entity.$entity_type_id.revision", $view_route);
}
return $collection;
}
/**
* Gets the entity revision view route.
*
* @param \Drupal\Core\Entity\EntityTypeInterface $entity_type
* The entity type.
*
* @return \Symfony\Component\Routing\Route|null
* The generated route, if available.
*/
protected function getRevisionViewRoute(EntityTypeInterface $entity_type) {
if ($entity_type->hasLinkTemplate('revision')) {
$entity_type_id = $entity_type->id();
$route = new Route($entity_type->getLinkTemplate('revision'));
$route->addDefaults([
'_controller' => '\Drupal\entity\Controller\RevisionController::view',
'_title_callback' => '\Drupal\Core\Entity\Controller\EntityController::title',
]);
$route->addRequirements([
'_entity_access_revision' => "$entity_type_id.view",
]);
$route->setOption('parameters', [
$entity_type->id() => [
'type' => 'entity:' . $entity_type->id(),
],
$entity_type->id() . '_revision' => [
'type' => 'entity_revision:' . $entity_type->id(),
],
]);
return $route;
}
}
}
...@@ -7,7 +7,7 @@ ...@@ -7,7 +7,7 @@
namespace Drupal\Tests\entity\Kernel; namespace Drupal\Tests\entity\Kernel;
use Drupal\entity_module_test\Entity\EntityWithRevisionLog; use Drupal\entity_module_test\Entity\EnhancedEntity;
use Drupal\KernelTests\KernelTestBase; use Drupal\KernelTests\KernelTestBase;
use Drupal\user\Entity\User; use Drupal\user\Entity\User;
...@@ -42,7 +42,7 @@ class EntityRevisionLogTraitTest extends KernelTestBase { ...@@ -42,7 +42,7 @@ class EntityRevisionLogTraitTest extends KernelTestBase {
$user2->save(); $user2->save();
/** @var \Drupal\entity\Revision\EntityRevisionLogInterface $entity */ /** @var \Drupal\entity\Revision\EntityRevisionLogInterface $entity */
$entity = EntityWithRevisionLog::create([ $entity = EnhancedEntity::create([
'revision_user' => $user->id(), 'revision_user' => $user->id(),
'revision_created' => 1447941735, 'revision_created' => 1447941735,
'revision_log_message' => 'Test message', 'revision_log_message' => 'Test message',
......
<?php
/**
* @file
* Contains \Drupal\Tests\entity\Kernel\RevisionBasicUITest.
*/
namespace Drupal\Tests\entity\Kernel;
use Drupal\entity_module_test\Entity\EnhancedEntity;
use Drupal\KernelTests\KernelTestBase;
use Drupal\user\Entity\Role;
use Drupal\user\Entity\User;
use Symfony\Component\HttpFoundation\Request;
/**
* @group entity
*/
class RevisionBasicUITest extends KernelTestBase {
/**
* {@inheritdoc}
*/
public static $modules = ['entity_module_test', 'system', 'user', 'entity'];
/**
* {@inheritdoc}
*/
protected function setUp() {
parent::setUp();
$this->installEntitySchema('user');
$this->installEntitySchema('entity_test_enhanced');
$this->installSchema('system', 'router');
\Drupal::service('router.builder')->rebuild();
}
public function testRevisionView() {
$entity = EnhancedEntity::create([
'name' => 'rev 1',
]);
$entity->save();
$revision = clone $entity;
$revision->name->value = 'rev 2';
$revision->setNewRevision(TRUE);
$revision->isDefaultRevision(FALSE);
$revision->save();
/** @var \Symfony\Component\HttpKernel\HttpKernelInterface $http_kernel */
$http_kernel = \Drupal::service('http_kernel');
$request = Request::create($revision->url('revision'));
$response = $http_kernel->handle($request);
$this->assertEquals(403, $response->getStatusCode());
$role = Role::create(['id' => 'test_role']);
$role->grantPermission('view all entity_test_enhanced revisions');
$role->grantPermission('administer entity_test_enhanced');
$role->save();
$user = User::create([
'name' => 'Test user',
]);
$user->addRole($role->id());
\Drupal::service('account_switcher')->switchTo($user);
$request = Request::create($revision->url('revision'));
$response = $http_kernel->handle($request);
$this->assertEquals(200, $response->getStatusCode());
$this->assertNotContains('rev 1', $response->getContent());
$this->assertContains('rev 2', $response->getContent());
}
}
...@@ -2,7 +2,7 @@ ...@@ -2,7 +2,7 @@
/** /**
* @file * @file
* Contains \Drupal\entity_module_test\Entity\EntityWithRevisionLog. * Contains \Drupal\entity_module_test\Entity\EnhancedEntity.
*/ */
namespace Drupal\entity_module_test\Entity; namespace Drupal\entity_module_test\Entity;
...@@ -14,22 +14,35 @@ use Drupal\entity\EntityKeysFieldsTrait; ...@@ -14,22 +14,35 @@ use Drupal\entity\EntityKeysFieldsTrait;
use Drupal\entity\Revision\EntityRevisionLogTrait; use Drupal\entity\Revision\EntityRevisionLogTrait;
/** /**
* Provides a test entity which uses all the capabilities of entity module.
*
* @ContentEntityType( * @ContentEntityType(
* id = "entity_test__revision_log", * id = "entity_test_enhanced",
* label = @Translation("Entity test with revision log"), * label = @Translation("Entity test with enhancements"),
* handlers = { * handlers = {
* "storage" = "\Drupal\Core\Entity\Sql\SqlContentEntityStorage", * "storage" = "\Drupal\Core\Entity\Sql\SqlContentEntityStorage",
* "route_provider" = {
* "revision" = "\Drupal\entity\Routing\RevisionRouteProvider",
* },
* }, * },
* base_table = "entity_test_enhanced",
* data_table = "entity_test_enhanced_field_data",
* revision_table = "entity_test_enhanced_revision",
* revision_data_table = "entity_test_enhanced_field_revision",
* translatable = TRUE, * translatable = TRUE,
* revisionable = TRUE, * revisionable = TRUE,
* admin_permission = "administer entity_test_enhanced",
* entity_keys = { * entity_keys = {
* "id" = "id", * "id" = "id",
* "revision" = "vid", * "revision" = "vid",
* "langcode" = "langcode", * "langcode" = "langcode",
* },
* links = {
* "revision" = "/entity_test_enhanced/{entity_test_enhanced}/revisions/{entity_test_enhanced_revision}/view",
* } * }
* ) * )
*/ */
class EntityWithRevisionLog extends ContentEntityBase { class EnhancedEntity extends ContentEntityBase {
use EntityRevisionLogTrait; use EntityRevisionLogTrait;
use EntityKeysFieldsTrait; use EntityKeysFieldsTrait;
...@@ -43,6 +56,15 @@ class EntityWithRevisionLog extends ContentEntityBase { ...@@ -43,6 +56,15 @@ class EntityWithRevisionLog extends ContentEntityBase {
$fields += static::entityKeysBaseFieldDefinitions($entity_type); $fields += static::entityKeysBaseFieldDefinitions($entity_type);
$fields += static::entityRevisionLogBaseFieldDefinitions(); $fields += static::entityRevisionLogBaseFieldDefinitions();
$fields['name'] = BaseFieldDefinition::create('string')
->setLabel('Name')
->setRevisionable(TRUE)
->setDisplayOptions('view', [
'label' => 'hidden',
'type' => 'string',
'weight' => -5,
]);
return $fields; return $fields;
} }
......
0% Loading or .
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment