По умолчанию Drupal генерирует маршруты для каждого термина таксономии. Это означает, что у каждого термина появляется своя страница с URL вида /taxonomy/term/123
. В некоторых случаях такие страницы могут быть полезны, например, если вы используете термины как категории для материалов и хотите отображать подборку по ним. Но чаще всего — особенно для вспомогательных словарей (тегов, фильтров, FAQ-групп) — такие страницы не нужны вовсе.
Проблема в том, что Drupal создаёт эти маршруты автоматически, и если вы не предпримете действий, то такие страницы могут попасть в индекс поисковых систем. Это может привести к нежелательным точкам входа на сайт, дублированию контента и негативному SEO-эффекту.
В этой статье разберём, как отключить генерацию страниц терминов таксономии на уровне маршрутов. Мы будем выбрасывать 404 ошибку при попытке открыть термин определённого словаря — в данном примере словаря faqs
.
Общая идея
Вместо того чтобы отключать маршруты на уровне routing.yml
, мы воспользуемся подпиской на событие KernelEvents::REQUEST
и программно проверим, какой термин запрашивается. Если он относится к словарю faqs
, мы выбросим исключение NotFoundHttpException
, тем самым сымитировав поведение страницы с ошибкой 404.
Реализация
Создаём класс подписчика событий:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 |
<?php namespace Drupal\MY_MODULE\EventSubscriber; use Drupal\taxonomy\Entity\Term; use Symfony\Component\EventDispatcher\EventSubscriberInterface; use Symfony\Component\HttpKernel\Event\RequestEvent; use Symfony\Component\HttpKernel\Exception\NotFoundHttpException; use Symfony\Component\HttpKernel\KernelEvents; class RestrictRoutableSubscriber implements EventSubscriberInterface { public function forbideTermRoute(RequestEvent $event) { $route = \Drupal::routeMatch(); $term = $route->getParameter('taxonomy_term'); // Проверим, что запрашиваемый маршрут — canonical терминов // Нам нужно отключить только каноникал, а редактор оставить как есть if ($route->getRouteName() !== 'entity.taxonomy_term.canonical') { return; } if ($term instanceof Term && $term->bundle() === 'faqs') { // Генерируем 404 throw new NotFoundHttpException(); } } /** * {@inheritdoc} */ public static function getSubscribedEvents() { // подписываем на событие KernelEvents::REQUEST // нашу функцию редиректа return [KernelEvents::REQUEST => [['forbideTermRoute']]]; } } |
Регистрируем сервис
Чтобы подписчик начал работать, нужно зарегистрировать его как сервис с тегом event_subscriber
.
Добавьте в MY_MODULE.services.yml
:
1 2 3 4 5 6 |
services: my_module.restrict_routable_subscriber: class: Drupal\my_module\EventSubscriber\RestrictRoutableSubscriber tags: - { name: event_subscriber } |
Проверка
Очистите кеш маршрутов и кеш контейнера:
1 |
drush cr |
Теперь при попытке открыть страницу термина из словаря faqs
, например /taxonomy/term/123
, вы получите ошибку 404.
Отключение страниц терминов — это простой и надёжный способ предотвратить индексацию нежелательных маршрутов, сохранить чистоту структуры сайта и избежать дублирующего контента. Такой подход хорошо работает, если вы используете таксономию исключительно для внутренней фильтрации или связей, без необходимости выводить термины на отдельные страницы.
Если нужно отключить страницы терминов не только одного, а нескольких словарей, просто расширьте условие проверки в forbideTermRoute
.