AlkantarClanX12

Your IP : 18.118.32.7


Current Path : /home/thanudqk/www/wp-content/plugins/w3-total-cache/lib/Minify/Minify/CSS/
Upload File :
Current File : /home/thanudqk/www/wp-content/plugins/w3-total-cache/lib/Minify/Minify/CSS/UriRewriter.php

<?php
/**
 * File: UriRewriter.php
 *
 * NOTE: Fixes have been included in this file; look for "W3TC FIX".
 */

 namespace W3TCL\Minify;
/**
 * Class Minify_CSS_UriRewriter
 * @package Minify
 */

/**
 * Rewrite file-relative URIs as root-relative in CSS files
 *
 * @package Minify
 * @author Stephen Clay <steve@mrclay.org>
 */
class Minify_CSS_UriRewriter {

	/**
	 * rewrite() and rewriteRelative() append debugging information here
	 *
	 * @var string
	 */
	public static $debugText = '';

	/**
	 * In CSS content, rewrite file relative URIs as root relative
	 *
	 * @param string $css
	 *
	 * @param string $currentDir The directory of the current CSS file.
	 *
	 * @param string $docRoot The document root of the web site in which
	 * the CSS file resides (default = $_SERVER['DOCUMENT_ROOT']).
	 *
	 * @param array $symlinks (default = array()) If the CSS file is stored in
	 * a symlink-ed directory, provide an array of link paths to
	 * target paths, where the link paths are within the document root. Because
	 * paths need to be normalized for this to work, use "//" to substitute
	 * the doc root in the link paths (the array keys). E.g.:
	 * <code>
	 * array('//symlink' => '/real/target/path') // unix
	 * array('//static' => 'D:\\staticStorage')  // Windows
	 * </code>
	 *
	 * @return string
	 */
	public static function rewrite($css, $options) {
		self::$_prependPath = null;

		if (!isset($options['prependRelativePath']) && !isset($options['currentDir']))
			return $css;

		self::$_browserCacheId = (isset($options['browserCacheId']) ?
			$options['browserCacheId'] : 0);
		self::$_browserCacheExtensions =
			(isset($options['browserCacheExtensions']) ?
			$options['browserCacheExtensions'] : array());

		// W3TC FIX: Override $_SERVER['DOCUMENT_ROOT'] if enabled in settings.
		$docroot = \W3TC\Util_Environment::document_root();

		if (isset($options['currentDir'])) {
			$document_root = (isset($options['docRoot']) ? $options['docRoot'] : $docroot);
			$symlinks = (isset($options['symlinks']) ? $options['symlinks'] : array());
			$prependAbsolutePath = (isset($options['prependAbsolutePath']) ? $options['prependAbsolutePath'] : '');
			$prependAbsolutePathCallback = (isset($options['prependAbsolutePathCallback']) ? $options['prependAbsolutePathCallback'] : '');

			$css = self::_rewrite(
				$css,
				$options['currentDir'],
				$prependAbsolutePath,
				$prependAbsolutePathCallback,
				$document_root,
				$symlinks
			);
		} elseif (isset($options['prependRelativePath'])) {
			$css = self::prepend(
				$css,
				$options['prependRelativePath']
			);
		}

		return $css;
	}

	/**
	 * Rewrite file relative URIs as root relative in CSS files
	 *
	 * @param string $css
	 *
	 * @param string $currentDir The directory of the current CSS file.
	 *
	 * @param string $prependAbsolutePath
	 *
	 * @param string $prependAbsolutePathCallback
	 *
	 * @param string $docRoot The document root of the web site in which
	 * the CSS file resides (default = $_SERVER['DOCUMENT_ROOT']).
	 *
	 * @param array $symlinks (default = array()) If the CSS file is stored in
	 * a symlink-ed directory, provide an array of link paths to
	 * target paths, where the link paths are within the document root. Because
	 * paths need to be normalized for this to work, use "//" to substitute
	 * the doc root in the link paths (the array keys). E.g.:
	 * <code>
	 * array('//symlink' => '/real/target/path') // unix
	 * array('//static' => 'D:\\staticStorage')  // Windows
	 * </code>
	 *
	 * @param int $browserCacheId
	 *
	 * @param array $browserCacheExtensions
	 *
	 * @return string
	 */
	private static function _rewrite($css, $currentDir,
			$prependAbsolutePath = null, $prependAbsolutePathCallback = null,
			$docRoot = null, $symlinks = array()) {
				// W3TC FIX: Override $_SERVER['DOCUMENT_ROOT'] if enabled in settings.
				$docroot = \W3TC\Util_Environment::document_root();

		self::$_docRoot = self::_realpath(
			$docRoot ? $docRoot : $docroot
		);
		self::$_currentDir = self::_realpath($currentDir);
		self::$_prependAbsolutePath = $prependAbsolutePath;
		self::$_prependAbsolutePathCallback = $prependAbsolutePathCallback;
		self::$_symlinks = array();

		// normalize symlinks
		foreach ($symlinks as $link => $target) {
			$link = ($link === '//')
				? self::$_docRoot
				: str_replace('//', self::$_docRoot . '/', $link);
			$link = strtr($link, '/', DIRECTORY_SEPARATOR);
			self::$_symlinks[$link] = self::_realpath($target);
		}

		self::$debugText .= "docRoot    : " . self::$_docRoot . "\n"
						  . "currentDir : " . self::$_currentDir . "\n";
		if (self::$_symlinks) {
			self::$debugText .= "symlinks : " . var_export(self::$_symlinks, 1) . "\n";
		}
		self::$debugText .= "\n";

		$css = self::_trimUrls($css);

		// rewrite
		$css = preg_replace_callback('/@import\\s+([\'"])(.*?)[\'"]/'
			,array(self::$className, '_processUriCB'), $css);
		$css = preg_replace_callback('/url\\(\\s*([^\\)\\s]+)\\s*\\)/'
			,array(self::$className, '_processUriCB'), $css);

		return $css;
	}

	/**
	 * In CSS content, prepend a path to relative URIs
	 *
	 * @param string $css
	 *
	 * @param string $path The path to prepend.
	 *
	 * @return string
	 */
	public static function prepend($css, $path)
	{
		self::$_prependPath = $path;

		$css = self::_trimUrls($css);

		// append
		$css = preg_replace_callback('/@import\\s+([\'"])(.*?)[\'"]/'
			,array(self::$className, '_processUriCB'), $css);
		$css = preg_replace_callback('/url\\(\\s*([^\\)\\s]+)\\s*\\)/'
			,array(self::$className, '_processUriCB'), $css);

		return $css;
	}

	/**
	 * Get a root relative URI from a file relative URI
	 *
	 * <code>
	 * Minify_CSS_UriRewriter::rewriteRelative(
	 *       '../img/hello.gif'
	 *     , '/home/user/www/css'  // path of CSS file
	 *     , '/home/user/www'      // doc root
	 * );
	 * // returns '/img/hello.gif'
	 *
	 * // example where static files are stored in a symlinked directory
	 * Minify_CSS_UriRewriter::rewriteRelative(
	 *       'hello.gif'
	 *     , '/var/staticFiles/theme'
	 *     , '/home/user/www'
	 *     , array('/home/user/www/static' => '/var/staticFiles')
	 * );
	 * // returns '/static/theme/hello.gif'
	 * </code>
	 *
	 * @param string $uri file relative URI
	 *
	 * @param string $realCurrentDir realpath of the current file's directory.
	 *
	 * @param string $realDocRoot realpath of the site document root.
	 *
	 * @param array $symlinks (default = array()) If the file is stored in
	 * a symlink-ed directory, provide an array of link paths to
	 * real target paths, where the link paths "appear" to be within the document
	 * root. E.g.:
	 * <code>
	 * array('/home/foo/www/not/real/path' => '/real/target/path') // unix
	 * array('C:\\htdocs\\not\\real' => 'D:\\real\\target\\path')  // Windows
	 * </code>
	 *
	 * @return string
	 */
	private static function rewriteRelative($uri, $realCurrentDir, $realDocRoot, $symlinks = array())
	{
		if ('/' === substr($uri, 0, 1)) { // root-relative
			return $uri;
		}

		// prepend path with current dir separator (OS-independent)
		$path = strtr($realCurrentDir, '/', DIRECTORY_SEPARATOR)
			. DIRECTORY_SEPARATOR . strtr($uri, '/', DIRECTORY_SEPARATOR);

		self::$debugText .= "file-relative URI  : {$uri}\n"
						  . "path prepended     : {$path}\n";

		// "unresolve" a symlink back to doc root
		foreach ($symlinks as $link => $target) {
			if (0 === strpos($path, $target)) {
				// replace $target with $link
				$path = $link . substr($path, strlen($target));

				self::$debugText .= "symlink unresolved : {$path}\n";

				break;
			}
		}
		// strip doc root
		$path = substr($path, strlen($realDocRoot));

		self::$debugText .= "docroot stripped   : {$path}\n";

		// fix to root-relative URI
		$uri = strtr($path, '/\\', '//');
		$uri = self::removeDots($uri);

		self::$debugText .= "traversals removed : {$uri}\n\n";

		return $uri;
	}

	/**
	 * Remove instances of "./" and "../" where possible from a root-relative URI
	 *
	 * @param string $uri
	 *
	 * @return string
	 */
	public static function removeDots($uri)
	{
		$uri = str_replace('/./', '/', $uri);
		// inspired by patch from Oleg Cherniy
		do {
			$uri = preg_replace('@/[^/]+/\\.\\./@', '/', $uri, 1, $changed);
		} while ($changed);
		return $uri;
	}

	/**
	 * Defines which class to call as part of callbacks, change this
	 * if you extend Minify_CSS_UriRewriter
	 *
	 * @var string
	 */
	protected static $className = '\W3TCL\Minify\Minify_CSS_UriRewriter';

	/**
	 * Get realpath with any trailing slash removed. If realpath() fails,
	 * just remove the trailing slash.
	 *
	 * @param string $path
	 *
	 * @return mixed path with no trailing slash
	 */
	protected static function _realpath($path)
	{
		$realPath = realpath($path);
		if ($realPath !== false) {
			$path = $realPath;
		}
		return rtrim($path, '/\\');
	}

	/**
	 * Directory of this stylesheet
	 *
	 * @var string
	 */
	private static $_currentDir = '';

	/**
	 * DOC_ROOT
	 *
	 * @var string
	 */
	private static $_docRoot = '';

	/**
	 * directory replacements to map symlink targets back to their
	 * source (within the document root) E.g. '/var/www/symlink' => '/var/realpath'
	 *
	 * @var array
	 */
	private static $_symlinks = array();

	/**
	 * Path to prepend
	 *
	 * @var string
	 */
	private static $_prependPath = null;

	/**
	 * @var string
	 */
	private static $_prependAbsolutePath = null;

	/**
	 * @var string
	 */
	private static $_prependAbsolutePathCallback = null;

	/**
	 * @var int
	 */
	private static $_browserCacheId = 0;

	/**
	 * @var array
	 */
	private static $_browserCacheExtensions = array();


	/**
	 * @param string $css
	 *
	 * @return string
	 */
	private static function _trimUrls($css)
	{
		return preg_replace('/
			url\\(      # url(
			\\s*
			([^\\)]+?)  # 1 = URI (assuming does not contain ")")
			\\s*
			\\)         # )
		/x', 'url($1)', $css);
	}

	/**
	 * @param array $m
	 *
	 * @return string
	 */
	private static function _processUriCB($m)
	{
		// $m matched either '/@import\\s+([\'"])(.*?)[\'"]/' or '/url\\(\\s*([^\\)\\s]+)\\s*\\)/'
		$isImport = ($m[0][0] === '@');
		// determine URI and the quote character (if any)
		if ($isImport) {
			$quoteChar = $m[1];
			$uri = $m[2];
		} else {
			// $m[1] is either quoted or not
			$quoteChar = ($m[1][0] === "'" || $m[1][0] === '"')
				? $m[1][0]
				: '';
			$uri = ($quoteChar === '')
				? $m[1]
				: substr($m[1], 1, strlen($m[1]) - 2);
		}
		// analyze URI
		if ( !empty($uri)
			&& '/' !== substr($uri, 0, 1) // Root-relative (/).
			&& false === strpos( $uri, '//' )  // Protocol/non-data (//).
			&& 'data:' !== substr( $uri, 0, 5 ) // Data protocol.
			&& '%' !== substr( $uri, 0, 1 ) // URL-encoded entity.
			&& '#' !== substr( $uri, 0, 1 ) // URL fragment.
		) {
			// URI is file-relative: rewrite depending on options
			if (self::$_prependPath === null) {
				$uri = self::rewriteRelative($uri, self::$_currentDir, self::$_docRoot, self::$_symlinks);
				if (self::$_prependAbsolutePath) {
					$prependAbsolutePath = self::$_prependAbsolutePath;
				} elseif (self::$_prependAbsolutePathCallback) {
					$prependAbsolutePath = call_user_func(self::$_prependAbsolutePathCallback, $uri);
				} else {
					$prependAbsolutePath = '';
				}

				if ($prependAbsolutePath) {
					$uri = rtrim($prependAbsolutePath, '/') . $uri;
				}
			} else {
				if (!\W3TC\Util_Environment::is_url(self::$_prependPath)) {
					$uri = self::$_prependPath . $uri;

					if (substr($uri, 0, 1) === '/') {
						$root = '';
						$rootRelative = $uri;
						$uri = $root . self::removeDots($rootRelative);
					} elseif (preg_match('@^((https?\:)?//([^/]+))/@', $uri, $m) && (false !== strpos($m[3], '.'))) {
						$root = $m[1];
						$rootRelative = substr($uri, strlen($root));
						$uri = $root . self::removeDots($rootRelative);
					}
				} else {
					$parse_url = @parse_url(self::$_prependPath);

					if ($parse_url && isset($parse_url['host'])) {
						$scheme = array_key_exists('scheme', $parse_url) ? $parse_url['scheme'] : '';
						$host = $parse_url['host'];
						$port = (isset($parse_url['port']) && $parse_url['port'] != 80 ? ':' . (int) $parse_url['port'] : '');
						$path = (!empty($parse_url['path']) ? $parse_url['path'] : '/');
						$dir_css = preg_replace('~[^/]+$~', '', $path);
						$dir_obj = preg_replace('~[^/]+$~', '', $uri);
						$dir = (ltrim((strpos($dir_obj, '/') === 0 ? \W3TC\Util_Environment::realpath($dir_obj) : \W3TC\Util_Environment::realpath($dir_css . $dir_obj)), '/'));
						$file = basename($uri);

						$scheme_dot = ( empty( $scheme ) ? '' : $scheme . ':' );
						$uri = sprintf('%s//%s%s/%s/%s', $scheme_dot, $host,
							$port, $dir, $file);
					}
				}
			}

			if (self::$_browserCacheId && count(self::$_browserCacheExtensions)) {
				$matches = null;

				if (preg_match('~\.([a-z-_]+)(\?.*)?$~', $uri, $matches)) {
					$extension = $matches[1];

					if ($extension && in_array($extension, self::$_browserCacheExtensions)) {
						$uri = \W3TC\Util_Environment::remove_query($uri);
						$uri .= ( strpos( $uri, '?' ) !== false ? '&' : '?' ) . self::$_browserCacheId;
					}
				}
			}
		}
		return $isImport
			? "@import {$quoteChar}{$uri}{$quoteChar}"
			: "url({$quoteChar}{$uri}{$quoteChar})";
	}
}