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 .= '
' . Error::formatFlattenedBacktrace($backtrace) . ''; } 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, ); } }