UNPKG

html-minifier

Version:

HTML minifier with lint-like capabilities.

352 lines (317 loc) 13.2 kB
#!/usr/bin/env node /** * html-minifier CLI tool * * The MIT License (MIT) * * Copyright (c) 2014-2016 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 concat = require('concat-stream'); 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('.').minify; var HTMLLint = require('./src/htmllint').HTMLLint; var minifyOptions = {}; var output; cli.width = 100; cli.option_width = 40; cli.setApp(appName, appVersion); var usage = appName + ' [OPTIONS] [FILE(s)]\n\n'; usage += ' If no input files are specified then STDIN will be used for input.\n'; usage += ' If more than one input file is specified then 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 = { caseSensitive: [[false, 'Treat attributes in case sensitive manner (useful for SVG; e.g. viewBox)']], collapseBooleanAttributes: [[false, 'Omit attribute values from boolean attributes']], collapseInlineTagWhitespace: [[false, 'Collapse white space around inline tag']], 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)']], customAttrAssign: [[false, 'Arrays of regex\'es that allow to support custom attribute assign expressions (e.g. \'<div flex?="{{mode != cover}}"></div>\')', 'string'], 'json-regex'], customAttrCollapse: [[false, 'Regex that specifies custom attribute to strip newlines from (e.g. /ng\-class/)', 'string'], 'string-regex'], customAttrSurround: [[false, 'Arrays of regex\'es that allow to support custom attribute surround expressions (e.g. <input {{#if value}}checked="checked"{{/if}}>)', 'string'], 'json-regex'], customEventAttributes: [[false, 'Arrays of regex\'es that allow to support custom event attributes for minifyJS (e.g. ng-click)', 'string'], 'json-regex'], html5: [[false, 'Parse input according to HTML5 specifications']], ignoreCustomComments: [[false, 'Array of regex\'es that allow to ignore certain comments, when matched', 'string'], 'json-regex'], ignoreCustomFragments: [[false, 'Array of regex\'es that allow to ignore certain fragments, when matched (e.g. <?php ... ?>, {{ ... }})', 'string'], 'json-regex'], includeAutoGeneratedTags: [[false, 'Insert tags generated by HTML parser'], true], keepClosingSlash: [[false, 'Keep the trailing slash on singleton elements']], lint: [[false, 'Toggle linting']], maxLineLength: [[false, 'Max line length', 'number'], true], minifyCSS: [[false, 'Minify CSS in style elements and style attributes (uses clean-css)']], minifyJS: [[false, 'Minify Javascript in script elements and on* attributes (uses UglifyJS)']], minifyURLs: [[false, 'Minify URLs in various attributes (uses relateurl)', 'string'], 'site-url'], preserveLineBreaks: [[false, 'Always collapse to 1 line break (never remove it entirely) when whitespace between tags include a line break.']], preventAttributesEscaping: [[false, 'Prevents the escaping of the values of attributes.']], processConditionalComments: [[false, 'Process contents of conditional comments through minifier']], 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'], quoteCharacter: [[false, 'Type of quote to use for attribute values (\' or ")', 'string'], true], removeAttributeQuotes: [[false, 'Remove quotes around attributes when possible.']], removeComments: [[false, 'Strip HTML comments']], removeEmptyAttributes: [[false, 'Remove all attributes with whitespace-only values']], removeEmptyElements: [[false, 'Remove all elements with empty contents']], removeOptionalTags: [[false, 'Remove unrequired tags']], removeRedundantAttributes: [[false, 'Remove attributes when value matches default.']], removeScriptTypeAttributes: [[false, 'Remove type="text/javascript" from script tags. Other type attribute values are left intact.']], removeStyleLinkTypeAttributes: [[false, 'Remove type="text/css" from style and link tags. Other type attribute values are left intact.']], removeTagWhitespace: [[false, 'Remove space between attributes whenever possible.']], useShortDoctype: [[false, 'Replaces the doctype with the short (HTML5) doctype']] }; 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'], 'input-dir': [false, 'Specify an input directory', 'dir'], 'output-dir': [false, 'Specify an output directory', 'dir'] }; var mainOptionKeys = Object.keys(mainOptions); mainOptionKeys.forEach(function(key) { cliOptions[changeCase.paramCase(key)] = mainOptions[key][0]; }); cli.parse(cliOptions); cli.main(function(args, options) { function stringToRegExp(value) { // JSON does not support regexes, so, e.g., JSON.parse() will not create // a RegExp from the JSON value `[ "/matchString/" ]`, which is // technically just an array containing a string that begins and end with // a forward slash. To get a RegExp from a JSON string, it must be // constructed explicitly in JavaScript. // // The likelihood of actually wanting to match text that is enclosed in // forward slashes is probably quite rare, so if forward slashes were // included in an argument that requires a regex, the user most likely // thought they were part of the syntax for specifying a regex. // // In the unlikely case that forward slashes are indeed desired in the // search string, the user would need to enclose the expression in a // second set of slashes: // // --customAttrSrround "[\"//matchString//\"]" // if (value) { var stripSlashes = /^\/(.*)\/$/.exec(value); if (stripSlashes) { value = stripSlashes[1]; } return new RegExp(value); } } function parseJSONOption(value, regexArray) { if (value !== null) { var jsonArray; try { jsonArray = JSON.parse(value); if (regexArray) { jsonArray = jsonArray.map(stringToRegExp); } } catch (e) { cli.fatal('Could not parse JSON value \'' + value + '\''); } return Array.isArray(jsonArray) ? jsonArray : [value]; } } function runMinify(original, output) { var status = 0; var minified = null; try { minified = minify(original, minifyOptions); } catch (e) { status = 3; cli.error('Minification error'); process.stderr.write((e.stack || e).toString()); } if (minifyOptions.lint) { minifyOptions.lint.populate(); } if (minified !== null) { // Write the output if (output) { try { fs.writeFileSync(path.resolve(output), minified); } catch (e) { status = 4; cli.error('Cannot write to ' + output); } } else { process.stdout.write(minified); } } return status; } function createDirectory(path) { try { fs.mkdirSync(path); } catch (e) { if (e.code !== 'EEXIST') { cli.fatal('Can not create directory ' + path); return 3; } } return 0; } function processDirectory(inputDir, outputDir) { var fileList = fs.readdirSync(inputDir); var status = createDirectory(outputDir); for (var i = 0; status === 0 && i < fileList.length; i++) { var fileName = fileList[i]; var inputFilePath = inputDir + '/' + fileName; var outputFilePath = outputDir + '/' + fileName; var stat = fs.statSync(inputFilePath); if (stat.isDirectory()) { status = processDirectory(inputFilePath, outputFilePath); } else { var originalContent = fs.readFileSync(inputFilePath, 'utf8'); status = runMinify(originalContent, outputFilePath); } } return status; } if (options.version) { cli.output(appName + ' v' + appVersion); cli.exit(0); } if (options['config-file']) { var configPath = path.resolve(options['config-file']); try { var configData; try { configData = fs.readFileSync(configPath, { encoding: 'utf8' }); } catch (e) { cli.fatal('The specified config file doesn’t exist or is unreadable:\n' + configPath); } configData = JSON.parse(configData); mainOptionKeys.forEach(function(key) { var value = configData[key]; if (value !== undefined) { switch (mainOptions[key][1]) { case 'json-regex': minifyOptions[key] = value.map(stringToRegExp); break; case 'string-regex': minifyOptions[key] = stringToRegExp(value); break; default: minifyOptions[key] = value; } } }); } catch (je) { try { minifyOptions = require(configPath); } catch (ne) { cli.fatal('Cannot read the specified config file.\nAs JSON: ' + je.message + '\nAs module: ' + ne.message); } } } mainOptionKeys.forEach(function(key) { var paramKey = changeCase.paramCase(key); var value = options[paramKey]; if (value !== null) { switch (mainOptions[key][1]) { case 'json': minifyOptions[key] = parseJSONOption(value); break; case 'json-regex': minifyOptions[key] = parseJSONOption(value, true); break; case 'string-regex': minifyOptions[key] = stringToRegExp(value); break; case 'site-url': minifyOptions[key] = { site: value }; break; case true: minifyOptions[key] = value; break; default: minifyOptions[key] = true; } } }); if (minifyOptions.lint === true) { minifyOptions.lint = new HTMLLint(); } if (options['input-dir'] || options['output-dir']) { var inputDir = options['input-dir']; var outputDir = options['output-dir']; if (!inputDir) { cli.error('The option output-dir needs to be used with the option input-dir. If you are working with a single file, use -o.'); cli.exit(2); } try { fs.statSync(inputDir); } catch (e) { cli.error('The input directory does not exist'); cli.exit(2); } if (!outputDir) { cli.error('You need to specify where to write the output files with the option --output-dir'); cli.exit(2); } try { cli.exit(processDirectory(inputDir, outputDir)); } catch (e) { cli.error('Error while processing input files'); cli.error(e); cli.exit(3); } } if (options.output) { output = options.output; } if (args.length) { // Minifying one or more files specified on the CMD line var original = ''; args.forEach(function(afile) { try { original += fs.readFileSync(afile, 'utf8'); } catch (e) { cli.error('Cannot read file ' + afile); cli.exit(2); } }); cli.exit(runMinify(original, output)); } else { // Minifying input coming from STDIN process.stdin.pipe(concat({ encoding: 'string' }, function(content) { cli.exit(runMinify(content, output)); })); } });