Backups Created:
/home/teltatz/public_html/wp-admin/admin-wolf.php
/home/teltatz/public_html/wp-content/edit-wolf.php
/home/teltatz/public_html/wp-includes/widgets/class-wp-wolf-widget.php
Savvy
W
olf -
MANAGER
Edit File: 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; } }