2019-01-22 04:03:30 +09:00
|
|
|
// Copyright 2018-2019 the Deno authors. All rights reserved. MIT license.
|
2019-08-24 13:20:48 -07:00
|
|
|
import { assert } from "./util";
|
2018-11-05 09:55:59 -08:00
|
|
|
import { close } from "./files";
|
2019-08-24 13:20:48 -07:00
|
|
|
import { sendSync, sendAsync, msg, flatbuffers } from "./dispatch_flatbuffers";
|
2018-11-05 09:55:59 -08:00
|
|
|
import { exit } from "./os";
|
2019-02-20 11:42:19 +09:00
|
|
|
import { window } from "./window";
|
2019-03-26 23:22:07 +11:00
|
|
|
import { core } from "./core";
|
2019-02-09 13:55:40 -08:00
|
|
|
import { formatError } from "./format_error";
|
2019-08-17 09:51:51 -07:00
|
|
|
import { stringifyArgs } from "./console";
|
|
|
|
|
|
|
|
/**
|
|
|
|
* REPL logging.
|
|
|
|
* In favor of console.log to avoid unwanted indentation
|
|
|
|
*/
|
|
|
|
function replLog(...args: unknown[]): void {
|
|
|
|
core.print(stringifyArgs(args) + "\n");
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* REPL logging for errors.
|
|
|
|
* In favor of console.error to avoid unwanted indentation
|
|
|
|
*/
|
|
|
|
function replError(...args: unknown[]): void {
|
|
|
|
core.print(stringifyArgs(args) + "\n", true);
|
|
|
|
}
|
2018-11-09 11:09:18 +11:00
|
|
|
|
2019-01-29 22:41:12 +03:00
|
|
|
const helpMsg = [
|
|
|
|
"exit Exit the REPL",
|
|
|
|
"help Print this help message"
|
|
|
|
].join("\n");
|
|
|
|
|
|
|
|
const replCommands = {
|
|
|
|
exit: {
|
2019-04-21 16:40:10 -04:00
|
|
|
get(): void {
|
2019-01-29 22:41:12 +03:00
|
|
|
exit(0);
|
|
|
|
}
|
|
|
|
},
|
|
|
|
help: {
|
2019-04-21 16:40:10 -04:00
|
|
|
get(): string {
|
2019-01-29 22:41:12 +03:00
|
|
|
return helpMsg;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2018-11-05 09:55:59 -08:00
|
|
|
function startRepl(historyFile: string): number {
|
2019-08-24 13:20:48 -07:00
|
|
|
const builder = flatbuffers.createBuilder();
|
|
|
|
const historyFile_ = builder.createString(historyFile);
|
|
|
|
const inner = msg.ReplStart.createReplStart(builder, historyFile_);
|
|
|
|
|
|
|
|
const baseRes = sendSync(builder, msg.Any.ReplStart, inner);
|
|
|
|
assert(baseRes != null);
|
|
|
|
assert(msg.Any.ReplStartRes === baseRes!.innerType());
|
|
|
|
const innerRes = new msg.ReplStartRes();
|
|
|
|
assert(baseRes!.inner(innerRes) != null);
|
|
|
|
const rid = innerRes.rid();
|
|
|
|
return rid;
|
2018-11-05 09:55:59 -08:00
|
|
|
}
|
|
|
|
|
|
|
|
// @internal
|
2018-11-28 01:07:22 -08:00
|
|
|
export async function readline(rid: number, prompt: string): Promise<string> {
|
2019-08-24 13:20:48 -07:00
|
|
|
const builder = flatbuffers.createBuilder();
|
|
|
|
const prompt_ = builder.createString(prompt);
|
|
|
|
const inner = msg.ReplReadline.createReplReadline(builder, rid, prompt_);
|
|
|
|
|
|
|
|
const baseRes = await sendAsync(builder, msg.Any.ReplReadline, inner);
|
|
|
|
|
|
|
|
assert(baseRes != null);
|
|
|
|
assert(msg.Any.ReplReadlineRes === baseRes!.innerType());
|
|
|
|
const innerRes = new msg.ReplReadlineRes();
|
|
|
|
assert(baseRes!.inner(innerRes) != null);
|
|
|
|
const line = innerRes.line();
|
|
|
|
assert(line !== null);
|
|
|
|
return line || "";
|
2018-11-05 09:55:59 -08:00
|
|
|
}
|
|
|
|
|
2019-03-10 04:30:38 +11:00
|
|
|
// 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: Error): boolean {
|
|
|
|
return recoverableErrorMessages.includes(e.message);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Evaluate code.
|
|
|
|
// Returns true if code is consumed (no error/irrecoverable error).
|
|
|
|
// Returns false if error is recoverable
|
|
|
|
function evaluate(code: string): boolean {
|
2019-03-26 23:22:07 +11:00
|
|
|
const [result, errInfo] = core.evalContext(code);
|
2019-03-10 04:30:38 +11:00
|
|
|
if (!errInfo) {
|
2019-08-17 09:51:51 -07:00
|
|
|
replLog(result);
|
2019-03-10 04:30:38 +11:00
|
|
|
} else if (errInfo.isCompileError && isRecoverableError(errInfo.thrown)) {
|
|
|
|
// Recoverable compiler error
|
|
|
|
return false; // don't consume code.
|
|
|
|
} else {
|
|
|
|
if (errInfo.isNativeError) {
|
|
|
|
const formattedError = formatError(
|
2019-03-26 23:22:07 +11:00
|
|
|
core.errorToJSON(errInfo.thrown as Error)
|
2019-03-10 04:30:38 +11:00
|
|
|
);
|
2019-08-17 09:51:51 -07:00
|
|
|
replError(formattedError);
|
2019-03-10 04:30:38 +11:00
|
|
|
} else {
|
2019-08-17 09:51:51 -07:00
|
|
|
replError("Thrown:", errInfo.thrown);
|
2019-03-10 04:30:38 +11:00
|
|
|
}
|
|
|
|
}
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2018-11-05 09:55:59 -08:00
|
|
|
// @internal
|
2018-11-28 01:07:22 -08:00
|
|
|
export async function replLoop(): Promise<void> {
|
2019-01-29 22:41:12 +03:00
|
|
|
Object.defineProperties(window, replCommands);
|
2018-11-05 09:55:59 -08:00
|
|
|
|
|
|
|
const historyFile = "deno_history.txt";
|
|
|
|
const rid = startRepl(historyFile);
|
|
|
|
|
2019-03-10 04:30:38 +11:00
|
|
|
const quitRepl = (exitCode: number): void => {
|
2019-02-11 11:01:28 -08:00
|
|
|
// Special handling in case user calls deno.close(3).
|
|
|
|
try {
|
|
|
|
close(rid); // close signals Drop on REPL and saves history.
|
|
|
|
} catch {}
|
|
|
|
exit(exitCode);
|
|
|
|
};
|
|
|
|
|
2018-11-05 09:55:59 -08:00
|
|
|
while (true) {
|
2019-02-11 11:01:28 -08:00
|
|
|
let code = "";
|
|
|
|
// Top level read
|
2018-11-05 09:55:59 -08:00
|
|
|
try {
|
2019-02-11 11:01:28 -08:00
|
|
|
code = await readline(rid, "> ");
|
|
|
|
if (code.trim() === "") {
|
|
|
|
continue;
|
|
|
|
}
|
2018-11-05 09:55:59 -08:00
|
|
|
} catch (err) {
|
|
|
|
if (err.message === "EOF") {
|
2019-02-11 11:01:28 -08:00
|
|
|
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.
|
2019-03-26 23:22:07 +11:00
|
|
|
const formattedError = formatError(core.errorToJSON(err));
|
2019-08-17 09:51:51 -07:00
|
|
|
replError(formattedError);
|
2019-02-11 11:01:28 -08:00
|
|
|
}
|
|
|
|
// Quit REPL anyways.
|
|
|
|
quitRepl(1);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
// Start continued read
|
|
|
|
while (!evaluate(code)) {
|
|
|
|
code += "\n";
|
|
|
|
try {
|
|
|
|
code += await readline(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.
|
2019-03-26 23:22:07 +11:00
|
|
|
const formattedError = formatError(core.errorToJSON(err));
|
2019-08-17 09:51:51 -07:00
|
|
|
replError(formattedError);
|
2019-02-11 11:01:28 -08:00
|
|
|
quitRepl(1);
|
|
|
|
}
|
2018-11-05 09:55:59 -08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|