| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238 | #!/usr/bin/env node/** * html-minifier CLI tool * * The MIT License (MIT) * *  Copyright (c) 2014 Zoltan Frombach * *  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. * *  THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR *  IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS *  FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR *  COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER *  IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN *  CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. * */'use strict';var cli = require('cli');var changeCase = require('change-case');var path = require('path');var fs = require('fs');var appName = require('./package.json').name;var appVersion = require('./package.json').version;var minify = require('./dist/htmlminifier.min.js').minify;var minifyOptions = {};var input = null;var output = null;cli.width = 100;cli.option_width = 40;cli.setApp(appName, appVersion);var usage = appName + ' [OPTIONS] [FILE(s)]\n\n';usage += '  If no input file(s) specified then STDIN will be used for input.\n';usage += '  If more than one input file specified those will be concatenated and minified together.\n\n';usage += '  When you specify a config file with the --config-file option (see sample-cli-config-file.conf for format)\n';usage += '    you can still override some of its contents by providing individual command line options, too.\n\n';usage += '  When you want to provide an array of strings for --ignore-custom-comments or --process-scripts options\n';usage += '    on the command line you must escape those such as --ignore-custom-comments "[\\"string1\\",\\"string1\\"]"\n';cli.setUsage(usage);var mainOptions = {  removeComments: [[false, 'Strip HTML comments']],  removeCommentsFromCDATA: [[false, 'Strip HTML comments from scripts and styles']],  removeCDATASectionsFromCDATA: [[false, 'Remove CDATA sections from script and style elements']],  collapseWhitespace: [[false, 'Collapse white space that contributes to text nodes in a document tree.']],  conservativeCollapse: [[false, 'Always collapse to 1 space (never remove it entirely)']],  preserveLineBreaks: [[false, 'Always collapse to 1 line break (never remove it entirely) when whitespace between tags include a line break.']],  collapseBooleanAttributes: [[false, 'Omit attribute values from boolean attributes']],  removeAttributeQuotes: [[false, 'Remove quotes around attributes when possible.']],  removeRedundantAttributes: [[false, 'Remove attributes when value matches default.']],  useShortDoctype: [[false, 'Replaces the doctype with the short (HTML5) doctype']],  removeEmptyAttributes: [[false, 'Remove all attributes with whitespace-only values']],  removeOptionalTags: [[false, 'Remove unrequired tags']],  removeEmptyElements: [[false, 'Remove all elements with empty contents']],  lint: [[false, 'Toggle linting']],  keepClosingSlash: [[false, 'Keep the trailing slash on singleton elements']],  caseSensitive: [[false, 'Treat attributes in case sensitive manner (useful for SVG; e.g. viewBox)']],  minifyJS: [[false, 'Minify Javascript in script elements and on* attributes (uses UglifyJS)']],  minifyCSS: [[false, 'Minify CSS in style elements and style attributes (uses clean-css)']],  minifyURLs: [[false, 'Minify URLs in various attributes (uses relateurl)']],  ignoreCustomComments: [[false, 'Array of regex\'es that allow to ignore certain comments, when matched', 'string'], 'json'],  processScripts: [[false, 'Array of strings corresponding to types of script elements to process through minifier (e.g. "text/ng-template", "text/x-handlebars-template", etc.)', 'string'], 'json'],  maxLineLength: [[false, 'Max line length', 'number'], true]};var cliOptions = {  version: ['v', 'Version information'],  output: ['o', 'Specify output file (if not specified STDOUT will be used for output)', 'file'],  'config-file': ['c', 'Use config file', 'file']};var mainOptionKeys = Object.keys(mainOptions);mainOptionKeys.forEach(function(key) {  cliOptions[changeCase.paramCase(key)] = mainOptions[key][0];});cli.parse(cliOptions);cli.main(function(args, options) {  if (options.version) {    process.stderr.write(appName + ' v' + appVersion);    cli.exit(0);  }  if (options['config-file']) {    try {      var fileOptions = JSON.parse(fs.readFileSync(path.resolve(options['config-file']), 'utf8'));      if ((fileOptions !== null) && (typeof fileOptions === 'object')) {        minifyOptions = fileOptions;      }    }    catch (e) {      process.stderr.write('Error: Cannot read the specified config file');      cli.exit(1);    }  }  mainOptionKeys.forEach(function(key) {    var paramKey = changeCase.paramCase(key);    var value = options[paramKey];    if (options[paramKey] !== null) {      switch (mainOptions[key][1]) {        case 'json':          minifyOptions[key] = parseJSONOption(value);          break;        case true:          minifyOptions[key] = value;          break;        default:          minifyOptions[key] = true;      }    }  });  if (args.length) {    input = args;  }  if (options.output) {    output = options.output;  }  var original = '';  var status = 0;  if (input !== null) { // Minifying one or more files specified on the CMD line    input.forEach(function(afile) {      try {        original += fs.readFileSync(afile, 'utf8');      }      catch (e) {        status = 2;        process.stderr.write('Error: Cannot read file ' + afile);      }    });  }  else { // Minifying input coming from STDIN    var BUFSIZE = 4096;    var buf = new Buffer(BUFSIZE);    var bytesRead;    while (true) { // Loop as long as stdin input is available.      bytesRead = 0;      try {        bytesRead = fs.readSync(process.stdin.fd, buf, 0, BUFSIZE);      }      catch (e) {        if (e.code === 'EAGAIN') { // 'resource temporarily unavailable'          // Happens on OS X 10.8.3 (not Windows 7!), if there's no          // stdin input - typically when invoking a script without any          // input (for interactive stdin input).          // If you were to just continue, you'd create a tight loop.          process.stderr.write('ERROR: interactive stdin input not supported');          cli.exit(2);        }        else if (e.code === 'EOF') {          // Happens on Windows 7, but not OS X 10.8.3:          // simply signals the end of *piped* stdin input.          break;        }        throw e; // unexpected exception      }      if (bytesRead === 0) {        // No more stdin input available.        // OS X 10.8.3: regardless of input method, this is how the end        //   of input is signaled.        // Windows 7: this is how the end of input is signaled for        //   *interactive* stdin input.        break;      }      original += buf.toString('utf8', 0, bytesRead);    }  }  function parseJSONOption(value) {    if (value !== null) {      var jsonArray;      try {        jsonArray = JSON.parse(value);      }      catch (e) {}      if (jsonArray instanceof Array) {        return jsonArray;      }      else {        return [value];      }    }  }  // Run minify  var minified = null;  try {    minified = minify(original, minifyOptions);  }  catch (e) {    status = 3;    process.stderr.write('Error: Minification error');  }  if (minified !== null) {    // Write the output    try {      if (output !== null) {        fs.writeFileSync(path.resolve(output), minified);      }      else {        process.stdout.write(minified);      }    }    catch (e) {      status = 4;      process.stderr.write('Error: Cannot write to output');    }  }  cli.exit(status);});
 |