From d20be4957068eea915779b63a8ebf6fa2bf98c8d Mon Sep 17 00:00:00 2001 From: Sam Atkins Date: Fri, 12 Apr 2024 12:43:13 +0100 Subject: [PATCH 1/2] Stop including PathCommandProvider (and so node-pty) in web build This is only usable in a native terminal. Including it also makes rollup upset about having to include a binary file from the node-pty library. There is a small hack here: rollup will automatically include anything that's imported, but splitting the imported path into two strings concatenated is enough to fool it into ignoring the file. --- src/puter-shell/main.js | 10 +++++++--- 1 file changed, 7 insertions(+), 3 deletions(-) diff --git a/src/puter-shell/main.js b/src/puter-shell/main.js index c640194..1a70d54 100644 --- a/src/puter-shell/main.js +++ b/src/puter-shell/main.js @@ -28,7 +28,6 @@ import { Context } from "contextlink"; import { SHELL_VERSIONS } from "../meta/versions.js"; import { PuterShellParser } from "../ansi-shell/parsing/PuterShellParser.js"; import { BuiltinCommandProvider } from "./providers/BuiltinCommandProvider.js"; -import { PathCommandProvider } from "./providers/PathCommandProvider.js"; import { CreateChatHistoryPlugin } from './plugins/ChatHistoryPlugin.js'; import { Pipe } from '../ansi-shell/pipeline/Pipe.js'; import { Coupler } from '../ansi-shell/pipeline/Coupler.js'; @@ -83,10 +82,15 @@ export const launchPuterShell = async (ctx) => { await sdkv2.setAPIOrigin(source_without_trailing_slash); } + // PathCommandProvider is only compatible with node.js for now + // HACK: The import path is split to fool rollup into not including it. + const { PathCommandProvider } = (ctx.platform.name === 'node') + ? await import('./providers/' + 'PathCommandProvider.js') + : { PathCommandProvider: null }; + const commandProvider = new CompositeCommandProvider([ new BuiltinCommandProvider(), - // PathCommandProvider is only compatible with node.js for now - ...(ctx.platform.name === 'node' ? [new PathCommandProvider()] : []), + ...(PathCommandProvider ? [new PathCommandProvider()] : []), new ScriptCommandProvider(), ]); From 8ededf630ae3e6b9e013cb044105ced8c08235a7 Mon Sep 17 00:00:00 2001 From: Sam Atkins Date: Fri, 12 Apr 2024 12:48:23 +0100 Subject: [PATCH 2/2] Guard use of `process` with checks that we're running on Node `process` only exists when running in Node. Notably, `if (process)` is still a ReferenceError, so the existing check in readline.js needs modifying too. Unfortunately we don't have access to the platform name there, so we just check if `process` exists. --- src/ansi-shell/ANSIShell.js | 2 +- src/ansi-shell/pipeline/Pipeline.js | 6 ++++-- src/ansi-shell/readline/readline.js | 2 +- 3 files changed, 6 insertions(+), 4 deletions(-) diff --git a/src/ansi-shell/ANSIShell.js b/src/ansi-shell/ANSIShell.js index 3ee3fab..0eb151e 100644 --- a/src/ansi-shell/ANSIShell.js +++ b/src/ansi-shell/ANSIShell.js @@ -113,7 +113,7 @@ export class ANSIShell extends EventTarget { } async doPromptIteration() { - if ( globalThis.force_eot ) { + if ( globalThis.force_eot && this.ctx.platform.name === 'node' ) { process.exit(0); } const { readline } = this.ctx.externs; diff --git a/src/ansi-shell/pipeline/Pipeline.js b/src/ansi-shell/pipeline/Pipeline.js index 0314d9e..8ef1073 100644 --- a/src/ansi-shell/pipeline/Pipeline.js +++ b/src/ansi-shell/pipeline/Pipeline.js @@ -288,8 +288,10 @@ export class PreparedCommand { // but for some reason Node crashes first, unless we set this handler, // EVEN IF IT DOES NOTHING. I also can't find a place to safely remove it, // so apologies if it makes debugging promises harder. - const rejectionCatcher = (reason, promise) => {}; - process.on('unhandledRejection', rejectionCatcher); + if (ctx.platform.name === 'node') { + const rejectionCatcher = (reason, promise) => { }; + process.on('unhandledRejection', rejectionCatcher); + } let exit_code = 0; try { diff --git a/src/ansi-shell/readline/readline.js b/src/ansi-shell/readline/readline.js index 2ac42c7..e7f816d 100644 --- a/src/ansi-shell/readline/readline.js +++ b/src/ansi-shell/readline/readline.js @@ -74,7 +74,7 @@ const ReadlineProcessorBuilder = builder => builder externs.out.write('^C\n'); // Exit if input line is empty // FIXME: Check for 'process' is so we only do this on Node. How should we handle exiting in Puter terminal? - if ( process && ctx.vars.result.length === 0 ) { + if ( typeof process !== 'undefined' && ctx.vars.result.length === 0 ) { process.exit(1); return; }