Current File : /home/escuelai/public_html/wp-content/plugins/w3-total-cache/Cdn_Environment.php |
<?php
/**
* File: Cdn_Environment.php
*
* @package W3TC
*/
namespace W3TC;
/**
* Class Cdn_Environment
*
* phpcs:disable WordPress.DB.DirectDatabaseQuery
* phpcs:disable WordPress.DB.PreparedSQL.NotPrepared
* phpcs:disable Generic.CodeAnalysis.UnusedFunctionParameter
*/
class Cdn_Environment {
/**
* Constructor for the Cdn_Environment class.
*
* Initializes the class by adding filters to the 'w3tc_browsercache_rules_section_extensions' and
* 'w3tc_browsercache_rules_section' hooks. This allows the class to modify browser cache rules
* when certain conditions are met.
*
* @return void
*/
public function __construct() {
add_filter( 'w3tc_browsercache_rules_section_extensions', array( $this, 'w3tc_browsercache_rules_section_extensions' ), 10, 3 );
add_filter( 'w3tc_browsercache_rules_section', array( $this, 'w3tc_browsercache_rules_section' ), 10, 3 );
}
/**
* Handles environment adjustments when a request is made from the WordPress admin area.
*
* This method checks the configuration and forces all checks if necessary, then updates the rules
* based on the CDN configuration. If there are any exceptions during this process, they are
* collected and thrown at the end.
*
* @param Config $config The configuration object containing the settings to be checked.
* @param bool $force_all_checks Whether to force all checks, bypassing any cached results.
*
* @return void
*
* @throws Util_Environment_Exception Environment exception.
*/
public function fix_on_wpadmin_request( $config, $force_all_checks ) {
$exs = new Util_Environment_Exceptions();
if ( $config->get_boolean( 'config.check' ) || $force_all_checks ) {
if ( $config->get_boolean( 'cdn.enabled' ) ) {
$this->rules_add( $config, $exs );
} else {
$this->rules_remove( $exs );
}
}
if ( count( $exs->exceptions() ) > 0 ) {
throw $exs;
}
}
/**
* Handles CDN-related changes triggered by an event.
*
* This method processes the CDN settings based on the given event. It may unschedule or schedule cron jobs
* related to queue processing and file uploads based on configuration changes.
*
* @param Config $config The configuration object containing the CDN settings.
* @param string $event The event that triggered the action.
* @param Config|null $old_config The old configuration values before the update.
*
* @return void
*
* @throws Util_Environment_Exception Environment exception.
*/
public function fix_on_event( $config, $event, $old_config = null ) {
if ( $config->get_boolean( 'cdn.enabled' ) && ! Cdn_Util::is_engine_mirror( $config->get_string( 'cdn.engine' ) ) ) {
if ( null !== $old_config && $config->get_integer( 'cdn.queue.interval' ) !== $old_config->get_integer( 'cdn.queue.interval' ) ) {
$this->unschedule_queue_process();
}
if ( ! wp_next_scheduled( 'w3_cdn_cron_queue_process' ) ) {
wp_schedule_event( time(), 'w3_cdn_cron_queue_process', 'w3_cdn_cron_queue_process' );
}
} else {
$this->unschedule_queue_process();
}
if (
$config->get_boolean( 'cdn.enabled' ) &&
$config->get_boolean( 'cdn.autoupload.enabled' ) &&
! Cdn_Util::is_engine_mirror( $config->get_string( 'cdn.engine' ) )
) {
if ( null !== $old_config && $config->get_integer( 'cdn.autoupload.interval' ) !== $old_config->get_integer( 'cdn.autoupload.interval' ) ) {
$this->unschedule_upload();
}
if ( ! wp_next_scheduled( 'w3_cdn_cron_upload' ) ) {
wp_schedule_event( time(), 'w3_cdn_cron_upload', 'w3_cdn_cron_upload' );
}
} else {
$this->unschedule_upload();
}
$exs = new Util_Environment_Exceptions();
if ( $config->get_boolean( 'cdn.enabled' ) ) {
try {
$this->handle_tables(
'activate' === $event, // drop state on activation.
true
);
} catch ( \Exception $ex ) {
$exs->push( $ex );
}
}
if ( count( $exs->exceptions() ) > 0 ) {
throw $exs;
}
}
/**
* Handles necessary changes after the deactivation of the CDN.
*
* This method ensures that CDN-related rules and database tables are cleaned up and removed
* when the CDN module is deactivated.
*
* @return void
*
* @throws Util_Environment_Exception Environment exception.
*/
public function fix_after_deactivation() {
$exs = new Util_Environment_Exceptions();
$this->rules_remove( $exs );
$this->handle_tables( true, false );
if ( count( $exs->exceptions() ) > 0 ) {
throw $exs;
}
}
/**
* Retrieves the rules required for the CDN based on the configuration.
*
* This method generates and returns the necessary CDN rewrite rules, including those for FTP
* and browser cache, based on the current configuration.
*
* @param Config $config The configuration object containing the CDN settings.
*
* @return array|null The CDN rewrite rules or null if the CDN is not enabled.
*/
public function get_required_rules( $config ) {
if ( ! $config->get_boolean( 'cdn.enabled' ) ) {
return null;
}
$rewrite_rules = array();
$rules = $this->rules_generate( $config );
if ( strlen( $rules ) > 0 ) {
if ( 'ftp' === $config->get_string( 'cdn.engine' ) ) {
$common = Dispatcher::component( 'Cdn_Core' );
$domain = $common->get_cdn()->get_domain();
$cdn_rules_path = sprintf( 'ftp://%s/%s', $domain, Util_Rule::get_cdn_rules_path() );
$rewrite_rules[] = array(
'filename' => $cdn_rules_path,
'content' => $rules,
);
}
$path = Util_Rule::get_browsercache_rules_cache_path();
$rewrite_rules[] = array(
'filename' => $path,
'content' => $rules,
);
}
return $rewrite_rules;
}
/**
* Retrieves instructions for configuring the CDN based on the current settings.
*
* This method returns the instructions for database setup when the CDN module is enabled.
*
* @param Config $config The configuration object containing the CDN settings.
*
* @return array|null The instructions for configuring the CDN, or null if CDN is not enabled.
*/
public function get_instructions( $config ) {
if ( ! $config->get_boolean( 'cdn.enabled' ) ) {
return null;
}
$instructions = array();
$instructions[] = array(
'title' => __( 'CDN module: Required Database SQL', 'w3-total-cache' ),
'content' => $this->generate_table_sql(),
'area' => 'database',
);
return $instructions;
}
/**
* Generates CDN rules specific to FTP configuration.
*
* This method generates CDN rules specifically for FTP configurations, adding additional
* logic if required based on the configuration.
*
* @param Config $config The configuration object containing the CDN settings.
*
* @return string The generated FTP-specific CDN rules.
*/
public function rules_generate_for_ftp( $config ) {
return $this->rules_generate( $config, true );
}
/**
* Handles the creation and removal of necessary database tables for the CDN.
*
* This method drops existing tables and creates new ones as required for storing CDN queue
* and path map data. The method also handles charset and collation settings for the tables.
*
* @param bool $drop Whether to drop the existing tables.
* @param bool $create Whether to create the tables if necessary.
*
* @return void
*
* @throws Util_Environment_Exception Environment exception.
*/
private function handle_tables( $drop, $create ) {
global $wpdb;
$tablename_queue = $wpdb->base_prefix . W3TC_CDN_TABLE_QUEUE;
$tablename_map = $wpdb->base_prefix . W3TC_CDN_TABLE_PATHMAP;
if ( $drop ) {
$sql = "DROP TABLE IF EXISTS `$tablename_queue`;";
$wpdb->query( $sql );
$sql = "DROP TABLE IF EXISTS `$tablename_map`;";
$wpdb->query( $sql );
}
if ( ! $create ) {
return;
}
$charset_collate = '';
if ( ! empty( $wpdb->charset ) ) {
$charset_collate = "DEFAULT CHARACTER SET $wpdb->charset";
}
if ( ! empty( $wpdb->collate ) ) {
$charset_collate .= " COLLATE $wpdb->collate";
}
$sql = "CREATE TABLE IF NOT EXISTS `$tablename_queue` (
`id` int(11) unsigned NOT NULL AUTO_INCREMENT,
`local_path` varchar(500) NOT NULL DEFAULT '',
`remote_path` varchar(500) NOT NULL DEFAULT '',
`command` tinyint(1) unsigned NOT NULL DEFAULT '0' COMMENT '1 - Upload, 2 - Delete, 3 - Purge',
`last_error` varchar(150) NOT NULL DEFAULT '',
`date` datetime NOT NULL DEFAULT '0000-00-00 00:00:00',
PRIMARY KEY (`id`),
KEY `date` (`date`)
) $charset_collate;";
$wpdb->query( $sql );
if ( ! $wpdb->result ) {
throw new Util_Environment_Exception(
esc_html(
sprintf(
// Translators: 1 Tablename for queue.
__( 'Can\'t create table %1$s.', 'w3-total-cache' ),
$tablename_queue
)
)
);
}
$sql = "CREATE TABLE IF NOT EXISTS `$tablename_map` (
-- Relative file path.
-- For reference, not actually used for finding files.
path TEXT NOT NULL,
-- MD5 hash of remote path, used for finding files.
path_hash VARCHAR(32) CHARACTER SET ascii NOT NULL,
type tinyint(1) NOT NULL DEFAULT '0',
-- Google Drive: document identifier
remote_id VARCHAR(200) CHARACTER SET ascii,
PRIMARY KEY (path_hash),
KEY `remote_id` (`remote_id`)
) $charset_collate";
$wpdb->query( $sql );
if ( ! $wpdb->result ) {
throw new Util_Environment_Exception(
esc_html(
sprintf(
// Translators: 1 Tablename for map.
__( 'Can\'t create table %1$s.', 'w3-total-cache' ),
$tablename_map
)
)
);
}
}
/**
* Generates the SQL query to create the CDN queue table.
*
* This method constructs the SQL query needed to create the CDN queue table in the database. It includes the table's
* columns and their types, as well as any necessary character set and collation settings. The query ensures that the
* table is dropped if it exists and creates a new one with the appropriate schema.
*
* @return string The SQL query to create the CDN queue table.
*/
private function generate_table_sql() {
global $wpdb;
$charset_collate = '';
if ( ! empty( $wpdb->charset ) ) {
$charset_collate = "DEFAULT CHARACTER SET $wpdb->charset";
}
if ( ! empty( $wpdb->collate ) ) {
$charset_collate .= " COLLATE $wpdb->collate";
}
$sql = sprintf( 'DROP TABLE IF EXISTS `%s%s`;', $wpdb->base_prefix, W3TC_CDN_TABLE_QUEUE );
$sql .= "\n" . sprintf(
"CREATE TABLE IF NOT EXISTS `%s%s` (
`id` int(11) unsigned NOT NULL AUTO_INCREMENT,
`local_path` varchar(500) NOT NULL DEFAULT '',
`remote_path` varchar(500) NOT NULL DEFAULT '',
`command` tinyint(1) unsigned NOT NULL DEFAULT '0' COMMENT '1 - Upload, 2 - Delete, 3 - Purge',
`last_error` varchar(150) NOT NULL DEFAULT '',
`date` datetime NOT NULL DEFAULT '0000-00-00 00:00:00',
PRIMARY KEY (`id`),
KEY `date` (`date`)
) $charset_collate;",
$wpdb->base_prefix,
W3TC_CDN_TABLE_QUEUE
);
return $sql;
}
/**
* Unschedules the cron job for processing the CDN queue.
*
* This method checks if the cron job for processing the CDN queue is scheduled. If it is, it clears the scheduled hook
* to stop the cron job from running in the future.
*
* @return void
*/
private function unschedule_queue_process() {
if ( wp_next_scheduled( 'w3_cdn_cron_queue_process' ) ) {
wp_clear_scheduled_hook( 'w3_cdn_cron_queue_process' );
}
}
/**
* Unschedules the cron job for uploading to the CDN.
*
* This method checks if the cron job for uploading files to the CDN is scheduled. If it is, it clears the scheduled hook
* to stop the cron job from running in the future.
*
* @return void
*/
private function unschedule_upload() {
if ( wp_next_scheduled( 'w3_cdn_cron_upload' ) ) {
wp_clear_scheduled_hook( 'w3_cdn_cron_upload' );
}
}
/**
* Adds CDN-related rules to the configuration.
*
* This method adds CDN-related rules to the specified configuration. It leverages the `Util_Rule` class to insert these
* rules into the appropriate cache file. The rules are wrapped with markers to ensure they are added in the correct position.
*
* @param object $config The configuration object containing CDN settings.
* @param array $exs The existing rules to which the new rules will be added.
*
* @return void
*/
private function rules_add( $config, $exs ) {
Util_Rule::add_rules(
$exs,
Util_Rule::get_browsercache_rules_cache_path(),
$this->rules_generate( $config ),
W3TC_MARKER_BEGIN_CDN,
W3TC_MARKER_END_CDN,
array(
W3TC_MARKER_BEGIN_MINIFY_CORE => 0,
W3TC_MARKER_BEGIN_PGCACHE_CORE => 0,
W3TC_MARKER_BEGIN_BROWSERCACHE_CACHE => 0,
W3TC_MARKER_BEGIN_WORDPRESS => 0,
W3TC_MARKER_END_PGCACHE_CACHE => strlen( W3TC_MARKER_END_PGCACHE_CACHE ) + 1,
W3TC_MARKER_END_MINIFY_CACHE => strlen( W3TC_MARKER_END_MINIFY_CACHE ) + 1,
)
);
}
/**
* Removes CDN-related rules from the configuration.
*
* This method removes any CDN-related rules from the specified configuration. It uses the `Util_Rule` class to remove
* these rules, which are wrapped with markers to identify their location in the configuration file.
*
* @param array $exs The existing rules from which the CDN-related rules will be removed.
*
* @return void
*/
private function rules_remove( $exs ) {
Util_Rule::remove_rules(
$exs,
Util_Rule::get_browsercache_rules_cache_path(),
W3TC_MARKER_BEGIN_CDN,
W3TC_MARKER_END_CDN
);
}
/**
* Generates the appropriate rules based on the environment.
*
* This method generates rules for the CDN based on the server environment. It detects whether the server is using Nginx,
* LiteSpeed, or Apache and calls the corresponding method to generate the rules for that environment.
*
* @param object $config The configuration object containing CDN settings.
* @param bool $cdnftp Whether FTP settings should be included in the rules.
*
* @return string The generated CDN rules for the environment.
*/
private function rules_generate( $config, $cdnftp = false ) {
if ( Util_Environment::is_nginx() ) {
$o = new Cdn_Environment_Nginx( $config );
return $o->generate( $cdnftp );
} elseif ( Util_Environment::is_litespeed() ) {
$o = new Cdn_Environment_LiteSpeed( $config );
return $o->generate( $cdnftp );
} else {
return $this->rules_generate_apache( $config, $cdnftp );
}
}
/**
* Generates Apache-specific CDN rules.
*
* This method generates the CDN rules specifically for Apache-based environments. It adds rules for canonical URLs, CORS,
* and other CDN-specific headers.
*
* @param object $config The configuration object containing CDN settings.
* @param bool $cdnftp Whether FTP settings should be included in the rules.
*
* @return string The generated Apache-specific CDN rules.
*/
private function rules_generate_apache( $config, $cdnftp ) {
$rules = '';
if ( $config->get_boolean( 'cdn.canonical_header' ) ) {
$rules .= $this->canonical( $cdnftp, $config->get_boolean( 'cdn.cors_header' ) );
}
if ( $config->get_boolean( 'cdn.cors_header' ) ) {
$rules .= $this->allow_origin( $cdnftp );
}
if ( strlen( $rules ) > 0 ) {
$rules = W3TC_MARKER_BEGIN_CDN . "\n" . $rules . W3TC_MARKER_END_CDN . "\n";
}
return $rules;
}
/**
* Generates the canonical URL rule.
*
* This method generates the Apache or Nginx rewrite rules to enforce canonical URLs for the CDN. It ensures that requests
* are properly redirected to the correct HTTP or HTTPS version of the URL.
*
* @param bool $cdnftp Whether FTP settings should be included in the rule.
* @param bool $cors_header Whether CORS headers should be included in the rule.
*
* @return string The generated canonical URL rule.
*/
private function canonical( $cdnftp = false, $cors_header = true ) {
$mime_types = include W3TC_INC_DIR . '/mime/other.php';
$extensions = array_keys( $mime_types );
$extensions_lowercase = array_map( 'strtolower', $extensions );
$extensions_uppercase = array_map( 'strtoupper', $extensions );
$host = $cdnftp ? Util_Environment::home_url_host() : '%{HTTP_HOST}';
$rules = '<FilesMatch "\\.(' . implode( '|', array_merge( $extensions_lowercase, $extensions_uppercase ) ) . ")$\">\n";
$rules .= " <IfModule mod_rewrite.c>\n";
$rules .= " RewriteEngine On\n";
$rules .= " RewriteCond %{HTTPS} !=on\n";
$rules .= " RewriteRule .* - [E=CANONICAL:http://$host%{REQUEST_URI},NE]\n";
$rules .= " RewriteCond %{HTTPS} =on\n";
$rules .= " RewriteRule .* - [E=CANONICAL:https://$host%{REQUEST_URI},NE]\n";
$rules .= " </IfModule>\n";
$rules .= " <IfModule mod_headers.c>\n";
$rules .= ' Header set Link "<%{CANONICAL}e>; rel=\"canonical\""' . "\n";
$rules .= " </IfModule>\n";
$rules .= "</FilesMatch>\n";
return $rules;
}
/**
* Generates the CORS allow origin rule.
*
* This method generates the Apache or Nginx header rule for allowing cross-origin resource sharing (CORS). It sets the
* `Access-Control-Allow-Origin` header to allow any origin, which is useful for CDN assets.
*
* @param bool $cdnftp Whether FTP settings should be included in the rule.
*
* @return string The generated CORS allow origin rule.
*/
private function allow_origin( $cdnftp = false ) {
$r = "<IfModule mod_headers.c>\n";
$r .= " Header set Access-Control-Allow-Origin \"*\"\n";
$r .= "</IfModule>\n";
if ( ! $cdnftp ) {
return $r;
} else {
return "<FilesMatch \"\.(ttf|ttc|otf|eot|woff|woff2|font.css)$\">\n" . $r . "</FilesMatch>\n";
}
}
/**
* Adds browser cache extension rules for CDN in the configuration.
*
* This method adds browser caching rules for CDN extensions to the specified configuration. It detects the server type
* (Nginx or LiteSpeed) and generates the appropriate extension rules based on the server.
*
* @param array $extensions The existing list of extensions to be cached.
* @param object $config The configuration object containing CDN settings.
* @param string $section The section of the configuration to add the rules to.
*
* @return array The updated list of extensions with CDN-related rules.
*/
public function w3tc_browsercache_rules_section_extensions( $extensions, $config, $section ) {
if ( Util_Environment::is_nginx() ) {
$o = new Cdn_Environment_Nginx( $config );
$extensions = $o->w3tc_browsercache_rules_section_extensions( $extensions, $section );
} elseif ( Util_Environment::is_litespeed() ) {
$o = new Cdn_Environment_LiteSpeed( $config );
$extensions = $o->w3tc_browsercache_rules_section_extensions( $extensions, $section );
}
return $extensions;
}
/**
* Adds browser cache section rules for CDN in the configuration.
*
* This method adds browser caching rules for the CDN to the specified section in the configuration. It works specifically
* for LiteSpeed servers, ensuring that the correct rules are added for CDN caching.
*
* @param string $section_rules The existing section rules to be updated.
* @param object $config The configuration object containing CDN settings.
* @param string $section The section of the configuration to add the rules to.
*
* @return string The updated section rules with CDN-related browser cache rules.
*/
public function w3tc_browsercache_rules_section( $section_rules, $config, $section ) {
if ( Util_Environment::is_litespeed() ) {
$o = new Cdn_Environment_LiteSpeed( $config );
$section_rules = $o->w3tc_browsercache_rules_section( $section_rules, $section );
}
return $section_rules;
}
}