mirror of
https://github.com/quine-global/hyper.git
synced 2026-01-12 20:18:41 -09:00
245 lines
7 KiB
TypeScript
245 lines
7 KiB
TypeScript
import {EventEmitter} from 'events';
|
|
import {StringDecoder} from 'string_decoder';
|
|
import defaultShell from 'default-shell';
|
|
import {getDecoratedEnv} from './plugins';
|
|
import {productName, version} from './package.json';
|
|
import * as config from './config';
|
|
import {IPty, IWindowsPtyForkOptions, spawn as npSpawn} from 'node-pty';
|
|
import {cliScriptPath} from './config/paths';
|
|
import {dirname} from 'path';
|
|
import shellEnv from 'shell-env';
|
|
|
|
const createNodePtyError = () =>
|
|
new Error(
|
|
'`node-pty` failed to load. Typically this means that it was built incorrectly. Please check the `readme.md` to more info.'
|
|
);
|
|
|
|
let spawn: typeof npSpawn;
|
|
try {
|
|
// eslint-disable-next-line @typescript-eslint/no-var-requires
|
|
spawn = require('node-pty').spawn;
|
|
} catch (err) {
|
|
throw createNodePtyError();
|
|
}
|
|
|
|
const envFromConfig = config.getConfig().env || {};
|
|
const useConpty = config.getConfig().useConpty;
|
|
|
|
// Max duration to batch session data before sending it to the renderer process.
|
|
const BATCH_DURATION_MS = 16;
|
|
|
|
// Max size of a session data batch. Note that this value can be exceeded by ~4k
|
|
// (chunk sizes seem to be 4k at the most)
|
|
const BATCH_MAX_SIZE = 200 * 1024;
|
|
|
|
// Data coming from the pty is sent to the renderer process for further
|
|
// vt parsing and rendering. This class batches data to minimize the number of
|
|
// IPC calls. It also reduces GC pressure and CPU cost: each chunk is prefixed
|
|
// with the window ID which is then stripped on the renderer process and this
|
|
// overhead is reduced with batching.
|
|
class DataBatcher extends EventEmitter {
|
|
uid: string;
|
|
decoder: StringDecoder;
|
|
data!: string;
|
|
timeout!: NodeJS.Timeout | null;
|
|
constructor(uid: string) {
|
|
super();
|
|
this.uid = uid;
|
|
this.decoder = new StringDecoder('utf8');
|
|
|
|
this.reset();
|
|
}
|
|
|
|
reset() {
|
|
this.data = this.uid;
|
|
this.timeout = null;
|
|
}
|
|
|
|
write(chunk: Buffer) {
|
|
if (this.data.length + chunk.length >= BATCH_MAX_SIZE) {
|
|
// We've reached the max batch size. Flush it and start another one
|
|
if (this.timeout) {
|
|
clearTimeout(this.timeout);
|
|
this.timeout = null;
|
|
}
|
|
this.flush();
|
|
}
|
|
|
|
this.data += this.decoder.write(chunk);
|
|
|
|
if (!this.timeout) {
|
|
this.timeout = setTimeout(() => this.flush(), BATCH_DURATION_MS);
|
|
}
|
|
}
|
|
|
|
flush() {
|
|
// Reset before emitting to allow for potential reentrancy
|
|
const data = this.data;
|
|
this.reset();
|
|
|
|
this.emit('flush', data);
|
|
}
|
|
}
|
|
|
|
interface SessionOptions {
|
|
uid: string;
|
|
rows: number;
|
|
cols: number;
|
|
cwd: string;
|
|
shell: string;
|
|
shellArgs: string[];
|
|
}
|
|
export default class Session extends EventEmitter {
|
|
pty: IPty | null;
|
|
batcher: DataBatcher | null;
|
|
shell: string | null;
|
|
ended: boolean;
|
|
initTimestamp: number;
|
|
constructor(options: SessionOptions) {
|
|
super();
|
|
this.pty = null;
|
|
this.batcher = null;
|
|
this.shell = null;
|
|
this.ended = false;
|
|
this.initTimestamp = new Date().getTime();
|
|
this.init(options);
|
|
}
|
|
|
|
init({uid, rows, cols: columns, cwd, shell: _shell, shellArgs: _shellArgs}: SessionOptions) {
|
|
// eslint-disable-next-line @typescript-eslint/no-var-requires
|
|
const osLocale = require('os-locale') as typeof import('os-locale');
|
|
const cleanEnv =
|
|
process.env['APPIMAGE'] && process.env['APPDIR'] ? shellEnv.sync(_shell || defaultShell) : process.env;
|
|
const baseEnv = Object.assign(
|
|
{},
|
|
cleanEnv,
|
|
{
|
|
LANG: `${osLocale.sync().replace(/-/, '_')}.UTF-8`,
|
|
TERM: 'xterm-256color',
|
|
COLORTERM: 'truecolor',
|
|
TERM_PROGRAM: productName,
|
|
TERM_PROGRAM_VERSION: version
|
|
},
|
|
envFromConfig
|
|
);
|
|
|
|
// path to AppImage mount point is added to PATH environment variable automatically
|
|
// which conflicts with the cli
|
|
if (baseEnv['APPIMAGE'] && baseEnv['APPDIR']) {
|
|
baseEnv['PATH'] = [dirname(cliScriptPath)]
|
|
.concat((baseEnv['PATH'] || '').split(':').filter((val) => !val.startsWith(baseEnv['APPDIR'])))
|
|
.join(':');
|
|
}
|
|
|
|
// Electron has a default value for process.env.GOOGLE_API_KEY
|
|
// We don't want to leak this to the shell
|
|
// See https://github.com/vercel/hyper/issues/696
|
|
if (baseEnv.GOOGLE_API_KEY && process.env.GOOGLE_API_KEY === baseEnv.GOOGLE_API_KEY) {
|
|
delete baseEnv.GOOGLE_API_KEY;
|
|
}
|
|
|
|
const defaultShellArgs = ['--login'];
|
|
|
|
const options: IWindowsPtyForkOptions = {
|
|
cols: columns,
|
|
rows,
|
|
cwd,
|
|
env: getDecoratedEnv(baseEnv)
|
|
};
|
|
|
|
// if config do not set the useConpty, it will be judged by the node-pty
|
|
if (typeof useConpty === 'boolean') {
|
|
options.useConpty = useConpty;
|
|
}
|
|
|
|
const shell = _shell || defaultShell;
|
|
const shellArgs = _shellArgs || defaultShellArgs;
|
|
|
|
try {
|
|
this.pty = spawn(shell, shellArgs, options);
|
|
} catch (_err) {
|
|
const err = _err as {message: string};
|
|
if (/is not a function/.test(err.message)) {
|
|
throw createNodePtyError();
|
|
} else {
|
|
throw err;
|
|
}
|
|
}
|
|
|
|
this.batcher = new DataBatcher(uid);
|
|
this.pty.onData((chunk) => {
|
|
if (this.ended) {
|
|
return;
|
|
}
|
|
this.batcher?.write(chunk as any);
|
|
});
|
|
|
|
this.batcher.on('flush', (data: string) => {
|
|
this.emit('data', data);
|
|
});
|
|
|
|
this.pty.onExit((e) => {
|
|
if (!this.ended) {
|
|
// fall back to default shell config if the shell exits within 1 sec with non zero exit code
|
|
// this will inform users in case there are errors in the config instead of instant exit
|
|
const runDuration = new Date().getTime() - this.initTimestamp;
|
|
if (e.exitCode > 0 && runDuration < 1000) {
|
|
const defaultShellConfig = {shell: defaultShell, shellArgs: defaultShellArgs};
|
|
const msg = `
|
|
shell exited in ${runDuration} ms with exit code ${e.exitCode}
|
|
please check the shell config: ${JSON.stringify({shell, shellArgs}, undefined, 2)}
|
|
fallback to default shell config: ${JSON.stringify(defaultShellConfig, undefined, 2)}
|
|
`;
|
|
console.warn(msg);
|
|
this.batcher?.write(msg.replace(/\n/g, '\r\n') as any);
|
|
this.init({uid, rows, cols: columns, cwd, ...defaultShellConfig});
|
|
} else {
|
|
this.ended = true;
|
|
this.emit('exit');
|
|
}
|
|
}
|
|
});
|
|
|
|
this.shell = shell;
|
|
}
|
|
|
|
exit() {
|
|
this.destroy();
|
|
}
|
|
|
|
write(data: string) {
|
|
if (this.pty) {
|
|
this.pty.write(data);
|
|
} else {
|
|
console.warn('Warning: Attempted to write to a session with no pty');
|
|
}
|
|
}
|
|
|
|
resize({cols, rows}: {cols: number; rows: number}) {
|
|
if (this.pty) {
|
|
try {
|
|
this.pty.resize(cols, rows);
|
|
} catch (_err) {
|
|
const err = _err as {stack: any};
|
|
console.error(err.stack);
|
|
}
|
|
} else {
|
|
console.warn('Warning: Attempted to resize a session with no pty');
|
|
}
|
|
}
|
|
|
|
destroy() {
|
|
if (this.pty) {
|
|
try {
|
|
this.pty.kill();
|
|
} catch (_err) {
|
|
const err = _err as {stack: any};
|
|
console.error('exit error', err.stack);
|
|
}
|
|
} else {
|
|
console.warn('Warning: Attempted to destroy a session with no pty');
|
|
}
|
|
this.emit('exit');
|
|
this.ended = true;
|
|
}
|
|
}
|