|
|
|
|
|
|
|
|
|
|
|
import * as net from 'net'; |
|
import { Terminal, DEFAULT_COLS, DEFAULT_ROWS } from './terminal'; |
|
import { IProcessEnv, IPtyForkOptions, IPtyOpenOptions } from './interfaces'; |
|
import { ArgvOrCommandLine } from './types'; |
|
import { assign } from './utils'; |
|
|
|
let pty: IUnixNative; |
|
try { |
|
pty = require('../build/Release/pty.node'); |
|
} catch (outerError) { |
|
try { |
|
pty = require('../build/Debug/pty.node'); |
|
} catch (innerError) { |
|
console.error('innerError', innerError); |
|
|
|
throw outerError; |
|
} |
|
} |
|
|
|
const DEFAULT_FILE = 'sh'; |
|
const DEFAULT_NAME = 'xterm'; |
|
const DESTROY_SOCKET_TIMEOUT_MS = 200; |
|
|
|
export class UnixTerminal extends Terminal { |
|
protected _fd: number; |
|
protected _pty: string; |
|
|
|
protected _file: string; |
|
protected _name: string; |
|
|
|
protected _readable: boolean; |
|
protected _writable: boolean; |
|
|
|
private _boundClose: boolean; |
|
private _emittedClose: boolean; |
|
private _master: net.Socket; |
|
private _slave: net.Socket; |
|
|
|
public get master(): net.Socket { return this._master; } |
|
public get slave(): net.Socket { return this._slave; } |
|
|
|
constructor(file?: string, args?: ArgvOrCommandLine, opt?: IPtyForkOptions) { |
|
super(opt); |
|
|
|
if (typeof args === 'string') { |
|
throw new Error('args as a string is not supported on unix.'); |
|
} |
|
|
|
|
|
args = args || []; |
|
file = file || DEFAULT_FILE; |
|
opt = opt || {}; |
|
opt.env = opt.env || process.env; |
|
|
|
this._cols = opt.cols || DEFAULT_COLS; |
|
this._rows = opt.rows || DEFAULT_ROWS; |
|
const uid = opt.uid || -1; |
|
const gid = opt.gid || -1; |
|
const env = assign({}, opt.env); |
|
|
|
if (opt.env === process.env) { |
|
this._sanitizeEnv(env); |
|
} |
|
|
|
const cwd = opt.cwd || process.cwd(); |
|
const name = opt.name || env.TERM || DEFAULT_NAME; |
|
env.TERM = name; |
|
const parsedEnv = this._parseEnv(env); |
|
|
|
const encoding = (opt.encoding === undefined ? 'utf8' : opt.encoding); |
|
|
|
const onexit = (code: number, signal: number) => { |
|
|
|
|
|
if (!this._emittedClose) { |
|
if (this._boundClose) { |
|
return; |
|
} |
|
this._boundClose = true; |
|
|
|
|
|
|
|
let timeout = setTimeout(() => { |
|
timeout = null; |
|
|
|
this._socket.destroy(); |
|
}, DESTROY_SOCKET_TIMEOUT_MS); |
|
this.once('close', () => { |
|
if (timeout !== null) { |
|
clearTimeout(timeout); |
|
} |
|
this.emit('exit', code, signal); |
|
}); |
|
return; |
|
} |
|
this.emit('exit', code, signal); |
|
}; |
|
|
|
|
|
const term = pty.fork(file, args, parsedEnv, cwd, this._cols, this._rows, uid, gid, (encoding === 'utf8'), onexit); |
|
|
|
this._socket = new PipeSocket(term.fd); |
|
if (encoding !== null) { |
|
this._socket.setEncoding(encoding); |
|
} |
|
|
|
|
|
this._socket.on('error', (err: any) => { |
|
|
|
if (err.code) { |
|
if (~err.code.indexOf('EAGAIN')) { |
|
return; |
|
} |
|
} |
|
|
|
|
|
this._close(); |
|
|
|
if (!this._emittedClose) { |
|
this._emittedClose = true; |
|
this.emit('close'); |
|
} |
|
|
|
|
|
|
|
|
|
|
|
if (err.code) { |
|
if (~err.code.indexOf('errno 5') || ~err.code.indexOf('EIO')) { |
|
return; |
|
} |
|
} |
|
|
|
|
|
if (this.listeners('error').length < 2) { |
|
throw err; |
|
} |
|
}); |
|
|
|
this._pid = term.pid; |
|
this._fd = term.fd; |
|
this._pty = term.pty; |
|
|
|
this._file = file; |
|
this._name = name; |
|
|
|
this._readable = true; |
|
this._writable = true; |
|
|
|
this._socket.on('close', () => { |
|
if (this._emittedClose) { |
|
return; |
|
} |
|
this._emittedClose = true; |
|
this._close(); |
|
this.emit('close'); |
|
}); |
|
|
|
this._forwardEvents(); |
|
} |
|
|
|
protected _write(data: string): void { |
|
this._socket.write(data); |
|
} |
|
|
|
|
|
|
|
|
|
|
|
public static open(opt: IPtyOpenOptions): UnixTerminal { |
|
const self: UnixTerminal = Object.create(UnixTerminal.prototype); |
|
opt = opt || {}; |
|
|
|
if (arguments.length > 1) { |
|
opt = { |
|
cols: arguments[1], |
|
rows: arguments[2] |
|
}; |
|
} |
|
|
|
const cols = opt.cols || DEFAULT_COLS; |
|
const rows = opt.rows || DEFAULT_ROWS; |
|
const encoding = (opt.encoding === undefined ? 'utf8' : opt.encoding); |
|
|
|
|
|
const term: IUnixOpenProcess = pty.open(cols, rows); |
|
|
|
self._master = new PipeSocket(<number>term.master); |
|
if (encoding !== null) { |
|
self._master.setEncoding(encoding); |
|
} |
|
self._master.resume(); |
|
|
|
self._slave = new PipeSocket(term.slave); |
|
if (encoding !== null) { |
|
self._slave.setEncoding(encoding); |
|
} |
|
self._slave.resume(); |
|
|
|
self._socket = self._master; |
|
self._pid = null; |
|
self._fd = term.master; |
|
self._pty = term.pty; |
|
|
|
self._file = process.argv[0] || 'node'; |
|
self._name = process.env.TERM || ''; |
|
|
|
self._readable = true; |
|
self._writable = true; |
|
|
|
self._socket.on('error', err => { |
|
self._close(); |
|
if (self.listeners('error').length < 2) { |
|
throw err; |
|
} |
|
}); |
|
|
|
self._socket.on('close', () => { |
|
self._close(); |
|
}); |
|
|
|
return self; |
|
} |
|
|
|
public destroy(): void { |
|
this._close(); |
|
|
|
|
|
|
|
this._socket.once('close', () => { |
|
this.kill('SIGHUP'); |
|
}); |
|
|
|
this._socket.destroy(); |
|
} |
|
|
|
public kill(signal?: string): void { |
|
try { |
|
process.kill(this.pid, signal || 'SIGHUP'); |
|
} catch (e) { } |
|
} |
|
|
|
|
|
|
|
|
|
public get process(): string { |
|
return pty.process(this._fd, this._pty) || this._file; |
|
} |
|
|
|
|
|
|
|
|
|
|
|
public resize(cols: number, rows: number): void { |
|
if (cols <= 0 || rows <= 0 || isNaN(cols) || isNaN(rows) || cols === Infinity || rows === Infinity) { |
|
throw new Error('resizing must be done using positive cols and rows'); |
|
} |
|
pty.resize(this._fd, cols, rows); |
|
this._cols = cols; |
|
this._rows = rows; |
|
} |
|
|
|
private _sanitizeEnv(env: IProcessEnv): void { |
|
|
|
delete env['TMUX']; |
|
delete env['TMUX_PANE']; |
|
|
|
|
|
|
|
delete env['STY']; |
|
delete env['WINDOW']; |
|
|
|
|
|
delete env['WINDOWID']; |
|
delete env['TERMCAP']; |
|
delete env['COLUMNS']; |
|
delete env['LINES']; |
|
} |
|
} |
|
|
|
|
|
|
|
|
|
|
|
|
|
class PipeSocket extends net.Socket { |
|
constructor(fd: number) { |
|
const { Pipe, constants } = (<any>process).binding('pipe_wrap'); |
|
|
|
const handle = new Pipe(constants.SOCKET); |
|
handle.open(fd); |
|
super(<any>{ handle }); |
|
} |
|
} |
|
|