KernelEventListener.php in Permissions by Term 8.2
File
src/Listener/KernelEventListener.php
View source
<?php
namespace Drupal\permissions_by_term\Listener;
use Drupal\Component\EventDispatcher\ContainerAwareEventDispatcher;
use Drupal\Core\Config\ConfigFactoryInterface;
use Drupal\Core\PageCache\ResponsePolicy\KillSwitch;
use Drupal\Core\Url;
use Drupal\permissions_by_term\Event\PermissionsByTermDeniedEvent;
use Drupal\permissions_by_term\Service\AccessCheck;
use Drupal\permissions_by_term\Service\AccessStorage;
use Drupal\permissions_by_term\Service\TermHandler;
use Drupal\taxonomy\Entity\Term;
use Symfony\Component\EventDispatcher\EventSubscriberInterface;
use Symfony\Component\HttpFoundation\JsonResponse;
use Symfony\Component\HttpFoundation\Request;
use Symfony\Component\HttpFoundation\Response;
use Symfony\Component\HttpKernel\Event\FilterResponseEvent;
use Symfony\Component\HttpKernel\Event\GetResponseEvent;
use Symfony\Component\HttpKernel\Exception\AccessDeniedHttpException;
use Symfony\Component\HttpKernel\KernelEvents;
class KernelEventListener implements EventSubscriberInterface {
private $accessCheckService;
private $term;
private $eventDispatcher;
private $accessStorageService;
private $pageCacheKillSwitch;
private $disabledNodeAccessRecords;
public function __construct(AccessCheck $accessCheck, AccessStorage $accessStorage, TermHandler $termHandler, ContainerAwareEventDispatcher $eventDispatcher, KillSwitch $pageCacheKillSwitch, ConfigFactoryInterface $configFactory) {
$this->accessCheckService = $accessCheck;
$this->accessStorageService = $accessStorage;
$this->term = $termHandler;
$this->eventDispatcher = $eventDispatcher;
$this->pageCacheKillSwitch = $pageCacheKillSwitch;
$this->disabledNodeAccessRecords = $configFactory
->get('permissions_by_term.settings')
->get('disable_node_access_records');
}
public function onKernelRequest(GetResponseEvent $event) {
if ($event
->isMasterRequest()) {
if (($result = $this
->handleAccessToNodePages($event)) instanceof Response) {
$event
->setResponse($result);
}
if (($result = $this
->handleAccessToTermAutocompleteLists($event)) instanceof Response) {
$event
->setResponse($result);
}
if (($result = $this
->handleAccessToTaxonomyTermViewsPages()) instanceof Response) {
$event
->setResponse($result);
}
}
}
public function onKernelResponse(FilterResponseEvent $event) {
$this
->restrictTermAccessAtAutoCompletion($event);
}
private function restrictTermAccessAtAutoCompletion(FilterResponseEvent $event) {
if ($event
->getRequest()->attributes
->get('target_type') === 'taxonomy_term' && $event
->getRequest()->attributes
->get('_route') === 'system.entity_autocomplete') {
$json_suggested_terms = $event
->getResponse()
->getContent();
$suggested_terms = json_decode($json_suggested_terms);
$allowed_terms = [];
foreach ($suggested_terms as $term) {
$tid = $this->term
->getTermIdByName($term->label);
$termLangcode = \Drupal::languageManager()
->getCurrentLanguage()
->getId();
if ($this->term
->getTerm() instanceof Term) {
$termLangcode = $this->term
->getTerm()
->language()
->getId();
}
if ($this->accessCheckService
->isAccessAllowedByDatabase($tid, \Drupal::currentUser()
->id(), $termLangcode)) {
$allowed_terms[] = [
'value' => $term->value,
'label' => $term->label,
];
}
}
$json_response = new JsonResponse($allowed_terms);
$event
->setResponse($json_response);
}
}
public static function getSubscribedEvents() : array {
return [
KernelEvents::REQUEST => 'onKernelRequest',
KernelEvents::RESPONSE => 'onKernelResponse',
];
}
private function canRequestGetNode(Request $request) : bool {
if (method_exists($request->attributes, 'get') && !empty($request->attributes
->get('node'))) {
if (method_exists($request->attributes
->get('node'), 'get')) {
return TRUE;
}
}
return FALSE;
}
private function handleAccessToTaxonomyTermViewsPages() {
$url_object = \Drupal::service('path.validator')
->getUrlIfValid(\Drupal::service('path.current')
->getPath());
if ($url_object instanceof Url && $url_object
->getRouteName() === 'entity.taxonomy_term.canonical') {
$route_parameters = $url_object
->getrouteParameters();
$termLangcode = \Drupal::languageManager()
->getCurrentLanguage()
->getId();
if (!$this->accessCheckService
->isAccessAllowedByDatabase($route_parameters['taxonomy_term'], \Drupal::currentUser()
->id(), $termLangcode)) {
$this->pageCacheKillSwitch
->trigger();
throw new AccessDeniedHttpException();
}
}
}
private function handleAccessToNodePages(GetResponseEvent $event) {
if ($this
->canRequestGetNode($event
->getRequest())) {
$node = $event
->getRequest()->attributes
->get('node');
if (!$this->accessCheckService
->canUserAccessByNode($node, false, $this->accessStorageService
->getLangCode($node
->id()))) {
$accessDeniedEvent = new PermissionsByTermDeniedEvent($node
->id());
$this->eventDispatcher
->dispatch(PermissionsByTermDeniedEvent::NAME, $accessDeniedEvent);
if ($this->disabledNodeAccessRecords) {
$this->pageCacheKillSwitch
->trigger();
}
throw new AccessDeniedHttpException();
}
}
}
private function handleAccessToTermAutocompleteLists(GetResponseEvent $event) {
if ($event
->getRequest()->attributes
->get('target_type') === 'taxonomy_term' && $event
->getRequest()->attributes
->get('_route') === 'system.entity_autocomplete') {
$query_string = $event
->getRequest()
->get('q');
$query_string = trim($query_string);
$tid = $this->term
->getTermIdByName($query_string);
$term = $this->term
->getTerm();
$termLangcode = \Drupal::languageManager()
->getCurrentLanguage()
->getId();
if ($term instanceof Term) {
$termLangcode = $term
->language()
->getId();
}
if (!$this->accessCheckService
->isAccessAllowedByDatabase($tid, \Drupal::currentUser()
->id(), $termLangcode)) {
throw new AccessDeniedHttpException();
}
}
}
}