src/Subscriber/EventSubscriber/System/ErrorSubscriber.php line 40

  1. <?php
  2. namespace App\Subscriber\EventSubscriber\System;
  3. use Psr\Log\LoggerInterface;
  4. use Symfony\Component\ErrorHandler\Exception\FlattenException;
  5. use Symfony\Component\EventDispatcher\EventDispatcherInterface;
  6. use Symfony\Component\EventDispatcher\EventSubscriberInterface;
  7. use Symfony\Component\HttpFoundation\Request;
  8. use Symfony\Component\HttpKernel\Event\ControllerArgumentsEvent;
  9. use Symfony\Component\HttpKernel\Event\ExceptionEvent;
  10. use Symfony\Component\HttpKernel\Exception\HttpExceptionInterface;
  11. use Symfony\Component\HttpKernel\HttpKernelInterface;
  12. use Symfony\Component\HttpKernel\KernelEvents;
  13. use Symfony\Component\HttpKernel\Log\DebugLoggerInterface;
  14. /**
  15.  * @author Fabien Potencier <fabien@symfony.com>
  16.  */
  17. class ErrorSubscriber implements EventSubscriberInterface
  18. {
  19.     protected $controller;
  20.     protected $logger;
  21.     protected $debug;
  22.     
  23.     public function __construct($controllerLoggerInterface $logger null$debug false)
  24.     {
  25.         $this->controller $controller;
  26.         $this->logger $logger;
  27.         $this->debug $debug;
  28.     }
  29.     
  30.     public function logKernelException(ExceptionEvent $event)
  31.     {
  32.         $e FlattenException::createFromThrowable($event->getThrowable());
  33.         
  34.         $this->logException($event->getThrowable(), sprintf('Uncaught PHP Exception %s: "%s" at %s line %s'$e->getClass(), $e->getMessage(), $e->getFile(), $e->getLine()));
  35.     }
  36.     
  37.     public function onKernelException(ExceptionEvent $eventstring $eventName nullEventDispatcherInterface $eventDispatcher null)
  38.     {
  39.         
  40.         return;
  41.         if (null === $this->controller) {
  42.             return;
  43.         }
  44.         
  45. //        dd($this->controller);
  46.         $exception $event->getThrowable();
  47.         
  48.         
  49.         $request $this->duplicateRequest($exception$event->getRequest());
  50.         
  51.      
  52.         try {
  53.             $response $event->getKernel()->handle($requestHttpKernelInterface::SUB_REQUESTfalse);
  54.         } catch (\Exception $e) {
  55.             $f FlattenException::createFromThrowable($e);
  56.             
  57.             $this->logException($esprintf('Exception thrown when handling an exception (%s: %s at %s line %s)'$f->getClass(), $f->getMessage(), $e->getFile(), $e->getLine()));
  58.             
  59.          
  60.             $prev $e;
  61.             do {
  62.                 if ($exception === $wrapper $prev) {
  63.                     throw $e;
  64.                 }
  65.             } while ($prev $wrapper->getPrevious());
  66.             
  67.             
  68.             
  69.             $prev = new \ReflectionProperty($wrapper instanceof \Exception \Exception::class : \Error::class, 'previous');
  70.             $prev->setAccessible(true);
  71.             $prev->setValue($wrapper$exception);
  72.             
  73.             // TODO burada hataları alabiliyoruz. DB kaydı yaptırılabilir.
  74.             dump($exception);
  75.             exit;
  76.             
  77.             throw $exception;
  78.         }
  79.         
  80.         
  81.         $event->setResponse($response);
  82.         
  83.         if ($this->debug && $eventDispatcher instanceof EventDispatcherInterface) {
  84.             $cspRemovalListener = function ($event) use (&$cspRemovalListener$eventDispatcher) {
  85.                 $event->getResponse()->headers->remove('Content-Security-Policy');
  86.                 $eventDispatcher->removeListener(KernelEvents::RESPONSE$cspRemovalListener);
  87.             };
  88.             $eventDispatcher->addListener(KernelEvents::RESPONSE$cspRemovalListener, -128);
  89.         }
  90.     }
  91.     
  92.     public function onControllerArguments(ControllerArgumentsEvent $event)
  93.     {
  94.         $e $event->getRequest()->attributes->get('exception');
  95.         
  96.         if (!$e instanceof \Throwable || false === $k array_search($e$event->getArguments(), true)) {
  97.             return;
  98.         }
  99.         
  100.         $r = new \ReflectionFunction(\Closure::fromCallable($event->getController()));
  101.         $r $r->getParameters()[$k] ?? null;
  102.         
  103.         if ($r && (!$r->hasType() || \in_array($r->getType()->getName(), [FlattenException::class, LegacyFlattenException::class], true))) {
  104.             $arguments $event->getArguments();
  105.             $arguments[$k] = FlattenException::createFromThrowable($e);
  106.             $event->setArguments($arguments);
  107.         }
  108.     }
  109.     
  110.     public static function getSubscribedEvents(): array
  111.     {
  112.         return [
  113.             KernelEvents::CONTROLLER_ARGUMENTS => 'onControllerArguments',
  114.             KernelEvents::EXCEPTION => [
  115.                 ['logKernelException'0],
  116.                 ['onKernelException', -128],
  117.             ],
  118.         ];
  119.     }
  120.     
  121.     /**
  122.      * Logs an exception.
  123.      */
  124.     protected function logException(\Throwable $exceptionstring $message): void
  125.     {
  126.         if (null !== $this->logger) {
  127.             if (!$exception instanceof HttpExceptionInterface || $exception->getStatusCode() >= 500) {
  128.                 $this->logger->critical($message, ['exception' => $exception]);
  129.             } else {
  130.                 $this->logger->error($message, ['exception' => $exception]);
  131.             }
  132.         }
  133.     }
  134.     
  135.     /**
  136.      * Clones the request for the exception.
  137.      */
  138.     protected function duplicateRequest(\Throwable $exceptionRequest $request): Request
  139.     {
  140.         $attributes = [
  141.             '_controller' => $this->controller,
  142.             'exception' => $exception,
  143.             'logger' => $this->logger instanceof DebugLoggerInterface $this->logger null,
  144.         ];
  145.         $request $request->duplicate(nullnull$attributes);
  146.         $request->setMethod('GET');
  147.         
  148.         return $request;
  149.     }
  150. }