106 lines
3.3 KiB
JavaScript
106 lines
3.3 KiB
JavaScript
/* jshint esversion: 6, asi: true, node: true */
|
|
/* eslint no-unused-expressions: ["error", { "allowShortCircuit": true, "allowTernary": true }],
|
|
no-console: ["error", { allow: ["warn", "error", "info"] }] */
|
|
// app.js
|
|
|
|
// eslint-disable-next-line import/order
|
|
const config = require('./config');
|
|
const path = require('path');
|
|
|
|
const nodeRoot = path.dirname(require.main.filename);
|
|
const publicPath = path.join(nodeRoot, 'client', 'public');
|
|
const express = require('express');
|
|
const logger = require('morgan');
|
|
|
|
const app = express();
|
|
const server = require('http').Server(app);
|
|
const favicon = require('serve-favicon');
|
|
const io = require('socket.io')(server, config.socketio);
|
|
const session = require('express-session')(config.express);
|
|
|
|
const appSocket = require('./socket');
|
|
const { setDefaultCredentials, basicAuth } = require('./util');
|
|
const { webssh2debug } = require('./logging');
|
|
const { reauth, connect, notfound, handleErrors } = require('./routes');
|
|
|
|
setDefaultCredentials(config);
|
|
|
|
// safe shutdown
|
|
let remainingSeconds = config.safeShutdownDuration;
|
|
let shutdownMode = false;
|
|
let shutdownInterval;
|
|
let connectionCount = 0;
|
|
// eslint-disable-next-line consistent-return
|
|
function safeShutdownGuard(req, res, next) {
|
|
if (!shutdownMode) return next();
|
|
res.status(503).end('Service unavailable: Server shutting down');
|
|
}
|
|
// express
|
|
app.use(safeShutdownGuard);
|
|
app.use(session);
|
|
if (config.accesslog) app.use(logger('common'));
|
|
app.disable('x-powered-by');
|
|
app.use(favicon(path.join(publicPath, 'favicon.ico')));
|
|
app.use('/ssh', express.static(publicPath, config.express.ssh));
|
|
app.use(basicAuth);
|
|
app.get('/ssh/reauth', reauth);
|
|
app.get('/ssh/host/:host?', connect);
|
|
app.use(notfound);
|
|
app.use(handleErrors);
|
|
|
|
// clean stop
|
|
function stopApp(reason) {
|
|
shutdownMode = false;
|
|
if (reason) console.info(`Stopping: ${reason}`);
|
|
clearInterval(shutdownInterval);
|
|
io.close();
|
|
server.close();
|
|
}
|
|
|
|
// bring up socket
|
|
io.on('connection', appSocket);
|
|
|
|
// socket.io
|
|
// expose express session with socket.request.session
|
|
io.use((socket, next) => {
|
|
socket.request.res ? session(socket.request, socket.request.res, next) : next(next); // eslint disable-line
|
|
});
|
|
|
|
function countdownTimer() {
|
|
if (!shutdownMode) clearInterval(shutdownInterval);
|
|
remainingSeconds -= 1;
|
|
if (remainingSeconds <= 0) {
|
|
stopApp('Countdown is over');
|
|
} else io.emit('shutdownCountdownUpdate', remainingSeconds);
|
|
}
|
|
|
|
const signals = ['SIGTERM', 'SIGINT'];
|
|
signals.forEach((signal) =>
|
|
process.on(signal, () => {
|
|
if (shutdownMode) stopApp('Safe shutdown aborted, force quitting');
|
|
if (!connectionCount > 0) stopApp('All connections ended');
|
|
shutdownMode = true;
|
|
console.error(
|
|
`\r\n${connectionCount} client(s) are still connected.\r\nStarting a ${remainingSeconds} seconds countdown.\r\nPress Ctrl+C again to force quit`
|
|
);
|
|
if (!shutdownInterval) shutdownInterval = setInterval(countdownTimer, 1000);
|
|
})
|
|
);
|
|
|
|
module.exports = { server, config };
|
|
|
|
const onConnection = (socket) => {
|
|
connectionCount += 1;
|
|
socket.on('disconnect', () => {
|
|
connectionCount -= 1;
|
|
if (connectionCount <= 0 && shutdownMode) {
|
|
stopApp('All clients disconnected');
|
|
}
|
|
});
|
|
socket.on('geometry', (cols, rows) => {
|
|
socket.request.session.ssh.terminfo = { cols, rows };
|
|
webssh2debug(socket, `SOCKET GEOMETRY: termCols = ${cols}, termRows = ${rows}`);
|
|
});
|
|
};
|
|
|
|
io.on('connection', onConnection);
|