mirror of
https://github.com/jashkenas/coffeescript.git
synced 2022-11-09 12:23:24 -05:00
fb2be8e1e3
* Eliminate wrapper around “bound” (arrow) functions; output `=>` for such functions * Remove irrelevant (and breaking) tests * Minor cleanup * When a function parameter is a splat (i.e., it uses the ES2015 rest parameter syntax) output that parameter as ES2015 * Rearrange function parameters when one of the parameters is a splat and isn’t the last parameter (very WIP) * Handle params like `@param`, adding assignment expressions for them when they appear; ensure splat parameter is last * Add parameter names (not a text like `'\nValue IdentifierLiteral: a'`) to the scope, so that parameters can’t be deleted; move body-related lines together; more explanation of what’s going on * For parameters with a default value, correctly add the parameter name to the function scope * Handle expansions in function parameters: when an expansion is found, set the parameters to only be the original parameters left of the expansion, then an `...args` parameter; and in the function body define variables for the parameters to the right of the expansion, including setting default values * Handle splat parameters the same way we handle expansions: if a splat parameter is found, it becomes the last parameter in the function definition, and all following parameters get declared in the function body. Fix the splat/rest parameter values after the post-splat parameters have been extracted from it. Clean up `Code.compileNode` so that we loop through the parameters only once, and we create all expressions using calls like `new IdentifierLiteral` rather than `@makeCode`. * Fix parameter name when a parameter is a splat attached to `this` (e.g. `@param...`) * Rather than assigning post-splat parameters based on index, use slice; passes test “Functions with splats being called with too few arguments” * Dial back our w00t indentation * Better parsing of parameter names (WIP) * Refactor processing of splat/expansion parameters * Fix assignment of default parameters for parameters that come after a splat * Better check for whether a param is attached to `this` * More understandable variable names * For parameters after a splat or expansion, assign them similar to the 1.x destructuring method of using `arguments`, except only concern ourselves with the post-splat parameters instead of all parameters; and use the splat/expansion parameter name, since `arguments` in ES fat arrow functions refers to the parent function’s `arguments` rather than the fat arrow function’s arguments/parameters * Don’t add unnamed parameters (like `[]` as a parameter) to the function scope * Disallow multiple splat/expansion parameters in function definitions; disallow lone expansion parameters * Fix `this` params not getting assigned if the parameter is after a splat parameter * Allow names of function parameters attached to `this` to be reserved words * Always add a statement to the function body defining a variable with its default value, if it has one, if the variable `== null`; this covers the case when ES doesn’t apply the default value when `null` is passed in as a value, but CoffeeScript expects `null` and `undefined` to act interchangeably * Aftermath of having both `undefined` and `null` trigger the use of default values for parameters with default values * More careful parsing of destructured parameters * Fall back to processing destructured parameters in the function body, to account for `this` or default values within destructured objects * Clean up comments * Restore new bare function test, minus the arrow function part of it * Test that bound/arrow functions aren’t overwriting the `arguments` object, which should refer to the parent scope’s `arguments` (like `this`) * Follow ES2015 spec for parameter default values: `null` gets assigned as as `null`, not the default value * Mimic ES default parameters behavior for parameters after a splat or expansion parameter * Bound functions cannot be generators: remove no-longer-relevant test, add check to throw error if `yield` appears inside a bound (arrow) function * Error for bound generator functions should underline the `yield`
199 lines
6.4 KiB
JavaScript
199 lines
6.4 KiB
JavaScript
// Generated by CoffeeScript 2.0.0-alpha
|
|
(function() {
|
|
var CoffeeScript, addHistory, addMultilineHandler, fs, getCommandId, merge, nodeREPL, path, ref, replDefaults, runInContext, updateSyntaxError, vm;
|
|
|
|
fs = require('fs');
|
|
|
|
path = require('path');
|
|
|
|
vm = require('vm');
|
|
|
|
nodeREPL = require('repl');
|
|
|
|
CoffeeScript = require('./coffee-script');
|
|
|
|
ref = require('./helpers'), merge = ref.merge, updateSyntaxError = ref.updateSyntaxError;
|
|
|
|
replDefaults = {
|
|
prompt: 'coffee> ',
|
|
historyFile: process.env.HOME ? path.join(process.env.HOME, '.coffee_history') : void 0,
|
|
historyMaxInputSize: 10240,
|
|
"eval": function(input, context, filename, cb) {
|
|
var Assign, Block, Literal, Value, ast, err, js, ref1, referencedVars, token, tokens;
|
|
input = input.replace(/\uFF00/g, '\n');
|
|
input = input.replace(/^\(([\s\S]*)\n\)$/m, '$1');
|
|
ref1 = require('./nodes'), Block = ref1.Block, Assign = ref1.Assign, Value = ref1.Value, Literal = ref1.Literal;
|
|
try {
|
|
tokens = CoffeeScript.tokens(input);
|
|
referencedVars = (function() {
|
|
var i, len, results;
|
|
results = [];
|
|
for (i = 0, len = tokens.length; i < len; i++) {
|
|
token = tokens[i];
|
|
if (token[0] === 'IDENTIFIER') {
|
|
results.push(token[1]);
|
|
}
|
|
}
|
|
return results;
|
|
})();
|
|
ast = CoffeeScript.nodes(tokens);
|
|
ast = new Block([new Assign(new Value(new Literal('_')), ast, '=')]);
|
|
js = ast.compile({
|
|
bare: true,
|
|
locals: Object.keys(context),
|
|
referencedVars: referencedVars
|
|
});
|
|
return cb(null, runInContext(js, context, filename));
|
|
} catch (error) {
|
|
err = error;
|
|
updateSyntaxError(err, input);
|
|
return cb(err);
|
|
}
|
|
}
|
|
};
|
|
|
|
runInContext = function(js, context, filename) {
|
|
if (context === global) {
|
|
return vm.runInThisContext(js, filename);
|
|
} else {
|
|
return vm.runInContext(js, context, filename);
|
|
}
|
|
};
|
|
|
|
addMultilineHandler = function(repl) {
|
|
var inputStream, multiline, nodeLineListener, origPrompt, outputStream, ref1, rli;
|
|
rli = repl.rli, inputStream = repl.inputStream, outputStream = repl.outputStream;
|
|
origPrompt = (ref1 = repl._prompt) != null ? ref1 : repl.prompt;
|
|
multiline = {
|
|
enabled: false,
|
|
initialPrompt: origPrompt.replace(/^[^> ]*/, function(x) {
|
|
return x.replace(/./g, '-');
|
|
}),
|
|
prompt: origPrompt.replace(/^[^> ]*>?/, function(x) {
|
|
return x.replace(/./g, '.');
|
|
}),
|
|
buffer: ''
|
|
};
|
|
nodeLineListener = rli.listeners('line')[0];
|
|
rli.removeListener('line', nodeLineListener);
|
|
rli.on('line', function(cmd) {
|
|
if (multiline.enabled) {
|
|
multiline.buffer += cmd + "\n";
|
|
rli.setPrompt(multiline.prompt);
|
|
rli.prompt(true);
|
|
} else {
|
|
rli.setPrompt(origPrompt);
|
|
nodeLineListener(cmd);
|
|
}
|
|
});
|
|
return inputStream.on('keypress', function(char, key) {
|
|
if (!(key && key.ctrl && !key.meta && !key.shift && key.name === 'v')) {
|
|
return;
|
|
}
|
|
if (multiline.enabled) {
|
|
if (!multiline.buffer.match(/\n/)) {
|
|
multiline.enabled = !multiline.enabled;
|
|
rli.setPrompt(origPrompt);
|
|
rli.prompt(true);
|
|
return;
|
|
}
|
|
if ((rli.line != null) && !rli.line.match(/^\s*$/)) {
|
|
return;
|
|
}
|
|
multiline.enabled = !multiline.enabled;
|
|
rli.line = '';
|
|
rli.cursor = 0;
|
|
rli.output.cursorTo(0);
|
|
rli.output.clearLine(1);
|
|
multiline.buffer = multiline.buffer.replace(/\n/g, '\uFF00');
|
|
rli.emit('line', multiline.buffer);
|
|
multiline.buffer = '';
|
|
} else {
|
|
multiline.enabled = !multiline.enabled;
|
|
rli.setPrompt(multiline.initialPrompt);
|
|
rli.prompt(true);
|
|
}
|
|
});
|
|
};
|
|
|
|
addHistory = function(repl, filename, maxSize) {
|
|
var buffer, fd, lastLine, readFd, size, stat;
|
|
lastLine = null;
|
|
try {
|
|
stat = fs.statSync(filename);
|
|
size = Math.min(maxSize, stat.size);
|
|
readFd = fs.openSync(filename, 'r');
|
|
buffer = Buffer.alloc(size);
|
|
fs.readSync(readFd, buffer, 0, size, stat.size - size);
|
|
fs.closeSync(readFd);
|
|
repl.rli.history = buffer.toString().split('\n').reverse();
|
|
if (stat.size > maxSize) {
|
|
repl.rli.history.pop();
|
|
}
|
|
if (repl.rli.history[0] === '') {
|
|
repl.rli.history.shift();
|
|
}
|
|
repl.rli.historyIndex = -1;
|
|
lastLine = repl.rli.history[0];
|
|
} catch (error) {}
|
|
fd = fs.openSync(filename, 'a');
|
|
repl.rli.addListener('line', function(code) {
|
|
if (code && code.length && code !== '.history' && code !== '.exit' && lastLine !== code) {
|
|
fs.writeSync(fd, code + "\n");
|
|
return lastLine = code;
|
|
}
|
|
});
|
|
repl.on('exit', function() {
|
|
return fs.closeSync(fd);
|
|
});
|
|
return repl.commands[getCommandId(repl, 'history')] = {
|
|
help: 'Show command history',
|
|
action: function() {
|
|
repl.outputStream.write((repl.rli.history.slice(0).reverse().join('\n')) + "\n");
|
|
return repl.displayPrompt();
|
|
}
|
|
};
|
|
};
|
|
|
|
getCommandId = function(repl, commandName) {
|
|
var commandsHaveLeadingDot;
|
|
commandsHaveLeadingDot = repl.commands['.help'] != null;
|
|
if (commandsHaveLeadingDot) {
|
|
return "." + commandName;
|
|
} else {
|
|
return commandName;
|
|
}
|
|
};
|
|
|
|
module.exports = {
|
|
start: function(opts = {}) {
|
|
var build, major, minor, ref1, repl;
|
|
ref1 = process.versions.node.split('.').map(function(n) {
|
|
return parseInt(n);
|
|
}), major = ref1[0], minor = ref1[1], build = ref1[2];
|
|
if (major < 6) {
|
|
console.warn("Node 6+ required for CoffeeScript REPL");
|
|
process.exit(1);
|
|
}
|
|
CoffeeScript.register();
|
|
process.argv = ['coffee'].concat(process.argv.slice(2));
|
|
opts = merge(replDefaults, opts);
|
|
repl = nodeREPL.start(opts);
|
|
if (opts.prelude) {
|
|
runInContext(opts.prelude, repl.context, 'prelude');
|
|
}
|
|
repl.on('exit', function() {
|
|
if (!repl.rli.closed) {
|
|
return repl.outputStream.write('\n');
|
|
}
|
|
});
|
|
addMultilineHandler(repl);
|
|
if (opts.historyFile) {
|
|
addHistory(repl, opts.historyFile, opts.historyMaxInputSize);
|
|
}
|
|
repl.commands[getCommandId(repl, 'load')].help = 'Load code from a file into this REPL session';
|
|
return repl;
|
|
}
|
|
};
|
|
|
|
}).call(this);
|