Edit File by line
/home/zeestwma/ceyloniy.../wp-conte.../plugins/litespee.../src
File: core.cls.php
<?php
[0] Fix | Delete
/**
[1] Fix | Delete
* The core plugin class.
[2] Fix | Delete
*
[3] Fix | Delete
* This is the main class for the LiteSpeed Cache plugin, responsible for initializing
[4] Fix | Delete
* the plugin's core functionality, registering hooks, and handling cache-related operations.
[5] Fix | Delete
*
[6] Fix | Delete
* Note: Core doesn't allow $this->cls( 'Core' )
[7] Fix | Delete
*
[8] Fix | Delete
* @since 1.0.0
[9] Fix | Delete
* @package LiteSpeed
[10] Fix | Delete
*/
[11] Fix | Delete
[12] Fix | Delete
namespace LiteSpeed;
[13] Fix | Delete
[14] Fix | Delete
defined( 'WPINC' ) || exit();
[15] Fix | Delete
[16] Fix | Delete
/**
[17] Fix | Delete
* Class Core
[18] Fix | Delete
*
[19] Fix | Delete
* @since 1.0.0
[20] Fix | Delete
*/
[21] Fix | Delete
class Core extends Root {
[22] Fix | Delete
[23] Fix | Delete
const NAME = 'LiteSpeed Cache';
[24] Fix | Delete
const PLUGIN_NAME = 'litespeed-cache';
[25] Fix | Delete
const PLUGIN_FILE = 'litespeed-cache/litespeed-cache.php';
[26] Fix | Delete
const VER = LSCWP_V;
[27] Fix | Delete
[28] Fix | Delete
const ACTION_DISMISS = 'dismiss';
[29] Fix | Delete
const ACTION_PURGE_BY = 'PURGE_BY';
[30] Fix | Delete
const ACTION_PURGE_EMPTYCACHE = 'PURGE_EMPTYCACHE';
[31] Fix | Delete
const ACTION_QS_PURGE = 'PURGE';
[32] Fix | Delete
const ACTION_QS_PURGE_SINGLE = 'PURGESINGLE'; // This will be same as `ACTION_QS_PURGE` (purge single URL only)
[33] Fix | Delete
const ACTION_QS_SHOW_HEADERS = 'SHOWHEADERS';
[34] Fix | Delete
const ACTION_QS_PURGE_ALL = 'purge_all';
[35] Fix | Delete
const ACTION_QS_PURGE_EMPTYCACHE = 'empty_all';
[36] Fix | Delete
const ACTION_QS_NOCACHE = 'NOCACHE';
[37] Fix | Delete
[38] Fix | Delete
const HEADER_DEBUG = 'X-LiteSpeed-Debug';
[39] Fix | Delete
[40] Fix | Delete
/**
[41] Fix | Delete
* Whether to show debug headers.
[42] Fix | Delete
*
[43] Fix | Delete
* @var bool
[44] Fix | Delete
* @since 1.0.0
[45] Fix | Delete
*/
[46] Fix | Delete
protected static $debug_show_header = false;
[47] Fix | Delete
[48] Fix | Delete
/**
[49] Fix | Delete
* Footer comment buffer.
[50] Fix | Delete
*
[51] Fix | Delete
* @var string
[52] Fix | Delete
* @since 1.0.0
[53] Fix | Delete
*/
[54] Fix | Delete
private $footer_comment = '';
[55] Fix | Delete
[56] Fix | Delete
/**
[57] Fix | Delete
* Define the core functionality of the plugin.
[58] Fix | Delete
*
[59] Fix | Delete
* Set the plugin name and the plugin version that can be used throughout the plugin.
[60] Fix | Delete
* Load the dependencies, define the locale, and set the hooks for the admin area and
[61] Fix | Delete
* the public-facing side of the site.
[62] Fix | Delete
*
[63] Fix | Delete
* @since 1.0.0
[64] Fix | Delete
*/
[65] Fix | Delete
public function __construct() {
[66] Fix | Delete
! defined( 'LSCWP_TS_0' ) && define( 'LSCWP_TS_0', microtime( true ) );
[67] Fix | Delete
$this->cls( 'Conf' )->init();
[68] Fix | Delete
[69] Fix | Delete
/**
[70] Fix | Delete
* Load API hooks
[71] Fix | Delete
*
[72] Fix | Delete
* @since 3.0
[73] Fix | Delete
*/
[74] Fix | Delete
$this->cls( 'API' )->init();
[75] Fix | Delete
[76] Fix | Delete
if ( defined( 'LITESPEED_ON' ) ) {
[77] Fix | Delete
// Load third party detection if lscache enabled.
[78] Fix | Delete
include_once LSCWP_DIR . 'thirdparty/entry.inc.php';
[79] Fix | Delete
}
[80] Fix | Delete
[81] Fix | Delete
[82] Fix | Delete
if ( $this->conf( Base::O_DEBUG_DISABLE_ALL ) || Debug2::is_tmp_disable() ) {
[83] Fix | Delete
! defined( 'LITESPEED_DISABLE_ALL' ) && define( 'LITESPEED_DISABLE_ALL', true );
[84] Fix | Delete
}
[85] Fix | Delete
[86] Fix | Delete
/**
[87] Fix | Delete
* Register plugin activate/deactivate/uninstall hooks
[88] Fix | Delete
* NOTE: this can't be moved under after_setup_theme, otherwise activation will be bypassed
[89] Fix | Delete
*
[90] Fix | Delete
* @since 2.7.1 Disabled admin&CLI check to make frontend able to enable cache too
[91] Fix | Delete
*/
[92] Fix | Delete
$plugin_file = LSCWP_DIR . 'litespeed-cache.php';
[93] Fix | Delete
register_activation_hook( $plugin_file, [ __NAMESPACE__ . '\Activation', 'register_activation' ] );
[94] Fix | Delete
register_deactivation_hook( $plugin_file, [ __NAMESPACE__ . '\Activation', 'register_deactivation' ] );
[95] Fix | Delete
register_uninstall_hook( $plugin_file, __NAMESPACE__ . '\Activation::uninstall_litespeed_cache' );
[96] Fix | Delete
[97] Fix | Delete
if ( defined( 'LITESPEED_ON' ) ) {
[98] Fix | Delete
// Register purge_all actions
[99] Fix | Delete
$purge_all_events = $this->conf( Base::O_PURGE_HOOK_ALL );
[100] Fix | Delete
[101] Fix | Delete
// Purge all on upgrade
[102] Fix | Delete
if ( $this->conf( Base::O_PURGE_ON_UPGRADE ) ) {
[103] Fix | Delete
$purge_all_events[] = 'automatic_updates_complete';
[104] Fix | Delete
$purge_all_events[] = 'upgrader_process_complete';
[105] Fix | Delete
$purge_all_events[] = 'admin_action_do-plugin-upgrade';
[106] Fix | Delete
}
[107] Fix | Delete
foreach ( $purge_all_events as $event ) {
[108] Fix | Delete
// Don't allow hook to update_option because purge_all will cause infinite loop of update_option
[109] Fix | Delete
if ( in_array( $event, [ 'update_option' ], true ) ) {
[110] Fix | Delete
continue;
[111] Fix | Delete
}
[112] Fix | Delete
add_action( $event, __NAMESPACE__ . '\Purge::purge_all' );
[113] Fix | Delete
}
[114] Fix | Delete
[115] Fix | Delete
// Add headers to site health check for full page cache
[116] Fix | Delete
// @since 5.4
[117] Fix | Delete
add_filter( 'site_status_page_cache_supported_cache_headers', function ( $cache_headers ) {
[118] Fix | Delete
$is_cache_hit = function ( $header_value ) {
[119] Fix | Delete
return false !== strpos( strtolower( $header_value ), 'hit' );
[120] Fix | Delete
};
[121] Fix | Delete
$cache_headers['x-litespeed-cache'] = $is_cache_hit;
[122] Fix | Delete
$cache_headers['x-lsadc-cache'] = $is_cache_hit;
[123] Fix | Delete
$cache_headers['x-qc-cache'] = $is_cache_hit;
[124] Fix | Delete
return $cache_headers;
[125] Fix | Delete
} );
[126] Fix | Delete
}
[127] Fix | Delete
[128] Fix | Delete
add_action( 'after_setup_theme', [ $this, 'init' ] );
[129] Fix | Delete
[130] Fix | Delete
// Check if there is a purge request in queue
[131] Fix | Delete
if ( ! defined( 'LITESPEED_CLI' ) ) {
[132] Fix | Delete
$purge_queue = Purge::get_option( Purge::DB_QUEUE );
[133] Fix | Delete
if ( $purge_queue && '-1' !== $purge_queue ) {
[134] Fix | Delete
$this->http_header( $purge_queue );
[135] Fix | Delete
Debug2::debug( '[Core] Purge Queue found&sent: ' . $purge_queue );
[136] Fix | Delete
}
[137] Fix | Delete
if ( '-1' !== $purge_queue ) {
[138] Fix | Delete
Purge::update_option( Purge::DB_QUEUE, '-1' ); // Use -1 to bypass purge while still enable db update as WP's update_option will check value===false to bypass update
[139] Fix | Delete
}
[140] Fix | Delete
[141] Fix | Delete
$purge_queue = Purge::get_option( Purge::DB_QUEUE2 );
[142] Fix | Delete
if ( $purge_queue && '-1' !== $purge_queue ) {
[143] Fix | Delete
$this->http_header( $purge_queue );
[144] Fix | Delete
Debug2::debug( '[Core] Purge2 Queue found&sent: ' . $purge_queue );
[145] Fix | Delete
}
[146] Fix | Delete
if ( '-1' !== $purge_queue ) {
[147] Fix | Delete
Purge::update_option( Purge::DB_QUEUE2, '-1' );
[148] Fix | Delete
}
[149] Fix | Delete
}
[150] Fix | Delete
[151] Fix | Delete
/**
[152] Fix | Delete
* Hook internal REST
[153] Fix | Delete
*
[154] Fix | Delete
* @since 2.9.4
[155] Fix | Delete
*/
[156] Fix | Delete
$this->cls( 'REST' );
[157] Fix | Delete
[158] Fix | Delete
/**
[159] Fix | Delete
* Hook wpnonce function
[160] Fix | Delete
*
[161] Fix | Delete
* Note: ESI nonce won't be available until hook after_setup_theme ESI init due to Guest Mode concern
[162] Fix | Delete
*
[163] Fix | Delete
* @since 4.1
[164] Fix | Delete
*/
[165] Fix | Delete
if ( $this->cls( 'Router' )->esi_enabled() && ! function_exists( 'wp_create_nonce' ) ) {
[166] Fix | Delete
Debug2::debug( '[ESI] Overwrite wp_create_nonce()' );
[167] Fix | Delete
litespeed_define_nonce_func();
[168] Fix | Delete
}
[169] Fix | Delete
}
[170] Fix | Delete
[171] Fix | Delete
/**
[172] Fix | Delete
* The plugin initializer.
[173] Fix | Delete
*
[174] Fix | Delete
* This function checks if the cache is enabled and ready to use, then determines what actions need to be set up based on the type of user and page accessed. Output is buffered if the cache is enabled.
[175] Fix | Delete
*
[176] Fix | Delete
* NOTE: WP user doesn't init yet
[177] Fix | Delete
*
[178] Fix | Delete
* @since 1.0.0
[179] Fix | Delete
*/
[180] Fix | Delete
public function init() {
[181] Fix | Delete
/**
[182] Fix | Delete
* Added hook before init
[183] Fix | Delete
* 3rd party preload hooks will be fired here too (e.g. Divi disable all in edit mode)
[184] Fix | Delete
*
[185] Fix | Delete
* @since 1.6.6
[186] Fix | Delete
* @since 2.6 Added filter to all config values in Conf
[187] Fix | Delete
*/
[188] Fix | Delete
do_action( 'litespeed_init' );
[189] Fix | Delete
add_action( 'wp_ajax_async_litespeed', 'LiteSpeed\Task::async_litespeed_handler' );
[190] Fix | Delete
add_action( 'wp_ajax_nopriv_async_litespeed', 'LiteSpeed\Task::async_litespeed_handler' );
[191] Fix | Delete
[192] Fix | Delete
// In `after_setup_theme`, before `init` hook
[193] Fix | Delete
$this->cls( 'Activation' )->auto_update();
[194] Fix | Delete
[195] Fix | Delete
if ( is_admin() && ! wp_doing_ajax() ) {
[196] Fix | Delete
$this->cls( 'Admin' );
[197] Fix | Delete
}
[198] Fix | Delete
[199] Fix | Delete
if ( defined( 'LITESPEED_DISABLE_ALL' ) && LITESPEED_DISABLE_ALL ) {
[200] Fix | Delete
Debug2::debug( '[Core] Bypassed due to debug disable all setting' );
[201] Fix | Delete
return;
[202] Fix | Delete
}
[203] Fix | Delete
[204] Fix | Delete
do_action( 'litespeed_initing' );
[205] Fix | Delete
[206] Fix | Delete
ob_start( [ $this, 'send_headers_force' ] );
[207] Fix | Delete
add_action( 'shutdown', [ $this, 'send_headers' ], 0 );
[208] Fix | Delete
add_action( 'wp_footer', [ $this, 'footer_hook' ] );
[209] Fix | Delete
[210] Fix | Delete
/**
[211] Fix | Delete
* Check if is non-optimization simulator
[212] Fix | Delete
*
[213] Fix | Delete
* @since 2.9
[214] Fix | Delete
*/
[215] Fix | Delete
// phpcs:ignore WordPress.Security.NonceVerification.Recommended
[216] Fix | Delete
if ( ! empty( $_GET[ Router::ACTION ] ) && 'before_optm' === $_GET[ Router::ACTION ] && ! apply_filters( 'litespeed_qs_forbidden', false ) ) {
[217] Fix | Delete
Debug2::debug( '[Core] ⛑️ bypass_optm due to QS CTRL' );
[218] Fix | Delete
! defined( 'LITESPEED_NO_OPTM' ) && define( 'LITESPEED_NO_OPTM', true );
[219] Fix | Delete
}
[220] Fix | Delete
[221] Fix | Delete
/**
[222] Fix | Delete
* Register vary filter
[223] Fix | Delete
*
[224] Fix | Delete
* @since 1.6.2
[225] Fix | Delete
*/
[226] Fix | Delete
$this->cls( 'Control' )->init();
[227] Fix | Delete
[228] Fix | Delete
// Init Purge hooks
[229] Fix | Delete
$this->cls( 'Purge' )->init();
[230] Fix | Delete
[231] Fix | Delete
$this->cls( 'Tag' )->init();
[232] Fix | Delete
[233] Fix | Delete
// Load hooks that may be related to users
[234] Fix | Delete
add_action( 'init', [ $this, 'after_user_init' ], 5 );
[235] Fix | Delete
[236] Fix | Delete
// Load 3rd party hooks
[237] Fix | Delete
add_action( 'wp_loaded', [ $this, 'load_thirdparty' ], 2 );
[238] Fix | Delete
}
[239] Fix | Delete
[240] Fix | Delete
/**
[241] Fix | Delete
* Run hooks after user init
[242] Fix | Delete
*
[243] Fix | Delete
* @since 2.9.8
[244] Fix | Delete
*/
[245] Fix | Delete
public function after_user_init() {
[246] Fix | Delete
$this->cls( 'Router' )->is_role_simulation();
[247] Fix | Delete
[248] Fix | Delete
// Detect if is Guest mode or not
[249] Fix | Delete
$this->cls( 'Vary' )->after_user_init();
[250] Fix | Delete
[251] Fix | Delete
// Register attachment delete hook
[252] Fix | Delete
$this->cls( 'Media' )->after_user_init();
[253] Fix | Delete
[254] Fix | Delete
/**
[255] Fix | Delete
* Preload ESI functionality for ESI request URI recovery
[256] Fix | Delete
*
[257] Fix | Delete
* @since 1.8.1
[258] Fix | Delete
* @since 4.0 ESI init needs to be after Guest mode detection to bypass ESI if is under Guest mode
[259] Fix | Delete
*/
[260] Fix | Delete
$this->cls( 'ESI' )->init();
[261] Fix | Delete
[262] Fix | Delete
if ( ! is_admin() && ! defined( 'LITESPEED_GUEST_OPTM' ) ) {
[263] Fix | Delete
$result = $this->cls( 'Conf' )->in_optm_exc_roles();
[264] Fix | Delete
if ( $result ) {
[265] Fix | Delete
Debug2::debug( '[Core] ⛑️ bypass_optm: hit Role Excludes setting: ' . $result );
[266] Fix | Delete
! defined( 'LITESPEED_NO_OPTM' ) && define( 'LITESPEED_NO_OPTM', true );
[267] Fix | Delete
}
[268] Fix | Delete
}
[269] Fix | Delete
[270] Fix | Delete
// Heartbeat control
[271] Fix | Delete
$this->cls( 'Tool' )->heartbeat();
[272] Fix | Delete
[273] Fix | Delete
if ( ! defined( 'LITESPEED_NO_OPTM' ) || ! LITESPEED_NO_OPTM ) {
[274] Fix | Delete
// Check missing static files
[275] Fix | Delete
$this->cls( 'Router' )->serve_static();
[276] Fix | Delete
[277] Fix | Delete
$this->cls( 'Media' )->init();
[278] Fix | Delete
[279] Fix | Delete
$this->cls( 'Placeholder' )->init();
[280] Fix | Delete
[281] Fix | Delete
$this->cls( 'Router' )->can_optm() && $this->cls( 'Optimize' )->init();
[282] Fix | Delete
[283] Fix | Delete
$this->cls( 'Localization' )->init();
[284] Fix | Delete
[285] Fix | Delete
// Hook CDN for attachments
[286] Fix | Delete
$this->cls( 'CDN' )->init();
[287] Fix | Delete
[288] Fix | Delete
// Load cron tasks
[289] Fix | Delete
$this->cls( 'Task' )->init();
[290] Fix | Delete
}
[291] Fix | Delete
[292] Fix | Delete
// Load litespeed actions
[293] Fix | Delete
$action = Router::get_action();
[294] Fix | Delete
if ( $action ) {
[295] Fix | Delete
$this->proceed_action( $action );
[296] Fix | Delete
}
[297] Fix | Delete
[298] Fix | Delete
// Load frontend GUI
[299] Fix | Delete
if ( ! is_admin() ) {
[300] Fix | Delete
$this->cls( 'GUI' )->init();
[301] Fix | Delete
}
[302] Fix | Delete
}
[303] Fix | Delete
[304] Fix | Delete
/**
[305] Fix | Delete
* Run frontend actions
[306] Fix | Delete
*
[307] Fix | Delete
* @since 1.1.0
[308] Fix | Delete
* @param string $action The action to proceed.
[309] Fix | Delete
*/
[310] Fix | Delete
public function proceed_action( $action ) {
[311] Fix | Delete
$msg = false;
[312] Fix | Delete
// Handle actions
[313] Fix | Delete
switch ( $action ) {
[314] Fix | Delete
case self::ACTION_QS_SHOW_HEADERS:
[315] Fix | Delete
self::$debug_show_header = true;
[316] Fix | Delete
break;
[317] Fix | Delete
[318] Fix | Delete
case self::ACTION_QS_PURGE:
[319] Fix | Delete
case self::ACTION_QS_PURGE_SINGLE:
[320] Fix | Delete
Purge::set_purge_single();
[321] Fix | Delete
break;
[322] Fix | Delete
[323] Fix | Delete
case self::ACTION_QS_PURGE_ALL:
[324] Fix | Delete
Purge::purge_all();
[325] Fix | Delete
break;
[326] Fix | Delete
[327] Fix | Delete
case self::ACTION_PURGE_EMPTYCACHE:
[328] Fix | Delete
case self::ACTION_QS_PURGE_EMPTYCACHE:
[329] Fix | Delete
define( 'LSWCP_EMPTYCACHE', true ); // Clear all sites caches
[330] Fix | Delete
Purge::purge_all();
[331] Fix | Delete
$msg = __( 'Notified LiteSpeed Web Server to purge everything.', 'litespeed-cache' );
[332] Fix | Delete
break;
[333] Fix | Delete
[334] Fix | Delete
case self::ACTION_PURGE_BY:
[335] Fix | Delete
$this->cls( 'Purge' )->purge_list();
[336] Fix | Delete
$msg = __( 'Notified LiteSpeed Web Server to purge the list.', 'litespeed-cache' );
[337] Fix | Delete
break;
[338] Fix | Delete
[339] Fix | Delete
case self::ACTION_DISMISS:
[340] Fix | Delete
GUI::dismiss();
[341] Fix | Delete
break;
[342] Fix | Delete
[343] Fix | Delete
default:
[344] Fix | Delete
$msg = $this->cls( 'Router' )->handler( $action );
[345] Fix | Delete
break;
[346] Fix | Delete
}
[347] Fix | Delete
if ( $msg && ! Router::is_ajax() ) {
[348] Fix | Delete
Admin_Display::add_notice( Admin_Display::NOTICE_GREEN, $msg );
[349] Fix | Delete
Admin::redirect();
[350] Fix | Delete
return;
[351] Fix | Delete
}
[352] Fix | Delete
[353] Fix | Delete
if ( Router::is_ajax() ) {
[354] Fix | Delete
exit();
[355] Fix | Delete
}
[356] Fix | Delete
}
[357] Fix | Delete
[358] Fix | Delete
/**
[359] Fix | Delete
* Callback used to call the detect third party action.
[360] Fix | Delete
*
[361] Fix | Delete
* The detect action is used by third party plugin integration classes to determine if they should add the rest of their hooks.
[362] Fix | Delete
*
[363] Fix | Delete
* @since 1.0.5
[364] Fix | Delete
*/
[365] Fix | Delete
public function load_thirdparty() {
[366] Fix | Delete
do_action( 'litespeed_load_thirdparty' );
[367] Fix | Delete
}
[368] Fix | Delete
[369] Fix | Delete
/**
[370] Fix | Delete
* Mark wp_footer called
[371] Fix | Delete
*
[372] Fix | Delete
* @since 1.3
[373] Fix | Delete
*/
[374] Fix | Delete
public function footer_hook() {
[375] Fix | Delete
Debug2::debug( '[Core] Footer hook called' );
[376] Fix | Delete
if ( ! defined( 'LITESPEED_FOOTER_CALLED' ) ) {
[377] Fix | Delete
define( 'LITESPEED_FOOTER_CALLED', true );
[378] Fix | Delete
}
[379] Fix | Delete
}
[380] Fix | Delete
[381] Fix | Delete
/**
[382] Fix | Delete
* Trigger comment info display hook
[383] Fix | Delete
*
[384] Fix | Delete
* @since 1.3
[385] Fix | Delete
* @param string|null $buffer The buffer to check.
[386] Fix | Delete
* @return void
[387] Fix | Delete
*/
[388] Fix | Delete
private function check_is_html( $buffer = null ) {
[389] Fix | Delete
if ( ! defined( 'LITESPEED_FOOTER_CALLED' ) ) {
[390] Fix | Delete
Debug2::debug2( '[Core] CHK html bypass: miss footer const' );
[391] Fix | Delete
return;
[392] Fix | Delete
}
[393] Fix | Delete
[394] Fix | Delete
if ( wp_doing_ajax() ) {
[395] Fix | Delete
Debug2::debug2( '[Core] CHK html bypass: doing ajax' );
[396] Fix | Delete
return;
[397] Fix | Delete
}
[398] Fix | Delete
[399] Fix | Delete
if ( wp_doing_cron() ) {
[400] Fix | Delete
Debug2::debug2( '[Core] CHK html bypass: doing cron' );
[401] Fix | Delete
return;
[402] Fix | Delete
}
[403] Fix | Delete
[404] Fix | Delete
if ( empty( $_SERVER['REQUEST_METHOD'] ) || 'GET' !== $_SERVER['REQUEST_METHOD'] ) {
[405] Fix | Delete
// phpcs:ignore WordPress.Security.ValidatedSanitizedInput.InputNotSanitized
[406] Fix | Delete
Debug2::debug2( '[Core] CHK html bypass: not get method ' . wp_unslash( $_SERVER['REQUEST_METHOD'] ) );
[407] Fix | Delete
return;
[408] Fix | Delete
}
[409] Fix | Delete
[410] Fix | Delete
if ( null === $buffer ) {
[411] Fix | Delete
$buffer = ob_get_contents();
[412] Fix | Delete
}
[413] Fix | Delete
[414] Fix | Delete
// Double check to make sure it is an HTML file
[415] Fix | Delete
if ( strlen( $buffer ) > 300 ) {
[416] Fix | Delete
$buffer = substr( $buffer, 0, 300 );
[417] Fix | Delete
}
[418] Fix | Delete
if ( false !== strstr( $buffer, '<!--' ) ) {
[419] Fix | Delete
$buffer = preg_replace( '/<!--.*?-->/s', '', $buffer );
[420] Fix | Delete
}
[421] Fix | Delete
$buffer = trim( $buffer );
[422] Fix | Delete
[423] Fix | Delete
$buffer = File::remove_zero_space( $buffer );
[424] Fix | Delete
[425] Fix | Delete
$is_html = 0 === stripos( $buffer, '<html' ) || 0 === stripos( $buffer, '<!DOCTYPE' );
[426] Fix | Delete
[427] Fix | Delete
if ( ! $is_html ) {
[428] Fix | Delete
Debug2::debug( '[Core] Footer check failed: ' . ob_get_level() . '-' . substr( $buffer, 0, 100 ) );
[429] Fix | Delete
return;
[430] Fix | Delete
}
[431] Fix | Delete
[432] Fix | Delete
Debug2::debug( '[Core] Footer check passed' );
[433] Fix | Delete
[434] Fix | Delete
if ( ! defined( 'LITESPEED_IS_HTML' ) ) {
[435] Fix | Delete
define( 'LITESPEED_IS_HTML', true );
[436] Fix | Delete
}
[437] Fix | Delete
}
[438] Fix | Delete
[439] Fix | Delete
/**
[440] Fix | Delete
* For compatibility with plugins that have 'Bad' logic that forced all buffer output even if it is NOT their buffer.
[441] Fix | Delete
*
[442] Fix | Delete
* Usually this is called after send_headers() if following original WP process
[443] Fix | Delete
*
[444] Fix | Delete
* @since 1.1.5
[445] Fix | Delete
* @param string $buffer The buffer to process.
[446] Fix | Delete
* @return string The processed buffer.
[447] Fix | Delete
*/
[448] Fix | Delete
public function send_headers_force( $buffer ) {
[449] Fix | Delete
$this->check_is_html( $buffer );
[450] Fix | Delete
[451] Fix | Delete
// Hook to modify buffer before
[452] Fix | Delete
$buffer = apply_filters( 'litespeed_buffer_before', $buffer );
[453] Fix | Delete
[454] Fix | Delete
/**
[455] Fix | Delete
* Media: Image lazyload && WebP
[456] Fix | Delete
* GUI: Clean wrapper mainly for ESI block NOTE: this needs to be before optimizer to avoid wrapper being removed
[457] Fix | Delete
* Optimize
[458] Fix | Delete
* CDN
[459] Fix | Delete
*/
[460] Fix | Delete
if ( ! defined( 'LITESPEED_NO_OPTM' ) || ! LITESPEED_NO_OPTM ) {
[461] Fix | Delete
Debug2::debug( '[Core] run hook litespeed_buffer_finalize' );
[462] Fix | Delete
$buffer = apply_filters( 'litespeed_buffer_finalize', $buffer );
[463] Fix | Delete
}
[464] Fix | Delete
[465] Fix | Delete
/**
[466] Fix | Delete
* Replace ESI preserved list
[467] Fix | Delete
*
[468] Fix | Delete
* @since 3.3 Replace this in the end to avoid `Inline JS Defer` or other Page Optm features encoded ESI tags wrongly, which caused LSWS can't recognize ESI
[469] Fix | Delete
*/
[470] Fix | Delete
$buffer = $this->cls( 'ESI' )->finalize( $buffer );
[471] Fix | Delete
[472] Fix | Delete
$this->send_headers( true );
[473] Fix | Delete
[474] Fix | Delete
// Log ESI nonce buffer empty issue
[475] Fix | Delete
if ( defined( 'LSCACHE_IS_ESI' ) && 0 === strlen( $buffer ) && ! empty( $_SERVER['REQUEST_URI'] ) ) {
[476] Fix | Delete
// Log ref for debug purpose
[477] Fix | Delete
// phpcs:ignore WordPress.Security.ValidatedSanitizedInput.InputNotSanitized, WordPress.PHP.DevelopmentFunctions.error_log_error_log
[478] Fix | Delete
error_log( 'ESI buffer empty ' . wp_unslash( $_SERVER['REQUEST_URI'] ) );
[479] Fix | Delete
}
[480] Fix | Delete
[481] Fix | Delete
// Init comment info
[482] Fix | Delete
$running_info_showing = defined( 'LITESPEED_IS_HTML' ) || defined( 'LSCACHE_IS_ESI' );
[483] Fix | Delete
if ( defined( 'LSCACHE_ESI_SILENCE' ) ) {
[484] Fix | Delete
$running_info_showing = false;
[485] Fix | Delete
Debug2::debug( '[Core] ESI silence' );
[486] Fix | Delete
}
[487] Fix | Delete
/**
[488] Fix | Delete
* Silence comment for JSON request
[489] Fix | Delete
*
[490] Fix | Delete
* @since 2.9.3
[491] Fix | Delete
*/
[492] Fix | Delete
if ( REST::cls()->is_rest() || Router::is_ajax() ) {
[493] Fix | Delete
$running_info_showing = false;
[494] Fix | Delete
Debug2::debug( '[Core] Silence Comment due to REST/AJAX' );
[495] Fix | Delete
}
[496] Fix | Delete
$running_info_showing = apply_filters( 'litespeed_comment', $running_info_showing );
[497] Fix | Delete
if ( $running_info_showing && $this->footer_comment ) {
[498] Fix | Delete
$buffer .= $this->footer_comment;
[499] Fix | Delete
12
It is recommended that you Edit text format, this type of Fix handles quite a lot in one request
Function