class DefaultMenuLinkTreeManipulatorsTest

Same name and namespace in other branches
  1. 8.9.x core/tests/Drupal/Tests/Core/Menu/DefaultMenuLinkTreeManipulatorsTest.php \Drupal\Tests\Core\Menu\DefaultMenuLinkTreeManipulatorsTest
  2. 10 core/tests/Drupal/Tests/Core/Menu/DefaultMenuLinkTreeManipulatorsTest.php \Drupal\Tests\Core\Menu\DefaultMenuLinkTreeManipulatorsTest
  3. 11.x core/tests/Drupal/Tests/Core/Menu/DefaultMenuLinkTreeManipulatorsTest.php \Drupal\Tests\Core\Menu\DefaultMenuLinkTreeManipulatorsTest

Tests the default menu link tree manipulators.

@group Menu

@coversDefaultClass \Drupal\Core\Menu\DefaultMenuLinkTreeManipulators

Hierarchy

Expanded class hierarchy of DefaultMenuLinkTreeManipulatorsTest

File

core/tests/Drupal/Tests/Core/Menu/DefaultMenuLinkTreeManipulatorsTest.php, line 22

Namespace

Drupal\Tests\Core\Menu
View source
class DefaultMenuLinkTreeManipulatorsTest extends UnitTestCase {
    
    /**
     * The mocked access manager.
     *
     * @var \Drupal\Core\Access\AccessManagerInterface|\PHPUnit\Framework\MockObject\MockObject
     */
    protected $accessManager;
    
    /**
     * The mocked current user.
     *
     * @var \Drupal\Core\Session\AccountInterface|\PHPUnit\Framework\MockObject\MockObject
     */
    protected $currentUser;
    
    /**
     * The mocked entity type manager.
     *
     * @var \Drupal\Core\Entity\EntityTypeManagerInterface|\PHPUnit\Framework\MockObject\MockObject
     */
    protected $entityTypeManager;
    
    /**
     * The default menu link tree manipulators.
     *
     * @var \Drupal\Core\Menu\DefaultMenuLinkTreeManipulators
     */
    protected $defaultMenuTreeManipulators;
    
    /**
     * Mock cache context manager.
     *
     * @var \Drupal\Core\Cache\Context\CacheContextsManager|\Prophecy\Prophecy\ObjectProphecy
     */
    protected $cacheContextManager;
    
    /**
     * The original menu tree build in mockTree().
     *
     * @var \Drupal\Core\Menu\MenuLinkTreeElement[]
     */
    protected $originalTree = [];
    
    /**
     * Array of menu link instances.
     *
     * @var \Drupal\Core\Menu\MenuLinkInterface[]
     */
    protected $links = [];
    
    /**
     * {@inheritdoc}
     */
    protected function setUp() : void {
        parent::setUp();
        $this->accessManager = $this->createMock('\\Drupal\\Core\\Access\\AccessManagerInterface');
        $this->currentUser = $this->createMock('Drupal\\Core\\Session\\AccountInterface');
        $this->currentUser
            ->method('isAuthenticated')
            ->willReturn(TRUE);
        $this->entityTypeManager = $this->createMock(EntityTypeManagerInterface::class);
        $this->defaultMenuTreeManipulators = new DefaultMenuLinkTreeManipulators($this->accessManager, $this->currentUser, $this->entityTypeManager);
        $this->cacheContextManager = $this->prophesize(CacheContextsManager::class);
        $container = new Container();
        $container->set('cache_contexts_manager', $this->cacheContextManager
            ->reveal());
        \Drupal::setContainer($container);
    }
    
    /**
     * Creates a mock tree.
     *
     * This mocks a tree with the following structure:
     * - 1
     * - 2
     *   - 3
     *     - 4
     * - 5
     *   - 7
     * - 6
     * - 8
     * - 9
     *
     * With link 6 being the only external link.
     */
    protected function mockTree() {
        $this->links = [
            1 => MenuLinkMock::create([
                'id' => 'test.example1',
                'route_name' => 'example1',
                'title' => 'foo',
                'parent' => '',
            ]),
            2 => MenuLinkMock::create([
                'id' => 'test.example2',
                'route_name' => 'example2',
                'title' => 'bar',
                'parent' => 'test.example1',
                'route_parameters' => [
                    'foo' => 'bar',
                ],
            ]),
            3 => MenuLinkMock::create([
                'id' => 'test.example3',
                'route_name' => 'example3',
                'title' => 'baz',
                'parent' => 'test.example2',
                'route_parameters' => [
                    'baz' => 'qux',
                ],
            ]),
            4 => MenuLinkMock::create([
                'id' => 'test.example4',
                'route_name' => 'example4',
                'title' => 'qux',
                'parent' => 'test.example3',
            ]),
            5 => MenuLinkMock::create([
                'id' => 'test.example5',
                'route_name' => 'example5',
                'title' => 'foofoo',
                'parent' => '',
            ]),
            6 => MenuLinkMock::create([
                'id' => 'test.example6',
                'route_name' => '',
                'url' => 'https://www.drupal.org/',
                'title' => 'barbar',
                'parent' => '',
            ]),
            7 => MenuLinkMock::create([
                'id' => 'test.example7',
                'route_name' => 'example7',
                'title' => 'bazbaz',
                'parent' => '',
            ]),
            8 => MenuLinkMock::create([
                'id' => 'test.example8',
                'route_name' => 'example8',
                'title' => 'quxqux',
                'parent' => '',
            ]),
            9 => DynamicMenuLinkMock::create([
                'id' => 'test.example9',
                'parent' => '',
            ])->setCurrentUser($this->currentUser),
        ];
        $this->originalTree = [];
        $this->originalTree[1] = new MenuLinkTreeElement($this->links[1], FALSE, 1, FALSE, []);
        $this->originalTree[2] = new MenuLinkTreeElement($this->links[2], TRUE, 1, FALSE, [
            3 => new MenuLinkTreeElement($this->links[3], TRUE, 2, FALSE, [
                4 => new MenuLinkTreeElement($this->links[4], FALSE, 3, FALSE, []),
            ]),
        ]);
        $this->originalTree[5] = new MenuLinkTreeElement($this->links[5], TRUE, 1, FALSE, [
            7 => new MenuLinkTreeElement($this->links[7], FALSE, 2, FALSE, []),
        ]);
        $this->originalTree[6] = new MenuLinkTreeElement($this->links[6], FALSE, 1, FALSE, []);
        $this->originalTree[8] = new MenuLinkTreeElement($this->links[8], FALSE, 1, FALSE, []);
        $this->originalTree[9] = new MenuLinkTreeElement($this->links[9], FALSE, 1, FALSE, []);
    }
    
    /**
     * Tests the generateIndexAndSort() tree manipulator.
     *
     * @covers ::generateIndexAndSort
     */
    public function testGenerateIndexAndSort() {
        $this->mockTree();
        $tree = $this->originalTree;
        $tree = $this->defaultMenuTreeManipulators
            ->generateIndexAndSort($tree);
        // Validate that parent elements #1, #2, #5 and #6 exist on the root level.
        $this->assertEquals($this->links[1]
            ->getPluginId(), $tree['50000 foo test.example1']->link
            ->getPluginId());
        $this->assertEquals($this->links[2]
            ->getPluginId(), $tree['50000 bar test.example2']->link
            ->getPluginId());
        $this->assertEquals($this->links[5]
            ->getPluginId(), $tree['50000 foofoo test.example5']->link
            ->getPluginId());
        $this->assertEquals($this->links[6]
            ->getPluginId(), $tree['50000 barbar test.example6']->link
            ->getPluginId());
        $this->assertEquals($this->links[8]
            ->getPluginId(), $tree['50000 quxqux test.example8']->link
            ->getPluginId());
        // Verify that child element #4 is at the correct location in the hierarchy.
        $this->assertEquals($this->links[4]
            ->getPluginId(), $tree['50000 bar test.example2']->subtree['50000 baz test.example3']->subtree['50000 qux test.example4']->link
            ->getPluginId());
        // Verify that child element #7 is at the correct location in the hierarchy.
        $this->assertEquals($this->links[7]
            ->getPluginId(), $tree['50000 foofoo test.example5']->subtree['50000 bazbaz test.example7']->link
            ->getPluginId());
    }
    
    /**
     * Tests the checkAccess() tree manipulator.
     *
     * @covers ::checkAccess
     * @covers ::menuLinkCheckAccess
     */
    public function testCheckAccess() {
        // Those menu links that are non-external will have their access checks
        // performed. 9 routes, but 1 is external, 2 already have their 'access'
        // property set, and 1 is a child if an inaccessible menu link, so only 5
        // calls will be made.
        $this->cacheContextManager
            ->assertValidTokens([
            'user.permissions',
        ])
            ->shouldBeCalled()
            ->willReturn(TRUE);
        $this->accessManager
            ->expects($this->exactly(5))
            ->method('checkNamedRoute')
            ->willReturnMap([
            [
                'example1',
                [],
                $this->currentUser,
                TRUE,
                AccessResult::forbidden(),
            ],
            [
                'example2',
                [
                    'foo' => 'bar',
                ],
                $this->currentUser,
                TRUE,
                AccessResult::allowed()->cachePerPermissions(),
            ],
            [
                'example3',
                [
                    'baz' => 'qux',
                ],
                $this->currentUser,
                TRUE,
                AccessResult::neutral(),
            ],
            [
                'example5',
                [],
                $this->currentUser,
                TRUE,
                AccessResult::allowed(),
            ],
            [
                'user.logout',
                [],
                $this->currentUser,
                TRUE,
                AccessResult::allowed(),
            ],
        ]);
        $this->mockTree();
        $this->originalTree[5]->subtree[7]->access = AccessResult::neutral();
        $this->cacheContextManager
            ->assertValidTokens([
            'user',
        ])
            ->shouldBeCalled()
            ->willReturn(TRUE);
        $this->originalTree[8]->access = AccessResult::allowed()->cachePerUser();
        // Since \Drupal\Core\Menu\DefaultMenuLinkTreeManipulators::checkAccess()
        // allows access to any link if the user has the 'link to any page'
        // permission, *every* single access result is varied by permissions.
        $tree = $this->defaultMenuTreeManipulators
            ->checkAccess($this->originalTree);
        // Menu link 1: route without parameters, access forbidden, but at level 0,
        // hence kept.
        $element = $tree[1];
        $this->assertEquals(AccessResult::forbidden()->cachePerPermissions(), $element->access);
        $this->assertInstanceOf('\\Drupal\\Core\\Menu\\InaccessibleMenuLink', $element->link);
        // Menu link 2: route with parameters, access granted.
        $element = $tree[2];
        $this->assertEquals(AccessResult::allowed()->cachePerPermissions(), $element->access);
        $this->assertNotInstanceOf('\\Drupal\\Core\\Menu\\InaccessibleMenuLink', $element->link);
        // Menu link 3: route with parameters, AccessResult::neutral(), top-level
        // inaccessible link, hence kept for its cacheability metadata.
        // Note that the permissions cache context is added automatically, because
        // we always check the "link to any page" permission.
        $element = $tree[2]->subtree[3];
        $this->assertEquals(AccessResult::neutral()->cachePerPermissions(), $element->access);
        $this->assertInstanceOf('\\Drupal\\Core\\Menu\\InaccessibleMenuLink', $element->link);
        // Menu link 4: child of menu link 3, which was AccessResult::neutral(),
        // hence menu link 3's subtree is removed, of which this menu link is one.
        $this->assertArrayNotHasKey(4, $tree[2]->subtree[3]->subtree);
        // Menu link 5: no route name, treated as external, hence access granted.
        $element = $tree[5];
        $this->assertEquals(AccessResult::allowed()->cachePerPermissions(), $element->access);
        $this->assertNotInstanceOf('\\Drupal\\Core\\Menu\\InaccessibleMenuLink', $element->link);
        // Menu link 6: external URL, hence access granted.
        $element = $tree[6];
        $this->assertEquals(AccessResult::allowed()->cachePerPermissions(), $element->access);
        $this->assertNotInstanceOf('\\Drupal\\Core\\Menu\\InaccessibleMenuLink', $element->link);
        // Menu link 7: 'access' already set: AccessResult::neutral(), top-level
        // inaccessible link, hence kept for its cacheability metadata.
        // Note that unlike for menu link 3, the permission cache context is absent,
        // because ::checkAccess() doesn't perform access checking when 'access' is
        // already set.
        $element = $tree[5]->subtree[7];
        $this->assertEquals(AccessResult::neutral(), $element->access);
        $this->assertInstanceOf('\\Drupal\\Core\\Menu\\InaccessibleMenuLink', $element->link);
        // Menu link 8: 'access' already set, note that 'per permissions' caching
        // is not added.
        $element = $tree[8];
        $this->assertEquals(AccessResult::allowed()->cachePerUser(), $element->access);
        $this->assertNotInstanceOf('\\Drupal\\Core\\Menu\\InaccessibleMenuLink', $element->link);
    }
    
    /**
     * Tests checkAccess() tree manipulator with 'link to any page' permission.
     *
     * @covers ::checkAccess
     * @covers ::menuLinkCheckAccess
     */
    public function testCheckAccessWithLinkToAnyPagePermission() {
        $this->mockTree();
        $this->currentUser
            ->expects($this->exactly(9))
            ->method('hasPermission')
            ->with('link to any page')
            ->willReturn(TRUE);
        $this->mockTree();
        $this->cacheContextManager
            ->assertValidTokens([
            'user.permissions',
        ])
            ->shouldBeCalled()
            ->willReturn(TRUE);
        $this->defaultMenuTreeManipulators
            ->checkAccess($this->originalTree);
        $expected_access_result = AccessResult::allowed()->cachePerPermissions();
        $this->assertEquals($expected_access_result, $this->originalTree[1]->access);
        $this->assertEquals($expected_access_result, $this->originalTree[2]->access);
        $this->assertEquals($expected_access_result, $this->originalTree[2]->subtree[3]->access);
        $this->assertEquals($expected_access_result, $this->originalTree[2]->subtree[3]->subtree[4]->access);
        $this->assertEquals($expected_access_result, $this->originalTree[5]->subtree[7]->access);
        $this->assertEquals($expected_access_result, $this->originalTree[6]->access);
        $this->assertEquals($expected_access_result, $this->originalTree[8]->access);
        $this->assertEquals($expected_access_result, $this->originalTree[9]->access);
    }
    
    /**
     * Tests the flatten() tree manipulator.
     *
     * @covers ::flatten
     */
    public function testFlatten() {
        $this->mockTree();
        $tree = $this->defaultMenuTreeManipulators
            ->flatten($this->originalTree);
        $this->assertEquals([
            1,
            2,
            5,
            6,
            8,
            9,
        ], array_keys($this->originalTree));
        $this->assertEquals([
            1,
            2,
            5,
            6,
            8,
            9,
            3,
            4,
            7,
        ], array_keys($tree));
    }
    
    /**
     * Tests the optimized node access checking.
     *
     * @covers ::checkNodeAccess
     * @covers ::collectNodeLinks
     * @covers ::checkAccess
     */
    public function testCheckNodeAccess() {
        $links = [
            1 => MenuLinkMock::create([
                'id' => 'node.1',
                'route_name' => 'entity.node.canonical',
                'title' => 'foo',
                'parent' => '',
                'route_parameters' => [
                    'node' => 1,
                ],
            ]),
            2 => MenuLinkMock::create([
                'id' => 'node.2',
                'route_name' => 'entity.node.canonical',
                'title' => 'bar',
                'parent' => '',
                'route_parameters' => [
                    'node' => 2,
                ],
            ]),
            3 => MenuLinkMock::create([
                'id' => 'node.3',
                'route_name' => 'entity.node.canonical',
                'title' => 'baz',
                'parent' => 'node.2',
                'route_parameters' => [
                    'node' => 3,
                ],
            ]),
            4 => MenuLinkMock::create([
                'id' => 'node.4',
                'route_name' => 'entity.node.canonical',
                'title' => 'qux',
                'parent' => 'node.3',
                'route_parameters' => [
                    'node' => 4,
                ],
            ]),
            5 => MenuLinkMock::create([
                'id' => 'test.1',
                'route_name' => 'test_route',
                'title' => 'qux',
                'parent' => '',
            ]),
            6 => MenuLinkMock::create([
                'id' => 'test.2',
                'route_name' => 'test_route',
                'title' => 'qux',
                'parent' => 'test.1',
            ]),
        ];
        $tree = [];
        $tree[1] = new MenuLinkTreeElement($links[1], FALSE, 1, FALSE, []);
        $tree[2] = new MenuLinkTreeElement($links[2], TRUE, 1, FALSE, [
            3 => new MenuLinkTreeElement($links[3], TRUE, 2, FALSE, [
                4 => new MenuLinkTreeElement($links[4], FALSE, 3, FALSE, []),
            ]),
        ]);
        $tree[5] = new MenuLinkTreeElement($links[5], TRUE, 1, FALSE, [
            6 => new MenuLinkTreeElement($links[6], FALSE, 2, FALSE, []),
        ]);
        $query = $this->createMock('Drupal\\Core\\Entity\\Query\\QueryInterface');
        $query->expects($this->once())
            ->method('accessCheck')
            ->with(TRUE);
        $query->expects($this->exactly(2))
            ->method('condition')
            ->withConsecutive([
            'nid',
            [
                1,
                2,
                3,
                4,
            ],
        ], [
            'status',
            NodeInterface::PUBLISHED,
        ]);
        $query->expects($this->once())
            ->method('execute')
            ->willReturn([
            1,
            2,
            4,
        ]);
        $storage = $this->createMock(EntityStorageInterface::class);
        $storage->expects($this->once())
            ->method('getQuery')
            ->willReturn($query);
        $this->entityTypeManager
            ->expects($this->once())
            ->method('getStorage')
            ->with('node')
            ->willReturn($storage);
        $this->cacheContextManager
            ->assertValidTokens([
            'user.permissions',
        ])
            ->shouldBeCalled()
            ->willReturn(TRUE);
        $this->cacheContextManager
            ->assertValidTokens([
            'user.permissions',
            'user.node_grants:view',
        ])
            ->shouldBeCalled()
            ->willReturn(TRUE);
        $node_access_result = AccessResult::allowed()->cachePerPermissions()
            ->addCacheContexts([
            'user.node_grants:view',
        ]);
        $tree = $this->defaultMenuTreeManipulators
            ->checkNodeAccess($tree);
        $this->assertEquals($node_access_result, $tree[1]->access);
        $this->assertEquals($node_access_result, $tree[2]->access);
        // Ensure that access denied is set.
        $this->assertEquals(AccessResult::neutral(), $tree[2]->subtree[3]->access);
        $this->assertEquals($node_access_result, $tree[2]->subtree[3]->subtree[4]->access);
        // Ensure that other routes than entity.node.canonical are set as well.
        $this->assertNull($tree[5]->access);
        $this->assertNull($tree[5]->subtree[6]->access);
        // On top of the node access checking now run the ordinary route based
        // access checkers.
        // Ensure that the access manager is just called for the non-node routes.
        $this->accessManager
            ->expects($this->exactly(2))
            ->method('checkNamedRoute')
            ->with('test_route', [], $this->currentUser, TRUE)
            ->willReturnOnConsecutiveCalls(AccessResult::allowed(), AccessResult::neutral());
        $tree = $this->defaultMenuTreeManipulators
            ->checkAccess($tree);
        $this->assertEquals($node_access_result, $tree[1]->access);
        $this->assertEquals($node_access_result, $tree[2]->access);
        $this->assertEquals(AccessResult::neutral(), $tree[2]->subtree[3]->access);
        $this->assertEquals(AccessResult::allowed()->cachePerPermissions(), $tree[5]->access);
        $this->assertEquals(AccessResult::neutral()->cachePerPermissions(), $tree[5]->subtree[6]->access);
    }

}

Members

Title Sort descending Deprecated Modifiers Object type Summary Overriden Title Overrides
DefaultMenuLinkTreeManipulatorsTest::$accessManager protected property The mocked access manager.
DefaultMenuLinkTreeManipulatorsTest::$cacheContextManager protected property Mock cache context manager.
DefaultMenuLinkTreeManipulatorsTest::$currentUser protected property The mocked current user.
DefaultMenuLinkTreeManipulatorsTest::$defaultMenuTreeManipulators protected property The default menu link tree manipulators.
DefaultMenuLinkTreeManipulatorsTest::$entityTypeManager protected property The mocked entity type manager.
DefaultMenuLinkTreeManipulatorsTest::$links protected property Array of menu link instances.
DefaultMenuLinkTreeManipulatorsTest::$originalTree protected property The original menu tree build in mockTree().
DefaultMenuLinkTreeManipulatorsTest::mockTree protected function Creates a mock tree.
DefaultMenuLinkTreeManipulatorsTest::setUp protected function Overrides UnitTestCase::setUp
DefaultMenuLinkTreeManipulatorsTest::testCheckAccess public function Tests the checkAccess() tree manipulator.
DefaultMenuLinkTreeManipulatorsTest::testCheckAccessWithLinkToAnyPagePermission public function Tests checkAccess() tree manipulator with 'link to any page' permission.
DefaultMenuLinkTreeManipulatorsTest::testCheckNodeAccess public function Tests the optimized node access checking.
DefaultMenuLinkTreeManipulatorsTest::testFlatten public function Tests the flatten() tree manipulator.
DefaultMenuLinkTreeManipulatorsTest::testGenerateIndexAndSort public function Tests the generateIndexAndSort() tree manipulator.
PhpUnitWarnings::$deprecationWarnings private static property Deprecation warnings from PHPUnit to raise with @trigger_error().
PhpUnitWarnings::addWarning public function Converts PHPUnit deprecation warnings to E_USER_DEPRECATED.
UnitTestCase::$randomGenerator protected property The random generator.
UnitTestCase::$root protected property The app root. 1
UnitTestCase::assertArrayEquals Deprecated protected function Asserts if two arrays are equal by sorting them first.
UnitTestCase::getClassResolverStub protected function Returns a stub class resolver.
UnitTestCase::getConfigFactoryStub public function Returns a stub config factory that behaves according to the passed array.
UnitTestCase::getConfigStorageStub public function Returns a stub config storage that returns the supplied configuration.
UnitTestCase::getContainerWithCacheTagsInvalidator protected function Sets up a container with a cache tags invalidator.
UnitTestCase::getRandomGenerator protected function Gets the random generator for the utility methods.
UnitTestCase::getStringTranslationStub public function Returns a stub translation manager that just returns the passed string.
UnitTestCase::randomMachineName public function Generates a unique random string containing letters and numbers.
UnitTestCase::setUpBeforeClass public static function

Buggy or inaccurate documentation? Please file an issue. Need support? Need help programming? Connect with the Drupal community.