3 * Minify a file or set of files
5 * This program is free software; you can redistribute it and/or modify
6 * it under the terms of the GNU General Public License as published by
7 * the Free Software Foundation; either version 2 of the License, or
8 * (at your option) any later version.
10 * This program is distributed in the hope that it will be useful,
11 * but WITHOUT ANY WARRANTY; without even the implied warranty of
12 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
13 * GNU General Public License for more details.
15 * You should have received a copy of the GNU General Public License along
16 * with this program; if not, write to the Free Software Foundation, Inc.,
17 * 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
18 * http://www.gnu.org/copyleft/gpl.html
21 * @ingroup Maintenance
24 require_once( dirname( __FILE__
) . '/Maintenance.php' );
26 class MinifyScript
extends Maintenance
{
29 public function __construct() {
30 parent
::__construct();
31 $this->addOption( 'outfile',
32 'File for output. Only a single file may be specified for input.',
34 $this->addOption( 'outdir',
35 "Directory for output. If this is not specified, and neither is --outfile, then the\n" .
36 "output files will be sent to the same directories as the input files.",
38 $this->addOption( 'js-statements-on-own-line',
39 "Boolean value for putting statements on their own line when minifying JavaScript.",
41 $this->addOption( 'js-max-line-length',
42 "Maximum line length for JavaScript minification.",
44 $this->mDescription
= "Minify a file or set of files.\n\n" .
45 "If --outfile is not specified, then the output file names will have a .min extension\n" .
46 "added, e.g. jquery.js -> jquery.min.js.";
50 public function execute() {
51 if ( !count( $this->mArgs
) ) {
52 $this->error( "minify.php: At least one input file must be specified." );
56 if ( $this->hasOption( 'outfile' ) ) {
57 if ( count( $this->mArgs
) > 1 ) {
58 $this->error( '--outfile may only be used with a single input file.' );
63 $this->minify( $this->getArg( 0 ), $this->getOption( 'outfile' ) );
67 $outDir = $this->getOption( 'outdir', false );
69 foreach ( $this->mArgs
as $arg ) {
70 $inPath = realpath( $arg );
71 $inName = basename( $inPath );
72 $inDir = dirname( $inPath );
74 if ( strpos( $inName, '.min.' ) !== false ) {
75 $this->error( "Skipping $inName\n" );
79 if ( !file_exists( $inPath ) ) {
80 $this->error( "File does not exist: $arg", true );
83 $extension = $this->getExtension( $inName );
84 $outName = substr( $inName, 0, -strlen( $extension ) ) . 'min.' . $extension;
85 if ( $outDir === false ) {
86 $outPath = $inDir . '/' . $outName;
88 $outPath = $outDir . '/' . $outName;
91 $this->minify( $inPath, $outPath );
95 public function getExtension( $fileName ) {
96 $dotPos = strrpos( $fileName, '.' );
97 if ( $dotPos === false ) {
98 $this->error( "No file extension, cannot determine type: $fileName" );
101 return substr( $fileName, $dotPos +
1 );
104 public function minify( $inPath, $outPath ) {
105 global $wgResourceLoaderMinifierStatementsOnOwnLine, $wgResourceLoaderMinifierMaxLineLength;
107 $extension = $this->getExtension( $inPath );
108 $this->output( basename( $inPath ) . ' -> ' . basename( $outPath ) . '...' );
110 $inText = file_get_contents( $inPath );
111 if ( $inText === false ) {
112 $this->error( "Unable to open file $inPath for reading." );
115 $outFile = fopen( $outPath, 'w' );
117 $this->error( "Unable to open file $outPath for writing." );
121 switch ( $extension ) {
123 $outText = JavaScriptMinifier
::minify( $inText,
124 $this->getOption( 'js-statements-on-own-line', $wgResourceLoaderMinifierStatementsOnOwnLine ),
125 $this->getOption( 'js-max-line-length', $wgResourceLoaderMinifierMaxLineLength )
129 $outText = CSSMin
::minify( $inText );
132 $this->error( "No minifier defined for extension \"$extension\"" );
135 fwrite( $outFile, $outText );
137 $this->output( " ok\n" );
141 $maintClass = 'MinifyScript';
142 require_once( RUN_MAINTENANCE_IF_MAIN
);