class Fast404 in Fast 404 8.2
Same name and namespace in other branches
- 8 src/Fast404.php \Drupal\fast404\Fast404
Fast404: A value object for manager Fast 404 logic.
@package Drupal\fast404
Hierarchy
- class \Drupal\fast404\Fast404 uses StringTranslationTrait
Expanded class hierarchy of Fast404
2 files declare their use of Fast404
- fast404.inc in ./
fast404.inc - Optional include that enables Fast 404 logic.
- Fast404EventSubscriber.php in src/
EventSubscriber/ Fast404EventSubscriber.php
File
- src/
Fast404.php, line 18
Namespace
Drupal\fast404View source
class Fast404 {
use StringTranslationTrait;
/**
* Whether Fast 404 logic should be used.
*
* @var bool
*/
public $respond404 = FALSE;
/**
* The current request.
*
* @var \Symfony\Component\HttpFoundation\Request
*/
public $request;
/**
* Whether to load html or respond otherwise.
*
* @var bool
*/
public $loadHtml = TRUE;
/**
* Fast404 constructor.
*
* @param \Symfony\Component\HttpFoundation\Request $request
* The current request.
*/
public function __construct(Request $request) {
$this->request = $request;
}
/**
* Extension check.
*
* A strategy for handling Fast 404 settings.
*/
public function extensionCheck() {
// Get the path from the request.
$path = $this->request
->getPathInfo();
// Ignore calls to the homepage, to avoid unnecessary processing.
if (!isset($path) || $path == '/') {
return;
}
// Check to see if the URL is that of an image derivative.
// If this file does not already exist, it will be handled via Drupal.
if (strpos($path, 'styles/')) {
// Check to see if we will allow anon users to access this page.
if (!Settings::get('fast404_allow_anon_imagecache', TRUE)) {
$cookies = $this->request->cookies
->all();
// At this stage of the game we don't know if the user is logged in via
// regular function calls. Simply look for a session cookie. If we find
// one we'll assume they're logged in.
if (isset($cookies) && is_array($cookies)) {
foreach ($cookies as $cookie) {
if (stristr($cookie, 'SESS')) {
return;
}
}
}
}
else {
return;
}
}
// If we are using URL whitelisting then determine if the current URL is
// whitelisted before running the extension check.
// Check for exact URL matches and assume it's fine if we get one.
if (Settings::get('fast404_url_whitelisting', FALSE)) {
$trimmed_path = ltrim($path, '/');
$allowed = Settings::get('fast404_whitelist', []);
if (in_array($trimmed_path, $allowed)) {
// URL is whitelisted. Assumed good.
return TRUE;
}
}
// Check for whitelisted strings in the URL.
$string_whitelist = Settings::get('fast404_string_whitelisting', FALSE);
if (is_array($string_whitelist)) {
foreach ($string_whitelist as $str) {
if (strstr($path, $str) !== FALSE) {
return;
}
}
}
$extensions = Settings::get('fast404_exts', '/^(?!\\/robots)^(?!\\/system\\/files).*\\.(txt|png|gif|jpe?g|css|js|ico|swf|flv|cgi|bat|pl|dll|exe|asp)$/i');
// Determine if URL contains a blacklisted extension.
if (isset($extensions) && preg_match($extensions, $path, $m)) {
$this->loadHtml = FALSE;
$this
->blockPath();
return;
}
}
/**
* Path check.
*
* Since the path check is a lot more aggressive in its blocking we should
* actually check that the user wants it to be done.
*/
public function pathCheck() {
if (!Settings::get('fast404_path_check', FALSE)) {
return;
}
// Get the path from the request.
$path = $this->request
->getPathInfo();
// Ignore calls to the homepage, to avoid unnecessary processing.
if (!isset($path) || $path == '/') {
return;
}
// If we have a database connection we can use it, otherwise we might be
// initialising it. We remove '/' from the list of possible patterns as it
// exists in the router by default. This means that the query would match
// any path (/%) which is undesirable.
$sql = "SELECT pattern_outline FROM {router} WHERE :path LIKE CONCAT(pattern_outline, '%') AND pattern_outline != '/'";
$result = Database::getConnection()
->query($sql, [
':path' => $path,
])
->fetchField();
if ($result) {
return;
}
// Check the URL alias table for anything that's not a standard Drupal path.
// Remove any trailing slash found in the request path.
$path_noslash = rtrim($path, '/');
$sql = "SELECT id FROM {path_alias} WHERE alias = :alias";
$result = Database::getConnection()
->query($sql, [
':alias' => $path_noslash,
])
->fetchField();
if ($result) {
return;
}
// If we get to here it means nothing has matched the request so we assume
// it's a bad path and block it.
$this
->blockPath();
}
/**
* Block the delivery of this 404 response.
*/
public function blockPath() {
$this->respond404 = TRUE;
}
/**
* Make sure cli calls are not blocked.
*
* @return bool
* Whether the path is blocked or not.
*/
public function isPathBlocked() {
if ($this
->isCli()) {
return FALSE;
}
return $this->respond404;
}
/**
* Prepare a 404 response.
*
* @param bool $return
* Decide whether to return the response object or simply send it.
*
* @return \Symfony\Component\HttpFoundation\Response
* If this returns anything, it will be a response object.
*/
public function response($return = FALSE) {
$message = Settings::get('fast404_html', '<!DOCTYPE html PUBLIC "-//W3C//DTD XHTML+RDFa 1.0//EN" "http://www.w3.org/MarkUp/DTD/xhtml-rdfa-1.dtd"><html xmlns="http://www.w3.org/1999/xhtml"><head><title>404 Not Found</title></head><body><h1>Not Found</h1><p>The requested URL "@path" was not found on this server.</p></body></html>');
$return_gone = Settings::get('fast404_return_gone', FALSE);
$custom_404_path = Settings::get('fast404_HTML_error_page', FALSE);
// If a file is set to provide us with Fast 404 joy, load it.
if (($this->loadHtml || Settings::get('fast404_HTML_error_all_paths', FALSE) === TRUE) && file_exists($custom_404_path)) {
$message = @file_get_contents($custom_404_path, FALSE);
}
$headers = [
Settings::get('fast404_HTTP_status_method', 'mod_php') === 'FastCGI' ? 'Status:' : 'HTTP/1.0' => $return_gone ? '410 Gone' : '404 Not Found',
];
$response = new Response(new FormattableMarkup($message, [
'@path' => $this->request
->getPathInfo(),
]), $return_gone ? 410 : 404, $headers);
if ($return) {
return $response;
}
$response
->send();
throw new ServiceUnavailableHttpException(3, 'The requested URL "@path" was not found on this server. Try again shortly.', [
'@path' => $this->request
->getPathInfo(),
]);
}
/**
* Check the type of interface between web server and PHP is CLI.
*
* @return bool
* Whether or not the Server API for this build of PHP is CLI.
*/
protected function isCli() {
return PHP_SAPI === 'cli';
}
}
Members
Name | Modifiers | Type | Description | Overrides |
---|---|---|---|---|
Fast404:: |
public | property | Whether to load html or respond otherwise. | |
Fast404:: |
public | property | The current request. | |
Fast404:: |
public | property | Whether Fast 404 logic should be used. | |
Fast404:: |
public | function | Block the delivery of this 404 response. | |
Fast404:: |
public | function | Extension check. | |
Fast404:: |
protected | function | Check the type of interface between web server and PHP is CLI. | |
Fast404:: |
public | function | Make sure cli calls are not blocked. | |
Fast404:: |
public | function | Path check. | |
Fast404:: |
public | function | Prepare a 404 response. | |
Fast404:: |
public | function | Fast404 constructor. | |
StringTranslationTrait:: |
protected | property | The string translation service. | 1 |
StringTranslationTrait:: |
protected | function | Formats a string containing a count of items. | |
StringTranslationTrait:: |
protected | function | Returns the number of plurals supported by a given language. | |
StringTranslationTrait:: |
protected | function | Gets the string translation service. | |
StringTranslationTrait:: |
public | function | Sets the string translation service to use. | 2 |
StringTranslationTrait:: |
protected | function | Translates a string to the current language or to a given language. |