oont-contents/plugins/wp-rocket/inc/Engine/Optimization/AssetsLocalCache.php
2025-02-08 15:10:23 +01:00

188 lines
4.2 KiB
PHP

<?php
namespace WP_Rocket\Engine\Optimization;
use WP_Filesystem_Direct;
/**
* Cache locally 3rd party assets.
*
* @since 3.1
*/
class AssetsLocalCache {
/**
* 3rd party assets cache folder path
*
* @since 3.1
*
* @var string
*/
protected $cache_path;
/**
* Filesystem instance.
*
* @var WP_Filesystem_Direct
*/
private $filesystem;
/**
* Cache contents with url.
*
* @var array
*/
private $url_cache = [];
/**
* Constructor
*
* @since 3.1
*
* @param string $cache_path 3rd party assets cache folder path.
* @param WP_Filesystem_Direct $filesystem Filesysten instance.
*/
public function __construct( $cache_path, $filesystem ) {
$this->cache_path = "{$cache_path}3rd-party/";
$this->filesystem = $filesystem;
}
/**
* Get remote file contents.
*
* @param string $url Url of the file to get contents for.
*
* @return string Raw file contents.
*/
private function get_raw_content( $url ) {
$cache_key = md5( $url );
if ( ! isset( $this->url_cache[ $cache_key ] ) ) {
$this->url_cache[ $cache_key ] = wp_remote_retrieve_body( wp_remote_get( $url ) );
}
return $this->url_cache[ $cache_key ];
}
/**
* Gets content for the provided URL.
* Use the local cache file if it exists, else get it from the 3rd party URL and save it locally for future use.
*
* @since 3.1
*
* @param string $url URL to get the content from.
*
* @return string
*/
public function get_content( $url ) {
$filepath = $this->get_filepath( $url );
if ( empty( $filepath ) ) {
return '';
}
if ( $this->filesystem->is_readable( $filepath ) ) {
return $this->filesystem->get_contents( $filepath );
}
$content = $this->get_raw_content( $url );
if ( empty( $content ) ) {
return '';
}
$this->write_file( $content, $filepath );
return $content;
}
/**
* Gets the filepath of the local copy for the given URL
*
* @since 3.7
*
* @param string $url URL to get filepath for.
* @return string
*/
public function get_filepath( $url ) {
$parts = wp_parse_url( $url );
if ( empty( $parts['path'] ) ) {
return '';
}
$filename = $parts['host'] . str_replace( '/', '-', $parts['path'] );
return $this->cache_path . $filename;
}
/**
* Writes the content to a file
*
* @since 3.1
*
* @param string $content Content to write.
* @param string $file Path to the file to write in.
* @return bool
*/
protected function write_file( $content, $file ) {
if ( $this->filesystem->is_readable( $file ) ) {
return true;
}
if ( ! rocket_mkdir_p( dirname( $file ) ) ) {
return false;
}
return rocket_put_content( $file, $content );
}
/**
* Check if this link HTML has integrity attribute or not?
*
* @since 3.7.5
*
* @param string $asset Link HTML to be tested.
*
* @return array|false Matched array with integrityhashmethod, integrityhash keys.
*/
private function has_integrity( $asset ) {
if ( ! preg_match( '#\s*integrity\s*=[\'"](?<integrityhashmethod>.*)-(?<integrityhash>.*)[\'"]#Ui', $asset, $integrity_matches ) ) {
return false;
}
if ( ! isset( $integrity_matches['integrityhashmethod'], $integrity_matches['integrityhash'] ) ) {
return false;
}
return $integrity_matches;
}
/**
* Validate the integrity attribute if the content matches with the hashed integrity value.
*
* @param array $asset_matched the matched array which has 0, url keys.
*
* @return bool|string
*/
public function validate_integrity( $asset_matched ) {
$integrity_matches = $this->has_integrity( $asset_matched[0] );
if ( false === $integrity_matches ) {
return $asset_matched[0];
}
// validate the hash algorithm.
if ( ! in_array( $integrity_matches['integrityhashmethod'], hash_algos(), true ) ) {
return false;
}
$content = $this->get_raw_content( $asset_matched['url'] );
$content_hash = base64_encode( hash( $integrity_matches['integrityhashmethod'], $content, true ) );// phpcs:ignore WordPress.PHP.DiscouragedPHPFunctions.obfuscation_base64_encode
if ( $integrity_matches['integrityhash'] !== $content_hash ) {
return false;
}
return str_replace( $integrity_matches[0], '', $asset_matched[0] );
}
}