Возврат альтернативных HTTP-кодов для неопубликованного узла

8

Я пытаюсь вернуть страницу 404 вместо ответа 403 для неопубликованных узлов в Drupal 8.

Я протестировал подписчик ответа ядра , но обнаружил, что код, который я использовал, изменит код состояния только с 403 на 404, а не на страницу 404. Так что, может быть, кто-то может показать мне, как создать объект Response на 404 страницы?

Это код, который я использовал:

class ResponseSubscriber implements EventSubscriberInterface {

  /**
   * {@inheritdoc}
   */
  public static function getSubscribedEvents() {
    return [KernelEvents::RESPONSE => [['alterResponse']]];
  }

  /**
   * Change status code to 404 from 403 if page is an unpublished node.
   *
   * @param FilterResponseEvent $event
   *   The route building event.
   */
  public function alterResponse(FilterResponseEvent $event) {
    if ($event->getResponse()->getStatusCode() == 403) {
      /** @var \Symfony\Component\HttpFoundation\Request $request */
      $request = $event->getRequest();
      $node = $request->attributes->get('node');
      if ($node instanceof Node && !$node->isPublished()) {
        $response = $event->getResponse();
        // This changes the code, but doesn't return a 404 page.
        $response->setStatusCode(404);

        $event->setResponse($response);
      }
    }
  }

}

Я наконец прибег к полному удалению этого подписчика ответа и использовал hook_node_access следующим образом:

use Symfony\Component\HttpKernel\Exception\NotFoundHttpException;
use Drupal\Core\Access\AccessResult;

function unpublished_404_node_access(\Drupal\node\NodeInterface $node, $op, \Drupal\Core\Session\AccountInterface $account) {

  if ($op == 'view' && !$node->isPublished()) {
    if (\Drupal::moduleHandler()->moduleExists('workbench_moderation') && $account->hasPermission('view any unpublished content')) {
      return AccessResult::neutral();
    }
    elseif (\Drupal::routeMatch()->getRouteName() == 'entity.node.canonical' && \Drupal::routeMatch()->getRawParameter('node') == $node->id()) {
      throw new NotFoundHttpException();
      return AccessResult::neutral();
    }
  }

  return AccessResult::neutral();
}

Похоже, что это соответствует нескольким ответам на этом сайте для Drupal 7. Но я хотел посмотреть, есть ли у кого-нибудь лучший способ сделать это либо с подписчиком KernelEvent, чем с hook_node_access. Похоже, что я хочу сделать, это проверить, возвращает ли узел 403, а затем сгенерировать новый ответ со страницей 404 и кодом состояния 404. Я не уверен, как это сделать.

oknate
источник

Ответы:

6

Вы можете попытаться сделать это раньше в исключительной ситуации вместо ответного подписчика. Расширьте HttpExceptionSubscriberBase, поэтому вам нужно меньше кода для этого. Затем замените 403 с исключением 404 методом$event->setException()

/src/EventSubscriber/Unpublished404Subscriber.php

<?php

namespace Drupal\mymodule\EventSubscriber;

use Drupal\Core\EventSubscriber\HttpExceptionSubscriberBase;
use Symfony\Component\HttpKernel\Event\GetResponseForExceptionEvent;
use Symfony\Component\HttpKernel\Exception\NotFoundHttpException;

class Unpublished404Subscriber extends HttpExceptionSubscriberBase {

  protected static function getPriority() {
    // set priority higher than 50 if you want to log "page not found"
    return 0;
  }

  protected function getHandledFormats() {
    return ['html'];
  }

  public function on403(GetResponseForExceptionEvent $event) {
    $request = $event->getRequest();
    if ($request->attributes->get('_route') == 'entity.node.canonical') {
      $event->setException(new NotFoundHttpException());
    }
  }

}

mymodule.services.yml:

services:
  mymodule.404:
    class: Drupal\mymodule\EventSubscriber\Unpublished404Subscriber
    arguments: []
    tags:
      - { name: event_subscriber }

Это заменяет все 403 исключения для канонических узловых маршрутов. Вы можете получить объект узла, $request->attributes->get('node')если хотите проверить, действительно ли это так, потому что узел не опубликован.

4k4
источник
Спасибо, я проверил это, и он прекрасно работает! Это именно то, что я искал.
окнать