diff --git a/core/core.services.yml b/core/core.services.yml index f8c428f2d5cfe504fa233eed667b603e3ea4f48e..8f19eb47742ed252a05648b2ad96c4ac12f560fc 100644 --- a/core/core.services.yml +++ b/core/core.services.yml @@ -196,6 +196,11 @@ services: factory_method: get factory_service: logger.factory arguments: ['system'] + logger.channel.php: + class: Drupal\Core\Logger\LoggerChannel + factory_method: get + factory_service: logger.factory + arguments: ['php'] logger.channel.image: class: Drupal\Core\Logger\LoggerChannel factory_method: get @@ -736,16 +741,44 @@ services: tags: - { name: event_subscriber } arguments: ['@config.manager', '@config.storage', '@config.storage.snapshot'] - exception_controller: - class: Drupal\Core\Controller\ExceptionController - arguments: ['@content_negotiation', '@title_resolver', '@html_page_renderer', '@html_fragment_renderer', '@string_translation', '@url_generator', '@logger.factory'] - calls: - - [setContainer, ['@service_container']] - exception_listener: - class: Drupal\Core\EventSubscriber\ExceptionListener + exception.default_json: + class: Drupal\Core\EventSubscriber\ExceptionJsonSubscriber + tags: + - { name: event_subscriber } + exception.default_html: + class: Drupal\Core\EventSubscriber\DefaultExceptionHtmlSubscriber + tags: + - { name: event_subscriber } + arguments: ['@html_fragment_renderer', '@html_page_renderer'] + exception.default: + class: Drupal\Core\EventSubscriber\DefaultExceptionSubscriber + tags: + - { name: event_subscriber } + arguments: ['@html_fragment_renderer', '@html_page_renderer', '@config.factory'] + exception.logger: + class: Drupal\Core\EventSubscriber\ExceptionLoggingSubscriber + tags: + - { name: event_subscriber } + arguments: ['@logger.factory'] + exception.custom_page_json: + class: Drupal\Core\EventSubscriber\ExceptionJsonSubscriber + tags: + - { name: event_subscriber } + arguments: ['@config.factory', '@path.alias_manager', '@http_kernel'] + exception.custom_page_html: + class: Drupal\Core\EventSubscriber\CustomPageExceptionHtmlSubscriber + tags: + - { name: event_subscriber } + arguments: ['@config.factory', '@path.alias_manager', '@http_kernel', '@logger.channel.php'] + exception.fast_404_html: + class: Drupal\Core\EventSubscriber\Fast404ExceptionHtmlSubscriber + tags: + - { name: event_subscriber } + arguments: ['@config.factory', '@http_kernel'] + exception.test_site: + class: Drupal\Core\EventSubscriber\ExceptionTestSiteSubscriber tags: - { name: event_subscriber } - arguments: [['@exception_controller', execute]] route_processor_manager: class: Drupal\Core\RouteProcessor\RouteProcessorManager tags: diff --git a/core/includes/menu.inc b/core/includes/menu.inc index 12d2a239faf97a61a5a443beb82872153e01b21b..a731750aa581acf346f616e3a6a81d007e26e1b2 100644 --- a/core/includes/menu.inc +++ b/core/includes/menu.inc @@ -573,7 +573,7 @@ function menu_local_tasks($level = 0) { $data['actions'] = array(); $route_name = \Drupal::routeMatch()->getRouteName(); - if (!empty($route_name)) { + if (!\Drupal::request()->attributes->has('exception') && !empty($route_name)) { $manager = \Drupal::service('plugin.manager.menu.local_task'); $local_tasks = $manager->getTasksBuild($route_name); foreach ($local_tasks as $level => $items) { diff --git a/core/lib/Drupal/Core/ContentNegotiation.php b/core/lib/Drupal/Core/ContentNegotiation.php index 500200cf75135c828d20e3eadad2379b389a67bc..857f9d59f643d58965b1f1d86e89bb77665559c2 100644 --- a/core/lib/Drupal/Core/ContentNegotiation.php +++ b/core/lib/Drupal/Core/ContentNegotiation.php @@ -23,7 +23,7 @@ class ContentNegotiation { * The normalized type is a short, lowercase version of the format, such as * 'html', 'json' or 'atom'. * - * @param Symfony\Component\HttpFoundation\Request $request + * @param \Symfony\Component\HttpFoundation\Request $request * The request object from which to extract the content type. * * @return string diff --git a/core/lib/Drupal/Core/Controller/ExceptionController.php b/core/lib/Drupal/Core/Controller/ExceptionController.php deleted file mode 100644 index 69a477b4ccc04014448ba35392cd08790c50bb64..0000000000000000000000000000000000000000 --- a/core/lib/Drupal/Core/Controller/ExceptionController.php +++ /dev/null @@ -1,452 +0,0 @@ -<?php - -/** - * @file - * Contains \Drupal\Core\Controller\ExceptionController. - */ - -namespace Drupal\Core\Controller; - -use Drupal\Core\Logger\LoggerChannelFactoryInterface; -use Drupal\Core\Page\DefaultHtmlPageRenderer; -use Drupal\Core\Page\HtmlFragmentRendererInterface; -use Drupal\Core\Page\HtmlPageRendererInterface; -use Drupal\Core\Routing\UrlGeneratorInterface; -use Symfony\Component\DependencyInjection\ContainerAwareInterface; -use Symfony\Component\DependencyInjection\ContainerInterface; -use Symfony\Component\HttpFoundation\Request; -use Symfony\Component\HttpFoundation\Response; -use Symfony\Component\HttpFoundation\JsonResponse; -use Symfony\Component\HttpKernel\HttpKernelInterface; -use Drupal\Component\Utility\SafeMarkup; -use Drupal\Component\Utility\String; -use Symfony\Component\Debug\Exception\FlattenException; -use Drupal\Core\ContentNegotiation; -use Drupal\Core\Utility\Error; -use Drupal\Core\StringTranslation\StringTranslationTrait; -use Drupal\Core\StringTranslation\TranslationInterface; - -/** - * This controller handles HTTP errors generated by the routing system. - */ -class ExceptionController extends HtmlControllerBase implements ContainerAwareInterface { - use StringTranslationTrait; - - /** - * The content negotiation library. - * - * @var \Drupal\Core\ContentNegotiation - */ - protected $negotiation; - - /** - * The service container. - * - * @var \Symfony\Component\DependencyInjection\ContainerInterface - */ - protected $container; - - /** - * The page rendering service. - * - * @var \Drupal\Core\Page\HtmlPageRendererInterface - */ - protected $htmlPageRenderer; - - /** - * The fragment rendering service. - * - * @var \Drupal\Core\Page\HtmlFragmentRendererInterface - */ - protected $fragmentRenderer; - - /** - * The logger factory service. - * - * @var \Drupal\Core\Logger\LoggerChannelFactoryInterface - */ - protected $loggerFactory; - - /** - * Constructor. - * - * @param \Drupal\Core\ContentNegotiation $negotiation - * The content negotiation library to use to determine the correct response - * format. - * @param \Drupal\Core\Controller\TitleResolverInterface $title_resolver - * The title resolver. - * @param \Drupal\Core\Page\HtmlPageRendererInterface $renderer - * The page renderer. - * @param \Drupal\Core\Page\HtmlFragmentRendererInterface $fragment_renderer - * The fragment rendering service. - * @param \Drupal\Core\StringTranslation\TranslationInterface $string_translation - * The url generator. - * @param \Drupal\Core\Routing\UrlGeneratorInterface $url_generator - * The URL generator. - * @param \Drupal\Core\Logger\LoggerChannelFactoryInterface $logger_factory - * The logger factory. - */ - public function __construct(ContentNegotiation $negotiation, TitleResolverInterface $title_resolver, HtmlPageRendererInterface $renderer, HtmlFragmentRendererInterface $fragment_renderer, TranslationInterface $string_translation, UrlGeneratorInterface $url_generator, LoggerChannelFactoryInterface $logger_factory) { - parent::__construct($title_resolver, $url_generator); - $this->negotiation = $negotiation; - $this->htmlPageRenderer = $renderer; - $this->fragmentRenderer = $fragment_renderer; - $this->stringTranslation = $string_translation; - $this->loggerFactory = $logger_factory; - } - - /** - * Sets the Container associated with this Controller. - * - * @param \Symfony\Component\DependencyInjection\ContainerInterface $container - * A ContainerInterface instance. - * - * @api - */ - public function setContainer(ContainerInterface $container = NULL) { - $this->container = $container; - } - - /** - * Handles an exception on a request. - * - * @param \Symfony\Component\Debug\Exception\FlattenException $exception - * The flattened exception. - * @param \Symfony\Component\HttpFoundation\Request $request - * The request that generated the exception. - * - * @return \Symfony\Component\HttpFoundation\Response - * A response object. - */ - public function execute(FlattenException $exception, Request $request) { - $method = 'on' . $exception->getStatusCode() . $this->negotiation->getContentType($request); - - if (method_exists($this, $method)) { - return $this->$method($exception, $request); - } - - return new Response('A fatal error occurred: ' . $exception->getMessage(), $exception->getStatusCode(), $exception->getHeaders()); - } - - /** - * Processes a MethodNotAllowed exception into an HTTP 405 response. - * - * @param \Symfony\Component\Debug\Exception\FlattenException $exception - * The flattened exception. - * @param \Symfony\Component\HttpFoundation\Request $request - * The request object that triggered this exception. - * - * @return \Symfony\Component\HttpFoundation\Response - * A response object. - */ - public function on405Html(FlattenException $exception, Request $request) { - return new Response('Method Not Allowed', 405); - } - - /** - * Processes an AccessDenied exception into an HTTP 403 response. - * - * @param \Symfony\Component\Debug\Exception\FlattenException $exception - * The flattened exception. - * @param \Symfony\Component\HttpFoundation\Request $request - * The request object that triggered this exception. - * - * @return \Symfony\Component\HttpFoundation\Response - * A response object. - */ - public function on403Html(FlattenException $exception, Request $request) { - // @todo Remove dependency on the internal _system_path attribute: - // https://www.drupal.org/node/2293523. - $system_path = $request->attributes->get('_system_path'); - $this->loggerFactory->get('access denied')->warning($system_path); - - $system_config = $this->container->get('config.factory')->get('system.site'); - $path = $this->container->get('path.alias_manager')->getPathByAlias($system_config->get('page.403')); - if ($path && $path != $system_path) { - if ($request->getMethod() === 'POST') { - $subrequest = Request::create($request->getBaseUrl() . '/' . $path, 'POST', array('destination' => $system_path, '_exception_statuscode' => 403) + $request->request->all(), $request->cookies->all(), array(), $request->server->all()); - } - else { - $subrequest = Request::create($request->getBaseUrl() . '/' . $path, 'GET', array('destination' => $system_path, '_exception_statuscode' => 403), $request->cookies->all(), array(), $request->server->all()); - } - - $response = $this->container->get('http_kernel')->handle($subrequest, HttpKernelInterface::SUB_REQUEST); - $response->setStatusCode(403, 'Access denied'); - } - else { - $page_content = array( - '#markup' => $this->t('You are not authorized to access this page.'), - '#title' => $this->t('Access denied'), - ); - - $fragment = $this->createHtmlFragment($page_content, $request); - $page = $this->fragmentRenderer->render($fragment, 403); - $response = new Response($this->htmlPageRenderer->render($page), $page->getStatusCode()); - return $response; - } - - return $response; - } - - /** - * Processes a NotFound exception into an HTTP 404 response. - * - * @param \Symfony\Component\Debug\Exception\FlattenException $exception - * The flattened exception. - * @param \Symfony\Component\HttpFoundation\Request $request - * The request object that triggered this exception. - * - * @return \Symfony\Component\HttpFoundation\Response - * A response object. - */ - public function on404Html(FlattenException $exception, Request $request) { - $this->loggerFactory->get('page not found')->warning(String::checkPlain($request->attributes->get('_system_path'))); - - // Check for and return a fast 404 page if configured. - $config = \Drupal::config('system.performance'); - - $exclude_paths = $config->get('fast_404.exclude_paths'); - if ($config->get('fast_404.enabled') && $exclude_paths && !preg_match($exclude_paths, $request->getPathInfo())) { - $fast_paths = $config->get('fast_404.paths'); - if ($fast_paths && preg_match($fast_paths, $request->getPathInfo())) { - $fast_404_html = $config->get('fast_404.html'); - $fast_404_html = strtr($fast_404_html, array('@path' => String::checkPlain($request->getUri()))); - return new Response($fast_404_html, 404); - } - } - - // @todo Remove dependency on the internal _system_path attribute: - // https://www.drupal.org/node/2293523. - $system_path = $request->attributes->get('_system_path'); - - $path = $this->container->get('path.alias_manager')->getPathByAlias(\Drupal::config('system.site')->get('page.404')); - if ($path && $path != $system_path) { - // @todo Um, how do I specify an override URL again? Totally not clear. Do - // that and sub-call the kernel rather than using meah(). - // @todo The create() method expects a slash-prefixed path, but we store a - // normal system path in the site_404 variable. - if ($request->getMethod() === 'POST') { - $subrequest = Request::create($request->getBaseUrl() . '/' . $path, 'POST', array('destination' => $system_path, '_exception_statuscode' => 404) + $request->request->all(), $request->cookies->all(), array(), $request->server->all()); - } - else { - $subrequest = Request::create($request->getBaseUrl() . '/' . $path, 'GET', array('destination' => $system_path, '_exception_statuscode' => 404), $request->cookies->all(), array(), $request->server->all()); - } - - $response = $this->container->get('http_kernel')->handle($subrequest, HttpKernelInterface::SUB_REQUEST); - $response->setStatusCode(404, 'Not Found'); - } - else { - $page_content = array( - '#markup' => $this->t('The requested page "@path" could not be found.', array('@path' => $request->getPathInfo())), - '#title' => $this->t('Page not found'), - ); - - $fragment = $this->createHtmlFragment($page_content, $request); - $page = $this->fragmentRenderer->render($fragment, 404); - $response = new Response($this->htmlPageRenderer->render($page), $page->getStatusCode()); - return $response; - } - - return $response; - } - - /** - * Processes a generic exception into an HTTP 500 response. - * - * @param \Symfony\Component\Debug\Exception\FlattenException $exception - * Metadata about the exception that was thrown. - * @param \Symfony\Component\HttpFoundation\Request $request - * The request object that triggered this exception. - * - * @return \Symfony\Component\HttpFoundation\Response - * A response object. - */ - public function on500Html(FlattenException $exception, Request $request) { - $error = $this->decodeException($exception); - - // Because the kernel doesn't run until full bootstrap, we know that - // most subsystems are already initialized. - - $headers = array(); - - // When running inside the testing framework, we relay the errors - // to the tested site by the way of HTTP headers. - if (DRUPAL_TEST_IN_CHILD_SITE && !headers_sent() && (!defined('SIMPLETEST_COLLECT_ERRORS') || SIMPLETEST_COLLECT_ERRORS)) { - // $number does not use drupal_static as it should not be reset - // as it uniquely identifies each PHP error. - static $number = 0; - $assertion = array( - $error['!message'], - $error['%type'], - array( - 'function' => $error['%function'], - 'file' => $error['%file'], - 'line' => $error['%line'], - ), - ); - $headers['X-Drupal-Assertion-' . $number] = rawurlencode(serialize($assertion)); - $number++; - } - - $this->loggerFactory->get('php')->log($error['severity_level'], '%type: !message in %function (line %line of %file).', $error); - - // Display the message if the current error reporting level allows this type - // of message to be displayed, and unconditionnaly in update.php. - if (error_displayable($error)) { - $class = 'error'; - - // If error type is 'User notice' then treat it as debug information - // instead of an error message. - // @see debug() - if ($error['%type'] == 'User notice') { - $error['%type'] = 'Debug'; - $class = 'status'; - } - - // Attempt to reduce verbosity by removing DRUPAL_ROOT from the file path - // in the message. This does not happen for (false) security. - $root_length = strlen(DRUPAL_ROOT); - if (substr($error['%file'], 0, $root_length) == DRUPAL_ROOT) { - $error['%file'] = substr($error['%file'], $root_length + 1); - } - // Should not translate the string to avoid errors producing more errors. - $message = String::format('%type: !message in %function (line %line of %file).', $error); - - // Check if verbose error reporting is on. - $error_level = $this->container->get('config.factory')->get('system.logging')->get('error_level'); - - if ($error_level == ERROR_REPORTING_DISPLAY_VERBOSE) { - $backtrace_exception = $exception; - while ($backtrace_exception->getPrevious()) { - $backtrace_exception = $backtrace_exception->getPrevious(); - } - $backtrace = $backtrace_exception->getTrace(); - // First trace is the error itself, already contained in the message. - // While the second trace is the error source and also contained in the - // message, the message doesn't contain argument values, so we output it - // once more in the backtrace. - array_shift($backtrace); - // Generate a backtrace containing only scalar argument values. - $message .= '<pre class="backtrace">' . Error::formatFlattenedBacktrace($backtrace) . '</pre>'; - } - drupal_set_message(SafeMarkup::set($message), $class, TRUE); - } - - $content = $this->t('The website has encountered an error. Please try again later.'); - $output = DefaultHtmlPageRenderer::renderPage($content, $this->t('Error')); - $response = new Response($output); - $response->setStatusCode(500, '500 Service unavailable (with message)'); - - return $response; - } - - /** - * Processes an AccessDenied exception that occurred on a JSON request. - * - * @param \Symfony\Component\Debug\Exception\FlattenException $exception - * The flattened exception. - * @param \Symfony\Component\HttpFoundation\Request $request - * The request object that triggered this exception. - * - * @return \Symfony\Component\HttpFoundation\JsonResponse - * A JSON response object. - */ - public function on403Json(FlattenException $exception, Request $request) { - $response = new JsonResponse(); - $response->setStatusCode(403, 'Access Denied'); - return $response; - } - - /** - * Processes a NotFound exception that occurred on a JSON request. - * - * @param \Symfony\Component\Debug\Exception\FlattenException $exception - * The flattened exception. - * @param \Symfony\Component\HttpFoundation\Request $request - * The request object that triggered this exception. - * - * @return \Symfony\Component\HttpFoundation\JsonResponse - * A JSON response object. - */ - public function on404Json(FlattenException $exception, Request $request) { - $response = new JsonResponse(); - $response->setStatusCode(404, 'Not Found'); - return $response; - } - - /** - * Processes a MethodNotAllowed exception that occurred on a JSON request. - * - * @param \Symfony\Component\Debug\Exception\FlattenException $exception - * The flattened exception. - * @param \Symfony\Component\HttpFoundation\Request $request - * The request object that triggered this exception. - * - * @return \Symfony\Component\HttpFoundation\JsonResponse - * A JSON response object. - */ - public function on405Json(FlattenException $exception, Request $request) { - $response = new JsonResponse(); - $response->setStatusCode(405, 'Method Not Allowed'); - return $response; - } - - - /** - * This method is a temporary port of _drupal_decode_exception(). - * - * @todo This should get refactored. FlattenException could use some - * improvement as well. - * - * @param \Symfony\Component\Debug\Exception\FlattenException $exception - * The flattened exception. - * - * @return array - * An array of string-substitution tokens for formatting a message about the - * exception. - */ - protected function decodeException(FlattenException $exception) { - $message = $exception->getMessage(); - - $backtrace = $exception->getTrace(); - - // This value is missing from the stack for some reason in the - // FlattenException version of the backtrace. - $backtrace[0]['line'] = $exception->getLine(); - - // For database errors, we try to return the initial caller, - // skipping internal functions of the database layer. - if (strpos($exception->getClass(), 'DatabaseExceptionWrapper') !== FALSE) { - // A DatabaseExceptionWrapper exception is actually just a courier for - // the original PDOException. It's the stack trace from that exception - // that we care about. - $backtrace = $exception->getPrevious()->getTrace(); - $backtrace[0]['line'] = $exception->getLine(); - - // The first element in the stack is the call, the second element gives us the caller. - // We skip calls that occurred in one of the classes of the database layer - // or in one of its global functions. - $db_functions = array('db_query', 'db_query_range'); - while (!empty($backtrace[1]) && ($caller = $backtrace[1]) && - ((strpos($caller['namespace'], 'Drupal\Core\Database') !== FALSE || strpos($caller['class'], 'PDO') !== FALSE)) || - in_array($caller['function'], $db_functions)) { - // We remove that call. - array_shift($backtrace); - } - } - - $caller = Error::getLastCaller($backtrace); - - return array( - '%type' => $exception->getClass(), - // The standard PHP exception handler considers that the exception message - // is plain-text. We mimick this behavior here. - '!message' => String::checkPlain($message), - '%function' => $caller['function'], - '%file' => $caller['file'], - '%line' => $caller['line'], - 'severity_level' => WATCHDOG_ERROR, - ); - } - -} diff --git a/core/lib/Drupal/Core/EventSubscriber/AuthenticationSubscriber.php b/core/lib/Drupal/Core/EventSubscriber/AuthenticationSubscriber.php index 87a3cfe1ae495a29f831308b7470fbf36f63c2c8..80f77962f67f6f0921a93629090fbe471ea79186 100644 --- a/core/lib/Drupal/Core/EventSubscriber/AuthenticationSubscriber.php +++ b/core/lib/Drupal/Core/EventSubscriber/AuthenticationSubscriber.php @@ -70,8 +70,8 @@ public function onException(GetResponseForExceptionEvent $event) { * Cookie provider to send all relevant session data to the user. */ public static function getSubscribedEvents() { - $events[KernelEvents::RESPONSE][] = array('onRespond', 0); - $events[KernelEvents::EXCEPTION][] = array('onException', 0); + $events[KernelEvents::RESPONSE][] = ['onRespond', 0]; + $events[KernelEvents::EXCEPTION][] = ['onException', 75]; return $events; } } diff --git a/core/lib/Drupal/Core/EventSubscriber/CustomPageExceptionHtmlSubscriber.php b/core/lib/Drupal/Core/EventSubscriber/CustomPageExceptionHtmlSubscriber.php new file mode 100644 index 0000000000000000000000000000000000000000..ae3d9447632e1281c506f59bf4ae08966d0de02d --- /dev/null +++ b/core/lib/Drupal/Core/EventSubscriber/CustomPageExceptionHtmlSubscriber.php @@ -0,0 +1,149 @@ +<?php + +/** + * @file + * Contains \Drupal\Core\EventSubscriber\DefaultExceptionHtmlSubscriber. + */ + +namespace Drupal\Core\EventSubscriber; + +use Drupal\Core\Config\ConfigFactoryInterface; +use Drupal\Core\Path\AliasManagerInterface; +use Drupal\Core\Utility\Error; +use Psr\Log\LoggerInterface; +use Symfony\Component\HttpFoundation\Request; +use Symfony\Component\HttpFoundation\Response; +use Symfony\Component\HttpKernel\Event\GetResponseForExceptionEvent; +use Symfony\Component\HttpKernel\HttpKernelInterface; + +/** + * Exception subscriber for handling core custom error pages. + */ +class CustomPageExceptionHtmlSubscriber extends HttpExceptionSubscriberBase { + + /** + * The configuration factory. + * + * @var \Drupal\Core\Config\ConfigFactoryInterface + */ + protected $configFactory; + + /** + * The page alias manager. + * + * @var \Drupal\Core\Path\AliasManagerInterface + */ + protected $aliasManager; + + /** + * The HTTP kernel. + * + * @var \Symfony\Component\HttpKernel\HttpKernelInterface + */ + protected $httpKernel; + + /** + * The logger instance. + * + * @var \Psr\Log\LoggerInterface + */ + protected $logger; + + /** + * Constructs a new CustomPageExceptionHtmlSubscriber. + * + * @param \Drupal\Core\Config\ConfigFactoryInterface $config_factory + * The configuration factory. + * @param \Drupal\Core\Path\AliasManagerInterface $alias_manager + * The alias manager service. + * @param \Symfony\Component\HttpKernel\HttpKernelInterface $http_kernel + * The HTTP Kernel service. + * @param \Psr\Log\LoggerInterface $logger + * The logger service. + */ + public function __construct(ConfigFactoryInterface $config_factory, AliasManagerInterface $alias_manager, HttpKernelInterface $http_kernel, LoggerInterface $logger) { + $this->configFactory = $config_factory; + $this->aliasManager = $alias_manager; + $this->httpKernel = $http_kernel; + $this->logger = $logger; + } + + /** + * {@inheritdoc} + */ + protected static function getPriority() { + return -50; + } + + /** + * {@inheritDoc} + */ + protected function getHandledFormats() { + return ['html']; + } + + /** + * Handles a 403 error for HTML. + * + * @param \Symfony\Component\HttpKernel\Event\GetResponseForExceptionEvent $event + * The event to process. + */ + public function on403(GetResponseForExceptionEvent $event) { + $path = $this->aliasManager->getPathByAlias($this->configFactory->get('system.site')->get('page.403')); + $this->makeSubrequest($event, $path, Response::HTTP_FORBIDDEN); + } + + /** + * Handles a 404 error for HTML. + * + * @param \Symfony\Component\HttpKernel\Event\GetResponseForExceptionEvent $event + * The event to process. + */ + public function on404(GetResponseForExceptionEvent $event) { + $path = $this->aliasManager->getPathByAlias($this->configFactory->get('system.site')->get('page.404')); + $this->makeSubrequest($event, $path, Response::HTTP_NOT_FOUND); + } + + /** + * Makes a subrequest to retrieve a custom error page. + * + * @param \Symfony\Component\HttpKernel\Event\GetResponseForExceptionEvent $event + * The event to process + * @param string $path + * The path to which to make a subrequest for this error message. + * @param int $status_code + * The status code for the error being handled. + */ + protected function makeSubrequest(GetResponseForExceptionEvent $event, $path, $status_code) { + $request = $event->getRequest(); + + // @todo Remove dependency on the internal _system_path attribute: + // https://www.drupal.org/node/2293523. + $system_path = $request->attributes->get('_system_path'); + + if ($path && $path != $system_path) { + // @todo The create() method expects a slash-prefixed path, but we store a + // normal system path in the site_404 variable. + if ($request->getMethod() === 'POST') { + $sub_request = Request::create($request->getBaseUrl() . '/' . $path, 'POST', ['destination' => $system_path, '_exception_statuscode' => $status_code] + $request->request->all(), $request->cookies->all(), [], $request->server->all()); + } + else { + $sub_request = Request::create($request->getBaseUrl() . '/' . $path, 'GET', $request->query->all() + ['destination' => $system_path, '_exception_statuscode' => $status_code], $request->cookies->all(), [], $request->server->all()); + } + + try { + $response = $this->httpKernel->handle($sub_request, HttpKernelInterface::SUB_REQUEST); + $response->setStatusCode($status_code); + $event->setResponse($response); + } + catch (\Exception $e) { + // If an error happened in the subrequest we can't do much else. + // Instead, just log it. The DefaultExceptionHandler will catch the + // original exception and handle it normally. + $error = Error::decodeException($e); + $this->logger->log($error['severity_level'], '%type: !message in %function (line %line of %file).', $error); + } + } + } + +} diff --git a/core/lib/Drupal/Core/EventSubscriber/DefaultExceptionHtmlSubscriber.php b/core/lib/Drupal/Core/EventSubscriber/DefaultExceptionHtmlSubscriber.php new file mode 100644 index 0000000000000000000000000000000000000000..0df6a3073a1c1c663e89e86fa899e11902f6b18b --- /dev/null +++ b/core/lib/Drupal/Core/EventSubscriber/DefaultExceptionHtmlSubscriber.php @@ -0,0 +1,121 @@ +<?php + +/** + * @file + * Contains \Drupal\Core\EventSubscriber\DefaultExceptionHtmlSubscriber. + */ + +namespace Drupal\Core\EventSubscriber; + +use Drupal\Core\Page\HtmlFragment; +use Drupal\Core\Page\HtmlFragmentRendererInterface; +use Drupal\Core\Page\HtmlPageRendererInterface; +use Drupal\Core\StringTranslation\StringTranslationTrait; +use Symfony\Component\HttpFoundation\Response; +use Symfony\Component\HttpKernel\Event\GetResponseForExceptionEvent; + +/** + * Handle most HTTP errors for HTML. + */ +class DefaultExceptionHtmlSubscriber extends HttpExceptionSubscriberBase { + use StringTranslationTrait; + + /** + * The HTML fragment renderer. + * + * @var \Drupal\Core\Page\HtmlFragmentRendererInterface + */ + protected $fragmentRenderer; + + /** + * The HTML page renderer. + * + * @var \Drupal\Core\Page\HtmlPageRendererInterface + */ + protected $htmlPageRenderer; + + /** + * Constructs a new DefaultExceptionHtmlSubscriber. + * + * @param \Drupal\Core\Page\HtmlFragmentRendererInterface $fragment_renderer + * The fragment renderer. + * @param \Drupal\Core\Page\HtmlPageRendererInterface $page_renderer + * The page renderer. + */ + public function __construct(HtmlFragmentRendererInterface $fragment_renderer, HtmlPageRendererInterface $page_renderer) { + $this->fragmentRenderer = $fragment_renderer; + $this->htmlPageRenderer = $page_renderer; + } + + /** + * {@inheritdoc} + */ + protected static function getPriority() { + // A very low priority so that custom handlers are almost certain to fire + // before it, even if someone forgets to set a priority. + return -128; + } + + /** + * {@inheritDoc} + */ + protected function getHandledFormats() { + return ['html']; + } + + /** + * Handles a 403 error for HTML. + * + * @param \Symfony\Component\HttpKernel\Event\GetResponseForExceptionEvent $event + * The event to process. + */ + public function on403(GetResponseForExceptionEvent $event) { + $response = $this->createResponse($this->t('Access denied'), $this->t('You are not authorized to access this page.'), Response::HTTP_FORBIDDEN); + $response->headers->set('Content-type', 'text/html'); + $event->setResponse($response); + } + + /** + * Handles a 404 error for HTML. + * + * @param \Symfony\Component\HttpKernel\Event\GetResponseForExceptionEvent $event + * The event to process. + */ + public function on404(GetResponseForExceptionEvent $event) { + $path = $event->getRequest()->getPathInfo(); + $response = $this->createResponse($this->t('Page not found'), $this->t('The requested page "@path" could not be found.', ['@path' => $path]), Response::HTTP_NOT_FOUND); + $response->headers->set('Content-type', 'text/html'); + $event->setResponse($response); + } + + /** + * Handles a 405 error for HTML. + * + * @param \Symfony\Component\HttpKernel\Event\GetResponseForExceptionEvent $event + * The event to process. + */ + public function on405(GetResponseForExceptionEvent $event) { + $response = new Response('Method Not Allowed', Response::HTTP_METHOD_NOT_ALLOWED); + $response->headers->set('Content-type', 'text/html'); + $event->setResponse($response); + } + + /** + * @param $title + * The page title of the response. + * @param $body + * The body of the error page. + * @param $response_code + * The HTTP response code of the response. + * @return Response + * An error Response object ready to return to the browser. + */ + protected function createResponse($title, $body, $response_code) { + $fragment = new HtmlFragment($body); + $fragment->setTitle($title); + + $page = $this->fragmentRenderer->render($fragment, $response_code); + return new Response($this->htmlPageRenderer->render($page), $page->getStatusCode()); + } + +} diff --git a/core/lib/Drupal/Core/EventSubscriber/DefaultExceptionSubscriber.php b/core/lib/Drupal/Core/EventSubscriber/DefaultExceptionSubscriber.php new file mode 100644 index 0000000000000000000000000000000000000000..b8dc94e85c6f44520aef1392debf5682c3499d2a --- /dev/null +++ b/core/lib/Drupal/Core/EventSubscriber/DefaultExceptionSubscriber.php @@ -0,0 +1,276 @@ +<?php + +/** + * @file + * Contains \Drupal\Core\EventSubscriber\DefaultExceptionSubscriber. + */ + +namespace Drupal\Core\EventSubscriber; + +use Drupal\Component\Utility\SafeMarkup; +use Drupal\Component\Utility\String; +use Drupal\Core\Config\ConfigFactoryInterface; +use Drupal\Core\ContentNegotiation; +use Drupal\Core\Page\DefaultHtmlPageRenderer; +use Drupal\Core\Page\HtmlFragment; +use Drupal\Core\Page\HtmlFragmentRendererInterface; +use Drupal\Core\Page\HtmlPageRendererInterface; +use Drupal\Core\StringTranslation\StringTranslationTrait; +use Drupal\Core\Utility\Error; +use Symfony\Component\Debug\Exception\FlattenException; +use Symfony\Component\EventDispatcher\EventSubscriberInterface; +use Symfony\Component\HttpFoundation\Request; +use Symfony\Component\HttpFoundation\Response; +use Symfony\Component\HttpKernel\Event\GetResponseForExceptionEvent; +use Symfony\Component\HttpKernel\Exception\HttpExceptionInterface; +use Symfony\Component\HttpKernel\KernelEvents; + +/** + * Last-chance handler for exceptions. + * + * This handler will catch any exceptions not caught elsewhere and report + * them as an error page. + */ +class DefaultExceptionSubscriber implements EventSubscriberInterface { + use StringTranslationTrait; + + /** + * The fragment renderer. + * + * @var \Drupal\Core\Page\HtmlFragmentRendererInterface + */ + protected $fragmentRenderer; + + /** + * The page renderer. + * + * @var \Drupal\Core\Page\HtmlPageRendererInterface + */ + protected $htmlPageRenderer; + + /** + * @var string + * + * One of the error level constants defined in bootstrap.inc. + */ + protected $errorLevel; + + /** + * The config factory. + * + * @var \Drupal\Core\Config\ConfigFactoryInterface + */ + protected $configFactory; + + /** + * Constructs a new DefaultExceptionHtmlSubscriber. + * + * @param \Drupal\Core\Page\HtmlFragmentRendererInterface $fragment_renderer + * The fragment renderer. + * @param \Drupal\Core\Page\HtmlPageRendererInterface $page_renderer + * The page renderer. + * @param \Drupal\Core\Config\ConfigFactoryInterface $config_factory + * The configuration factory. + */ + public function __construct(HtmlFragmentRendererInterface $fragment_renderer, HtmlPageRendererInterface $page_renderer, ConfigFactoryInterface $config_factory) { + $this->fragmentRenderer = $fragment_renderer; + $this->htmlPageRenderer = $page_renderer; + $this->configFactory = $config_factory; + } + + /** + * Gets the configured error level. + * + * @return string + */ + protected function getErrorLevel() { + if (!isset($this->errorLevel)) { + $this->errorLevel = $this->configFactory->get('system.logging')->get('error_level'); + } + return $this->errorLevel; + } + + /** + * Handles any exception as a generic error page for HTML. + * + * @param \Symfony\Component\HttpKernel\Event\GetResponseForExceptionEvent $event + * The event to process. + */ + protected function onHtml(GetResponseForExceptionEvent $event) { + $exception = $event->getException(); + $error = Error::decodeException($exception); + $flatten_exception = FlattenException::create($exception, 500); + + // Display the message if the current error reporting level allows this type + // of message to be displayed, and unconditionally in update.php. + if (error_displayable($error)) { + $class = 'error'; + + // If error type is 'User notice' then treat it as debug information + // instead of an error message. + // @see debug() + if ($error['%type'] == 'User notice') { + $error['%type'] = 'Debug'; + $class = 'status'; + } + + // Attempt to reduce verbosity by removing DRUPAL_ROOT from the file path + // in the message. This does not happen for (false) security. + $root_length = strlen(DRUPAL_ROOT); + if (substr($error['%file'], 0, $root_length) == DRUPAL_ROOT) { + $error['%file'] = substr($error['%file'], $root_length + 1); + } + // Do not translate the string to avoid errors producing more errors. + unset($error['backtrace']); + $message = String::format('%type: !message in %function (line %line of %file).', $error); + + // Check if verbose error reporting is on. + if ($this->getErrorLevel() == ERROR_REPORTING_DISPLAY_VERBOSE) { + $backtrace_exception = $flatten_exception; + while ($backtrace_exception->getPrevious()) { + $backtrace_exception = $backtrace_exception->getPrevious(); + } + $backtrace = $backtrace_exception->getTrace(); + // First trace is the error itself, already contained in the message. + // While the second trace is the error source and also contained in the + // message, the message doesn't contain argument values, so we output it + // once more in the backtrace. + array_shift($backtrace); + + // Generate a backtrace containing only scalar argument values. + $message .= '<pre class="backtrace">' . Error::formatFlattenedBacktrace($backtrace) . '</pre>'; + } + drupal_set_message(SafeMarkup::set($message), $class, TRUE); + } + + $content = $this->t('The website has encountered an error. Please try again later.'); + $output = DefaultHtmlPageRenderer::renderPage($content, $this->t('Error')); + $response = new Response($output); + + if ($exception instanceof HttpExceptionInterface) { + $response->setStatusCode($exception->getStatusCode()); + } + else { + $response->setStatusCode(Response::HTTP_INTERNAL_SERVER_ERROR, '500 Service unavailable (with message)'); + } + + $event->setResponse($response); + } + + /** + * Handles any exception as a generic error page for JSON. + * + * @todo This should probably check the error reporting level. + * + * @param \Symfony\Component\HttpKernel\Event\GetResponseForExceptionEvent $event + * The event to process. + */ + protected function onJson(GetResponseForExceptionEvent $event) { + $exception = $event->getException(); + $error = Error::decodeException($exception); + + // Display the message if the current error reporting level allows this type + // of message to be displayed, + $message = error_displayable($error) ? 'A fatal error occurred: ' . $exception->getMessage() : ''; + + // @todo We would prefer to use JsonResponse here, but existing code and + // tests are not prepared for parsing a JSON response when there are quotes + // or other values that would cause escaping issues. Instead, return a + // plain string and mark it as such. + $response = new Response($message, Response::HTTP_INTERNAL_SERVER_ERROR, [ + 'Content-type' => 'text/plain' + ]); + if ($exception instanceof HttpExceptionInterface) { + $response->setStatusCode($exception->getStatusCode()); + } + + $event->setResponse($response); + } + + /** + * Creates an Html response for the provided criteria. + * + * @param $title + * The page title of the response. + * @param $body + * The body of the error page. + * @param $response_code + * The HTTP response code of the response. + * @return \Symfony\Component\HttpFoundation\Response + * An error Response object ready to return to the browser. + */ + protected function createHtmlResponse($title, $body, $response_code) { + $fragment = new HtmlFragment($body); + $fragment->setTitle($title); + + $page = $this->fragmentRenderer->render($fragment, $response_code); + return new Response($this->htmlPageRenderer->render($page), $page->getStatusCode()); + } + + /** + * Handles errors for this subscriber. + * + * @param \Symfony\Component\HttpKernel\Event\GetResponseForExceptionEvent $event + * The event to process. + */ + public function onException(GetResponseForExceptionEvent $event) { + $format = $this->getFormat($event->getRequest()); + + // If it's an unrecognized format, assume HTML. + $method = 'on' . $format; + if (!method_exists($this, $method)) { + $method = 'onHtml'; + } + $this->$method($event); + } + + /** + * Gets the error-relevant format from the request. + * + * @param \Symfony\Component\HttpFoundation\Request $request + * The request object. + * + * @return string + * The format as which to treat the exception. + */ + protected function getFormat(Request $request) { + // @todo We are trying to switch to a more robust content negotiation + // library in https://www.drupal.org/node/1505080 that will make + // $request->getRequestFormat() reliable as a better alternative + // to this code. We therefore use this style for now on the expectation + // that it will get replaced with better code later. This approach makes + // that change easier when we get to it. + $conneg = new ContentNegotiation(); + $format = $conneg->getContentType($request); + + // These are all JSON errors for our purposes. Any special handling for + // them can/should happen in earlier listeners if desired. + if (in_array($format, ['drupal_modal', 'drupal_dialog', 'drupal_ajax'])) { + $format = 'json'; + } + + // Make an educated guess that any Accept header type that includes "json" + // can probably handle a generic JSON response for errors. As above, for + // any format this doesn't catch or that wants custom handling should + // register its own exception listener. + foreach ($request->getAcceptableContentTypes() as $mime) { + if (strpos($mime, 'html') === FALSE && strpos($mime, 'json') !== FALSE) { + $format = 'json'; + } + } + + return $format; + } + + /** + * Registers the methods in this class that should be listeners. + * + * @return array + * An array of event listener definitions. + */ + public static function getSubscribedEvents() { + $events[KernelEvents::EXCEPTION][] = ['onException', -256]; + return $events; + } + +} diff --git a/core/lib/Drupal/Core/EventSubscriber/ExceptionJsonSubscriber.php b/core/lib/Drupal/Core/EventSubscriber/ExceptionJsonSubscriber.php new file mode 100644 index 0000000000000000000000000000000000000000..7befb00bec27c1094bf27e031c88c621e97a27e2 --- /dev/null +++ b/core/lib/Drupal/Core/EventSubscriber/ExceptionJsonSubscriber.php @@ -0,0 +1,68 @@ +<?php + +/** + * @file + * Contains \Drupal\Core\EventSubscriber\ExceptionJsonSubscriber. + */ + +namespace Drupal\Core\EventSubscriber; + +use Symfony\Component\HttpFoundation\JsonResponse; +use Symfony\Component\HttpFoundation\Response; +use Symfony\Component\HttpKernel\Event\GetResponseForExceptionEvent; + +/** + * Default handling for JSON errors. + */ +class ExceptionJsonSubscriber extends HttpExceptionSubscriberBase { + + /** + * {@inheritDoc} + */ + protected function getHandledFormats() { + return ['json']; + } + + /** + * {@inheritdoc} + */ + protected static function getPriority() { + // This will fire after the most common HTML handler, since HTML requests + // are still more common than JSON requests. + return -75; + } + + /** + * Handles a 403 error for JSON. + * + * @param \Symfony\Component\HttpKernel\Event\GetResponseForExceptionEvent $event + * The event to process. + */ + public function on403(GetResponseForExceptionEvent $event) { + $response = new JsonResponse(NULL, Response::HTTP_FORBIDDEN); + $event->setResponse($response); + } + + /** + * Handles a 404 error for JSON. + * + * @param \Symfony\Component\HttpKernel\Event\GetResponseForExceptionEvent $event + * The event to process. + */ + public function on404(GetResponseForExceptionEvent $event) { + $response = new JsonResponse(NULL, Response::HTTP_NOT_FOUND); + $event->setResponse($response); + } + + /** + * Handles a 405 error for JSON. + * + * @param \Symfony\Component\HttpKernel\Event\GetResponseForExceptionEvent $event + * The event to process. + */ + public function on405(GetResponseForExceptionEvent $event) { + $response = new JsonResponse(NULL, Response::HTTP_METHOD_NOT_ALLOWED); + $event->setResponse($response); + } + +} diff --git a/core/lib/Drupal/Core/EventSubscriber/ExceptionListener.php b/core/lib/Drupal/Core/EventSubscriber/ExceptionListener.php deleted file mode 100644 index 09447036a90502125365b4da2064388c073ea99d..0000000000000000000000000000000000000000 --- a/core/lib/Drupal/Core/EventSubscriber/ExceptionListener.php +++ /dev/null @@ -1,36 +0,0 @@ -<?php - -/** - * @file - * Contains \Drupal\Core\EventSubscriber\ExceptionListener. - */ - -namespace Drupal\Core\EventSubscriber; - -use Symfony\Component\Debug\Exception\FlattenException; -use Symfony\Component\HttpFoundation\Request; -use Symfony\Component\HttpKernel\EventListener\ExceptionListener as ExceptionListenerBase; -use Symfony\Component\HttpKernel\Log\DebugLoggerInterface; - -/** - * Extends the symfony exception listener to support POST subrequests. - */ -class ExceptionListener extends ExceptionListenerBase { - - /** - * {@inheritdoc} - * - * In contrast to the symfony base class, do not override POST requests to GET - * requests. - */ - protected function duplicateRequest(\Exception $exception, Request $request) { - $attributes = array( - '_controller' => $this->controller, - 'exception' => FlattenException::create($exception), - 'logger' => $this->logger instanceof DebugLoggerInterface ? $this->logger : NULL, - 'format' => $request->getRequestFormat(), - ); - return $request->duplicate(NULL, NULL, $attributes); - } - -} diff --git a/core/lib/Drupal/Core/EventSubscriber/ExceptionLoggingSubscriber.php b/core/lib/Drupal/Core/EventSubscriber/ExceptionLoggingSubscriber.php new file mode 100644 index 0000000000000000000000000000000000000000..3d421763aac032aa7ab585b206804f6bb9f059be --- /dev/null +++ b/core/lib/Drupal/Core/EventSubscriber/ExceptionLoggingSubscriber.php @@ -0,0 +1,113 @@ +<?php + +/** + * @file + * Contains \Drupal\Core\EventSubscriber\ExceptionLoggingSubscriber. + */ + +namespace Drupal\Core\EventSubscriber; + +use Drupal\Component\Utility\String; +use Drupal\Core\Logger\LoggerChannelFactoryInterface; +use Drupal\Core\Utility\Error; +use Symfony\Component\EventDispatcher\EventSubscriberInterface; +use Symfony\Component\HttpKernel\Event\GetResponseForExceptionEvent; +use Symfony\Component\HttpKernel\Exception\HttpExceptionInterface; +use Symfony\Component\HttpKernel\KernelEvents; + +/** + * Log exceptions without further handling. + */ +class ExceptionLoggingSubscriber implements EventSubscriberInterface { + + /** + * The logger channel factory. + * + * @var \Drupal\Core\Logger\LoggerChannelFactoryInterface + */ + protected $logger; + + /** + * Constructs a new ExceptionLoggingSubscriber. + * + * @param \Drupal\Core\Logger\LoggerChannelFactoryInterface $logger + * The logger channel factory. + */ + public function __construct(LoggerChannelFactoryInterface $logger) { + $this->logger = $logger; + } + + /** + * Log 403 errors. + * + * @param \Symfony\Component\HttpKernel\Event\GetResponseForExceptionEvent $event + * The event to process. + */ + public function on403(GetResponseForExceptionEvent $event) { + $request = $event->getRequest(); + // @todo Remove dependency on the internal _system_path attribute: + // https://www.drupal.org/node/2293523. + $this->logger->get('access denied')->warning(String::checkPlain($request->attributes->get('_system_path'))); + } + + /** + * Log 404 errors. + * + * @param \Symfony\Component\HttpKernel\Event\GetResponseForExceptionEvent $event + * The event to process. + */ + public function on404(GetResponseForExceptionEvent $event) { + $request = $event->getRequest(); + // @todo Remove dependency on the internal _system_path attribute: + // https://www.drupal.org/node/2293523. + $this->logger->get('page not found')->warning(String::checkPlain($request->attributes->get('_system_path'))); + } + + /** + * Log not-otherwise-specified errors, including HTTP 500. + * + * @param \Symfony\Component\HttpKernel\Event\GetResponseForExceptionEvent $event + * The event to process. + */ + public function onError(GetResponseForExceptionEvent $event) { + $exception = $event->getException(); + $error = Error::decodeException($exception); + $this->logger->get('php')->log($error['severity_level'], '%type: !message in %function (line %line of %file).', $error); + + $is_critical = !$exception instanceof HttpExceptionInterface || $exception->getStatusCode() >= 500; + if ($is_critical) { + error_log(sprintf('Uncaught PHP Exception %s: "%s" at %s line %s', get_class($exception), $exception->getMessage(), $exception->getFile(), $exception->getLine())); + } + } + + /** + * Log all exceptions. + * + * @param \Symfony\Component\HttpKernel\Event\GetResponseForExceptionEvent $event + * The event to process. + */ + public function onException(GetResponseForExceptionEvent $event) { + $exception = $event->getException(); + + $method = 'onError'; + + // Treat any non-HTTP exception as if it were one, so we log it the same. + if ($exception instanceof HttpExceptionInterface) { + $possible_method = 'on' . $exception->getStatusCode(); + if (method_exists($this, $possible_method)) { + $method = $possible_method; + } + } + + $this->$method($event); + } + + /** + * {@inheritdoc} + */ + public static function getSubscribedEvents() { + $events[KernelEvents::EXCEPTION][] = ['onException', 50]; + return $events; + } + +} diff --git a/core/lib/Drupal/Core/EventSubscriber/ExceptionTestSiteSubscriber.php b/core/lib/Drupal/Core/EventSubscriber/ExceptionTestSiteSubscriber.php new file mode 100644 index 0000000000000000000000000000000000000000..d8f1a61339a534b66783950662b822b8c3240b90 --- /dev/null +++ b/core/lib/Drupal/Core/EventSubscriber/ExceptionTestSiteSubscriber.php @@ -0,0 +1,67 @@ +<?php + +/** + * @file + * Contains \Drupal\Core\EventSubscriber\ExceptionTestSiteSubscriber. + */ + +namespace Drupal\Core\EventSubscriber; + +use Drupal\Core\Utility\Error; +use Symfony\Component\HttpKernel\Event\GetResponseForExceptionEvent; + +/** + * Custom handling of errors when in a system-under-test. + */ +class ExceptionTestSiteSubscriber extends HttpExceptionSubscriberBase { + + /** + * {@inheritdoc} + */ + protected static function getPriority() { + return 3; + } + + /** + * {@inheritDoc} + */ + protected function getHandledFormats() { + return ['html']; + } + + /** + * Checks for special handling of errors inside Simpletest. + * + * @todo The $headers array appears to not actually get used at all in the + * original code. It's quite possible that this entire method is now + * vestigial and can be removed. + * + * @param \Symfony\Component\HttpKernel\Event\GetResponseForExceptionEvent $event + */ + public function on500(GetResponseForExceptionEvent $event) { + $exception = $event->getException(); + $error = Error::decodeException($exception); + + $headers = array(); + + // When running inside the testing framework, we relay the errors + // to the tested site by the way of HTTP headers. + if (DRUPAL_TEST_IN_CHILD_SITE && !headers_sent() && (!defined('SIMPLETEST_COLLECT_ERRORS') || SIMPLETEST_COLLECT_ERRORS)) { + // $number does not use drupal_static as it should not be reset + // as it uniquely identifies each PHP error. + static $number = 0; + $assertion = array( + $error['!message'], + $error['%type'], + array( + 'function' => $error['%function'], + 'file' => $error['%file'], + 'line' => $error['%line'], + ), + ); + $headers['X-Drupal-Assertion-' . $number] = rawurlencode(serialize($assertion)); + $number++; + } + } + +} diff --git a/core/lib/Drupal/Core/EventSubscriber/Fast404ExceptionHtmlSubscriber.php b/core/lib/Drupal/Core/EventSubscriber/Fast404ExceptionHtmlSubscriber.php new file mode 100644 index 0000000000000000000000000000000000000000..c5d18dbeb2f465939693863950574e7d016eeaac --- /dev/null +++ b/core/lib/Drupal/Core/EventSubscriber/Fast404ExceptionHtmlSubscriber.php @@ -0,0 +1,89 @@ +<?php + +/** + * @file + * Contains \Drupal\Core\EventSubscriber\Fast404ExceptionHtmlSubscriber. + */ + +namespace Drupal\Core\EventSubscriber; + +use Drupal\Core\Config\ConfigFactoryInterface; +use Drupal\Component\Utility\String; +use Symfony\Component\HttpFoundation\Response; +use Symfony\Component\HttpKernel\Event\GetResponseForExceptionEvent; +use Symfony\Component\HttpKernel\HttpKernelInterface; + +/** + * High-performance 404 exception subscriber. + * + * This subscriber will return a minimalist 404 response for HTML requests + * without running a full page theming operation. + */ +class Fast404ExceptionHtmlSubscriber extends HttpExceptionSubscriberBase { + + /** + * The HTTP kernel. + * + * @var \Symfony\Component\HttpKernel\HttpKernelInterface + */ + protected $httpKernel; + + /** + * The config factory. + * + * @var \Drupal\Core\Config\ConfigFactoryInterface + */ + protected $configFactory; + + /** + * Constructs a new CustomPageExceptionHtmlSubscriber. + * + * @param \Drupal\Core\Config\ConfigFactoryInterface $config_factory + * The configuration factory. + * @param \Symfony\Component\HttpKernel\HttpKernelInterface $http_kernel + * The HTTP Kernel service. + */ + public function __construct(ConfigFactoryInterface $config_factory, HttpKernelInterface $http_kernel) { + $this->configFactory = $config_factory; + $this->httpKernel = $http_kernel; + } + + + /** + * {@inheritdoc} + */ + protected static function getPriority() { + // A very high priority so that it can take precedent over anything else, + // and thus be fast. + return 200; + } + + /** + * {@inheritDoc} + */ + protected function getHandledFormats() { + return ['html']; + } + + /** + * Handles a 404 error for HTML. + * + * @param \Symfony\Component\HttpKernel\Event\GetResponseForExceptionEvent $event + * The event to process. + */ + public function on404(GetResponseForExceptionEvent $event) { + $request = $event->getRequest(); + + $config = $this->configFactory->get('system.performance'); + $exclude_paths = $config->get('fast_404.exclude_paths'); + if ($config->get('fast_404.enabled') && $exclude_paths && !preg_match($exclude_paths, $request->getPathInfo())) { + $fast_paths = $config->get('fast_404.paths'); + if ($fast_paths && preg_match($fast_paths, $request->getPathInfo())) { + $fast_404_html = strtr($config->get('fast_404.html'), ['@path' => String::checkPlain($request->getUri())]); + $response = new Response($fast_404_html, Response::HTTP_NOT_FOUND); + $event->setResponse($response); + } + } + } + +} diff --git a/core/lib/Drupal/Core/EventSubscriber/HttpExceptionSubscriberBase.php b/core/lib/Drupal/Core/EventSubscriber/HttpExceptionSubscriberBase.php new file mode 100644 index 0000000000000000000000000000000000000000..832505834746aca068fa48178ba80ad5611c63ba --- /dev/null +++ b/core/lib/Drupal/Core/EventSubscriber/HttpExceptionSubscriberBase.php @@ -0,0 +1,129 @@ +<?php + +/** + * @file + * Contains \Drupal\Core\EventSubscriber\HttpExceptionSubscriberBase. + */ + +namespace Drupal\Core\EventSubscriber; + +use Drupal\Core\ContentNegotiation; +use Symfony\Component\Debug\Exception\FlattenException; +use Symfony\Component\EventDispatcher\EventSubscriberInterface; +use Symfony\Component\HttpKernel\Event\GetResponseForExceptionEvent; +use Symfony\Component\HttpKernel\Exception\HttpExceptionInterface; +use Symfony\Component\HttpKernel\KernelEvents; + +/** + * Utility base class for exception subscribers. + * + * A subscriber may extend this class and implement getHandledFormats() to + * indicate which request formats it will respond to. Then implement an on*() + * method for any error code (HTTP response code) that should be handled. For + * example, to handle 404 Not Found messages add a method: + * + * @code + * public function on404(GetResponseForExceptionEvent $event) {} + * @endcode + * + * That method should then call $event->setResponse() to set the response object + * for the exception. Alternatively, it may opt not to do so and then other + * listeners will have the opportunity to handle the exception. + * + * Note: Core provides several important exception listeners by default. In most + * cases, setting the priority of a contrib listener to the default of 0 will + * do what you expect and handle the exceptions you'd expect it to handle. + * If a custom priority is set, be aware of the following core-registered + * listeners. + * + * - Fast404ExceptionHtmlSubscriber: 200. This subscriber will return a + * minimalist, high-performance 404 page for HTML requests. It is not + * recommended to have a priority higher than this one as it will only slow + * down that use case. + * - ExceptionLoggingSubscriber: 50. This subscriber logs all exceptions but + * does not handle them. Do not register a listener with a higher priority + * unless you want exceptions to not get logged, which makes debugging more + * difficult. + * - DefaultExceptionSubscriber: -256. The subscriber of last resort, this will + * provide generic handling for any exception. A listener with a lower + * priority will never get called. + * + * All other core-provided exception handlers have negative priorities so most + * module-provided listeners will naturally take precedence over them. + */ +abstract class HttpExceptionSubscriberBase implements EventSubscriberInterface { + + /** + * Specifies the request formats this subscriber will respond to. + * + * @return array + * An indexed array of the format machine names that this subscriber will + * attempt ot process,such as "html" or "json". Returning an empty array + * will apply to all formats. + * + * @see \Symfony\Component\HttpFoundation\Request + */ + abstract protected function getHandledFormats(); + + /** + * Specifies the priority of all listeners in this class. + * + * The default priority is 1, which is very low. To have listeners that have + * a "first attempt" at handling exceptions return a higher priority. + * + * @return int + * The event priority of this subscriber. + */ + protected static function getPriority() { + return 0; + } + + /** + * Handles errors for this subscriber. + * + * @param \Symfony\Component\HttpKernel\Event\GetResponseForExceptionEvent $event + * The event to process. + */ + public function onException(GetResponseForExceptionEvent $event) { + $exception = $event->getException(); + + // Make the exception available for example when rendering a block. + $event->getRequest()->attributes->set('exception', FlattenException::create($exception)); + + $handled_formats = $this->getHandledFormats(); + + // @todo Injecting this service would force all implementing classes to also + // handle its injection. However, we are trying to switch to a more robust + // content negotiation library in https://www.drupal.org/node/1505080 that + // will make $request->getRequestFormat() reliable as a better alternative + // to this code. We therefore use this style for now on the expectation + // that it will get replaced with better code later. That change will NOT + // be an API change for any implementing classes. (Whereas if we injected + // this class it would be an API change. That's why we're not doing it.) + $conneg = new ContentNegotiation(); + $format = $conneg->getContentType($event->getRequest()); + + if ($exception instanceof HttpExceptionInterface && (empty($handled_formats) || in_array($format, $handled_formats))) { + $method = 'on' . $exception->getStatusCode(); + // We want to allow the method to be called and still not set a response + // if it has additional filtering logic to determine when it will apply. + // It is therefore the method's responsibility to set the response on the + // event if appropriate. + if (method_exists($this, $method)) { + $this->$method($event); + } + } + } + + /** + * Registers the methods in this class that should be listeners. + * + * @return array + * An array of event listener definitions. + */ + public static function getSubscribedEvents() { + $events[KernelEvents::EXCEPTION][] = ['onException', static::getPriority()]; + return $events; + } + +} diff --git a/core/lib/Drupal/Core/EventSubscriber/MaintenanceModeSubscriber.php b/core/lib/Drupal/Core/EventSubscriber/MaintenanceModeSubscriber.php index 8556f9b7923401a704e104761bdb4f2333f9e94d..f09e3e318d2daee03aa030a713e0cf14087c08e0 100644 --- a/core/lib/Drupal/Core/EventSubscriber/MaintenanceModeSubscriber.php +++ b/core/lib/Drupal/Core/EventSubscriber/MaintenanceModeSubscriber.php @@ -130,6 +130,7 @@ protected function drupalSetMessage($message = NULL, $type = 'status', $repeat = */ public static function getSubscribedEvents() { $events[KernelEvents::REQUEST][] = array('onKernelRequestMaintenance', 30); + $events[KernelEvents::EXCEPTION][] = array('onKernelRequestMaintenance'); return $events; } diff --git a/core/lib/Drupal/Core/Routing/Enhancer/ParamConversionEnhancer.php b/core/lib/Drupal/Core/Routing/Enhancer/ParamConversionEnhancer.php index 6785b95c0772e34a96b59fdb779199ffd4c83f06..a8c115935820c0f772217f4ea80e3034368c001c 100644 --- a/core/lib/Drupal/Core/Routing/Enhancer/ParamConversionEnhancer.php +++ b/core/lib/Drupal/Core/Routing/Enhancer/ParamConversionEnhancer.php @@ -86,7 +86,7 @@ public function onException(GetResponseForExceptionEvent $event) { * {@inheritdoc} */ public static function getSubscribedEvents() { - $events[KernelEvents::EXCEPTION][] = array('onException', 0); + $events[KernelEvents::EXCEPTION][] = array('onException', 75); return $events; } diff --git a/core/lib/Drupal/Core/Utility/Error.php b/core/lib/Drupal/Core/Utility/Error.php index 2bfee623e8f4fe9be5e29910be83e951b17ad096..e3bca6dbbd45561c53a30601763b6d428871bfbc 100644 --- a/core/lib/Drupal/Core/Utility/Error.php +++ b/core/lib/Drupal/Core/Utility/Error.php @@ -10,6 +10,7 @@ use Drupal\Component\Utility\String; use Drupal\Component\Utility\Xss; use Drupal\Component\Utility\SafeMarkup; +use Drupal\Core\Database\DatabaseExceptionWrapper; /** * Drupal error utility class. @@ -48,7 +49,7 @@ public static function decodeException(\Exception $exception) { // For PDOException errors, we try to return the initial caller, // skipping internal functions of the database layer. - if ($exception instanceof \PDOException) { + if ($exception instanceof \PDOException || $exception instanceof DatabaseExceptionWrapper) { // The first element in the stack is the call, the second element gives us // the caller. We skip calls that occurred in one of the classes of the // database layer or in one of its global functions. diff --git a/core/modules/shortcut/shortcut.module b/core/modules/shortcut/shortcut.module index 9dc7d5a667d7edc2f95d956ad2ba57d8456ccd9e..13c827bc754376e887ea489fe386a7995dc229f7 100644 --- a/core/modules/shortcut/shortcut.module +++ b/core/modules/shortcut/shortcut.module @@ -320,15 +320,9 @@ function shortcut_preprocess_page(&$variables) { // shortcuts and if the page's actual content is being shown (for example, // we do not want to display it on "access denied" or "page not found" // pages). - $item = array(); - if (\Drupal::routeMatch()->getRouteObject()) { - // @todo What should be done on a 404/403 page? - $item['access'] = TRUE; - } - - if (shortcut_set_edit_access() && !empty($item['access'])) { + if (shortcut_set_edit_access() && !\Drupal::request()->attributes->has('exception')) { $link = current_path(); - if (!($url = Url::createFromPath($link))) { + if (!($url = \Drupal::pathValidator()->getUrlIfValid($link))) { // Bail out early if we couldn't find a matching route. return; } diff --git a/core/modules/simpletest/src/WebTestBase.php b/core/modules/simpletest/src/WebTestBase.php index dc3771361cd8ca3b7c8750b3cc0d850752ffaa72..5a6bba80ff1cb5cba06d6cf56a652be469f3af44 100644 --- a/core/modules/simpletest/src/WebTestBase.php +++ b/core/modules/simpletest/src/WebTestBase.php @@ -1827,6 +1827,10 @@ protected function drupalProcessAjaxResponse($content, array $ajax_response, arr // XPath allows for finding wrapper nodes better than DOM does. $xpath = new \DOMXPath($dom); foreach ($ajax_response as $command) { + // Error messages might be not commands. + if (!is_array($command)) { + continue; + } switch ($command['command']) { case 'settings': $drupal_settings = drupal_merge_js_settings(array($drupal_settings, $command['settings'])); diff --git a/core/modules/system/src/Tests/Routing/ExceptionHandlingTest.php b/core/modules/system/src/Tests/Routing/ExceptionHandlingTest.php new file mode 100644 index 0000000000000000000000000000000000000000..b6ed513ac9087f28e6a8effd1d8bf999eaf0d62c --- /dev/null +++ b/core/modules/system/src/Tests/Routing/ExceptionHandlingTest.php @@ -0,0 +1,120 @@ +<?php + +/** + * @file + * Contains \Drupal\system\Tests\Routing\ExceptionHandlingTest. + */ + +namespace Drupal\system\Tests\Routing; + +use Drupal\simpletest\KernelTestBase; +use Symfony\Component\HttpFoundation\Request; +use Symfony\Component\HttpFoundation\Response; + +/** + * Tests the exception handling for various cases. + * + * @group Routing + */ +class ExceptionHandlingTest extends KernelTestBase { + + /** + * {@inheritdoc} + */ + public static $modules = ['system', 'router_test']; + + /** + * {@inheritdoc} + */ + protected function setUp() { + parent::setUp(); + + $this->installSchema('system', ['router']); + \Drupal::service('router.builder')->rebuild(); + } + + /** + * Tests the exception handling for json and 403 status code. + */ + public function testJson403() { + $request = Request::create('/router_test/test15'); + $request->headers->set('Accept', 'application/json'); + $request->setFormat('json', ['application/json']); + + /** @var \Symfony\Component\HttpKernel\HttpKernelInterface $kernel */ + $kernel = \Drupal::getContainer()->get('http_kernel'); + $response = $kernel->handle($request); + + $this->assertEqual($response->getStatusCode(), Response::HTTP_FORBIDDEN); + $this->assertEqual($response->headers->get('Content-type'), 'application/json'); + $this->assertEqual('{}', $response->getContent()); + } + + /** + * Tests the exception handling for json and 404 status code. + */ + public function testJson404() { + $request = Request::create('/not-found'); + $request->headers->set('Accept', 'application/json'); + $request->setFormat('json', ['application/json']); + + /** @var \Symfony\Component\HttpKernel\HttpKernelInterface $kernel */ + $kernel = \Drupal::getContainer()->get('http_kernel'); + $response = $kernel->handle($request); + + $this->assertEqual($response->getStatusCode(), Response::HTTP_NOT_FOUND); + $this->assertEqual($response->headers->get('Content-type'), 'application/json'); + $this->assertEqual('{}', $response->getContent()); + } + + /** + * Tests the exception handling for HTML and 403 status code. + */ + public function testHtml403() { + $request = Request::create('/router_test/test15'); + $request->headers->set('Accept', 'text/html'); + $request->setFormat('html', ['text/html']); + + /** @var \Symfony\Component\HttpKernel\HttpKernelInterface $kernel */ + $kernel = \Drupal::getContainer()->get('http_kernel'); + $response = $kernel->handle($request); + + $this->assertEqual($response->getStatusCode(), Response::HTTP_FORBIDDEN); + $this->assertEqual($response->headers->get('Content-type'), 'text/html'); + } + + /** + * Tests the exception handling for HTML and 403 status code. + */ + public function testHtml404() { + $request = Request::create('/not-found'); + $request->headers->set('Accept', 'text/html'); + $request->setFormat('html', ['text/html']); + + /** @var \Symfony\Component\HttpKernel\HttpKernelInterface $kernel */ + $kernel = \Drupal::getContainer()->get('http_kernel'); + $response = $kernel->handle($request); + + $this->assertEqual($response->getStatusCode(), Response::HTTP_NOT_FOUND); + $this->assertEqual($response->headers->get('Content-type'), 'text/html'); + } + + /** + * Tests the exception handling for HTML and 405 status code. + */ + public function testHtml405() { + $request = Request::create('/admin', 'NOT_EXISTING'); + $request->headers->set('Accept', 'text/html'); + $request->setFormat('html', ['text/html']); + + /** @var \Symfony\Component\HttpKernel\HttpKernelInterface $kernel */ + $kernel = \Drupal::getContainer()->get('http_kernel'); + $response = $kernel->handle($request); + + $this->assertEqual($response->getStatusCode(), Response::HTTP_METHOD_NOT_ALLOWED); + $this->assertEqual($response->headers->get('Content-type'), 'text/html'); + $this->assertEqual($response->getContent(), 'Method Not Allowed'); + } + +} + diff --git a/core/tests/Drupal/Tests/Core/Controller/ExceptionControllerTest.php b/core/tests/Drupal/Tests/Core/Controller/ExceptionControllerTest.php deleted file mode 100644 index 3a796ec15609c74580e693e7758342a1ac096327..0000000000000000000000000000000000000000 --- a/core/tests/Drupal/Tests/Core/Controller/ExceptionControllerTest.php +++ /dev/null @@ -1,58 +0,0 @@ -<?php - -/** - * @file - * Contains \Drupal\Tests\Core\Controller\ExceptionControllerTest - */ - -namespace Drupal\Tests\Core\Controller { - -use Drupal\Core\Controller\ExceptionController; -use Drupal\Tests\UnitTestCase; -use Symfony\Component\HttpFoundation\Request; -use Symfony\Component\Debug\Exception\FlattenException; - -/** - * @coversDefaultClass \Drupal\Core\Controller\ExceptionController - * @group Controller - */ -class ExceptionControllerTest extends UnitTestCase { - - /** - * Ensure the execute() method returns a valid response on 405 exceptions. - */ - public function test405HTML() { - $exception = new \Exception('Test exception'); - $flat_exception = FlattenException::create($exception, 405); - $html_page_renderer = $this->getMock('Drupal\Core\Page\HtmlPageRendererInterface'); - $html_fragment_renderer = $this->getMock('Drupal\Core\Page\HtmlFragmentRendererInterface'); - $title_resolver = $this->getMock('Drupal\Core\Controller\TitleResolverInterface'); - $translation = $this->getMock('Drupal\Core\StringTranslation\TranslationInterface'); - $url_generator = $this->getMock('Drupal\Core\Routing\UrlGeneratorInterface'); - $logger_factory = $this->getMock('Drupal\Core\Logger\LoggerChannelFactoryInterface'); - - $content_negotiation = $this->getMock('Drupal\Core\ContentNegotiation'); - $content_negotiation->expects($this->any()) - ->method('getContentType') - ->will($this->returnValue('html')); - - $exception_controller = new ExceptionController($content_negotiation, $title_resolver, $html_page_renderer, $html_fragment_renderer, $translation, $url_generator, $logger_factory); - $response = $exception_controller->execute($flat_exception, new Request()); - $this->assertEquals($response->getStatusCode(), 405, 'HTTP status of response is correct.'); - $this->assertEquals($response->getContent(), 'Method Not Allowed', 'HTTP response body is correct.'); - } - -} - -} - -namespace { - use Drupal\Core\Language\Language; - - if (!function_exists('language_default')) { - function language_default() { - $language = new Language(array('langcode' => 'en')); - return $language; - } - } -} diff --git a/core/tests/Drupal/Tests/Core/EventSubscriber/CustomPageExceptionHtmlSubscriberTest.php b/core/tests/Drupal/Tests/Core/EventSubscriber/CustomPageExceptionHtmlSubscriberTest.php new file mode 100644 index 0000000000000000000000000000000000000000..1de4249cf7037389b54c6e189959537f7ec538d4 --- /dev/null +++ b/core/tests/Drupal/Tests/Core/EventSubscriber/CustomPageExceptionHtmlSubscriberTest.php @@ -0,0 +1,141 @@ +<?php + +/** + * @file + * Contains + * \Drupal\Tests\Core\EventSubscriber\CustomPageExceptionHtmlSubscriberTest. + */ + +namespace Drupal\Tests\Core\EventSubscriber; + +use Drupal\Component\Utility\UrlHelper; +use Drupal\Core\EventSubscriber\CustomPageExceptionHtmlSubscriber; +use Drupal\Tests\UnitTestCase; +use Symfony\Component\HttpFoundation\Request; +use Symfony\Component\HttpFoundation\Response; +use Symfony\Component\HttpKernel\Event\GetResponseForExceptionEvent; +use Symfony\Component\HttpKernel\Exception\NotFoundHttpException; + +/** + * @coversDefaultClass \Drupal\Core\EventSubscriber\CustomPageExceptionHtmlSubscriber + * @group EventSubscriber + */ +class CustomPageExceptionHtmlSubscriberTest extends UnitTestCase { + + /** + * The mocked HTTP kernel. + * + * @var \Symfony\Component\HttpKernel\HttpKernelInterface|\PHPUnit_Framework_MockObject_MockObject + */ + protected $kernel; + + /** + * The mocked config factory + * + * @var \Drupal\Core\Config\ConfigFactoryInterface|\PHPUnit_Framework_MockObject_MockObject + */ + protected $configFactory; + + /** + * The mocked alias manager. + * + * @var \Drupal\Core\Path\AliasManagerInterface|\PHPUnit_Framework_MockObject_MockObject + */ + protected $aliasManager; + + /** + * The mocked logger. + * + * @var \Psr\Log\LoggerInterface + */ + protected $logger; + + /** + * The PHP error log settings before the test. + * + * @var string + */ + protected $errorLog; + + /** + * The tested custom page exception subscriber. + * + * @var \Drupal\Core\EventSubscriber\CustomPageExceptionHtmlSubscriber + */ + protected $customPageSubscriber; + + /** + * {@inheritdoc} + */ + protected function setUp() { + $this->configFactory = $this->getConfigFactoryStub(['system.site' => ['page.403' => 'access-denied-page', 'page.404' => 'not-found-page']]); + + $this->aliasManager = $this->getMock('Drupal\Core\Path\AliasManagerInterface'); + $this->kernel = $this->getMock('Symfony\Component\HttpKernel\HttpKernelInterface'); + $this->logger = $this->getMock('Psr\Log\LoggerInterface'); + $this->customPageSubscriber = new CustomPageExceptionHtmlSubscriber($this->configFactory, $this->aliasManager, $this->kernel, $this->logger); + + // You can't create an exception in PHP without throwing it. Store the + // current error_log, and disable it temporarily. + $this->errorLog = ini_set('error_log', file_exists('/dev/null') ? '/dev/null' : 'nul'); + } + + /** + * {@inheritdoc} + */ + protected function tearDown() { + ini_set('error_log', $this->errorLog); + } + + /** + * Sets up an alias manager that does nothing. + */ + protected function setupStubAliasManager() { + $this->aliasManager->expects($this->any()) + ->method('getPathByAlias') + ->willReturnArgument(0); + } + + /** + * Tests onHandleException with a POST request. + */ + public function testHandleWithPostRequest() { + $this->setupStubAliasManager(); + + $request = Request::create('/test', 'POST', array('name' => 'druplicon', 'pass' => '12345')); + + $this->kernel->expects($this->once())->method('handle')->will($this->returnCallback(function (Request $request) { + return new Response($request->getMethod()); + })); + + $event = new GetResponseForExceptionEvent($this->kernel, $request, 'foo', new NotFoundHttpException('foo')); + + $this->customPageSubscriber->onException($event); + + $response = $event->getResponse(); + $result = $response->getContent() . " " . UrlHelper::buildQuery($request->request->all()); + $this->assertEquals('POST name=druplicon&pass=12345', $result); + } + + /** + * Tests onHandleException with a GET request. + */ + public function testHandleWithGetRequest() { + $this->setupStubAliasManager(); + + $request = Request::create('/test', 'GET', array('name' => 'druplicon', 'pass' => '12345')); + $request->attributes->set('_system_path', 'test'); + + $this->kernel->expects($this->once())->method('handle')->will($this->returnCallback(function (Request $request) { + return new Response($request->getMethod() . ' ' . UrlHelper::buildQuery($request->query->all())); + })); + + $event = new GetResponseForExceptionEvent($this->kernel, $request, 'foo', new NotFoundHttpException('foo')); + $this->customPageSubscriber->onException($event); + + $response = $event->getResponse(); + $result = $response->getContent() . " " . UrlHelper::buildQuery($request->request->all()); + $this->assertEquals('GET name=druplicon&pass=12345&destination=test&_exception_statuscode=404 ', $result); + } + +} diff --git a/core/tests/Drupal/Tests/Core/EventSubscriber/ExceptionListenerTest.php b/core/tests/Drupal/Tests/Core/EventSubscriber/ExceptionListenerTest.php deleted file mode 100644 index 5815ee167f9239a4cbeb3f24d10e4e8bcfc753fd..0000000000000000000000000000000000000000 --- a/core/tests/Drupal/Tests/Core/EventSubscriber/ExceptionListenerTest.php +++ /dev/null @@ -1,98 +0,0 @@ -<?php - -/** - * @file - * Contains \Drupal\Tests\Core\EventSubscriber\ExceptionListenerTest. - */ - -namespace Drupal\Tests\Core\EventSubscriber; - -use Drupal\Component\Utility\UrlHelper; -use Drupal\Core\EventSubscriber\ExceptionListener; -use Drupal\Tests\UnitTestCase; -use Symfony\Component\HttpFoundation\Request; -use Symfony\Component\HttpFoundation\Response; -use Symfony\Component\HttpKernel\Event\GetResponseForExceptionEvent; - -/** - * @coversDefaultClass \Drupal\Core\EventSubscriber\ExceptionListener - * @group EventSubscriber - */ -class ExceptionListenerTest extends UnitTestCase { - - /** - * The tested exception listener. - * - * @var \Drupal\Core\EventSubscriber\ExceptionListener - */ - protected $exceptionListener; - - /** - * The mocked HTTP kernel. - * - * @var \Symfony\Component\HttpKernel\HttpKernelInterface|\PHPUnit_Framework_MockObject_MockObject - */ - protected $kernel; - - /** - * The PHP error log settings before the test. - * - * @var string - */ - protected $errorLog; - - /** - * {@inheritdoc} - */ - protected function setUp() { - $this->exceptionListener = new ExceptionListener('example'); - $this->kernel = $this->getMock('Symfony\Component\HttpKernel\HttpKernelInterface'); - - // You can't create an exception in PHP without throwing it. Store the - // current error_log, and disable it temporarily. - $this->errorLog = ini_set('error_log', file_exists('/dev/null') ? '/dev/null' : 'nul'); - } - - /** - * {@inheritdoc} - */ - protected function tearDown() { - ini_set('error_log', $this->errorLog); - } - - /** - * Tests onHandleException with a POST request. - */ - public function testHandleWithPostRequest() { - $request = Request::create('/test', 'POST', array('name' => 'druplicon', 'pass' => '12345')); - - $this->kernel->expects($this->once())->method('handle')->will($this->returnCallback(function (Request $request) { - return new Response($request->getMethod()); - })); - - $event = new GetResponseForExceptionEvent($this->kernel, $request, 'foo', new \Exception('foo')); - - $this->exceptionListener->onKernelException($event); - - $response = $event->getResponse(); - $this->assertEquals('POST name=druplicon&pass=12345', $response->getContent() . " " . UrlHelper::buildQuery($request->request->all())); - } - - /** - * Tests onHandleException with a GET request. - */ - public function testHandleWithGetRequest() { - $request = Request::create('/test', 'GET', array('name' => 'druplicon', 'pass' => '12345')); - - $this->kernel->expects($this->once())->method('handle')->will($this->returnCallback(function (Request $request) { - return new Response($request->getMethod() . ' ' . UrlHelper::buildQuery($request->query->all())); - })); - - $event = new GetResponseForExceptionEvent($this->kernel, $request, 'foo', new \Exception('foo')); - $this->exceptionListener->onKernelException($event); - - $response = $event->getResponse(); - $this->assertEquals('GET name=druplicon&pass=12345 ', $response->getContent() . " " . UrlHelper::buildQuery($request->request->all())); - } - -}