<?php /** * Combines multiple javascript files and serve up as gzip if possible. * Allowable scripts and script dependencies can be specified in a "packages.php" file * with the following format. This "packages.php" is optional, if absent the filenames * without ".js" extension are used. * * <code> * <?php * $packages = array( * 'package1' => array('file1.js', 'file2.js'), * 'package2' => array('file3.js', 'file4.js')); * * $dependencies = array( * 'package1' => array('package1'), * 'package2' => array('package1', 'package2')); //package2 requires package1 first. * </code> * * To serve up 'package1', specify the url, a maxium of 25 packages separated with commas is allows. * * clientscripts.php?js=package1 * * for 'package2' (automatically resolves 'package1') dependency * * clientscripts.php?js=package2 * * The scripts comments are removed and whitespaces removed appropriately. The * scripts may be served as zipped if the browser and php server allows it. Cache * headers are also sent to inform the browser and proxies to cache the file. * Moreover, the post-processed (comments removed and zipped) are saved in this * current directory for the next requests. * * If the url contains the parameter "mode=debug", the comments are not removed * and headers invalidating the cache are sent. In debug mode, the script can still * be zipped if the browser and server supports it. * * E.g. clientscripts.php?js=package2&mode=debug * * @link http://www.pradosoft.com/ * @copyright Copyright © 2007 PradoSoft * @license http://www.pradosoft.com/license/ * @author Wei Zhuo<weizhuo[at]gmail[dot]com> * @version $Id$ * @package System.Web.Javascripts * @since 3.1 */ @error_reporting(E_ERROR | E_WARNING | E_PARSE); function get_client_script_files() { $package_file = dirname(__FILE__).'/packages.php'; if(is_file($package_file)) return get_package_files($package_file, get_script_requests()); else return get_javascript_files(get_script_requests()); } /** * @param array list of requested libraries */ function get_script_requests($max=25) { $param = isset($_GET['js']) ? $_GET['js'] : ''; if(preg_match('/([a-zA-z0-9\-_])+(,[a-zA-z0-9\-_]+)*/', $param)) return array_unique(explode(',', $param, $max)); return array(); } /** * @param string packages.php filename * @param array packages requests */ function get_package_files($package_file, $request) { list($packages, $dependencies) = include($package_file); if(!(is_array($packages) && is_array($dependencies))) { error_log('Prado client script: invalid package file '.$package_file); return array(); } $result = array(); $found = array(); foreach($request as $library) { if(isset($dependencies[$library])) { foreach($dependencies[$library] as $dep) { if(isset($found[$dep])) continue; $result = array_merge($result, (array)$packages[$dep]); $found[$dep]=true; } } else error_log('Prado client script: no such javascript library "'.$library.'"'); } return $result; } /** * @param string requested javascript files * @array array list of javascript files. */ function get_javascript_files($request) { $result = array(); foreach($request as $file) $result[] = $file.'.js'; return $result; } /** * @param array list of javascript files. * @return string combined the available javascript files. */ function combine_javascript($files) { $content = ''; $base = dirname(__FILE__); foreach($files as $file) { $filename = $base.'/'.$file; if(is_file($filename)) //relies on get_client_script_files() for security $content .= "\x0D\x0A".file_get_contents($filename); //add CR+LF else error_log('Prado client script: missing file '.$filename); } return $content; } /** * @param string javascript code * @param array files names * @return string javascript code without comments. */ function minify_javascript($content, $files) { return jsm_minify($content); } /** * @param boolean true if in debug mode. */ function is_debug_mode() { return isset($_GET['mode']) && $_GET['mode']==='debug'; } /** * @param string javascript code * @param string gzip code */ function gzip_content($content) { return gzencode($content, 9, FORCE_GZIP); } /** * @param string javascript code. * @param string filename */ function save_javascript($content, $filename) { file_put_contents($filename, $content); if(supports_gzip_encoding()) file_put_contents($filename.'.gz', gzip_content($content)); } /** * @param string comprssed javascript file to be read * @param string javascript code, null if file is not found. */ function get_saved_javascript($filename) { $fn=$filename; if(supports_gzip_encoding()) $fn .= '.gz'; if(!is_file($fn)) save_javascript(get_javascript_code(true), $filename); return file_get_contents($fn); } /** * @return string compressed javascript file name. */ function compressed_js_filename() { $files = get_client_script_files(); if(count($files) > 0) { $filename = sprintf('%x',crc32(implode(',',($files)))); return dirname(__FILE__).'/clientscript_'.$filename.'.js'; } } /** * @param boolean true to strip comments from javascript code * @return string javascript code */ function get_javascript_code($minify=false) { $files = get_client_script_files(); if(count($files) > 0) { $content = combine_javascript($files); if($minify) return minify_javascript($content, $files); else return $content; } } /** * Prints headers to serve javascript */ function print_headers() { $expiresOffset = is_debug_mode() ? -10000 : 3600 * 24 * 10; //no cache header("Content-type: text/javascript"); header("Vary: Accept-Encoding"); // Handle proxies header("Expires: " . @gmdate("D, d M Y H:i:s", @time() + $expiresOffset) . " GMT"); if(($enc = supports_gzip_encoding()) !== null) header("Content-Encoding: " . $enc); } /** * @return string 'x-gzip' or 'gzip' if php supports gzip and browser supports gzip response, null otherwise. */ function supports_gzip_encoding() { if(isset($_GET['gzip']) && $_GET['gzip']==='false') return false; if (isset($_SERVER['HTTP_ACCEPT_ENCODING'])) { $encodings = explode(',', strtolower(preg_replace("/\s+/", "", $_SERVER['HTTP_ACCEPT_ENCODING']))); $allowsZipEncoding = in_array('gzip', $encodings) || in_array('x-gzip', $encodings) || isset($_SERVER['---------------']); $hasGzip = function_exists('ob_gzhandler'); $noZipBuffer = !ini_get('zlib.output_compression'); $noZipBufferHandler = ini_get('output_handler') != 'ob_gzhandler'; if ( $allowsZipEncoding && $hasGzip && $noZipBuffer && $noZipBufferHandler) $enc = in_array('x-gzip', $encodings) ? "x-gzip" : "gzip"; return $enc; } } /** * JSMin: the following jsm_* functions are adapted from JSMin_lib.php * written by Douglas Crockford. Below is the original copyright notice: * * PHP adaptation of JSMin, published by Douglas Crockford as jsmin.c, also based * on its Java translation by John Reilly. * * Permission is hereby granted to use the PHP version under the same conditions * as jsmin.c, which has the following notice : * * ---------------------------------------------------------------------------- * * Copyright (c) 2002 Douglas Crockford (www.crockford.com) * * Permission is hereby granted, free of charge, to any person obtaining a copy of * this software and associated documentation files (the "Software"), to deal in * the Software without restriction, including without limitation the rights to * use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies * of the Software, and to permit persons to whom the Software is furnished to do * so, subject to the following conditions: * * The above copyright notice and this permission notice shall be included in all * copies or substantial portions of the Software. */ $jsm_in=''; $jsm_out=''; $jsm_length=0; $jsm_pos=0; $jsm_c1=''; $jsm_c2=''; function jsm_minify($jsCode) { global $jsm_in, $jsm_out, $jsm_length, $jsm_pos, $jsm_c1, $jsm_c2; $jsm_in=$jsCode; $jsm_out=''; $jsm_length=strlen($jsCode); $jsm_pos=0; // Initialize A and run the first (minimal) action $jsm_c1="\n"; jsm_next(); // Proceed all the way to the end of the input file while($jsm_c1!==false) { if($jsm_c1===' ') { if(jsm_isAlphaNum($jsm_c2)) jsm_write($jsm_c1); else jsm_buffer(); } else if($jsm_c1==="\n") { if($jsm_c2===' ') jsm_next(); else if(jsm_isAlphaNum($jsm_c2) || $jsm_c2==='{' || $jsm_c2==='[' || $jsm_c2==='(' || $jsm_c2==='+' || $jsm_c2==='-') jsm_write($jsm_c1); else jsm_buffer(); } else { if($jsm_c2===' ') { if(jsm_isAlphaNum($jsm_c1)) jsm_write($jsm_c1); else jsm_next(); } else if($jsm_c2==="\n") { if(jsm_isAlphaNum($jsm_c1) || $jsm_c1==='}' || $jsm_c1===']' || $jsm_c1===')' || $jsm_c1==='+' || $jsm_c1==='-' || $jsm_c1==='"' || $jsm_c1==='\'') jsm_write($jsm_c1); else jsm_next(); } else jsm_write($jsm_c1); } } return $jsm_out; } function jsm_write() { global $jsm_c1; jsm_put($jsm_c1); jsm_buffer(); } function jsm_buffer() { global $jsm_c1, $jsm_c2; $tmpA=$jsm_c1=$jsm_c2; // Treating a string as a single char : outputting it whole // Note that the string-opening char (" or ') is memorized in B if($tmpA==='\'' || $tmpA==='"') { while (true) { // Output string contents jsm_put($tmpA); // Get next character, watching out for termination of the current string, // new line & co (then the string is not terminated !), or a backslash // (upon which the following char is directly output to serve the escape mechanism) $tmpA=$jsm_c1=jsm_get(); if($tmpA===$jsm_c2) // String terminated break; // from while(true) if(ord($tmpA) <= ord("\n")) die('Unterminated string literal'); if($tmpA==='\\') { // Escape next char immediately jsm_put($tmpA); $tmpA=$jsm_c1=jsm_get(); } } } jsm_next(); } function jsm_next() { global $jsm_c1, $jsm_c2; // Get the next B $jsm_c2=jsm_get2(); // Special case of recognising regular expressions (beginning with /) that are // preceded by '(', ',' or '=' $tmpA=$jsm_c1; if($jsm_c2==='/' && ($tmpA==='(' || $tmpA===',' || $tmpA==='=')) { // Output the two successive chars jsm_put($tmpA); jsm_put($jsm_c2); // Look for the end of the RE literal, watching out for escaped chars or a control / // end of line char (the RE literal then being unterminated !) while (true) { $tmpA=$jsm_c1=jsm_get(); if($tmpA==='/') break; // from while(true) if($tmpA==='\\') { // Escape next char immediately jsm_put($tmpA); $tmpA=$jsm_c1=jsm_get(); } else if(ord($tmpA) <= ord("\n")) die('Unterminated regexp literal'); // Output RE characters jsm_put($tmpA); } // Move forward after the RE literal $jsm_c2=jsm_get2(); } } function jsm_get() { global $jsm_pos, $jsm_length, $jsm_in; // Get next input character and advance position in file if($jsm_pos < $jsm_length) { $c=$jsm_in[$jsm_pos++]; if($c==="\n" || ord($c) >= ord(' ')) return $c; else if($c==="\r") return "\n"; else return ' '; } else return false; } function jsm_peek() { global $jsm_pos, $jsm_length, $jsm_in; if($jsm_pos < $jsm_length) return $jsm_in[$jsm_pos]; else return false; } function jsm_put($c) { global $jsm_out; $jsm_out .= $c; } /** * Get the next character from the input stream, excluding comments. */ function jsm_get2() { // Get next char from input, translated if necessary if(($c=jsm_get())!=='/') return $c; // Look ahead : a comment is two slashes or slashes followed by asterisk (to be closed) if(($c=jsm_peek())==='/') { // Comment is up to the end of the line while (true) { $c=jsm_get(); if(ord($c)<=ord("\n")) return $c; } } else if($c==='*') { // Comment is up to comment close. // Might not be terminated, if we hit the end of file. while(true) { // N.B. not using switch() because of having to test EOF with === $c=jsm_get(); if($c==='*') { // Comment termination if the char ahead is a slash if(jsm_peek()==='/') { // Advance again and make into a single space jsm_get(); return ' '; } } else if($c===false) die('Unterminated comment'); } } else return '/'; } /** * Indicates whether a character is alphanumeric or _, $, \ or non-ASCII. */ function jsm_isAlphaNum($c) { return ctype_alnum($c) || $c==='_' || $c==='$' || $c==='\\' || ord($c)>126; } /************** OUTPUT *****************/ if(count(get_script_requests()) > 0) { if(is_debug_mode()) { if(($code = get_javascript_code()) !== null) { print_headers(); echo supports_gzip_encoding() ? gzip_content($code) : $code; } } else { if(($filename = compressed_js_filename()) !== null) { print_headers(); echo get_saved_javascript($filename); } } } ?>