mirror of
https://github.com/beautifier/js-beautify.git
synced 2024-12-02 17:56:29 +00:00
323 lines
9.0 KiB
JavaScript
Executable File
323 lines
9.0 KiB
JavaScript
Executable File
#!/usr/bin/env node
|
|
|
|
var debug = process.env.DEBUG_JSBEAUTIFY || process.env.JSBEAUTIFY_DEBUG
|
|
? function () { console.error.apply(console, arguments); }
|
|
: function () {};
|
|
|
|
var fs = require('fs'),
|
|
cc = require('config-chain'),
|
|
beautify = require('./beautify').js_beautify,
|
|
nopt = require('nopt'),
|
|
path = require('path'),
|
|
knownOpts = {
|
|
// Beautifier
|
|
"indent_size": Number,
|
|
"indent_char": String,
|
|
"indent_level": Number,
|
|
"indent_with_tabs": Boolean,
|
|
"preserve_newlines": Boolean,
|
|
"max_preserve_newlines": Number,
|
|
"jslint_happy": Boolean,
|
|
"brace_style": ["collapse", "expand", "end-expand", "expand-strict"],
|
|
"break_chained_methods": Boolean,
|
|
"keep_array_indentation": Boolean,
|
|
"unescape_strings": Boolean,
|
|
"wrap_line_length": Number,
|
|
// CLI
|
|
"version": Boolean,
|
|
"help": Boolean,
|
|
"files": [path, Array],
|
|
"outfile": path,
|
|
"replace": Boolean,
|
|
"quiet": Boolean,
|
|
"config": path
|
|
},
|
|
// dasherizeShorthands provides { "indent-size": ["--indent_size"] }
|
|
// translation, allowing more convenient dashes in CLI arguments
|
|
shortHands = dasherizeShorthands({
|
|
// Beautifier
|
|
"s": ["--indent_size"],
|
|
"c": ["--indent_char"],
|
|
"l": ["--indent_level"],
|
|
"t": ["--indent_with_tabs"],
|
|
"p": ["--preserve_newlines"],
|
|
"m": ["--max_preserve_newlines"],
|
|
"j": ["--jslint_happy"],
|
|
"b": ["--brace_style"],
|
|
"B": ["--break_chained_methods"],
|
|
"k": ["--keep_array_indentation"],
|
|
"x": ["--unescape_strings"],
|
|
"w": ["--wrap_line_length"],
|
|
// non-dasherized hybrid shortcuts
|
|
"good-stuff": [
|
|
"--keep_array_indentation",
|
|
"--keep_function_indentation",
|
|
"--jslint_happy"
|
|
],
|
|
// CLI
|
|
"v": ["--version"],
|
|
"h": ["--help"],
|
|
"f": ["--files"],
|
|
"o": ["--outfile"],
|
|
"r": ["--replace"],
|
|
"q": ["--quiet"]
|
|
// no shorthand for "config"
|
|
}),
|
|
parsed = nopt(knownOpts, shortHands);
|
|
|
|
if (parsed.version) {
|
|
console.log(require('./package.json').version);
|
|
process.exit(0);
|
|
}
|
|
else if (parsed.help) {
|
|
usage();
|
|
process.exit(0);
|
|
}
|
|
|
|
var cfg = cc(
|
|
parsed,
|
|
cleanOptions(cc.env('jsbeautify_'), knownOpts),
|
|
parsed.config,
|
|
cc.find('.jsbeautifyrc'),
|
|
__dirname + '/config/defaults.json'
|
|
).snapshot;
|
|
|
|
|
|
try {
|
|
// Verify arguments
|
|
checkFiles(cfg);
|
|
checkIndent(cfg);
|
|
debug(cfg);
|
|
|
|
// Process files synchronously to avoid EMFILE error
|
|
cfg.files.forEach(processInputSync, { cfg: cfg });
|
|
logToStdout('\nBeautified ' + cfg.files.length + ' files');
|
|
}
|
|
catch (ex) {
|
|
debug(cfg);
|
|
// usage(ex);
|
|
console.error(ex);
|
|
console.error('Run `js-beautify -h` for help.');
|
|
process.exit(1);
|
|
}
|
|
|
|
|
|
function usage(err) {
|
|
var msg = [
|
|
'js-beautify@' + require('./package.json').version,
|
|
'',
|
|
'CLI Options:',
|
|
' -f, --file Input file(s) (Pass \'-\' for stdin)',
|
|
' -r, --replace Write output in-place, replacing input',
|
|
' -o, --outfile Write output to file (default stdout)',
|
|
' --config Path to config file',
|
|
' -q, --quiet Suppress output to stdout',
|
|
' -h, --help Show this help',
|
|
' -v, --version Show the version',
|
|
'',
|
|
'Beautifier Options:',
|
|
' -s, --indent-size Indentation size [4]',
|
|
' -c, --indent-char Indentation character [" "]',
|
|
' -l, --indent-level Initial indentation level [0]',
|
|
' -t, --indent-with-tabs Indent with tabs, overrides -s and -c',
|
|
' -p, --preserve-newlines Preserve existing line-breaks (--no-preserve-newlines disables)',
|
|
' -m, --max-preserve-newlines Number of line-breaks to be preserved in one chunk [10]',
|
|
' -j, --jslint-happy Enable jslint-stricter mode',
|
|
' -b, --brace-style [collapse|expand|end-expand|expand-strict] ["collapse"]',
|
|
' -B, --break-chained-methods Break chained method calls across subsequent lines',
|
|
' -k, --keep-array-indentation Preserve array indentation',
|
|
' -x, --unescape-strings Decode printable characters encoded in xNN notation',
|
|
' -w, --wrap-line-length Wrap lines at next opportunity after N characters [0]',
|
|
' --good-stuff Warm the cockles of Crockford\'s heart',
|
|
''
|
|
];
|
|
|
|
if (err) {
|
|
msg.push(err);
|
|
msg.push('');
|
|
console.error(msg.join('\n'));
|
|
} else {
|
|
console.log(msg.join('\n'));
|
|
}
|
|
}
|
|
|
|
// main iterator, {cfg} passed as thisArg of forEach call
|
|
function processInputSync(filepath) {
|
|
var data = '',
|
|
config = this.cfg,
|
|
outfile = config.outfile,
|
|
input;
|
|
|
|
// -o passed with no value overwrites
|
|
if (outfile === true || config.replace) {
|
|
outfile = filepath;
|
|
}
|
|
|
|
if (filepath === '-') {
|
|
input = process.stdin;
|
|
input.resume();
|
|
input.setEncoding('utf8');
|
|
|
|
input.on('data', function (chunk) {
|
|
data += chunk;
|
|
});
|
|
|
|
input.on('end', function () {
|
|
makePretty(data, config, outfile, writePretty);
|
|
});
|
|
}
|
|
else {
|
|
data = fs.readFileSync(filepath, 'utf8');
|
|
makePretty(data, config, outfile, writePretty);
|
|
}
|
|
}
|
|
|
|
function makePretty(code, config, outfile, callback) {
|
|
try {
|
|
var pretty = beautify(code, config);
|
|
|
|
// ensure newline at end of beautified output
|
|
pretty += '\n';
|
|
|
|
callback(null, pretty, outfile);
|
|
}
|
|
catch (ex) {
|
|
callback(ex);
|
|
}
|
|
}
|
|
|
|
function writePretty(err, pretty, outfile) {
|
|
if (err) {
|
|
console.error(err);
|
|
process.exit(1);
|
|
}
|
|
|
|
if (outfile) {
|
|
try {
|
|
fs.writeFileSync(outfile, pretty, 'utf8');
|
|
logToStdout('beautified ' + path.relative(process.cwd(), outfile));
|
|
}
|
|
catch (ex) {
|
|
onOutputError(ex);
|
|
}
|
|
}
|
|
else {
|
|
process.stdout.write(pretty);
|
|
}
|
|
}
|
|
|
|
// workaround the fact that nopt.clean doesn't return the object passed in :P
|
|
function cleanOptions(data, types) {
|
|
nopt.clean(data, types);
|
|
return data;
|
|
}
|
|
|
|
// error handler for output stream that swallows errors silently,
|
|
// allowing the loop to continue over unwritable files.
|
|
function onOutputError(err) {
|
|
if (err.code === 'EACCES') {
|
|
console.error(err.path + " is not writable. Skipping!");
|
|
}
|
|
else {
|
|
console.error(err);
|
|
process.exit(0);
|
|
}
|
|
}
|
|
|
|
// turn "--foo_bar" into "foo-bar"
|
|
function dasherizeFlag(str) {
|
|
return str.replace(/^\-+/, '').replace(/_/g, '-');
|
|
}
|
|
|
|
// translate weird python underscored keys into dashed argv,
|
|
// avoiding single character aliases.
|
|
function dasherizeShorthands(hash) {
|
|
// operate in-place
|
|
Object.keys(hash).forEach(function (key) {
|
|
// each key value is an array
|
|
var val = hash[key][0];
|
|
// only dasherize one-character shorthands
|
|
if (key.length === 1 && val.indexOf('_') > -1) {
|
|
hash[dasherizeFlag(val)] = val;
|
|
}
|
|
});
|
|
|
|
return hash;
|
|
}
|
|
|
|
function checkIndent(parsed) {
|
|
if (parsed["indent_with_tabs"]) {
|
|
parsed["indent_size"] = 1;
|
|
parsed["indent_char"] = "\t";
|
|
}
|
|
|
|
return parsed;
|
|
}
|
|
|
|
function checkFiles(parsed) {
|
|
var argv = parsed.argv;
|
|
|
|
if (!parsed.files) {
|
|
parsed.files = [];
|
|
}
|
|
else {
|
|
if (argv.cooked.indexOf('-') > -1) {
|
|
// strip stdin path eagerly added by nopt in '-f -' case
|
|
parsed.files.some(removeDashedPath);
|
|
}
|
|
}
|
|
|
|
if (argv.remain.length) {
|
|
// assume any remaining args are files
|
|
argv.remain.forEach(function (f) {
|
|
parsed.files.push(path.resolve(f));
|
|
});
|
|
}
|
|
|
|
if ('string' === typeof parsed.outfile && !parsed.files.length) {
|
|
// use outfile as input when no other files passed in args
|
|
parsed.files.push(parsed.outfile);
|
|
// operation is now an implicit overwrite
|
|
parsed.replace = true;
|
|
}
|
|
|
|
if (argv.original.indexOf('-') > -1) {
|
|
// ensure '-' without '-f' still consumes stdin
|
|
parsed.files.push('-');
|
|
}
|
|
|
|
if (!parsed.files.length) {
|
|
throw 'Must define at least one file.';
|
|
}
|
|
debug('files.length ' + parsed.files.length);
|
|
|
|
parsed.files.forEach(testFilePath);
|
|
|
|
return parsed;
|
|
}
|
|
|
|
function removeDashedPath(filepath, i, arr) {
|
|
var found = filepath.lastIndexOf('-') === (filepath.length - 1);
|
|
if (found) {
|
|
arr.splice(i, 1);
|
|
}
|
|
return found;
|
|
}
|
|
|
|
function testFilePath(filepath) {
|
|
try {
|
|
if (filepath !== "-") {
|
|
fs.statSync(filepath);
|
|
}
|
|
}
|
|
catch (err) {
|
|
throw 'Unable to open path "' + filepath + '"';
|
|
}
|
|
}
|
|
|
|
function logToStdout(str) {
|
|
if (typeof parsed.quiet === "undefined" || !parsed.quiet) {
|
|
console.log(str);
|
|
}
|
|
}
|