request = $request; $this->dbConnection = $dbConnection; $this->routeCollection = $routeCollection; $this->appConfig = $appConfig; $this->method = strtolower($requestMethod); $this->parsedUrl = parse_url($requestUrl); $this->rawUrl = $requestUrl; } public function exceptionsErrorHandler($severity, $message, $filename, $lineno) { throw new ErrorException($message, 0, $severity, $filename, $lineno); } public function render(): void { $match = $this->routeCollection->match($this->method, $this->parsedUrl['path']); if ($match === null) { $this->render404(); return; } list($route, $params) = $match; $this->request->setParsedRouteParams($params); $handler = $route->getHandler(); $controller = new $handler[0](); if ( $controller instanceof IAuthenticationRequired && $controller->isAuthenticationRequired() && $this->request->user() === null ) { $this->redirectToLogin(); return; } if ( $this->method === 'post' && !in_array($this->parsedUrl['path'], $this->appConfig['antiCsrfTokenExceptions']) && $this->request->post($this->appConfig['antiCsrfTokenName']) !== $this->request->session()->get($this->appConfig['antiCsrfTokenName']) ) { $this->renderAntiCsrfError(); return; } if ($controller instanceof ISecured && !$controller->authorize()) { $this->render404(); return; } $this->dbConnection->startTransaction(); try { $response = call_user_func([$controller, $handler[1]]); } catch (Exception $exception) { $this->dbConnection->rollback(); $this->render500($exception); return; } $this->dbConnection->commit(); if ($response instanceof IContent) { header('Content-Type: ' . $response->getContentType() . '; charset=UTF-8'); $response->render(); } elseif ($response instanceof IRedirect) { header('Location: ' . $this->getRedirectUrl($response), true, $response->getHttpCode()); } else { $this->render404(); } } private function redirectToLogin(): void { $this->request->session()->set('redirect_after_login', $this->rawUrl); $response = new Redirect($this->routeCollection->getRoute($this->appConfig['loginRouteId'])->generateLink(), IRedirect::TEMPORARY); header('Location: ' . $this->getRedirectUrl($response), true, $response->getHttpCode()); } private function renderAntiCsrfError(): void { $content = new JsonContent($this->appConfig['antiCsrfTokenErrorResponse']); header('Content-Type: text/html; charset=UTF-8', true, 403); $content->render(); } private function render404(): void { $content = new HtmlContent($this->appConfig['error404View']); header('Content-Type: text/html; charset=UTF-8', true, 404); $content->render(); } private function render500(Exception $exception): void { if (empty($_ENV['DEV'])) { $exceptionToPrint = null; } else { $exceptionToPrint = (string)$exception; } $content = new HtmlContent($this->appConfig['error500View'], ['exceptionToPrint' => $exceptionToPrint]); header('Content-Type: text/html; charset=UTF-8', true, 500); $content->render(); } private function getRedirectUrl(IRedirect $redirect): string { $url = $redirect->getTarget(); if (preg_match('/^http(s)?/', $url) !== 1) { $url = $this->request->getBase() . $url; } return $url; } }