#!/usr/bin/env node
"use strict";
Object.defineProperty(exports, "__esModule", { value: true });
var path_1 = require("path");
var repl_1 = require("repl");
var util_1 = require("util");
var arrify = require("arrify");
var Module = require("module");
var minimist = require("minimist");
var diff_1 = require("diff");
var vm_1 = require("vm");
var fs_1 = require("fs");
var index_1 = require("./index");
var argv = minimist(process.argv.slice(2), {
stopEarly: true,
string: ['eval', 'print', 'compiler', 'project', 'ignoreDiagnostics', 'require', 'cacheDirectory', 'ignore'],
boolean: ['help', 'transpileOnly', 'typeCheck', 'version', 'files', 'cache', 'pretty', 'skipProject', 'skipIgnore'],
alias: {
eval: ['e'],
print: ['p'],
require: ['r'],
help: ['h'],
version: ['v'],
typeCheck: ['type-check'],
transpileOnly: ['T', 'transpile-only'],
cacheDirectory: ['cache-directory'],
ignore: ['I'],
project: ['P'],
skipIgnore: ['skip-ignore'],
skipProject: ['skip-project'],
compiler: ['C'],
ignoreDiagnostics: ['D', 'ignore-diagnostics'],
compilerOptions: ['O', 'compiler-options']
},
default: {
cache: index_1.DEFAULTS.cache,
files: index_1.DEFAULTS.files
}
});
if (argv.help) {
console.log("\nUsage: ts-node [options] [ -e script | script.ts ] [arguments]\n\nOptions:\n\n -e, --eval [code] Evaluate code\n -p, --print [code] Evaluate code and print result\n -r, --require [path] Require a node module before execution\n\n -h, --help Print CLI usage\n -v, --version Print module version information\n\n -T, --transpile-only Use TypeScript's faster `transpileModule`\n --cache-directory Configure the output file cache directory\n -I, --ignore [pattern] Override the path patterns to skip compilation\n -P, --project [path] Path to TypeScript JSON project file\n -C, --compiler [name] Specify a custom TypeScript compiler\n -D, --ignoreDiagnostics [code] Ignore TypeScript warnings by diagnostic code\n -O, --compilerOptions [opts] JSON object to merge with compiler options\n\n --files Load files from `tsconfig.json` on startup\n --pretty Use pretty diagnostic formatter\n --no-cache Disable the local TypeScript Node cache\n --skip-project Skip reading `tsconfig.json`\n --skip-ignore Skip `--ignore` checks\n");
process.exit(0);
}
var cwd = process.cwd();
var code = argv.eval === undefined ? argv.print : argv.eval;
var isEval = typeof argv.eval === 'string' || !!argv.print; // Minimist struggles with empty strings.
var isPrinted = argv.print !== undefined;
// Register the TypeScript compiler instance.
var service = index_1.register({
files: argv.files || index_1.DEFAULTS.files,
pretty: argv.pretty || index_1.DEFAULTS.pretty,
typeCheck: argv.typeCheck || index_1.DEFAULTS.typeCheck,
transpileOnly: argv.transpileOnly || index_1.DEFAULTS.transpileOnly,
cache: argv.cache || index_1.DEFAULTS.cache,
cacheDirectory: argv.cacheDirectory || index_1.DEFAULTS.cacheDirectory,
ignore: argv.ignore || index_1.DEFAULTS.ignore,
project: argv.project || index_1.DEFAULTS.project,
skipIgnore: argv.skipIgnore || index_1.DEFAULTS.skipIgnore,
skipProject: argv.skipProject || index_1.DEFAULTS.skipProject,
compiler: argv.compiler || index_1.DEFAULTS.compiler,
ignoreDiagnostics: argv.ignoreDiagnostics || index_1.DEFAULTS.ignoreDiagnostics,
compilerOptions: index_1.parse(argv.compilerOptions) || index_1.DEFAULTS.compilerOptions,
readFile: isEval ? readFileEval : undefined,
fileExists: isEval ? fileExistsEval : undefined
});
// Output project information.
if (argv.version) {
console.log("ts-node v" + index_1.VERSION);
console.log("node " + process.version);
console.log("typescript v" + service.ts.version);
console.log("cache " + JSON.stringify(service.cachedir));
process.exit(0);
}
// Require specified modules before start-up.
Module._preloadModules(arrify(argv.require));
/**
* Eval helpers.
*/
var EVAL_FILENAME = "[eval].ts";
var EVAL_PATH = path_1.join(cwd, EVAL_FILENAME);
var EVAL_INSTANCE = { input: '', output: '', version: 0, lines: 0 };
// Execute the main contents (either eval, script or piped).
if (isEval) {
evalAndExit(code, isPrinted);
}
else {
if (argv._.length) {
process.argv = ['node'].concat(path_1.resolve(cwd, argv._[0])).concat(argv._.slice(1));
process.execArgv.unshift(__filename);
Module.runMain();
}
else {
// Piping of execution _only_ occurs when no other script is specified.
if (process.stdin.isTTY) {
startRepl();
}
else {
var code_1 = '';
process.stdin.on('data', function (chunk) { return code_1 += chunk; });
process.stdin.on('end', function () { return evalAndExit(code_1, isPrinted); });
}
}
}
/**
* Evaluate a script.
*/
function evalAndExit(code, isPrinted) {
var module = new Module(EVAL_FILENAME);
module.filename = EVAL_FILENAME;
module.paths = Module._nodeModulePaths(cwd);
global.__filename = EVAL_FILENAME;
global.__dirname = cwd;
global.exports = module.exports;
global.module = module;
global.require = module.require.bind(module);
var result;
try {
result = _eval(code);
}
catch (error) {
if (error instanceof index_1.TSError) {
console.error(error.diagnosticText);
process.exit(1);
}
throw error;
}
if (isPrinted) {
console.log(typeof result === 'string' ? result : util_1.inspect(result));
}
}
/**
* Evaluate the code snippet.
*/
function _eval(input) {
var lines = EVAL_INSTANCE.lines;
var isCompletion = !/\n$/.test(input);
var undo = appendEval(input);
var output;
try {
output = service.compile(EVAL_INSTANCE.input, EVAL_PATH, -lines);
}
catch (err) {
undo();
throw err;
}
// Use `diff` to check for new JavaScript to execute.
var changes = diff_1.diffLines(EVAL_INSTANCE.output, output);
if (isCompletion) {
undo();
}
else {
EVAL_INSTANCE.output = output;
}
return changes.reduce(function (result, change) {
return change.added ? exec(change.value, EVAL_FILENAME) : result;
}, undefined);
}
/**
* Execute some code.
*/
function exec(code, filename) {
var script = new vm_1.Script(code, { filename: filename });
return script.runInThisContext();
}
/**
* Start a CLI REPL.
*/
function startRepl() {
var repl = repl_1.start({
prompt: '> ',
input: process.stdin,
output: process.stdout,
terminal: process.stdout.isTTY,
eval: replEval,
useGlobal: true
});
// Bookmark the point where we should reset the REPL state.
var resetEval = appendEval('');
function reset() {
resetEval();
// Hard fix for TypeScript forcing `Object.defineProperty(exports, ...)`.
exec('exports = module.exports', EVAL_FILENAME);
}
reset();
repl.on('reset', reset);
repl.defineCommand('type', {
help: 'Check the type of a TypeScript identifier',
action: function (identifier) {
if (!identifier) {
repl.displayPrompt();
return;
}
var undo = appendEval(identifier);
var _a = service.getTypeInfo(EVAL_INSTANCE.input, EVAL_PATH, EVAL_INSTANCE.input.length), name = _a.name, comment = _a.comment;
undo();
repl.outputStream.write(name + "\n" + (comment ? comment + "\n" : ''));
repl.displayPrompt();
}
});
}
/**
* Eval code from the REPL.
*/
function replEval(code, _context, _filename, callback) {
var err;
var result;
// TODO: Figure out how to handle completion here.
if (code === '.scope') {
callback();
return;
}
try {
result = _eval(code);
}
catch (error) {
if (error instanceof index_1.TSError) {
// Support recoverable compilations using >= node 6.
if (repl_1.Recoverable && isRecoverable(error)) {
err = new repl_1.Recoverable(error);
}
else {
console.error(error.diagnosticText);
err = undefined;
}
}
else {
err = error;
}
}
callback(err, result);
}
/**
* Append to the eval instance and return an undo function.
*/
function appendEval(input) {
var undoInput = EVAL_INSTANCE.input;
var undoVersion = EVAL_INSTANCE.version;
var undoOutput = EVAL_INSTANCE.output;
var undoLines = EVAL_INSTANCE.lines;
// Handle ASI issues with TypeScript re-evaluation.
if (undoInput.charAt(undoInput.length - 1) === '\n' && /^\s*[\[\(\`]/.test(input) && !/;\s*$/.test(undoInput)) {
EVAL_INSTANCE.input = EVAL_INSTANCE.input.slice(0, -1) + ";\n";
}
EVAL_INSTANCE.input += input;
EVAL_INSTANCE.lines += lineCount(input);
EVAL_INSTANCE.version++;
return function () {
EVAL_INSTANCE.input = undoInput;
EVAL_INSTANCE.output = undoOutput;
EVAL_INSTANCE.version = undoVersion;
EVAL_INSTANCE.lines = undoLines;
};
}
/**
* Count the number of lines.
*/
function lineCount(value) {
var count = 0;
for (var _i = 0, value_1 = value; _i < value_1.length; _i++) {
var char = value_1[_i];
if (char === '\n') {
count++;
}
}
return count;
}
/**
* Get the file text, checking for eval first.
*/
function readFileEval(path) {
if (path === EVAL_PATH)
return EVAL_INSTANCE.input;
try {
return fs_1.readFileSync(path, 'utf8');
}
catch (err) { /* Ignore. */ }
}
/**
* Get whether the file exists.
*/
function fileExistsEval(path) {
if (path === EVAL_PATH)
return true;
try {
var stats = fs_1.statSync(path);
return stats.isFile() || stats.isFIFO();
}
catch (err) {
return false;
}
}
var RECOVERY_CODES = new Set([
1003,
1005,
1109,
1126,
1160,
1161 // "Unterminated regular expression literal."
]);
/**
* Check if a function can recover gracefully.
*/
function isRecoverable(error) {
return error.diagnosticCodes.every(function (code) { return RECOVERY_CODES.has(code); });
}
//# sourceMappingURL=bin.js.map