ssh forward moved to extra process #253

This commit is contained in:
Jan Prochazka
2022-04-03 09:40:46 +02:00
parent 1a76cc0979
commit 1a32d88312
2 changed files with 100 additions and 28 deletions

View File

@@ -0,0 +1,68 @@
const fs = require('fs-extra');
const platformInfo = require('../utility/platformInfo');
const childProcessChecker = require('../utility/childProcessChecker');
const { SSHConnection } = require('node-ssh-forward');
const { handleProcessCommunication } = require('../utility/processComm');
async function getSshConnection(connection) {
const sshConfig = {
endHost: connection.sshHost || '',
endPort: connection.sshPort || 22,
bastionHost: connection.sshBastionHost || '',
agentForward: connection.sshMode == 'agent',
passphrase: connection.sshMode == 'keyFile' ? connection.sshKeyfilePassword : undefined,
username: connection.sshLogin,
password: connection.sshMode == 'userPassword' ? connection.sshPassword : undefined,
agentSocket: connection.sshMode == 'agent' ? platformInfo.sshAuthSock : undefined,
privateKey:
connection.sshMode == 'keyFile' && connection.sshKeyfile ? await fs.readFile(connection.sshKeyfile) : undefined,
skipAutoPrivateKey: true,
noReadline: true,
};
const sshConn = new SSHConnection(sshConfig);
return sshConn;
}
async function handleStart({ connection, tunnelConfig }) {
try {
const sshConn = await getSshConnection(connection);
await sshConn.forward(tunnelConfig);
process.send({
msgtype: 'connected',
connection,
tunnelConfig,
});
} catch (err) {
process.send({
msgtype: 'error',
connection,
tunnelConfig,
errorMessage: err.message,
});
}
}
const messageHandlers = {
connect: handleStart,
};
async function handleMessage({ msgtype, ...other }) {
const handler = messageHandlers[msgtype];
await handler(other);
}
function start() {
childProcessChecker();
process.on('message', async message => {
if (handleProcessCommunication(message)) return;
try {
await handleMessage(message);
} catch (e) {
console.error('sshForwardProcess - unhandled error', e);
}
});
}
module.exports = { start };

View File

@@ -1,13 +1,12 @@
const { SSHConnection } = require('node-ssh-forward');
const fs = require('fs-extra');
const portfinder = require('portfinder'); const portfinder = require('portfinder');
const stableStringify = require('json-stable-stringify'); const stableStringify = require('json-stable-stringify');
const _ = require('lodash'); const _ = require('lodash');
const platformInfo = require('./platformInfo');
const AsyncLock = require('async-lock'); const AsyncLock = require('async-lock');
const lock = new AsyncLock(); const lock = new AsyncLock();
const { fork } = require('child_process');
const processArgs = require('../utility/processArgs');
const sshConnectionCache = {}; // const sshConnectionCache = {};
const sshTunnelCache = {}; const sshTunnelCache = {};
const CONNECTION_FIELDS = [ const CONNECTION_FIELDS = [
@@ -22,37 +21,41 @@ const CONNECTION_FIELDS = [
]; ];
const TUNNEL_FIELDS = [...CONNECTION_FIELDS, 'server', 'port']; const TUNNEL_FIELDS = [...CONNECTION_FIELDS, 'server', 'port'];
async function getSshConnection(connection) { function callForwardProcess(connection, tunnelConfig) {
const connectionCacheKey = stableStringify(_.pick(connection, CONNECTION_FIELDS)); let subprocess = fork(global['API_PACKAGE'] || process.argv[1], [
if (sshConnectionCache[connectionCacheKey]) return sshConnectionCache[connectionCacheKey]; '--is-forked-api',
'--start-process',
'sshForwardProcess',
...processArgs.getPassArgs(),
]);
const sshConfig = { subprocess.send({
endHost: connection.sshHost || '', msgtype: 'connect',
endPort: connection.sshPort || 22, connection,
bastionHost: connection.sshBastionHost || '', tunnelConfig,
agentForward: connection.sshMode == 'agent', });
passphrase: connection.sshMode == 'keyFile' ? connection.sshKeyfilePassword : undefined, return new Promise((resolve, reject) => {
username: connection.sshLogin, subprocess.on('message', resp => {
password: connection.sshMode == 'userPassword' ? connection.sshPassword : undefined, // @ts-ignore
agentSocket: connection.sshMode == 'agent' ? platformInfo.sshAuthSock : undefined, const { msgtype, errorMessage } = resp;
privateKey: if (msgtype == 'connected') {
connection.sshMode == 'keyFile' && connection.sshKeyfile ? await fs.readFile(connection.sshKeyfile) : undefined, resolve(resp);
skipAutoPrivateKey: true, }
noReadline: true, if (msgtype == 'error') {
}; reject(errorMessage);
}
const sshConn = new SSHConnection(sshConfig); });
sshConnectionCache[connectionCacheKey] = sshConn; subprocess.on('exit', code => {
return sshConn; console.log('SSH forward process exited');
});
});
} }
async function getSshTunnel(connection) { async function getSshTunnel(connection) {
const tunnelCacheKey = stableStringify(_.pick(connection, TUNNEL_FIELDS)); const tunnelCacheKey = stableStringify(_.pick(connection, TUNNEL_FIELDS));
return await lock.acquire(tunnelCacheKey, async () => { return await lock.acquire(tunnelCacheKey, async () => {
const sshConn = await getSshConnection(connection);
if (sshTunnelCache[tunnelCacheKey]) return sshTunnelCache[tunnelCacheKey]; if (sshTunnelCache[tunnelCacheKey]) return sshTunnelCache[tunnelCacheKey];
const localPort = await portfinder.getPortPromise({ port: 10000, stopPort: 60000 }); const localPort = await portfinder.getPortPromise({ port: 10000, stopPort: 60000 });
// workaround for `getPortPromise` not releasing the port quickly enough // workaround for `getPortPromise` not releasing the port quickly enough
await new Promise(resolve => setTimeout(resolve, 500)); await new Promise(resolve => setTimeout(resolve, 500));
@@ -66,7 +69,8 @@ async function getSshTunnel(connection) {
`Creating SSH tunnel to ${connection.sshHost}-${connection.server}:${connection.port}, using local port ${localPort}` `Creating SSH tunnel to ${connection.sshHost}-${connection.server}:${connection.port}, using local port ${localPort}`
); );
const tunnel = await sshConn.forward(tunnelConfig); await callForwardProcess(connection, tunnelConfig);
console.log( console.log(
`Created SSH tunnel to ${connection.sshHost}-${connection.server}:${connection.port}, using local port ${localPort}` `Created SSH tunnel to ${connection.sshHost}-${connection.server}:${connection.port}, using local port ${localPort}`
); );