Merge "Disambiguate buildForm docs, add TODO"
[mediawiki.git] / maintenance / minify.php
blob270536b4d948d5489837b7fd91f0ebab0e0e329c
1 <?php
2 /**
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
20 * @file
21 * @ingroup Maintenance
24 require_once( __DIR__ . '/Maintenance.php' );
26 /**
27 * Maintenance script that minifies a file or set of files.
29 * @ingroup Maintenance
31 class MinifyScript extends Maintenance {
32 public $outDir;
34 public function __construct() {
35 parent::__construct();
36 $this->addOption( 'outfile',
37 'File for output. Only a single file may be specified for input.',
38 false, true );
39 $this->addOption( 'outdir',
40 "Directory for output. If this is not specified, and neither is --outfile, then the\n" .
41 "output files will be sent to the same directories as the input files.",
42 false, true );
43 $this->addOption( 'js-statements-on-own-line',
44 "Boolean value for putting statements on their own line when minifying JavaScript.",
45 false, true );
46 $this->addOption( 'js-max-line-length',
47 "Maximum line length for JavaScript minification.",
48 false, true );
49 $this->mDescription = "Minify a file or set of files.\n\n" .
50 "If --outfile is not specified, then the output file names will have a .min extension\n" .
51 "added, e.g. jquery.js -> jquery.min.js.";
55 public function execute() {
56 if ( !count( $this->mArgs ) ) {
57 $this->error( "minify.php: At least one input file must be specified." );
58 exit( 1 );
61 if ( $this->hasOption( 'outfile' ) ) {
62 if ( count( $this->mArgs ) > 1 ) {
63 $this->error( '--outfile may only be used with a single input file.' );
64 exit( 1 );
67 // Minify one file
68 $this->minify( $this->getArg( 0 ), $this->getOption( 'outfile' ) );
69 return;
72 $outDir = $this->getOption( 'outdir', false );
74 foreach ( $this->mArgs as $arg ) {
75 $inPath = realpath( $arg );
76 $inName = basename( $inPath );
77 $inDir = dirname( $inPath );
79 if ( strpos( $inName, '.min.' ) !== false ) {
80 $this->error( "Skipping $inName\n" );
81 continue;
84 if ( !file_exists( $inPath ) ) {
85 $this->error( "File does not exist: $arg", true );
88 $extension = $this->getExtension( $inName );
89 $outName = substr( $inName, 0, -strlen( $extension ) ) . 'min.' . $extension;
90 if ( $outDir === false ) {
91 $outPath = $inDir . '/' . $outName;
92 } else {
93 $outPath = $outDir . '/' . $outName;
96 $this->minify( $inPath, $outPath );
100 public function getExtension( $fileName ) {
101 $dotPos = strrpos( $fileName, '.' );
102 if ( $dotPos === false ) {
103 $this->error( "No file extension, cannot determine type: $fileName" );
104 exit( 1 );
106 return substr( $fileName, $dotPos + 1 );
109 public function minify( $inPath, $outPath ) {
110 global $wgResourceLoaderMinifierStatementsOnOwnLine, $wgResourceLoaderMinifierMaxLineLength;
112 $extension = $this->getExtension( $inPath );
113 $this->output( basename( $inPath ) . ' -> ' . basename( $outPath ) . '...' );
115 $inText = file_get_contents( $inPath );
116 if ( $inText === false ) {
117 $this->error( "Unable to open file $inPath for reading." );
118 exit( 1 );
120 $outFile = fopen( $outPath, 'w' );
121 if ( !$outFile ) {
122 $this->error( "Unable to open file $outPath for writing." );
123 exit( 1 );
126 switch ( $extension ) {
127 case 'js':
128 $outText = JavaScriptMinifier::minify( $inText,
129 $this->getOption( 'js-statements-on-own-line', $wgResourceLoaderMinifierStatementsOnOwnLine ),
130 $this->getOption( 'js-max-line-length', $wgResourceLoaderMinifierMaxLineLength )
132 break;
133 case 'css':
134 $outText = CSSMin::minify( $inText );
135 break;
136 default:
137 $this->error( "No minifier defined for extension \"$extension\"" );
140 fwrite( $outFile, $outText );
141 fclose( $outFile );
142 $this->output( " ok\n" );
146 $maintClass = 'MinifyScript';
147 require_once RUN_MAINTENANCE_IF_MAIN;