mirror of
https://github.com/mrclay/minify.git
synced 2025-02-22 16:04:08 +01:00
Encoder.php : added option to leave out compress Fixed unit tests for new Minify behavior and to show verbose when called directly on Windows.
267 lines
7.9 KiB
PHP
267 lines
7.9 KiB
PHP
<?php
|
|
/**
|
|
* Class HTTP_Encoder
|
|
* @package Minify
|
|
* @subpackage HTTP
|
|
*/
|
|
|
|
/**
|
|
* Encode and send gzipped/deflated content
|
|
*
|
|
* <code>
|
|
* // Send a CSS file, compressed if possible
|
|
* $he = new HTTP_Encoder(array(
|
|
* 'content' => file_get_contents($cssFile)
|
|
* ,'type' => 'text/css'
|
|
* ));
|
|
* $he->encode();
|
|
* $he->sendAll();
|
|
* </code>
|
|
*
|
|
* <code>
|
|
* // Just sniff for the accepted encoding
|
|
* $encoding = HTTP_Encoder::getAcceptedEncoding();
|
|
* </code>
|
|
*
|
|
* For more control over headers, use getHeaders() and getData() and send your
|
|
* own output.
|
|
*
|
|
* Note: If you don't need header mgmt, use PHP's native gzencode, gzdeflate,
|
|
* and gzcompress functions for gzip, deflate, and compress-encoding
|
|
* respectively.
|
|
*
|
|
* @package Minify
|
|
* @subpackage HTTP
|
|
* @author Stephen Clay <steve@mrclay.org>
|
|
*/
|
|
class HTTP_Encoder {
|
|
|
|
/**
|
|
* Default compression level for zlib operations
|
|
*
|
|
* This level is used if encode() is not given a $compressionLevel
|
|
*
|
|
* @var int
|
|
*/
|
|
public static $compressionLevel = 6;
|
|
|
|
/**
|
|
* Get an HTTP Encoder object
|
|
*
|
|
* @param array $spec options
|
|
*
|
|
* 'content': (string required) content to be encoded
|
|
*
|
|
* 'type': (string) if set, the Content-Type header will have this value.
|
|
*
|
|
* 'method: (string) only set this if you are forcing a particular encoding
|
|
* method. If not set, the best method will be chosen by getAcceptedEncoding()
|
|
* The available methods are 'gzip', 'deflate', 'compress', and '' (no
|
|
* encoding)
|
|
*
|
|
* @return null
|
|
*/
|
|
public function __construct($spec)
|
|
{
|
|
$this->_content = $spec['content'];
|
|
$this->_headers['Content-Length'] = (string)strlen($this->_content);
|
|
if (isset($spec['type'])) {
|
|
$this->_headers['Content-Type'] = $spec['type'];
|
|
}
|
|
if (isset($spec['method'])
|
|
&& in_array($spec['method'], array('gzip', 'deflate', 'compress', '')))
|
|
{
|
|
$this->_encodeMethod = array($spec['method'], $spec['method']);
|
|
} else {
|
|
$this->_encodeMethod = self::getAcceptedEncoding();
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Get content in current form
|
|
*
|
|
* Call after encode() for encoded content.
|
|
*
|
|
* return string
|
|
*/
|
|
public function getContent()
|
|
{
|
|
return $this->_content;
|
|
}
|
|
|
|
/**
|
|
* Get array of output headers to be sent
|
|
*
|
|
* E.g.
|
|
* <code>
|
|
* array(
|
|
* 'Content-Length' => '615'
|
|
* ,'Content-Encoding' => 'x-gzip'
|
|
* ,'Vary' => 'Accept-Encoding'
|
|
* )
|
|
* </code>
|
|
*
|
|
* @return array
|
|
*/
|
|
public function getHeaders()
|
|
{
|
|
return $this->_headers;
|
|
}
|
|
|
|
/**
|
|
* Send output headers
|
|
*
|
|
* You must call this before headers are sent and it probably cannot be
|
|
* used in conjunction with zlib output buffering / mod_gzip. Errors are
|
|
* not handled purposefully.
|
|
*
|
|
* @see getHeaders()
|
|
*
|
|
* @return null
|
|
*/
|
|
public function sendHeaders()
|
|
{
|
|
foreach ($this->_headers as $name => $val) {
|
|
header($name . ': ' . $val);
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Send output headers and content
|
|
*
|
|
* A shortcut for sendHeaders() and echo getContent()
|
|
*
|
|
* You must call this before headers are sent and it probably cannot be
|
|
* used in conjunction with zlib output buffering / mod_gzip. Errors are
|
|
* not handled purposefully.
|
|
*
|
|
* @return null
|
|
*/
|
|
public function sendAll()
|
|
{
|
|
$this->sendHeaders();
|
|
echo $this->_content;
|
|
}
|
|
|
|
/**
|
|
* Determine the client's best encoding method from the HTTP Accept-Encoding
|
|
* header.
|
|
*
|
|
* If no Accept-Encoding header is set, or the browser is IE before v6 SP2,
|
|
* this will return ('', ''), the "identity" encoding.
|
|
*
|
|
* A syntax-aware scan is done of the Accept-Encoding, so the method must
|
|
* be non 0. The methods are favored in order of deflate, gzip, then
|
|
* compress. Yes, deflate is always smaller and faster!
|
|
*
|
|
* @param bool $allowCompress allow the older compress encoding
|
|
*
|
|
* @return array two values, 1st is the actual encoding method, 2nd is the
|
|
* alias of that method to use in the Content-Encoding header (some browsers
|
|
* call gzip "x-gzip" etc.)
|
|
*/
|
|
public static function getAcceptedEncoding($allowCompress = true)
|
|
{
|
|
// @link http://www.w3.org/Protocols/rfc2616/rfc2616-sec14.html
|
|
|
|
if (! isset($_SERVER['HTTP_ACCEPT_ENCODING'])
|
|
|| self::_isBuggyIe())
|
|
{
|
|
return array('', '');
|
|
}
|
|
$ae = $_SERVER['HTTP_ACCEPT_ENCODING'];
|
|
$aeRev = strrev($ae);
|
|
// Fast tests for common AEs. If these don't pass we have to do
|
|
// slow regex parsing
|
|
if (0 === strpos($aeRev, 'etalfed ,') // ie, webkit
|
|
|| 0 === strpos($aeRev, 'etalfed,') // gecko
|
|
|| 0 === strpos($ae, 'deflate,') // opera 9.5b
|
|
// slow parsing
|
|
|| preg_match(
|
|
'@(?:^|,)\\s*deflate\\s*(?:$|,|;\\s*q=(?:0\\.|1))@', $ae)) {
|
|
return array('deflate', 'deflate');
|
|
}
|
|
if (preg_match(
|
|
'@(?:^|,)\\s*((?:x-)?gzip)\\s*(?:$|,|;\\s*q=(?:0\\.|1))@'
|
|
,$ae
|
|
,$m)) {
|
|
return array('gzip', $m[1]);
|
|
}
|
|
if ($allowCompress && preg_match(
|
|
'@(?:^|,)\\s*((?:x-)?compress)\\s*(?:$|,|;\\s*q=(?:0\\.|1))@'
|
|
,$ae
|
|
,$m)) {
|
|
return array('compress', $m[1]);
|
|
}
|
|
return array('', '');
|
|
}
|
|
|
|
/**
|
|
* Encode (compress) the content
|
|
*
|
|
* If the encode method is '' (none) or compression level is 0, or the 'zlib'
|
|
* extension isn't loaded, we return false.
|
|
*
|
|
* Then the appropriate gz_* function is called to compress the content. If
|
|
* this fails, false is returned.
|
|
*
|
|
* If successful, the Content-Length header is updated, and Content-Encoding
|
|
* and Vary headers are added.
|
|
*
|
|
* @param int $compressionLevel given to zlib functions. If not given, the
|
|
* class default will be used.
|
|
*
|
|
* @return bool success true if the content was actually compressed
|
|
*/
|
|
public function encode($compressionLevel = null)
|
|
{
|
|
if (null === $compressionLevel) {
|
|
$compressionLevel = self::$compressionLevel;
|
|
}
|
|
if ('' === $this->_encodeMethod[0]
|
|
|| ($compressionLevel == 0)
|
|
|| !extension_loaded('zlib'))
|
|
{
|
|
return false;
|
|
}
|
|
if ($this->_encodeMethod[0] === 'deflate') {
|
|
$encoded = gzdeflate($this->_content, $compressionLevel);
|
|
} elseif ($this->_encodeMethod[0] === 'gzip') {
|
|
$encoded = gzencode($this->_content, $compressionLevel);
|
|
} else {
|
|
$encoded = gzcompress($this->_content, $compressionLevel);
|
|
}
|
|
if (false === $encoded) {
|
|
return false;
|
|
}
|
|
$this->_headers['Content-Length'] = strlen($encoded);
|
|
$this->_headers['Content-Encoding'] = $this->_encodeMethod[1];
|
|
$this->_headers['Vary'] = 'Accept-Encoding';
|
|
$this->_content = $encoded;
|
|
return true;
|
|
}
|
|
|
|
protected $_content = '';
|
|
protected $_headers = array();
|
|
protected $_encodeMethod = array('', '');
|
|
|
|
/**
|
|
* Is the browser an IE version earlier than 6 SP2?
|
|
*/
|
|
protected static function _isBuggyIe()
|
|
{
|
|
$ua = $_SERVER['HTTP_USER_AGENT'];
|
|
// quick escape for non-IEs
|
|
if (0 !== strpos($ua, 'Mozilla/4.0 (compatible; MSIE ')
|
|
|| false !== strpos($ua, 'Opera')) {
|
|
return false;
|
|
}
|
|
// no regex = faaast
|
|
$version = (float)substr($ua, 30);
|
|
return (
|
|
$version < 6
|
|
|| ($version == 6 && false === strpos($ua, 'SV1'))
|
|
);
|
|
}
|
|
}
|