mirror of
https://github.com/denoland/deno.git
synced 2025-01-03 12:58:54 -05:00
186 lines
5.7 KiB
JavaScript
186 lines
5.7 KiB
JavaScript
// Copyright 2018-2020 the Deno authors. All rights reserved. MIT license.
|
|
|
|
((window) => {
|
|
const core = window.Deno.core;
|
|
const exit = window.__bootstrap.os.exit;
|
|
const version = window.__bootstrap.version.version;
|
|
const dispatchJson = window.__bootstrap.dispatchJson;
|
|
const close = window.__bootstrap.resources.close;
|
|
const inspectArgs = window.__bootstrap.console.inspectArgs;
|
|
|
|
function opStartRepl(historyFile) {
|
|
return dispatchJson.sendSync("op_repl_start", { historyFile });
|
|
}
|
|
|
|
function opReadline(rid, prompt) {
|
|
return dispatchJson.sendAsync("op_repl_readline", { rid, prompt });
|
|
}
|
|
|
|
function replLog(...args) {
|
|
core.print(inspectArgs(args) + "\n");
|
|
}
|
|
|
|
function replError(...args) {
|
|
core.print(inspectArgs(args) + "\n", true);
|
|
}
|
|
|
|
// Error messages that allow users to continue input
|
|
// instead of throwing an error to REPL
|
|
// ref: https://github.com/v8/v8/blob/master/src/message-template.h
|
|
// TODO(kevinkassimo): this list might not be comprehensive
|
|
const recoverableErrorMessages = [
|
|
"Unexpected end of input", // { or [ or (
|
|
"Missing initializer in const declaration", // const a
|
|
"Missing catch or finally after try", // try {}
|
|
"missing ) after argument list", // console.log(1
|
|
"Unterminated template literal", // `template
|
|
// TODO(kevinkassimo): need a parser to handling errors such as:
|
|
// "Missing } in template expression" // `${ or `${ a 123 }`
|
|
];
|
|
|
|
function isRecoverableError(e) {
|
|
return recoverableErrorMessages.includes(e.message);
|
|
}
|
|
|
|
// Returns `true` if `close()` is called in REPL.
|
|
// We should quit the REPL when this function returns `true`.
|
|
function isCloseCalled() {
|
|
return globalThis.closed;
|
|
}
|
|
|
|
let lastEvalResult = undefined;
|
|
let lastThrownError = undefined;
|
|
|
|
// Evaluate code.
|
|
// Returns true if code is consumed (no error/irrecoverable error).
|
|
// Returns false if error is recoverable
|
|
function evaluate(code) {
|
|
// each evalContext is a separate function body, and we want strict mode to
|
|
// work, so we should ensure that the code starts with "use strict"
|
|
const [result, errInfo] = core.evalContext(`"use strict";\n\n${code}`);
|
|
if (!errInfo) {
|
|
// when a function is eval'ed with just "use strict" sometimes the result
|
|
// is "use strict" which should be discarded
|
|
lastEvalResult = typeof result === "string" && result === "use strict"
|
|
? undefined
|
|
: result;
|
|
if (!isCloseCalled()) {
|
|
replLog(lastEvalResult);
|
|
}
|
|
} else if (errInfo.isCompileError && isRecoverableError(errInfo.thrown)) {
|
|
// Recoverable compiler error
|
|
return false; // don't consume code.
|
|
} else {
|
|
lastThrownError = errInfo.thrown;
|
|
if (errInfo.isNativeError) {
|
|
const formattedError = core.formatError(errInfo.thrown);
|
|
replError(formattedError);
|
|
} else {
|
|
replError("Thrown:", errInfo.thrown);
|
|
}
|
|
}
|
|
return true;
|
|
}
|
|
|
|
async function replLoop() {
|
|
const { console } = globalThis;
|
|
|
|
const historyFile = "deno_history.txt";
|
|
const rid = opStartRepl(historyFile);
|
|
|
|
const quitRepl = (exitCode) => {
|
|
// Special handling in case user calls deno.close(3).
|
|
try {
|
|
close(rid); // close signals Drop on REPL and saves history.
|
|
} catch {}
|
|
exit(exitCode);
|
|
};
|
|
|
|
// Configure globalThis._ to give the last evaluation result.
|
|
Object.defineProperty(globalThis, "_", {
|
|
configurable: true,
|
|
get: () => lastEvalResult,
|
|
set: (value) => {
|
|
Object.defineProperty(globalThis, "_", {
|
|
value: value,
|
|
writable: true,
|
|
enumerable: true,
|
|
configurable: true,
|
|
});
|
|
console.log("Last evaluation result is no longer saved to _.");
|
|
},
|
|
});
|
|
|
|
// Configure globalThis._error to give the last thrown error.
|
|
Object.defineProperty(globalThis, "_error", {
|
|
configurable: true,
|
|
get: () => lastThrownError,
|
|
set: (value) => {
|
|
Object.defineProperty(globalThis, "_error", {
|
|
value: value,
|
|
writable: true,
|
|
enumerable: true,
|
|
configurable: true,
|
|
});
|
|
console.log("Last thrown error is no longer saved to _error.");
|
|
},
|
|
});
|
|
|
|
replLog(`Deno ${version.deno}`);
|
|
replLog("exit using ctrl+d or close()");
|
|
|
|
while (true) {
|
|
if (isCloseCalled()) {
|
|
quitRepl(0);
|
|
}
|
|
|
|
let code = "";
|
|
// Top level read
|
|
try {
|
|
code = await opReadline(rid, "> ");
|
|
if (code.trim() === "") {
|
|
continue;
|
|
}
|
|
} catch (err) {
|
|
if (err.message === "EOF") {
|
|
quitRepl(0);
|
|
} else {
|
|
// If interrupted, don't print error.
|
|
if (err.message !== "Interrupted") {
|
|
// e.g. this happens when we have deno.close(3).
|
|
// We want to display the problem.
|
|
const formattedError = core.formatError(err);
|
|
replError(formattedError);
|
|
}
|
|
// Quit REPL anyways.
|
|
quitRepl(1);
|
|
}
|
|
}
|
|
// Start continued read
|
|
while (!evaluate(code)) {
|
|
code += "\n";
|
|
try {
|
|
code += await opReadline(rid, " ");
|
|
} catch (err) {
|
|
// If interrupted on continued read,
|
|
// abort this read instead of quitting.
|
|
if (err.message === "Interrupted") {
|
|
break;
|
|
} else if (err.message === "EOF") {
|
|
quitRepl(0);
|
|
} else {
|
|
// e.g. this happens when we have deno.close(3).
|
|
// We want to display the problem.
|
|
const formattedError = core.formatError(err);
|
|
replError(formattedError);
|
|
quitRepl(1);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
window.__bootstrap.repl = {
|
|
replLoop,
|
|
};
|
|
})(this);
|