_config = Dispatcher::config(); $this->debug = $this->_config->get_boolean( 'cdn.debug' ); } /** * Adds file to queue * * @param string $local_path * @param string $remote_path * @param integer $command * @param string $last_error * @return integer */ function queue_add( $local_path, $remote_path, $command, $last_error ) { global $wpdb; $table = $wpdb->base_prefix . W3TC_CDN_TABLE_QUEUE; $rows = $wpdb->get_results( $wpdb->prepare( 'SELECT id, command '. "FROM $table " . 'WHERE local_path = %s AND remote_path = %s', $local_path, $remote_path ) ); $already_exists = false; foreach ( $rows as $row ) { if ( $row->command != $command ) $wpdb->query( $wpdb->prepare( "DELETE FROM $table " . 'WHERE id = %d', $row->id ) ); else $already_exists = true; } if ( $already_exists ) return true; // insert if not yet there return $wpdb->query( $wpdb->prepare( "INSERT INTO $table " . '(local_path, remote_path, command, last_error, date) ' . 'VALUES (%s, %s, %d, %s, NOW())', $local_path, $remote_path, $command, $last_error ) ); } /** * Returns array of array('local_path' => '', 'remote_path' => '') for specified file * * @param string $file * @return array */ function get_files_for_upload( $file ) { $files = array(); $upload_info = Util_Http::upload_info(); if ( $upload_info ) { $file = $this->normalize_attachment_file( $file ); $local_file = $upload_info['basedir'] . '/' . $file; $parsed = parse_url( rtrim( $upload_info['baseurl'], '/' ) . '/' . $file ); $local_uri = $parsed['path']; $remote_uri = $this->uri_to_cdn_uri( $local_uri ); $remote_file = ltrim( $remote_uri, '/' ); $files[] = $this->build_file_descriptor( $local_file, $remote_file ); } return $files; } /** * Returns array of files from sizes array * * @param string $attached_file * @param array $sizes * @return array */ function _get_sizes_files( $attached_file, $sizes ) { $files = array(); $base_dir = Util_File::dirname( $attached_file ); foreach ( (array) $sizes as $size ) { if ( isset( $size['file'] ) ) { if ( $base_dir ) { $file = $base_dir . '/' . $size['file']; } else { $file = $size['file']; } $files = array_merge( $files, $this->get_files_for_upload( $file ) ); } } return $files; } /** * Returns attachment files by metadata * * @param array $metadata * @return array */ function get_metadata_files( $metadata ) { $files = array(); if ( isset( $metadata['file'] ) && isset( $metadata['sizes'] ) ) { $files = array_merge( $files, $this->_get_sizes_files( $metadata['file'], $metadata['sizes'] ) ); } return $files; } /** * Returns attachment files by attachment ID * * @param integer $attachment_id * @return array */ function get_attachment_files( $attachment_id ) { $files = array(); /** * Get attached file */ $attached_file = get_post_meta( $attachment_id, '_wp_attached_file', true ); if ( $attached_file != '' ) { $files = array_merge( $files, $this->get_files_for_upload( $attached_file ) ); /** * Get backup sizes files */ $attachment_backup_sizes = get_post_meta( $attachment_id, '_wp_attachment_backup_sizes', true ); if ( is_array( $attachment_backup_sizes ) ) { $files = array_merge( $files, $this->_get_sizes_files( $attached_file, $attachment_backup_sizes ) ); } } /** * Get files from metadata */ $attachment_metadata = get_post_meta( $attachment_id, '_wp_attachment_metadata', true ); if ( is_array( $attachment_metadata ) ) { $files = array_merge( $files, $this->get_metadata_files( $attachment_metadata ) ); } return $files; } /** * Uploads files to CDN * * @param array $files * @param boolean $queue_failed * @param array $results * @return boolean */ function upload( $files, $queue_failed, &$results, $timeout_time = NULL ) { if ( $this->debug ) { Util_Debug::log( 'cdn', 'upload: ' . json_encode( $files, JSON_PRETTY_PRINT ) ); } $cdn = $this->get_cdn(); $force_rewrite = $this->_config->get_boolean( 'cdn.force.rewrite' ); @set_time_limit( $this->_config->get_integer( 'timelimit.cdn_upload' ) ); $engine = $this->_config->get_string( 'cdn.engine' ); $return = $cdn->upload( $files, $results, $force_rewrite, $timeout_time ); if ( !$return && $queue_failed ) { foreach ( $results as $result ) { if ( $result['result'] != W3TC_CDN_RESULT_OK ) { $this->queue_add( $result['local_path'], $result['remote_path'], W3TC_CDN_COMMAND_UPLOAD, $result['error'] ); } } } return $return; } /** * Deletes files frrom CDN * * @param array $files * @param boolean $queue_failed * @param array $results * @return boolean */ function delete( $files, $queue_failed, &$results ) { $cdn = $this->get_cdn(); @set_time_limit( $this->_config->get_integer( 'timelimit.cdn_delete' ) ); $return = $cdn->delete( $files, $results ); if ( $this->debug ) { Util_Debug::log( 'cdn', 'delete: ' . json_encode( $files, JSON_PRETTY_PRINT ) ); } if ( !$return && $queue_failed ) { foreach ( $results as $result ) { if ( $result['result'] != W3TC_CDN_RESULT_OK ) { $this->queue_add( $result['local_path'], $result['remote_path'], W3TC_CDN_COMMAND_DELETE, $result['error'] ); } } } return $return; } /** * Purges files from CDN * * @param array $files consisting of array('local_path'=>'', 'remote_path'=>'') * @param boolean $queue_failed * @param array $results * @return boolean */ function purge( $files, &$results ) { if ( $this->debug ) { Util_Debug::log( 'cdn', 'purge: ' . json_encode( $files, JSON_PRETTY_PRINT ) ); } /** * Purge varnish servers before mirror purging */ if ( Cdn_Util::is_engine_mirror( $this->_config->get_string( 'cdn.engine' ) ) && $this->_config->get_boolean( 'varnish.enabled' ) ) { $varnish = Dispatcher::component( 'Varnish_Flush' ); foreach ( $files as $file ) { $remote_path = $file['remote_path']; $varnish->flush_url( network_site_url( $remote_path ) ); } } /** * Purge CDN */ $cdn = $this->get_cdn(); @set_time_limit( $this->_config->get_integer( 'timelimit.cdn_purge' ) ); $return = $cdn->purge( $files, $results ); if ( !$return ) { foreach ( $results as $result ) { if ( $result['result'] != W3TC_CDN_RESULT_OK ) { $this->queue_add( $result['local_path'], $result['remote_path'], W3TC_CDN_COMMAND_PURGE, $result['error'] ); } } } return $return; } /** * Purge CDN completely * * @param unknown $results * @return mixed */ function purge_all( &$results ) { /** * Purge CDN */ $cdn = $this->get_cdn(); @set_time_limit( $this->_config->get_integer( 'timelimit.cdn_purge' ) ); $return = $cdn->purge_all( $results ); return $return; } /** * Queues file upload. * Links wp_cron call to do that by the end of request processing * * @param string $url * @return void */ function queue_upload_url( $url ) { $docroot_filename = Util_Environment::url_to_docroot_filename( $url ); if ( is_null( $docroot_filename ) ) { return; } $filename = Util_Environment::docroot_to_full_filename( $docroot_filename ); $a = parse_url( $url ); $uri = $a['path']; $remote_file_name = $this->uri_to_cdn_uri( $uri ); $this->queue_add( $filename, $remote_file_name, W3TC_CDN_COMMAND_UPLOAD, 'Pending' ); } /** * Normalizes attachment file * * @param string $file * @return string */ function normalize_attachment_file( $file ) { $upload_info = Util_Http::upload_info(); if ( $upload_info ) { $file = ltrim( str_replace( $upload_info['basedir'], '', $file ), '/\\' ); $matches = null; if ( preg_match( '~(\d{4}/\d{2}/)?[^/]+$~', $file, $matches ) ) { $file = $matches[0]; } } return $file; } /** * Returns CDN object */ function get_cdn() { static $cdn = null; if ( is_null( $cdn ) ) { $c = $this->_config; $engine = $c->get_string( 'cdn.engine' ); $compression = ( $c->get_boolean( 'browsercache.enabled' ) && $c->get_boolean( 'browsercache.html.compression' ) ); switch ( $engine ) { case 'akamai': $engine_config = array( 'username' => $c->get_string( 'cdn.akamai.username' ), 'password' => $c->get_string( 'cdn.akamai.password' ), 'zone' => $c->get_string( 'cdn.akamai.zone' ), 'domain' => $c->get_array( 'cdn.akamai.domain' ), 'ssl' => $c->get_string( 'cdn.akamai.ssl' ), 'email_notification' => $c->get_array( 'cdn.akamai.email_notification' ), 'compression' => false ); break; case 'att': $engine_config = array( 'account' => $c->get_string( 'cdn.att.account' ), 'token' => $c->get_string( 'cdn.att.token' ), 'domain' => $c->get_array( 'cdn.att.domain' ), 'ssl' => $c->get_string( 'cdn.att.ssl' ), 'compression' => false ); break; case 'azure': $engine_config = array( 'user' => $c->get_string( 'cdn.azure.user' ), 'key' => $c->get_string( 'cdn.azure.key' ), 'container' => $c->get_string( 'cdn.azure.container' ), 'cname' => $c->get_array( 'cdn.azure.cname' ), 'ssl' => $c->get_string( 'cdn.azure.ssl' ), 'compression' => false ); break; case 'cf': $engine_config = array( 'key' => $c->get_string( 'cdn.cf.key' ), 'secret' => $c->get_string( 'cdn.cf.secret' ), 'bucket' => $c->get_string( 'cdn.cf.bucket' ), 'bucket_location' => $c->get_string( 'cdn.cf.bucket.location' ), 'id' => $c->get_string( 'cdn.cf.id' ), 'cname' => $c->get_array( 'cdn.cf.cname' ), 'ssl' => $c->get_string( 'cdn.cf.ssl' ), 'public_objects' => $c->get_string( 'cdn.cf.public_objects' ), 'compression' => $compression ); break; case 'cf2': $engine_config = array( 'key' => $c->get_string( 'cdn.cf2.key' ), 'secret' => $c->get_string( 'cdn.cf2.secret' ), 'id' => $c->get_string( 'cdn.cf2.id' ), 'cname' => $c->get_array( 'cdn.cf2.cname' ), 'ssl' => $c->get_string( 'cdn.cf2.ssl' ), 'compression' => false ); break; case 'cotendo': $engine_config = array( 'username' => $c->get_string( 'cdn.cotendo.username' ), 'password' => $c->get_string( 'cdn.cotendo.password' ), 'zones' => $c->get_array( 'cdn.cotendo.zones' ), 'domain' => $c->get_array( 'cdn.cotendo.domain' ), 'ssl' => $c->get_string( 'cdn.cotendo.ssl' ), 'compression' => false ); break; case 'edgecast': $engine_config = array( 'account' => $c->get_string( 'cdn.edgecast.account' ), 'token' => $c->get_string( 'cdn.edgecast.token' ), 'domain' => $c->get_array( 'cdn.edgecast.domain' ), 'ssl' => $c->get_string( 'cdn.edgecast.ssl' ), 'compression' => false ); break; case 'ftp': $engine_config = array( 'host' => $c->get_string( 'cdn.ftp.host' ), 'type' => $c->get_string( 'cdn.ftp.type' ), 'user' => $c->get_string( 'cdn.ftp.user' ), 'pass' => $c->get_string( 'cdn.ftp.pass' ), 'path' => $c->get_string( 'cdn.ftp.path' ), 'pasv' => $c->get_boolean( 'cdn.ftp.pasv' ), 'domain' => $c->get_array( 'cdn.ftp.domain' ), 'ssl' => $c->get_string( 'cdn.ftp.ssl' ), 'compression' => false, 'docroot' => Util_Environment::document_root() ); break; case 'google_drive': $state = Dispatcher::config_state(); $engine_config = array( 'client_id' => $c->get_string( 'cdn.google_drive.client_id' ), 'access_token' => $state->get_string( 'cdn.google_drive.access_token' ), 'refresh_token' => $c->get_string( 'cdn.google_drive.refresh_token' ), 'root_url' => $c->get_string( 'cdn.google_drive.folder.url' ), 'root_folder_id' => $c->get_string( 'cdn.google_drive.folder.id' ), 'new_access_token_callback' => array( $this, 'on_google_drive_new_access_token' ) ); break; case 'highwinds': $state = Dispatcher::config_state(); $engine_config = array( 'domains' => $c->get_array( 'cdn.highwinds.host.domains' ), 'ssl' => $c->get_string( 'cdn.highwinds.ssl' ), 'api_token' => $c->get_string( 'cdn.highwinds.api_token' ), 'account_hash' => $c->get_string( 'cdn.highwinds.account_hash' ), 'host_hash_code' => $c->get_string( 'cdn.highwinds.host.hash_code' ) ); break; case 'limelight': $engine_config = array( 'short_name' => $c->get_string( 'cdn.limelight.short_name' ), 'username' => $c->get_string( 'cdn.limelight.username' ), 'api_key' => $c->get_string( 'cdn.limelight.api_key' ), 'domains' => $c->get_array( 'cdn.limelight.host.domains' ), 'debug' => $c->get_string( 'cdn.debug' ) ); break; case 'mirror': $engine_config = array( 'domain' => $c->get_array( 'cdn.mirror.domain' ), 'ssl' => $c->get_string( 'cdn.mirror.ssl' ), 'compression' => false ); break; case 'rackspace_cdn': $state = Dispatcher::config_state(); $engine_config = array( 'user_name' => $c->get_string( 'cdn.rackspace_cdn.user_name' ), 'api_key' => $c->get_string( 'cdn.rackspace_cdn.api_key' ), 'region' => $c->get_string( 'cdn.rackspace_cdn.region' ), 'service_access_url' => $c->get_string( 'cdn.rackspace_cdn.service.access_url' ), 'service_id' => $c->get_string( 'cdn.rackspace_cdn.service.id' ), 'service_protocol' => $c->get_string( 'cdn.rackspace_cdn.service.protocol' ), 'domains' => $c->get_array( 'cdn.rackspace_cdn.domains' ), 'access_state' => $state->get_string( 'cdn.rackspace_cdn.access_state' ), 'new_access_state_callback' => array( $this, 'on_rackspace_cdn_new_access_state' ) ); break; case 'rscf': $state = Dispatcher::config_state(); $engine_config = array( 'user_name' => $c->get_string( 'cdn.rscf.user' ), 'api_key' => $c->get_string( 'cdn.rscf.key' ), 'region' => $c->get_string( 'cdn.rscf.location' ), 'container' => $c->get_string( 'cdn.rscf.container' ), 'cname' => $c->get_array( 'cdn.rscf.cname' ), 'ssl' => $c->get_string( 'cdn.rscf.ssl' ), 'compression' => false, 'access_state' => $state->get_string( 'cdn.rackspace_cf.access_state' ), 'new_access_state_callback' => array( $this, 'on_rackspace_cf_new_access_state' ) ); break; case 's3': $engine_config = array( 'key' => $c->get_string( 'cdn.s3.key' ), 'secret' => $c->get_string( 'cdn.s3.secret' ), 'bucket' => $c->get_string( 'cdn.s3.bucket' ), 'bucket_location' => $c->get_string( 'cdn.s3.bucket.location' ), 'cname' => $c->get_array( 'cdn.s3.cname' ), 'ssl' => $c->get_string( 'cdn.s3.ssl' ), 'public_objects' => $c->get_string( 'cdn.s3.public_objects' ), 'compression' => $compression ); break; case 's3_compatible': $engine_config = array( 'key' => $c->get_string( 'cdn.s3.key' ), 'secret' => $c->get_string( 'cdn.s3.secret' ), 'bucket' => $c->get_string( 'cdn.s3.bucket' ), 'cname' => $c->get_array( 'cdn.s3.cname' ), 'ssl' => $c->get_string( 'cdn.s3.ssl' ), 'compression' => $compression, 'api_host' => $c->get_string( 'cdn.s3_compatible.api_host' ) ); break; case 'stackpath': $engine_config = array( 'authorization_key' => $c->get_string( 'cdn.stackpath.authorization_key' ), 'zone_id' => $c->get_integer( 'cdn.stackpath.zone_id' ), 'domain' => $c->get_array( 'cdn.stackpath.domain' ), 'ssl' => $c->get_string( 'cdn.stackpath.ssl' ), 'compression' => false ); break; case 'stackpath2': $state = Dispatcher::config_state(); $engine_config = array( 'client_id' => $c->get_string( 'cdn.stackpath2.client_id' ), 'client_secret' => $c->get_string( 'cdn.stackpath2.client_secret' ), 'stack_id' => $c->get_string( 'cdn.stackpath2.stack_id' ), 'site_root_domain' => $c->get_string( 'cdn.stackpath2.site_root_domain' ), 'domain' => $c->get_array( 'cdn.stackpath2.domain' ), 'ssl' => $c->get_string( 'cdn.stackpath2.ssl' ), 'access_token' => $state->get_string( 'cdn.stackpath2.access_token' ), 'on_new_access_token' => array( $this, 'on_stackpath2_new_access_token' ) ); break; } $engine_config = array_merge( $engine_config, array( 'debug' => $c->get_boolean( 'cdn.debug' ), 'headers' => apply_filters( 'w3tc_cdn_config_headers', array() ) ) ); $cdn = CdnEngine::instance( $engine, $engine_config ); } return $cdn; } /** * Called when new access token is issued by cdnengine */ public function on_google_drive_new_access_token( $access_token ) { $state = Dispatcher::config_state(); $state->set( 'cdn.google_drive.access_token', $access_token ); $state->save(); } /** * Called when new access state is issued by cdnengine */ public function on_rackspace_cdn_new_access_state( $access_state ) { $state = Dispatcher::config_state(); $state->set( 'cdn.rackspace_cdn.access_state', $access_state ); $state->save(); } /** * Called when new access state is issued by cdnengine */ public function on_rackspace_cf_new_access_state( $access_state ) { $state = Dispatcher::config_state(); $state->set( 'cdn.rackspace_cf.access_state', $access_state ); $state->save(); } public function on_stackpath2_new_access_token( $access_token ) { $state = Dispatcher::config_state(); $state->set( 'cdn.stackpath2.access_token', $access_token ); $state->save(); } /** * Convert relative file which is relative to ABSPATH (wp folder on disc) to path uri * * @param unknown $file * @return string */ function docroot_filename_to_uri( $file ) { $file = ltrim( $file, '/' ); // Translate multisite subsite uploads paths $file = str_replace( basename( WP_CONTENT_DIR ) . '/blogs.dir/' . Util_Environment::blog_id() . '/', '', $file ); return $file; } /** * Convert a relative path (relative to ABSPATH (wp folder on disc) into a absolute path * * @param unknown $file * @return string */ function docroot_filename_to_absolute_path( $file ) { if ( is_file( $file ) ) return $file; if ( DIRECTORY_SEPARATOR != '/' ) $file = str_replace( '/', DIRECTORY_SEPARATOR, $file ); return rtrim( Util_Environment::document_root(), '/\\' ) . DIRECTORY_SEPARATOR . ltrim( $file, '/\\' ); } /** * Convert local uri path to CDN type specific path * * @param unknown $local_uri_path * @return string */ function uri_to_cdn_uri( $local_uri ) { $local_uri = ltrim( $local_uri, '/' ); $remote_uri = $local_uri; if ( Util_Environment::is_wpmu() && defined( 'DOMAIN_MAPPING' ) && DOMAIN_MAPPING ) $remote_uri = str_replace( site_url(), '', $local_uri ); $engine = $this->_config->get_string( 'cdn.engine' ); if ( Cdn_Util::is_engine_mirror( $engine ) ) { if ( Util_Environment::is_wpmu() && strpos( $local_uri, 'files' ) === 0 ) { $upload_dir = Util_Environment::wp_upload_dir(); $remote_uri = $this->abspath_to_relative_path( dirname( $upload_dir['basedir'] ) ) . '/' . $local_uri; } } elseif ( Util_Environment::is_wpmu() && !Util_Environment::is_wpmu_subdomain() && Util_Environment::is_using_master_config() && Cdn_Util::is_engine_push( $engine ) ) { // in common config mode files are uploaded for network home url // so mirror will not contain /subblog/ path in uri // // since upload process is not blog-specific and // wp-content/plugins/../*.jpg files are common $home = trim( home_url( '', 'relative' ), '/' ) . '/'; $network_home = trim( network_home_url( '', 'relative' ), '/' ) . '/'; if ( $home != $network_home && substr( $local_uri, 0, strlen( $home ) ) == $home ) { $remote_uri = $network_home . substr( $local_uri, strlen( $home ) ); } } return apply_filters( 'w3tc_uri_cdn_uri', ltrim( $remote_uri, '/' ) ); } /** * Need to pass full URL and it's URI * URI passed to prevent redundant parsing, normally it's available for caller **/ function url_to_cdn_url( $url, $path ) { $cdn = $this->get_cdn(); $remote_path = $this->uri_to_cdn_uri( $path ); $new_url = $cdn->format_url( $remote_path ); if ( !$new_url ) { return null; } $is_engine_mirror = Cdn_Util::is_engine_mirror( $this->_config->get_string( 'cdn.engine' ) ); $new_url = apply_filters( 'w3tc_cdn_url', $new_url, $url, $is_engine_mirror ); return $new_url; } /** * Returns the sitepath for multisite subfolder or subdomain path for multisite subdomain * * @return string */ private function _get_multisite_url_identifier() { if ( defined( 'DOMAIN_MAPPING' ) && DOMAIN_MAPPING ) { $parsedUrl = parse_url( site_url() ); return $parsedUrl['host']; } elseif ( Util_Environment::is_wpmu_subdomain() ) { $parsedUrl = parse_url( Util_Environment::home_domain_root_url() ); $urlparts = explode( '.', $parsedUrl['host'] ); if ( sizeof( $urlparts ) > 2 ) { $subdomain = array_shift( $urlparts ); return trim( $subdomain, '/' ); } } return trim( Util_Environment::site_url_uri(), '/' ); } /** * Taks an absolute path and converts to a relative path to root * * @param unknown $path * @return mixed */ function abspath_to_relative_path( $path ) { return str_replace( Util_Environment::document_root(), '', $path ); } /** * Takes a root relative path and converts to a full uri * * @param unknown $path * @return string */ function relative_path_to_url( $path ) { $cdnuri = $this->docroot_filename_to_uri( ltrim( $path, "/" ) ); return rtrim( Util_Environment::home_domain_root_url(), "/" ) . '/' . $cdnuri; } /** * Constructs a CDN file descriptor * * @param unknown $local_path * @param unknown $remote_path * @return array */ function build_file_descriptor( $local_path, $remote_path ) { $file = array( 'local_path' => $local_path, 'remote_path' => $remote_path, 'original_url' => $this->relative_path_to_url( $local_path ) ); $file = apply_filters( 'w3tc_build_cdn_file_array', $file ); return $file; } }