hyper/app/session.js

132 lines
3.1 KiB
JavaScript
Raw Normal View History

const { app } = require('electron');
2016-06-30 22:01:04 -08:00
const { EventEmitter } = require('events');
const { exec } = require('child_process');
const defaultShell = require('default-shell');
const { getDecoratedEnv } = require('./plugins');
const { productName, version } = require('./package');
const config = require('./config');
2016-06-30 22:01:04 -08:00
2016-07-13 13:06:05 -08:00
let spawn;
try {
spawn = require('child_pty').spawn;
} catch (err) {
console.error(
'A native module failed to load. Typically this means ' +
'you installed the modules incorrectly.\n Use `scripts/install.sh` ' +
'to trigger the installation.\n ' +
'More information: https://github.com/zeit/hyperterm/issues/72'
);
}
2016-07-08 13:26:44 -08:00
const TITLE_POLL_INTERVAL = 500;
2016-06-30 22:01:04 -08:00
const envFromConfig = config.getConfig().env || {};
2016-06-30 22:01:04 -08:00
module.exports = class Session extends EventEmitter {
constructor ({ rows, cols: columns, cwd, shell, shellArgs }) {
2016-06-30 22:01:04 -08:00
super();
const baseEnv = Object.assign({}, process.env, {
LANG: app.getLocale().replace('-', '_') + '.UTF-8',
TERM: 'xterm-256color',
TERM_PROGRAM: productName,
TERM_PROGRAM_VERSION: version
}, envFromConfig);
const defaultShellArgs = ['--login'];
this.pty = spawn(shell || defaultShell, shellArgs || defaultShellArgs, {
2016-07-03 12:35:45 -08:00
columns,
2016-06-30 22:01:04 -08:00
rows,
cwd,
env: getDecoratedEnv(baseEnv)
2016-06-30 22:01:04 -08:00
});
this.pty.stdout.on('data', (data) => {
this.emit('data', data.toString('utf8'));
});
this.pty.on('exit', () => {
if (!this.ended) {
this.ended = true;
this.emit('exit');
}
});
2016-07-23 16:57:47 -08:00
this.shell = shell || defaultShell;
2016-06-30 22:01:04 -08:00
this.getTitle();
}
focus () {
2016-07-14 15:16:44 -08:00
this.subscribed = true;
this.getTitle();
2016-06-30 22:01:04 -08:00
}
blur () {
2016-07-14 15:16:44 -08:00
this.subscribed = false;
2016-06-30 22:01:04 -08:00
clearTimeout(this.titlePoll);
}
2016-07-14 15:16:44 -08:00
getTitle () {
2016-06-30 22:01:04 -08:00
if ('win32' === process.platform) return;
2016-07-14 15:16:44 -08:00
if (this.fetching) return;
this.fetching = true;
2016-06-30 22:01:04 -08:00
let tty = this.pty.stdout.ttyname;
tty = tty.replace(/^\/dev\/tty/, '');
// try to exclude grep from the results
// by grepping for `[s]001` instead of `s001`
tty = `[${tty[0]}]${tty.substr(1)}`;
// TODO: limit the concurrency of how many processes we run?
// TODO: only tested on mac
2016-07-08 13:26:44 -08:00
exec(`ps uxac | grep ${tty} | head -n 1`, (err, out) => {
2016-07-14 15:16:44 -08:00
this.fetching = false;
2016-06-30 22:01:04 -08:00
if (this.ended) return;
if (err) return;
let title = out.split(' ').pop();
if (title) {
title = title.replace(/^\(/, '');
title = title.replace(/\)?\n$/, '');
if (title !== this.lastTitle) {
this.emit('title', title);
this.lastTitle = title;
}
}
2016-07-14 15:16:44 -08:00
if (this.subscribed) {
this.titlePoll = setTimeout(() => this.getTitle(), TITLE_POLL_INTERVAL);
2016-06-30 22:01:04 -08:00
}
});
}
exit () {
this.destroy();
}
write (data) {
this.pty.stdin.write(data);
}
resize ({ cols: columns, rows }) {
try {
this.pty.stdout.resize({ columns, rows });
} catch (err) {
2016-06-30 22:25:19 -08:00
console.error(err.stack);
2016-06-30 22:01:04 -08:00
}
}
destroy () {
2016-07-03 12:35:45 -08:00
try {
this.pty.kill('SIGHUP');
} catch (err) {
console.error('exit error', err.stack);
}
2016-06-30 22:01:04 -08:00
this.emit('exit');
this.ended = true;
2016-07-14 15:16:44 -08:00
this.blur();
2016-06-30 22:01:04 -08:00
}
};