Current File : /home/escuelai/public_html/wp-content/plugins/w3-total-cache/Minify_Plugin.php |
<?php
/**
* File: Minify_Plugin.php
*
* @package W3TC
*/
namespace W3TC;
/**
* Class Minify_Plugin
*
* phpcs:disable PSR2.Classes.PropertyDeclaration.Underscore
* phpcs:disable PSR2.Methods.MethodDeclaration.Underscore
* phpcs:disable Generic.CodeAnalysis.UnusedFunctionParameter
*/
class Minify_Plugin {
/**
* Minify reject reason
*
* @var string
*/
public $minify_reject_reason = '';
/**
* Error
*
* @var string
*/
public $error = '';
/**
* Array of replaced styles
*
* @var array
*/
public $replaced_styles = array();
/**
* Array of replaced scripts
*
* @var array
*/
public $replaced_scripts = array();
/**
* Array of printed scripts.
*
* @var array
*/
public $printed_scripts = array();
/**
* Array of printed styles.
*
* @var array
*/
public $printed_styles = array();
/**
* Helper object to use
*
* @var _W3_MinifyHelpers
*/
private $minify_helpers;
/**
* Config.
*
* @var Config Configuration.
*/
private $_config = null;
/**
* Constructor for the Minify_Plugin class.
*
* @return void
*/
public function __construct() {
$this->_config = Dispatcher::config();
}
/**
* Initializes the plugin by registering filters and actions.
*
* @return void
*/
public function run() {
add_action( 'init', array( $this, 'init' ) );
add_filter( 'cron_schedules', array( $this, 'cron_schedules' ) ); // phpcs:ignore WordPress.WP.CronInterval.ChangeDetected
add_action( 'w3tc_minifycache_purge_wpcron', array( $this, 'w3tc_minifycache_purge_wpcron' ) );
add_filter( 'w3tc_admin_bar_menu', array( $this, 'w3tc_admin_bar_menu' ) );
add_filter( 'w3tc_footer_comment', array( $this, 'w3tc_footer_comment' ) );
if ( 'file' === $this->_config->get_string( 'minify.engine' ) ) {
add_action( 'w3_minify_cleanup', array( $this, 'cleanup' ) );
}
add_filter( 'w3tc_pagecache_set_header', array( $this, 'w3tc_pagecache_set_header' ), 20, 2 );
// usage statistics handling.
add_action( 'w3tc_usage_statistics_of_request', array( $this, 'w3tc_usage_statistics_of_request' ), 10, 1 );
add_filter( 'w3tc_usage_statistics_metrics', array( $this, 'w3tc_usage_statistics_metrics' ) );
// Start minify.
if ( $this->can_minify() ) {
Util_Bus::add_ob_callback( 'minify', array( $this, 'ob_callback' ) );
}
}
/**
* Initializes the Minify Plugin during the `init` action.
*
* @return void
*/
public function init() {
$url = Util_Environment::filename_to_url( W3TC_CACHE_MINIFY_DIR );
$parsed = wp_parse_url( $url );
$prefix = '/' . trim( $parsed['path'], '/' ) . '/';
$request_uri = isset( $_SERVER['REQUEST_URI'] ) ? esc_url_raw( wp_unslash( $_SERVER['REQUEST_URI'] ) ) : '';
if ( substr( $request_uri, 0, strlen( $prefix ) ) === $prefix ) {
$w3_minify = Dispatcher::component( 'Minify_MinifiedFileRequestHandler' );
$filename = Util_Environment::remove_query_all( substr( $request_uri, strlen( $prefix ) ) );
$w3_minify->process( $filename );
exit();
}
if ( ! empty( Util_Request::get_string( 'w3tc_minify' ) ) ) {
$w3_minify = Dispatcher::component( 'Minify_MinifiedFileRequestHandler' );
$w3_minify->process( Util_Request::get_string( 'w3tc_minify' ) );
exit();
}
}
/**
* Cleans up the minify cache.
*
* @return void
*/
public function cleanup() {
$a = Dispatcher::component( 'Minify_Plugin_Admin' );
$a->cleanup();
}
/**
* Adds custom schedules for cron jobs related to minify cache cleanup.
*
* @param array $schedules The existing cron schedules.
*
* @return array The updated cron schedules.
*/
public function cron_schedules( $schedules ) {
$c = $this->_config;
$minify_enabled = $c->get_boolean( 'minify.enabled' );
$engine = $c->get_string( 'minify.engine' );
if ( $minify_enabled && ( 'file' === $engine || 'file_generic' === $engine ) ) {
$interval = $c->get_integer( 'minify.file.gc' );
$schedules['w3_minify_cleanup'] = array(
'interval' => $interval,
'display' => sprintf(
// translators: 1 interval in seconds.
__( '[W3TC] Minify Cache file GC (every %d seconds)', 'w3-total-cache' ),
$interval
),
);
}
return $schedules;
}
/**
* Purges the minify cache via WP-Cron.
*
* @since 2.8.0
*
* @return void
*/
public function w3tc_minifycache_purge_wpcron() {
$flusher = Dispatcher::component( 'CacheFlush' );
$flusher->minifycache_flush();
}
/**
* Handles output buffering for minification and optimization.
*
* @param string $buffer The output buffer content.
*
* @return string The processed buffer after minification.
*
* @throws \Exception If an error occurs during the minification process.
*/
public function ob_callback( $buffer ) {
$enable = Util_Content::is_html( $buffer ) && $this->can_minify2( $buffer );
$enable = apply_filters( 'w3tc_minify_enable', $enable );
if ( ! $enable ) {
return $buffer;
}
$this->minify_helpers = new _W3_MinifyHelpers( $this->_config );
// Replace script and style tags.
$js_enable = $this->_config->get_boolean( 'minify.js.enable' );
$css_enable = $this->_config->get_boolean( 'minify.css.enable' );
$html_enable = $this->_config->get_boolean( 'minify.html.enable' );
if ( function_exists( 'is_feed' ) && is_feed() ) {
$js_enable = false;
$css_enable = false;
}
$js_enable = apply_filters( 'w3tc_minify_js_enable', $js_enable );
$css_enable = apply_filters( 'w3tc_minify_css_enable', $css_enable );
$html_enable = apply_filters( 'w3tc_minify_html_enable', $html_enable );
$head_prepend = '';
$body_prepend = '';
$body_append = '';
$embed_extsrcjs = false;
$buffer = apply_filters( 'w3tc_minify_before', $buffer );
// If the minify cache folder is missing minify fails. This will generate the minify folder path if missing.
$minify_environment = Dispatcher::component( 'Minify_Environment' );
try {
$minify_environment->fix_on_wpadmin_request( $this->_config, true );
} catch ( \Exception $e ) { // phpcs:ignore Generic.CodeAnalysis.EmptyStatement.DetectedCatch
// Exception.
}
if ( $this->_config->get_boolean( 'minify.auto' ) ) {
if ( $js_enable ) {
$minifier = new Minify_AutoJs( $this->_config, $buffer, $this->minify_helpers );
$buffer = $minifier->execute();
$this->replaced_scripts = $minifier->get_debug_minified_urls();
}
if ( $css_enable ) {
$minifier = new Minify_AutoCss( $this->_config, $buffer, $this->minify_helpers );
$buffer = $minifier->execute();
}
$buffer = apply_filters( 'w3tc_minify_processed', $buffer );
} else {
if ( $css_enable ) {
$style = $this->get_style_group( 'include' );
if ( $style['body'] ) {
if ( $this->_custom_location_does_not_exist( '/<!-- W3TC-include-css -->/', $buffer, $style['body'] ) ) {
$head_prepend .= $style['body'];
}
$this->remove_styles_group( $buffer, 'include' );
}
if ( $this->_config->getf_boolean( 'minify.css.http2push' ) ) {
$this->minify_helpers->http2_header_add( $style['url'], 'style' );
}
}
if ( $js_enable ) {
$embed_type = $this->_config->get_string( 'minify.js.header.embed_type' );
$http2push = $this->_config->getf_boolean( 'minify.js.http2push' );
$script = $this->get_script_group( 'include', $embed_type );
if ( $script['body'] ) {
$embed_extsrcjs = 'extsrc' === $embed_type || 'asyncsrc' === $embed_type ? true : $embed_extsrcjs;
if ( $this->_custom_location_does_not_exist( '/<!-- W3TC-include-js-head -->/', $buffer, $script['body'] ) ) {
$head_prepend .= $script['body'];
}
$this->remove_scripts_group( $buffer, 'include' );
}
if ( $http2push ) {
$this->minify_helpers->http2_header_add( $script['url'], 'script' );
}
$embed_type = $this->_config->get_string( 'minify.js.body.embed_type' );
$script = $this->get_script_group( 'include-body', $embed_type );
if ( $script['body'] ) {
$embed_extsrcjs = 'extsrc' === $embed_type || 'asyncsrc' === $embed_type ? true : $embed_extsrcjs;
if ( $this->_custom_location_does_not_exist( '/<!-- W3TC-include-js-body-start -->/', $buffer, $script['body'] ) ) {
$body_prepend .= $script['body'];
}
$this->remove_scripts_group( $buffer, 'include-body' );
}
if ( $http2push ) {
$this->minify_helpers->http2_header_add( $script['url'], 'script' );
}
$embed_type = $this->_config->get_string( 'minify.js.footer.embed_type' );
$script = $this->get_script_group( 'include-footer', $embed_type );
if ( $script['body'] ) {
$embed_extsrcjs = 'extsrc' === $embed_type || 'asyncsrc' === $embed_type ? true : $embed_extsrcjs;
if ( $this->_custom_location_does_not_exist( '/<!-- W3TC-include-js-body-end -->/', $buffer, $script['body'] ) ) {
$body_append .= $script['body'];
}
$this->remove_scripts_group( $buffer, 'include-footer' );
}
if ( $http2push ) {
$this->minify_helpers->http2_header_add( $script['url'], 'script' );
}
}
}
if ( '' !== $head_prepend ) {
$buffer = preg_replace( '~<head(\s+[^>]*)*>~Ui', '\\0' . $head_prepend, $buffer, 1 );
}
if ( '' !== $body_prepend ) {
$buffer = preg_replace( '~<body(\s+[^>]*)*>~Ui', '\\0' . $body_prepend, $buffer, 1 );
}
if ( '' !== $body_append ) {
$buffer = preg_replace( '~<\\/body>~', $body_append . '\\0', $buffer, 1 );
}
if ( $embed_extsrcjs ) {
$script = '
<script>
var extsrc=null;
(function(){function j(){if(b&&g){document.write=k;document.writeln=l;var f=document.createElement("span");f.innerHTML=b;g.appendChild(f);b=""}}function d(){j();for(var f=document.getElementsByTagName("script"),c=0;c<f.length;c++){var e=f[c],h=e.getAttribute("asyncsrc");if(h){e.setAttribute("asyncsrc","");var a=document.createElement("script");a.async=!0;a.src=h;document.getElementsByTagName("head")[0].appendChild(a)}if(h=e.getAttribute("extsrc")){e.setAttribute("extsrc","");g=document.createElement("span");e.parentNode.insertBefore(g,e);document.write=function(a){b+=a};document.writeln=function(a){b+=a;b+="\n"};a=document.createElement("script");a.async=!0;a.src=h;/msie/i.test(navigator.userAgent)&&!/opera/i.test(navigator.userAgent)?a.onreadystatechange=function(){("loaded"==this.readyState||"complete"==this.readyState)&&d()}:-1!=navigator.userAgent.indexOf("Firefox")||"onerror"in a?(a.onload=d,a.onerror=d):(a.onload=d,a.onreadystatechange=d);document.getElementsByTagName("head")[0].appendChild(a);return}}j();document.write=k;document.writeln=l;for(c=0;c<extsrc.complete.funcs.length;c++)extsrc.complete.funcs[c]()}function i(){arguments.callee.done||(arguments.callee.done=!0,d())}extsrc={complete:function(b){this.complete.funcs.push(b)}};extsrc.complete.funcs=[];var k=document.write,l=document.writeln,b="",g="";document.addEventListener&&document.addEventListener("DOMContentLoaded",i,!1);if(/WebKit/i.test(navigator.userAgent))var m=setInterval(function(){/loaded|complete/.test(document.readyState)&&(clearInterval(m),i())},10);window.onload=i})();
</script>
';
$buffer = preg_replace( '~<head(\s+[^>]*)*>~Ui', '\\0' . $script, $buffer, 1 );
}
// Minify HTML/Feed.
if ( $html_enable ) {
try {
$buffer = $this->minify_html( $buffer );
} catch ( \Exception $exception ) {
$this->error = $exception->getMessage();
}
}
return $buffer;
}
/**
* Adds the minify flush item to the WordPress admin bar menu.
*
* @param array $menu_items Array of existing admin bar menu items.
*
* @return array Modified menu items with the minify cache option included.
*/
public function w3tc_admin_bar_menu( $menu_items ) {
$menu_items['20210.minify'] = array(
'id' => 'w3tc_flush_minify',
'parent' => 'w3tc_flush',
'title' => __( 'Minify Cache', 'w3-total-cache' ),
'href' => wp_nonce_url(
admin_url(
'admin.php?page=w3tc_dashboard&w3tc_flush_minify'
),
'w3tc'
),
);
return $menu_items;
}
/**
* Appends a footer comment regarding minification to the HTML strings.
*
* @param array $strings Array of footer comments to append.
*
* @return array Modified array of footer comments.
*/
public function w3tc_footer_comment( $strings ) {
$strings[] = sprintf(
// Translators: 1 engine name, 2 reject reason.
__(
'Minified using %1$s%2$s',
'w3-total-cache'
),
Cache::engine_name( $this->_config->get_string( 'minify.engine' ) ),
( '' !== $this->minify_reject_reason ? sprintf( ' (%s)', $this->minify_reject_reason ) : '' )
);
if ( $this->_config->get_boolean( 'minify.debug' ) ) {
$strings[] = '';
$strings[] = 'Minify debug info:';
$strings[] = sprintf( '%s%s', str_pad( 'Theme: ', 20 ), $this->get_theme() );
$strings[] = sprintf( '%s%s', str_pad( 'Template: ', 20 ), $this->get_template() );
if ( $this->error ) {
$strings[] = sprintf( '%s%s', str_pad( 'Errors: ', 20 ), $this->error );
}
if ( count( $this->replaced_styles ) ) {
$strings[] = 'Replaced CSS files:';
foreach ( $this->replaced_styles as $index => $file ) {
$strings[] = sprintf( '%d. %s', $index + 1, Util_Content::escape_comment( $file ) );
}
}
if ( count( $this->replaced_scripts ) ) {
$strings[] = 'Replaced JavaScript files:';
foreach ( $this->replaced_scripts as $index => $file ) {
$strings[] = sprintf( "%d. %s\r\n", $index + 1, Util_Content::escape_comment( $file ) );
}
}
$strings[] = '';
}
return $strings;
}
/**
* Checks if a custom minification location does not exist.
*
* @param string $pattern Regular expression pattern to search.
* @param string $source Source string to check.
* @param string $script Replacement script for the match.
*
* @return bool True if the location does not exist, false otherwise.
*/
public function _custom_location_does_not_exist( $pattern, &$source, $script ) {
$count = 0;
$source = preg_replace( $pattern, $script, $source, 1, $count );
return 0 === $count;
}
/**
* Removes specified CSS files from the provided content.
*
* @param string $content HTML content to search for CSS references.
* @param array $files List of CSS files to remove.
*
* @return void
*/
public function remove_styles( &$content, $files ) {
$regexps = array();
$home_url_regexp = Util_Environment::home_url_regexp();
$path = '';
if ( Util_Environment::is_wpmu() && ! Util_Environment::is_wpmu_subdomain() ) {
$path = ltrim( Util_Environment::home_url_uri(), '/' );
}
foreach ( $files as $file ) {
if ( $path && strpos( $file, $path ) === 0 ) {
$file = substr( $file, strlen( $path ) );
}
$this->replaced_styles[] = $file;
if ( Util_Environment::is_url( $file ) && ! preg_match( '~' . $home_url_regexp . '~i', $file ) ) {
// external CSS files.
$regexps[] = Util_Environment::preg_quote( $file );
} else {
// local CSS files.
$file = ltrim( $file, '/' );
if (
home_url() === site_url() &&
ltrim( Util_Environment::site_url_uri(), '/' ) &&
strpos( $file, ltrim( Util_Environment::site_url_uri(), '/' ) ) === 0
) {
$file = str_replace( ltrim( Util_Environment::site_url_uri(), '/' ), '', $file );
}
$file = ltrim( preg_replace( '~' . $home_url_regexp . '~i', '', $file ), '/\\' );
$regexps[] = '(' . $home_url_regexp . ')?/?' . Util_Environment::preg_quote( $file );
}
}
foreach ( $regexps as $regexp ) {
$content = preg_replace( '~<link\s+[^<>]*href=["\']?' . $regexp . '["\']?[^<>]*/?>(.*</link>)?~Uis', '', $content );
$content = preg_replace( '~@import\s+(url\s*)?\(?["\']?\s*' . $regexp . '\s*["\']?\)?[^;]*;?~is', '', $content );
}
$content = preg_replace( '~<style[^<>]*>\s*</style>~', '', $content );
}
/**
* Removes specified JavaScript files from the provided content.
*
* @param string $content HTML content to search for script references.
* @param array $files List of JavaScript files to remove.
*
* @return void
*/
public function remove_scripts( &$content, $files ) {
$regexps = array();
$home_url_regexp = Util_Environment::home_url_regexp();
$path = '';
if ( Util_Environment::is_wpmu() && ! Util_Environment::is_wpmu_subdomain() ) {
$path = ltrim( Util_Environment::network_home_url_uri(), '/' );
}
foreach ( $files as $file ) {
if ( $path && strpos( $file, $path ) === 0 ) {
$file = substr( $file, strlen( $path ) );
}
$this->replaced_scripts[] = $file;
if ( Util_Environment::is_url( $file ) && ! preg_match( '~' . $home_url_regexp . '~i', $file ) ) {
// external JS files.
$regexps[] = Util_Environment::preg_quote( $file );
} else {
// local JS files.
$file = ltrim( $file, '/' );
if (
home_url() === site_url() &&
ltrim( Util_Environment::site_url_uri(), '/' ) &&
strpos( $file, ltrim( Util_Environment::site_url_uri(), '/' ) ) === 0
) {
$file = str_replace( ltrim( Util_Environment::site_url_uri(), '/' ), '', $file );
}
$file = ltrim( preg_replace( '~' . $home_url_regexp . '~i', '', $file ), '/\\' );
$regexps[] = '(' . $home_url_regexp . ')?/?' . Util_Environment::preg_quote( $file );
}
}
foreach ( $regexps as $regexp ) {
$content = preg_replace( '~<script\s+[^<>]*src=["\']?' . $regexp . '["\']?[^<>]*>\s*</script>~Uis', '', $content );
}
}
/**
* Removes a group of CSS files for a specified location.
*
* @param string $content HTML content to search for CSS references.
* @param string $location Location identifier for the CSS group.
*
* @return void
*/
public function remove_styles_group( &$content, $location ) {
$theme = $this->get_theme();
$template = $this->get_template();
$files = array();
$groups = $this->_config->get_array( 'minify.css.groups' );
if ( isset( $groups[ $theme ]['default'][ $location ]['files'] ) ) {
$files = (array) $groups[ $theme ]['default'][ $location ]['files'];
}
if ( 'default' !== $template && isset( $groups[ $theme ][ $template ][ $location ]['files'] ) ) {
$files = array_merge( $files, (array) $groups[ $theme ][ $template ][ $location ]['files'] );
}
$this->remove_styles( $content, $files );
}
/**
* Removes a group of JavaScript files for a specified location.
*
* @param string $content HTML content to search for script references.
* @param string $location Location identifier for the script group.
*
* @return void
*/
public function remove_scripts_group( &$content, $location ) {
$theme = $this->get_theme();
$template = $this->get_template();
$files = array();
$groups = $this->_config->get_array( 'minify.js.groups' );
if ( isset( $groups[ $theme ]['default'][ $location ]['files'] ) ) {
$files = (array) $groups[ $theme ]['default'][ $location ]['files'];
}
if ( 'default' !== $template && isset( $groups[ $theme ][ $template ][ $location ]['files'] ) ) {
$files = array_merge( $files, (array) $groups[ $theme ][ $template ][ $location ]['files'] );
}
$this->remove_scripts( $content, $files );
}
/**
* Minifies the provided HTML content.
*
* @param string $html HTML content to minify.
*
* @return string Minified HTML content.
*/
public function minify_html( $html ) {
$w3_minifier = Dispatcher::component( 'Minify_ContentMinifier' );
$ignored_comments = $this->_config->get_array( 'minify.html.comments.ignore' );
if ( count( $ignored_comments ) ) {
$ignored_comments_preserver = new \W3TCL\Minify\Minify_IgnoredCommentPreserver();
$ignored_comments_preserver->setIgnoredComments( $ignored_comments );
$html = $ignored_comments_preserver->search( $html );
}
if ( $this->_config->get_boolean( 'minify.html.inline.js' ) ) {
$js_engine = $this->_config->get_string( 'minify.js.engine' );
if ( ! $w3_minifier->exists( $js_engine ) || ! $w3_minifier->available( $js_engine ) ) {
$js_engine = 'js';
}
$js_minifier = $w3_minifier->get_minifier( $js_engine );
$js_options = $w3_minifier->get_options( $js_engine );
$w3_minifier->init( $js_engine );
$html = \W3TCL\Minify\Minify_Inline_JavaScript::minify( $html, $js_minifier, $js_options );
}
if ( $this->_config->get_boolean( 'minify.html.inline.css' ) ) {
$css_engine = $this->_config->get_string( 'minify.css.engine' );
if ( ! $w3_minifier->exists( $css_engine ) || ! $w3_minifier->available( $css_engine ) ) {
$css_engine = 'css';
}
$css_minifier = $w3_minifier->get_minifier( $css_engine );
$css_options = $w3_minifier->get_options( $css_engine );
$w3_minifier->init( $css_engine );
$html = \W3TCL\Minify\Minify_Inline_CSS::minify( $html, $css_minifier, $css_options );
}
$engine = $this->_config->get_string( 'minify.html.engine' );
if ( ! $w3_minifier->exists( $engine ) || ! $w3_minifier->available( $engine ) ) {
$engine = 'html';
}
if ( function_exists( 'is_feed' ) && is_feed() ) {
$engine .= 'xml';
}
$minifier = $w3_minifier->get_minifier( $engine );
$options = $w3_minifier->get_options( $engine );
$w3_minifier->init( $engine );
$html = call_user_func( $minifier, $html, $options );
if ( isset( $ignored_comments_preserver ) ) {
$html = $ignored_comments_preserver->replace( $html );
}
return $html;
}
/**
* Retrieves the current theme identifier.
*
* @return string Theme identifier.
*/
public function get_theme() {
static $theme = null;
if ( null === $theme ) {
$theme = Util_Theme::get_theme_key( get_theme_root(), get_template(), get_stylesheet() );
}
return $theme;
}
/**
* Retrieves the current template identifier.
*
* phpcs:disable WordPress.CodeAnalysis.AssignmentInCondition.Found
* phpcs:disable Squiz.PHP.DisallowMultipleAssignments.Found
* phpcs:disable Generic.CodeAnalysis.AssignmentInCondition.Found
*
* @return string Template identifier.
*/
public function get_template() {
static $template = null;
if ( null === $template ) {
$template_file = 'index.php';
switch ( true ) {
case ( is_404() && ( $template_file = get_404_template() ) ):
case ( is_search() && ( $template_file = get_search_template() ) ):
case ( is_tax() && ( $template_file = get_taxonomy_template() ) ):
case ( is_front_page() && function_exists( 'get_front_page_template' ) && $template_file = get_front_page_template() ):
case ( is_home() && ( $template_file = get_home_template() ) ):
case ( is_attachment() && ( $template_file = get_attachment_template() ) ):
case ( is_single() && ( $template_file = get_single_template() ) ):
case ( is_page() && ( $template_file = get_page_template() ) ):
case ( is_category() && ( $template_file = get_category_template() ) ):
case ( is_tag() && ( $template_file = get_tag_template() ) ):
case ( is_author() && ( $template_file = get_author_template() ) ):
case ( is_date() && ( $template_file = get_date_template() ) ):
case ( is_archive() && ( $template_file = get_archive_template() ) ):
case ( is_paged() && ( $template_file = get_query_template( 'paged' ) ) ):
break;
default:
if ( function_exists( 'get_index_template' ) ) {
$template_file = get_index_template();
} else {
$template_file = 'index.php';
}
break;
}
$template = basename( $template_file, '.php' );
}
return $template;
}
/**
* Generates the HTML markup for including a stylesheet.
*
* phpcs:disable WordPress.WP.EnqueuedResources.NonEnqueuedStylesheet
*
* @param string $url URL of the stylesheet.
* @param bool $import Whether to use @import syntax.
* @param bool $use_style Whether to wrap @import in <style> tags.
*
* @return string Generated HTML markup.
*/
public function get_style( $url, $import = false, $use_style = true ) {
if ( $import && $use_style ) {
return '<style media="all">@import url("' . $url . "\");</style>\r\n";
} elseif ( $import && ! $use_style ) {
return '@import url("' . $url . "\");\r\n";
} else {
return '<link rel="stylesheet" href="' . str_replace( '&', '&', $url ) . "\" media=\"all\" />\r\n";
}
}
/**
* Retrieves the details of a grouped CSS style by location.
*
* @param string $location Location identifier for the style group.
*
* @return array Associative array containing 'url' and 'body' keys.
*/
public function get_style_group( $location ) {
$style = false;
$type = 'css';
$groups = $this->_config->get_array( 'minify.css.groups' );
$theme = $this->get_theme();
$template = $this->get_template();
if ( 'default' !== $template && empty( $groups[ $theme ][ $template ][ $location ]['files'] ) ) {
$template = 'default';
}
$return = array(
'url' => null,
'body' => '',
);
if ( ! empty( $groups[ $theme ][ $template ][ $location ]['files'] ) ) {
if ( $this->_config->get_boolean( 'minify.css.embed' ) ) {
$minify = Dispatcher::component( 'Minify_MinifiedFileRequestHandler' );
$minify_filename = $this->get_minify_manual_filename( $theme, $template, $location, $type );
$m = $minify->process( $minify_filename, true );
if ( isset( $m['content'] ) ) {
$style = $m['content'];
} else {
$style = 'not set';
}
$return['body'] = "<style media=\"all\">$style</style>\r\n";
} else {
$return['url'] = $this->get_minify_manual_url( $theme, $template, $location, $type );
if ( $return['url'] ) {
$import = (
isset( $groups[ $theme ][ $template ][ $location ]['import'] ) ?
(bool) $groups[ $theme ][ $template ][ $location ]['import'] :
false
);
$return['body'] = $this->get_style( $return['url'], $import );
}
}
}
return $return;
}
/**
* Retrieves the script group for the specified location and embed type.
*
* @param string $location The location of the scripts (e.g., 'header', 'footer').
* @param string $embed_type The embed type, defaults to 'blocking'.
*
* @return array Associative array containing 'url' and 'body'.
*/
public function get_script_group( $location, $embed_type = 'blocking' ) {
$script = false;
$file_type = 'js';
$theme = $this->get_theme();
$template = $this->get_template();
$groups = $this->_config->get_array( 'minify.js.groups' );
if ( 'default' !== $template && empty( $groups[ $theme ][ $template ][ $location ]['files'] ) ) {
$template = 'default';
}
$return = array(
'url' => null,
'body' => '',
);
if ( ! empty( $groups[ $theme ][ $template ][ $location ]['files'] ) ) {
$return['url'] = $this->get_minify_manual_url( $theme, $template, $location, $file_type );
if ( $return['url'] ) {
$return['body'] = $this->minify_helpers->generate_script_tag( $return['url'], $embed_type );
}
}
return $return;
}
/**
* Generates a style tag or URL for custom styles.
*
* @param array $files Array of file paths for the styles.
* @param bool $embed_to_html Whether to embed styles directly in HTML.
*
* @return array Associative array containing 'url' and 'body'.
*/
public function get_style_custom( $files, $embed_to_html = false ) {
return $this->minify_helpers->generate_css_style_tag( $files, $embed_to_html );
}
/**
* Retrieves the manual filename for minified resources.
*
* @param string $theme The theme name.
* @param string $template The template name.
* @param string $location The location identifier.
* @param string $type The resource type ('js' or 'css').
*
* @return string|false The generated filename or false if not available.
*/
public function get_minify_manual_filename( $theme, $template, $location, $type ) {
$minify = Dispatcher::component( 'Minify_MinifiedFileRequestHandler' );
$id = $minify->get_id_group( $theme, $template, $location, $type );
if ( ! $id ) {
return false;
}
return $theme . '.' . $template . '.' . $location . '.' . $id . '.' . $type;
}
/**
* Retrieves the manual URL for minified resources.
*
* @param string $theme The theme name.
* @param string $template The template name.
* @param string $location The location identifier.
* @param string $type The resource type ('js' or 'css').
*
* @return string The generated URL.
*/
public function get_minify_manual_url( $theme, $template, $location, $type ) {
return Minify_Core::minified_url( $this->get_minify_manual_filename( $theme, $template, $location, $type ) );
}
/**
* Retrieves an array of all URLs for minified resources.
*
* @return array Array of URLs.
*/
public function get_urls() {
$files = array();
$js_groups = $this->_config->get_array( 'minify.js.groups' );
$css_groups = $this->_config->get_array( 'minify.css.groups' );
foreach ( $js_groups as $js_theme => $js_templates ) {
foreach ( $js_templates as $js_template => $js_locations ) {
foreach ( (array) $js_locations as $js_location => $js_config ) {
if ( ! empty( $js_config['files'] ) ) {
$files[] = $this->get_minify_manual_url( $js_theme, $js_template, $js_location, 'js' );
}
}
}
}
foreach ( $css_groups as $css_theme => $css_templates ) {
foreach ( $css_templates as $css_template => $css_locations ) {
foreach ( (array) $css_locations as $css_location => $css_config ) {
if ( ! empty( $css_config['files'] ) ) {
$files[] = $this->get_minify_manual_url( $css_theme, $css_template, $css_location, 'css' );
}
}
}
}
return $files;
}
/**
* Checks whether minification can be applied based on the current environment.
*
* @return bool True if minification can proceed, false otherwise.
*/
public function can_minify() {
// Skip if doint AJAX.
if ( defined( 'DOING_AJAX' ) ) {
$this->minify_reject_reason = 'Doing AJAX';
return false;
}
// Skip if doing cron.
if ( defined( 'DOING_CRON' ) ) {
$this->minify_reject_reason = 'Doing cron';
return false;
}
// Skip if APP request.
if ( defined( 'APP_REQUEST' ) ) {
$this->minify_reject_reason = 'Application request';
return false;
}
// Skip if XMLRPC request.
if ( defined( 'XMLRPC_REQUEST' ) ) {
$this->minify_reject_reason = 'XMLRPC request';
return false;
}
// Skip if Admin.
if ( defined( 'WP_ADMIN' ) ) {
$this->minify_reject_reason = 'wp-admin';
return false;
}
// Check for WPMU's and WP's 3.0 short init.
if ( defined( 'SHORTINIT' ) && SHORTINIT ) {
$this->minify_reject_reason = 'Short init';
return false;
}
// Check User agent.
if ( ! $this->check_ua() ) {
$this->minify_reject_reason = 'User agent is rejected';
return false;
}
// Check request URI.
if ( ! $this->check_request_uri() ) {
$this->minify_reject_reason = 'Request URI is rejected';
return false;
}
// Skip if user is logged in.
if ( $this->_config->get_boolean( 'minify.reject.logged' ) && ! $this->check_logged_in() ) {
$this->minify_reject_reason = 'User is logged in';
return false;
}
return true;
}
/**
* Checks whether minification can be applied to the provided buffer.
*
* @param string $buffer The buffer to check.
*
* @return bool True if the buffer can be minified, false otherwise.
*/
public function can_minify2( $buffer ) {
// Check for DONOTMINIFY constant.
if ( defined( 'DONOTMINIFY' ) && DONOTMINIFY ) {
$this->minify_reject_reason = 'DONOTMINIFY constant is defined';
return false;
}
// Check feed minify.
if ( $this->_config->get_boolean( 'minify.html.reject.feed' ) && function_exists( 'is_feed' ) && is_feed() ) {
$this->minify_reject_reason = 'Feed is rejected';
return false;
}
return true;
}
/**
* Validates the user agent for minification.
*
* @return bool True if the user agent is allowed, false otherwise.
*/
public function check_ua() {
$uas = array_merge(
$this->_config->get_array( 'minify.reject.ua' ),
array(
W3TC_POWERED_BY,
)
);
foreach ( $uas as $ua ) {
if ( ! empty( $ua ) ) {
if ( stristr( isset( $_SERVER['HTTP_USER_AGENT'] ) ? sanitize_text_field( wp_unslash( $_SERVER['HTTP_USER_AGENT'] ) ) : '', $ua ) !== false ) {
return false;
}
}
}
return true;
}
/**
* Checks whether the current user is logged in.
*
* @return bool True if the user is not logged in, false otherwise.
*/
public function check_logged_in() {
foreach ( array_keys( $_COOKIE ) as $cookie_name ) {
if ( strpos( $cookie_name, 'wordpress_logged_in' ) === 0 ) {
return false;
}
}
return true;
}
/**
* Validates the request URI for minification.
*
* @return bool True if the URI is valid, false otherwise.
*/
public function check_request_uri() {
$auto_reject_uri = array(
'wp-login',
'wp-register',
);
foreach ( $auto_reject_uri as $uri ) {
if ( strstr( isset( $_SERVER['REQUEST_URI'] ) ? esc_url_raw( wp_unslash( $_SERVER['REQUEST_URI'] ) ) : '', $uri ) !== false ) {
return false;
}
}
$reject_uri = $this->_config->get_array( 'minify.reject.uri' );
$reject_uri = array_map( array( '\W3TC\Util_Environment', 'parse_path' ), $reject_uri );
foreach ( $reject_uri as $expr ) {
$expr = trim( $expr );
$expr = str_replace( '~', '\~', $expr );
if ( '' !== $expr && preg_match( '~' . $expr . '~i', isset( $_SERVER['REQUEST_URI'] ) ? esc_url_raw( wp_unslash( $_SERVER['REQUEST_URI'] ) ) : '' ) ) {
return false;
}
}
if ( Util_Request::get_string( 'wp_customize' ) ) {
return false;
}
return true;
}
/**
* Collects usage statistics for the current request.
*
* @param mixed $storage The storage object or array for the statistics.
*
* @return void
*/
public function w3tc_usage_statistics_of_request( $storage ) {
$o = Dispatcher::component( 'Minify_MinifiedFileRequestHandler' );
$o->w3tc_usage_statistics_of_request( $storage );
}
/**
* Adds minification-related metrics to the usage statistics.
*
* @param array $metrics Existing metrics array.
*
* @return array Modified metrics array.
*/
public function w3tc_usage_statistics_metrics( $metrics ) {
return array_merge(
$metrics,
array(
'minify_requests_total',
'minify_original_length_css',
'minify_output_length_css',
'minify_original_length_js',
'minify_output_length_js',
)
);
}
/**
* Modifies or stores preload Link headers for page caching.
*
* @param array $header The current header being processed.
* @param array $header_original The original header details.
*
* @return array The modified header.
*/
public function w3tc_pagecache_set_header( $header, $header_original ) {
if ( 'Link' === $header_original['n'] && false !== strpos( $header_original['v'], 'rel=preload' ) ) {
// store preload Link headers in cache.
$new = $header_original;
$new['files_match'] = '\\.html[_a-z]*$';
return $new;
}
return $header;
}
}
/**
* Class _W3_MinifyHelpers
*
* phpcs:disable Generic.Files.OneObjectStructurePerFile.MultipleFound
* phpcs:disable PEAR.NamingConventions.ValidClassName.StartWithCapital
*/
class _W3_MinifyHelpers {
/**
* Config
*
* @var Config
*/
private $config;
/**
* Debug flag
*
* @var bool
*/
private $debug = false;
/**
* Initializes the _W3_MinifyHelpers class.
*
* @param Config $config Configuration instance used for the class.
*
* @return void
*/
public function __construct( $config ) {
$this->config = $config;
$this->debug = $config->get_boolean( 'minify.debug' );
}
/**
* Retrieves the minified URL for a given set of files and type.
*
* @param array $files Array of file paths to be minified.
* @param string $type Type of files (e.g., 'css', 'js').
*
* @return string|null The minified URL or null if no URL is generated.
*/
public function get_minify_url_for_files( $files, $type ) {
$minify_filename = Minify_Core::urls_for_minification_to_minify_filename( $files, $type );
if ( is_null( $minify_filename ) ) {
return null;
}
$url = Minify_Core::minified_url( $minify_filename );
$url = Util_Environment::url_to_maybe_https( $url );
$url = apply_filters( 'w3tc_minify_url_for_files', $url, $files, $type );
return $url;
}
/**
* Retrieves the minified content for a given set of files and type.
*
* @param array $files Array of file paths to be minified.
* @param string $type Type of files (e.g., 'css', 'js').
*
* @return string|null The minified content wrapped in HTML or null if no content is available.
*/
public function get_minified_content_for_files( $files, $type ) {
$minify_filename = Minify_Core::urls_for_minification_to_minify_filename( $files, $type );
if ( is_null( $minify_filename ) ) {
return null;
}
$minify = Dispatcher::component( 'Minify_MinifiedFileRequestHandler' );
$m = $minify->process( $minify_filename, true );
if ( ! isset( $m['content'] ) ) {
return null;
}
if ( empty( $m['content'] ) ) {
return null;
}
$style = $m['content'];
return "<style media=\"all\">$style</style>\r\n";
}
/**
* Generates a script tag for a given URL and embed type.
*
* @param string $url URL of the script.
* @param string $embed_type Type of embed (e.g., 'blocking', 'nb-js', 'nb-async').
*
* @return string The generated script tag.
*/
public function generate_script_tag( $url, $embed_type = 'blocking' ) {
static $non_blocking_function = false;
$rocket_loader_ignore = '';
if ( $this->config->get_boolean( array( 'cloudflare', 'minify_js_rl_exclude' ) ) ) {
$rocket_loader_ignore = 'data-cfasync="false"';
}
if ( 'blocking' === $embed_type ) {
$script = '<script ' . $rocket_loader_ignore . ' src="' . str_replace( '&', '&', $url ) . '"></script>';
} else {
$script = '';
if ( 'nb-js' === $embed_type ) {
if ( ! $non_blocking_function ) {
$non_blocking_function = true;
$script = "<script>function w3tc_load_js(u){var d=document,p=d.getElementsByTagName('HEAD')[0],c=d.createElement('script');c.src=u;p.appendChild(c);}</script>";
}
$script .= "<script>w3tc_load_js('" . $url . "');</script>";
} elseif ( 'nb-async' === $embed_type ) {
$script = '<script ' . $rocket_loader_ignore . ' async src="' . str_replace( '&', '&', $url ) . '"></script>';
} elseif ( 'nb-defer' === $embed_type ) {
$script = '<script ' . $rocket_loader_ignore . ' defer src="' . str_replace( '&', '&', $url ) . '"></script>';
} elseif ( 'extsrc' === $embed_type ) {
$script = '<script ' . $rocket_loader_ignore . ' extsrc="' . str_replace( '&', '&', $url ) . '"></script>';
} elseif ( 'asyncsrc' === $embed_type ) {
$script = '<script ' . $rocket_loader_ignore . ' asyncsrc="' . str_replace( '&', '&', $url ) . '"></script>';
} else {
$script = '<script ' . $rocket_loader_ignore . ' src="' . str_replace( '&', '&', $url ) . '"></script>';
}
}
return $script . "\r\n";
}
/**
* Determines whether a given file or URL should be minified.
*
* @param string $url URL of the file to check.
* @param string $file File path to check (optional).
*
* @return string Indicates the type of minification ('url', 'file', or empty string).
*/
public function is_file_for_minification( $url, $file ) {
static $external;
static $external_regexp;
if ( ! isset( $external ) ) {
$external = $this->config->get_array( 'minify.cache.files' );
$external_regexp = $this->config->get_boolean( 'minify.cache.files_regexp' );
}
foreach ( $external as $item ) {
if ( empty( $item ) ) {
continue;
}
if ( $external_regexp ) {
$item = str_replace( '~', '\~', $item );
if ( ! preg_match( '~' . $item . '~', $url ) ) {
continue;
}
} elseif ( ! preg_match( '~^' . Util_Environment::get_url_regexp( $item ) . '~', $url ) ) {
continue;
}
if ( $this->debug ) {
Minify_Core::log( 'is_file_for_minification: whilelisted ' . $url . ' by ' . $item );
}
return 'url';
}
if ( is_null( $file ) ) {
if ( $this->debug ) {
Minify_Core::log( 'is_file_for_minification: external not whitelisted url ' . $url );
}
return '';
}
$file_normalized = Util_Environment::remove_query_all( $file );
$ext = strrchr( $file_normalized, '.' );
if ( '.js' !== $ext && '.css' !== $ext ) {
if ( $this->debug ) {
Minify_Core::log( 'is_file_for_minification: unknown extension ' . $ext . ' for ' . $file );
}
return '';
}
$path = Util_Environment::docroot_to_full_filename( $file );
if ( ! file_exists( $path ) ) {
if ( $this->debug ) {
Minify_Core::log( 'is_file_for_minification: file doesnt exists ' . $path );
}
return '';
}
if ( $this->debug ) {
Minify_Core::log( 'is_file_for_minification: true for file ' . $file . ' path ' . $path );
}
return 'file';
}
/**
* Adds an HTTP/2 header for preloading a given URL.
*
* @param string $url URL to be preloaded.
* @param string $type Resource type (e.g., 'script', 'style').
*
* @return void
*/
public function http2_header_add( $url, $type ) {
if ( empty( $url ) ) {
return;
}
// Cloudflare needs URI without host.
$uri = Util_Environment::url_to_uri( $url );
// priorities attached:
// 3000 - cdn
// 4000 - browsercache.
$data = apply_filters(
'w3tc_minify_http2_preload_url',
array(
'result_link' => $uri,
'original_url' => $url,
)
);
header( 'Link: <' . $data['result_link'] . '>; rel=preload; as=' . $type, false );
}
/**
* Generates a CSS style tag or URL for embedding styles.
*
* @param array $files Array of CSS file paths.
* @param bool $embed_to_html Whether to embed the CSS content directly into the HTML.
*
* @return array Contains 'url' (string|null) and 'body' (string) keys.
*/
public function generate_css_style_tag( $files, $embed_to_html ) {
$return = array(
'url' => null,
'body' => '',
);
if ( count( $files ) ) {
if ( $embed_to_html ) {
$body = $this->get_minified_content_for_files( $files, 'css' );
if ( ! is_null( $body ) ) {
$return['body'] = $body;
}
}
if ( empty( $return['body'] ) ) {
$return['url'] = $this->get_minify_url_for_files( $files, 'css' );
if ( ! is_null( $return['url'] ) ) {
$return['body'] = '<link rel="stylesheet" href="' . // phpcs:ignore WordPress.WP.EnqueuedResources.NonEnqueuedStylesheet
str_replace( '&', '&', $return['url'] ) . "\" media=\"all\" />\r\n";
}
}
}
return $return;
}
}