222 lines
6.9 KiB
PHP
222 lines
6.9 KiB
PHP
|
<?php
|
||
|
/**
|
||
|
* Class Minify_Controller_Base
|
||
|
* @package Minify
|
||
|
*/
|
||
|
|
||
|
/**
|
||
|
* Base class for Minify controller
|
||
|
*
|
||
|
* The controller class validates a request and uses it to create sources
|
||
|
* for minification and set options like contentType. It's also responsible
|
||
|
* for loading minifier code upon request.
|
||
|
*
|
||
|
* @package Minify
|
||
|
* @author Stephen Clay <steve@mrclay.org>
|
||
|
*/
|
||
|
abstract class Minify_Controller_Base {
|
||
|
|
||
|
/**
|
||
|
* Setup controller sources and set an needed options for Minify::source
|
||
|
*
|
||
|
* You must override this method in your subclass controller to set
|
||
|
* $this->sources. If the request is NOT valid, make sure $this->sources
|
||
|
* is left an empty array. Then strip any controller-specific options from
|
||
|
* $options and return it. To serve files, $this->sources must be an array of
|
||
|
* Minify_Source objects.
|
||
|
*
|
||
|
* @param array $options controller and Minify options
|
||
|
*
|
||
|
* @return array $options Minify::serve options
|
||
|
*/
|
||
|
abstract public function setupSources($options);
|
||
|
|
||
|
/**
|
||
|
* Get default Minify options for this controller.
|
||
|
*
|
||
|
* Override in subclass to change defaults
|
||
|
*
|
||
|
* @return array options for Minify
|
||
|
*/
|
||
|
public function getDefaultMinifyOptions() {
|
||
|
return array(
|
||
|
'isPublic' => true
|
||
|
,'encodeOutput' => function_exists('gzdeflate')
|
||
|
,'encodeMethod' => null // determine later
|
||
|
,'encodeLevel' => 9
|
||
|
,'minifierOptions' => array() // no minifier options
|
||
|
,'contentTypeCharset' => 'utf-8'
|
||
|
,'maxAge' => 1800 // 30 minutes
|
||
|
,'rewriteCssUris' => true
|
||
|
,'bubbleCssImports' => false
|
||
|
,'quiet' => false // serve() will send headers and output
|
||
|
,'debug' => false
|
||
|
,'concatOnly' => false
|
||
|
|
||
|
// if you override these, the response codes MUST be directly after
|
||
|
// the first space.
|
||
|
,'badRequestHeader' => 'HTTP/1.0 400 Bad Request'
|
||
|
,'errorHeader' => 'HTTP/1.0 500 Internal Server Error'
|
||
|
|
||
|
// callback function to see/modify content of all sources
|
||
|
,'postprocessor' => null
|
||
|
// file to require to load preprocessor
|
||
|
,'postprocessorRequire' => null
|
||
|
);
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Get default minifiers for this controller.
|
||
|
*
|
||
|
* Override in subclass to change defaults
|
||
|
*
|
||
|
* @return array minifier callbacks for common types
|
||
|
*/
|
||
|
public function getDefaultMinifers() {
|
||
|
$ret[Minify::TYPE_JS] = array('JSMin', 'minify');
|
||
|
$ret[Minify::TYPE_CSS] = array('Minify_CSS', 'minify');
|
||
|
$ret[Minify::TYPE_HTML] = array('Minify_HTML', 'minify');
|
||
|
return $ret;
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Is a user-given file within an allowable directory, existing,
|
||
|
* and having an extension js/css/html/txt ?
|
||
|
*
|
||
|
* This is a convenience function for controllers that have to accept
|
||
|
* user-given paths
|
||
|
*
|
||
|
* @param string $file full file path (already processed by realpath())
|
||
|
*
|
||
|
* @param array $safeDirs directories where files are safe to serve. Files can also
|
||
|
* be in subdirectories of these directories.
|
||
|
*
|
||
|
* @return bool file is safe
|
||
|
*
|
||
|
* @deprecated use checkAllowDirs, checkNotHidden instead
|
||
|
*/
|
||
|
public static function _fileIsSafe($file, $safeDirs)
|
||
|
{
|
||
|
$pathOk = false;
|
||
|
foreach ((array)$safeDirs as $safeDir) {
|
||
|
if (strpos($file, $safeDir) === 0) {
|
||
|
$pathOk = true;
|
||
|
break;
|
||
|
}
|
||
|
}
|
||
|
$base = basename($file);
|
||
|
if (! $pathOk || ! is_file($file) || $base[0] === '.') {
|
||
|
return false;
|
||
|
}
|
||
|
list($revExt) = explode('.', strrev($base));
|
||
|
return in_array(strrev($revExt), array('js', 'css', 'html', 'txt'));
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* @param string $file
|
||
|
* @param array $allowDirs
|
||
|
* @param string $uri
|
||
|
* @return bool
|
||
|
* @throws Exception
|
||
|
*/
|
||
|
public static function checkAllowDirs($file, $allowDirs, $uri)
|
||
|
{
|
||
|
foreach ((array)$allowDirs as $allowDir) {
|
||
|
if (strpos($file, $allowDir) === 0) {
|
||
|
return true;
|
||
|
}
|
||
|
}
|
||
|
throw new Exception("File '$file' is outside \$allowDirs. If the path is"
|
||
|
. " resolved via an alias/symlink, look into the \$min_symlinks option."
|
||
|
. " E.g. \$min_symlinks['/" . dirname($uri) . "'] = '" . dirname($file) . "';");
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* @param string $file
|
||
|
* @throws Exception
|
||
|
*/
|
||
|
public static function checkNotHidden($file)
|
||
|
{
|
||
|
$b = basename($file);
|
||
|
if (0 === strpos($b, '.')) {
|
||
|
throw new Exception("Filename '$b' starts with period (may be hidden)");
|
||
|
}
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* instances of Minify_Source, which provide content and any individual minification needs.
|
||
|
*
|
||
|
* @var Minify_Source[]
|
||
|
*/
|
||
|
public $sources = array();
|
||
|
|
||
|
/**
|
||
|
* Short name to place inside cache id
|
||
|
*
|
||
|
* The setupSources() method may choose to set this, making it easier to
|
||
|
* recognize a particular set of sources/settings in the cache folder. It
|
||
|
* will be filtered and truncated to make the final cache id <= 250 bytes.
|
||
|
*
|
||
|
* @var string
|
||
|
*/
|
||
|
public $selectionId = '';
|
||
|
|
||
|
/**
|
||
|
* Mix in default controller options with user-given options
|
||
|
*
|
||
|
* @param array $options user options
|
||
|
*
|
||
|
* @return array mixed options
|
||
|
*/
|
||
|
public final function mixInDefaultOptions($options)
|
||
|
{
|
||
|
$ret = array_merge(
|
||
|
$this->getDefaultMinifyOptions(), $options
|
||
|
);
|
||
|
if (! isset($options['minifiers'])) {
|
||
|
$options['minifiers'] = array();
|
||
|
}
|
||
|
$ret['minifiers'] = array_merge(
|
||
|
$this->getDefaultMinifers(), $options['minifiers']
|
||
|
);
|
||
|
return $ret;
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Analyze sources (if there are any) and set $options 'contentType'
|
||
|
* and 'lastModifiedTime' if they already aren't.
|
||
|
*
|
||
|
* @param array $options options for Minify
|
||
|
*
|
||
|
* @return array options for Minify
|
||
|
*/
|
||
|
public final function analyzeSources($options = array())
|
||
|
{
|
||
|
if ($this->sources) {
|
||
|
if (! isset($options['contentType'])) {
|
||
|
$options['contentType'] = Minify_Source::getContentType($this->sources);
|
||
|
}
|
||
|
// last modified is needed for caching, even if setExpires is set
|
||
|
if (! isset($options['lastModifiedTime'])) {
|
||
|
$max = 0;
|
||
|
foreach ($this->sources as $source) {
|
||
|
$max = max($source->lastModified, $max);
|
||
|
}
|
||
|
$options['lastModifiedTime'] = $max;
|
||
|
}
|
||
|
}
|
||
|
return $options;
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Send message to the Minify logger
|
||
|
*
|
||
|
* @param string $msg
|
||
|
*
|
||
|
* @return null
|
||
|
*/
|
||
|
public function log($msg) {
|
||
|
Minify_Logger::log($msg);
|
||
|
}
|
||
|
}
|