mirror of
https://github.com/jashkenas/coffeescript.git
synced 2022-11-09 12:23:24 -05:00
172 lines
No EOL
5.9 KiB
JavaScript
172 lines
No EOL
5.9 KiB
JavaScript
(function(){
|
|
var BANNER, SWITCHES, WATCH_INTERVAL, coffee, compile, compile_scripts, lint, option_parser, options, optparse, parse_options, path, posix, sources, tokenize, usage, version, write_js;
|
|
posix = require('posix');
|
|
path = require('path');
|
|
coffee = require('coffee-script');
|
|
optparse = require('./../../vendor/optparse-js/src/optparse');
|
|
BANNER = "coffee compiles CoffeeScript source files into JavaScript.\n\nUsage:\n coffee path/to/script.coffee";
|
|
SWITCHES = [['-i', '--interactive', 'run an interactive CoffeeScript REPL'], ['-r', '--run', 'compile and run a CoffeeScript'], ['-o', '--output [DIR]', 'set the directory for compiled JavaScript'], ['-w', '--watch', 'watch scripts for changes, and recompile'], ['-p', '--print', 'print the compiled JavaScript to stdout'], ['-l', '--lint', 'pipe the compiled JavaScript through JSLint'], ['-e', '--eval', 'compile a cli scriptlet or read from stdin'], ['-t', '--tokens', 'print the tokens that the lexer produces'], ['--tree', 'print the parse tree that Jison produces'], ['-n', '--no-wrap', 'raw output, no function safety wrapper'], ['-g', '--globals', 'attach all top-level variables as globals'], ['-v', '--version', 'display CoffeeScript version'], ['-h', '--help', 'display this help message']];
|
|
WATCH_INTERVAL = 0.5;
|
|
options = {};
|
|
sources = [];
|
|
option_parser = null;
|
|
// The CommandLine handles all of the functionality of the `coffee` utility.
|
|
exports.run = function run() {
|
|
parse_options();
|
|
if (options.interactive) {
|
|
return require('./repl');
|
|
}
|
|
if (!(sources.length)) {
|
|
usage();
|
|
}
|
|
compile_scripts();
|
|
return this;
|
|
};
|
|
// The "--help" usage message.
|
|
usage = function usage() {
|
|
puts('\n' + option_parser.toString() + '\n');
|
|
return process.exit(0);
|
|
};
|
|
// The "--version" message.
|
|
version = function version() {
|
|
puts("CoffeeScript version " + coffee.VERSION);
|
|
return process.exit(0);
|
|
};
|
|
// Compile a single source file to JavaScript.
|
|
compile = function compile(script, source) {
|
|
source = source || 'error';
|
|
options = {};
|
|
if (options.no_wrap) {
|
|
options.no_wrap = true;
|
|
}
|
|
if (options.globals) {
|
|
options.globals = true;
|
|
}
|
|
try {
|
|
return CoffeeScript.compile(script, options);
|
|
} catch (error) {
|
|
process.stdio.writeError(source + ': ' + error.toString());
|
|
if (!(options.watch)) {
|
|
process.exit(1);
|
|
}
|
|
return null;
|
|
}
|
|
};
|
|
// Compiles the source CoffeeScript, returning the desired JavaScript, tokens,
|
|
// or JSLint results.
|
|
compile_scripts = function compile_scripts() {
|
|
var opts, source;
|
|
if (!((source = sources.shift()))) {
|
|
return null;
|
|
}
|
|
opts = options;
|
|
return posix.cat(source).addCallback(function(code) {
|
|
var js;
|
|
if (opts.tokens) {
|
|
puts(tokenize(code));
|
|
} else if (opts.tree) {
|
|
puts(coffee.tree(code).toString());
|
|
} else {
|
|
js = coffee.compile(code);
|
|
if (opts.run) {
|
|
eval(js);
|
|
} else if (opts.print) {
|
|
puts(js);
|
|
} else if (opts.lint) {
|
|
lint(js);
|
|
} else {
|
|
write_js(source, coffee.compile(code));
|
|
}
|
|
}
|
|
return compile_scripts();
|
|
});
|
|
};
|
|
// Write out a JavaScript source file with the compiled code.
|
|
write_js = function write_js(source, js) {
|
|
var dir, filename, js_path;
|
|
filename = path.basename(source, path.extname(source)) + '.js';
|
|
dir = options.output || path.dirname(source);
|
|
js_path = path.join(dir, filename);
|
|
return posix.open(js_path, process.O_CREAT | process.O_WRONLY | process.O_TRUNC, parseInt('0755', 8)).addCallback(function(fd) {
|
|
return posix.write(fd, js);
|
|
});
|
|
};
|
|
// Pretty-print the token stream.
|
|
tokenize = function tokenize(code) {
|
|
var strings;
|
|
strings = coffee.tokenize(code).map(function(token) {
|
|
return '[' + token[0] + ' ' + token[1].toString().replace(/\n/, '\\n') + ']';
|
|
});
|
|
return strings.join(' ');
|
|
};
|
|
// Pipe compiled JS through JSLint (requires a working 'jsl' command).
|
|
lint = function lint(js) {
|
|
var jsl;
|
|
jsl = process.createChildProcess('jsl', ['-nologo', '-stdin']);
|
|
jsl.addListener('output', function(result) {
|
|
if (result) {
|
|
return puts(result.replace(/\n/g, ''));
|
|
}
|
|
});
|
|
jsl.addListener('error', function(result) {
|
|
if (result) {
|
|
return puts(result);
|
|
}
|
|
});
|
|
jsl.write(js);
|
|
return jsl.close();
|
|
};
|
|
// Use OptionParser for all the options.
|
|
parse_options = function parse_options() {
|
|
var oparser, opts, paths;
|
|
opts = (options = {});
|
|
oparser = (option_parser = new optparse.OptionParser(SWITCHES));
|
|
oparser.banner = BANNER;
|
|
oparser.add = oparser['on'];
|
|
oparser.add('interactive', function() {
|
|
return opts.interactive = true;
|
|
});
|
|
oparser.add('run', function() {
|
|
return opts.run = true;
|
|
});
|
|
oparser.add('output', function(n, dir) {
|
|
return opts.output = dir;
|
|
});
|
|
oparser.add('watch', function() {
|
|
return opts.watch = true;
|
|
});
|
|
oparser.add('print', function() {
|
|
return opts.print = true;
|
|
});
|
|
oparser.add('lint', function() {
|
|
return opts.lint = true;
|
|
});
|
|
oparser.add('eval', function() {
|
|
return opts.eval = true;
|
|
});
|
|
oparser.add('tokens', function() {
|
|
return opts.tokens = true;
|
|
});
|
|
oparser.add('tree', function() {
|
|
return opts.tree = true;
|
|
});
|
|
oparser.add('help', (function(__this) {
|
|
var __func = function() {
|
|
return usage();
|
|
};
|
|
return (function() {
|
|
return __func.apply(__this, arguments);
|
|
});
|
|
})(this));
|
|
oparser.add('version', (function(__this) {
|
|
var __func = function() {
|
|
return version();
|
|
};
|
|
return (function() {
|
|
return __func.apply(__this, arguments);
|
|
});
|
|
})(this));
|
|
paths = oparser.parse(process.ARGV);
|
|
return sources = paths.slice(2, paths.length);
|
|
};
|
|
})(); |