False access-denied-link with some links of Taxonomies

Created on 13 March 2025, 23 days ago

Problem/Motivation

The links to taxonomy term pages are classified as access-denied links when they do not have an alias. If they have an alias, the entity is correctly identified, and the access check returns true.

The problem is that if there is no alias, the URL follows the structure "/taxonomy/term/1", which identifies the view instead of the entity.

How to reproduce

Create taxonomy terms, some with a set alias and others without an alias. Generate links in the body of a node that refer to these taxonomies. Process the nodes with Entity Mesh, for example, by running the command:
drush entity-mesh:generate node

🐛 Bug report
Status

Active

Version

1.0

Component

Code

Created by

🇪🇸Spain lpeidro Madrid

Live updates comments and jobs are added and updated live.
Sign in to follow issues

Merge Requests

Comments & Activities

  • Issue created by @lpeidro
  • 🇪🇸Spain lpeidro Madrid

    The problem was in this part of the code in the class EntityRender:

      protected function setDataTargetFromRoute($target) {
        if (empty($target->getPath())) {
          return;
        }
    
        try {
          // @phpstan-ignore-next-line
          $route_match = \Drupal::service('router.no_access_checks')->match($target->getPath());
        }
        catch (\Exception $e) {
          $target->setSubcategory('broken-link');
          return;
        }
    
        // It is a view route.
        if (isset($route_match['view_id'])) {
          $target->setEntityType('view');
          $target->setEntityId($route_match['view_id'] . '.' . $route_match['display_id']);
          return;
        }
    
        // This case apply with entity canonical routes.
        $route_parts = explode('.', $route_match['_route']);
        if (count($route_parts) === 3 && $route_parts[0] === 'entity' && $route_parts[2] === 'canonical') {
          $entity_id = '';
          $entity = $route_parts[1];
          if (isset($route_match[$entity]) && $route_match[$entity] instanceof EntityInterface) {
            $entity_id = $route_match[$entity]->id();
          }
          $target->setEntityType($entity);
          $target->setEntityId((string) $entity_id);
        }
      }
    

    The taxonomy term page is built using a view. For this reason, when the code detected that the view_id parameter was set in the $route_match variable, it handled the link as a view, without considering other parameters such as route_name.

    When a taxonomy term had an alias, the entity was processed earlier, and this method was not called.

    I have modified the code to be more precise in this aspect:

    protected function setDataTargetFromRoute($target) {
        if (empty($target->getPath())) {
          return;
        }
    
        try {
          // @phpstan-ignore-next-line
          $route_match = \Drupal::service('router.no_access_checks')->match($target->getPath());
        }
        catch (\Exception $e) {
          $target->setSubcategory('broken-link');
          return;
        }
    
        if (empty($route_match['_route'])) {
          $target->setSubcategory('broken-link');
          return;
        }
    
        // @todo Maybe here is possible to get as well the entity object
        if (str_starts_with($route_match['_route'], 'view.')) {
          $target->setEntityType('view');
          $target->setEntityId($route_match['view_id'] . '.' . $route_match['display_id']);
          return;
        }
    
        // This case apply with entity canonical routes.
        $route_parts = explode('.', $route_match['_route']);
        if (count($route_parts) > 1) {
          $entity_id = '';
          $entity = $route_parts[1];
          if (isset($route_match[$entity]) && $route_match[$entity] instanceof EntityInterface) {
            $entity_id = $route_match[$entity]->id();
            $entity = $route_match[$entity]->getEntityTypeId();
          }
          $target->setEntityType($entity);
          $target->setEntityId((string) $entity_id);
        }
      }
    
  • 🇪🇸Spain lpeidro Madrid

    It is ready to review and to merge. But I am not create unit test for this case. Maybe we can create a new issue to generate the unit tests.

  • 🇪🇸Spain eduardo morales alberti Spain, 🇪🇺

    Adding tests to cover it

  • 🇪🇸Spain lpeidro Madrid

    I’m using this issue to introduce two minor changes:

    • Restrict access to the ability to launch the Batch.
    • Add the subcategory field and filter to the Entity Mesh view of nodes.

    It works as expected.
    I merge the issue.

    • bf1fa0b7 committed on 1.x
      Issue #3512879: Fix spell
      
    • 1141beb9 committed on 1.x
      Issue #3512879: Add suncategory field and filter in entity mesh node...
    • 9a03fa7c committed on 1.x
      Issue #3512879: Add administer permisión to the batch form
      
  • 🇪🇸Spain GeduR

    After testing it with real data, I have found some use cases where the taxonomy terms are found as access-denied because a view is detected instead of the taxonomy term itself (see image)

  • 🇪🇸Spain lpeidro Madrid

    lpeidro changed the visibility of the branch 3512879-false-access-denied-link-with to hidden.

  • 🇪🇸Spain lpeidro Madrid

    lpeidro changed the visibility of the branch 3512879-false-access-denied-link-with to hidden.

  • Merge request !24Resolve #3512879 "False access denied ii" → (Merged) created by lpeidro
  • 🇪🇸Spain lpeidro Madrid

    There are many cases to manage. To avoid reducing the cases, a method has been implemented as the first one executed to detect whether the route corresponds to an entity route in the most flexible way possible, regardless of the entity type.

    To achieve this, we use two criteria:

    • The route name starts with "entity".
    • There is a parameter of type entity converter.

    Additionally, this parameter provides the key to retrieving the entity from the $route_match variable.

    Ready for validation.

  • 🇪🇸Spain eduardo morales alberti Spain, 🇪🇺

    Merged!! Thank you

  • Automatically closed - issue fixed for 2 weeks with no activity.

Production build 0.71.5 2024