2023-11-11 15:18:37 +01:00
|
|
|
import { SocketHandler } from "../socket-handler.js";
|
|
|
|
import { DockgeServer } from "../dockge-server";
|
|
|
|
import { callbackError, checkLogin, DockgeSocket, ValidationError } from "../util-server";
|
|
|
|
import { log } from "../log";
|
|
|
|
import yaml from "yaml";
|
|
|
|
import path from "path";
|
|
|
|
import fs from "fs";
|
|
|
|
import {
|
|
|
|
allowedCommandList,
|
|
|
|
allowedRawKeys,
|
|
|
|
getComposeTerminalName, getContainerExecTerminalName,
|
|
|
|
isDev,
|
|
|
|
PROGRESS_TERMINAL_ROWS
|
|
|
|
} from "../util-common";
|
|
|
|
import { InteractiveTerminal, MainTerminal, Terminal } from "../terminal";
|
|
|
|
import { Stack } from "../stack";
|
|
|
|
|
|
|
|
export class TerminalSocketHandler extends SocketHandler {
|
|
|
|
create(socket : DockgeSocket, server : DockgeServer) {
|
|
|
|
|
|
|
|
socket.on("terminalInput", async (terminalName : unknown, cmd : unknown, errorCallback) => {
|
|
|
|
try {
|
|
|
|
checkLogin(socket);
|
|
|
|
|
|
|
|
if (typeof(terminalName) !== "string") {
|
|
|
|
throw new Error("Terminal name must be a string.");
|
|
|
|
}
|
|
|
|
|
|
|
|
if (typeof(cmd) !== "string") {
|
|
|
|
throw new Error("Command must be a string.");
|
|
|
|
}
|
|
|
|
|
|
|
|
let terminal = Terminal.getTerminal(terminalName);
|
|
|
|
if (terminal instanceof InteractiveTerminal) {
|
|
|
|
//log.debug("terminalInput", "Terminal found, writing to terminal.");
|
|
|
|
terminal.write(cmd);
|
|
|
|
} else {
|
|
|
|
throw new Error("Terminal not found or it is not a Interactive Terminal.");
|
|
|
|
}
|
|
|
|
} catch (e) {
|
2023-11-18 08:54:43 +01:00
|
|
|
if (e instanceof Error) {
|
|
|
|
errorCallback({
|
|
|
|
ok: false,
|
|
|
|
msg: e.message,
|
|
|
|
});
|
|
|
|
}
|
2023-11-11 15:18:37 +01:00
|
|
|
}
|
|
|
|
});
|
|
|
|
|
|
|
|
// Main Terminal
|
|
|
|
socket.on("mainTerminal", async (terminalName : unknown, callback) => {
|
|
|
|
try {
|
|
|
|
checkLogin(socket);
|
|
|
|
|
|
|
|
// TODO: Reset the name here, force one main terminal for now
|
|
|
|
terminalName = "console";
|
|
|
|
|
|
|
|
if (typeof(terminalName) !== "string") {
|
|
|
|
throw new ValidationError("Terminal name must be a string.");
|
|
|
|
}
|
|
|
|
|
|
|
|
log.debug("deployStack", "Terminal name: " + terminalName);
|
|
|
|
|
|
|
|
let terminal = Terminal.getTerminal(terminalName);
|
|
|
|
|
|
|
|
if (!terminal) {
|
|
|
|
terminal = new MainTerminal(server, terminalName);
|
|
|
|
terminal.rows = 50;
|
|
|
|
log.debug("deployStack", "Terminal created");
|
|
|
|
}
|
|
|
|
|
|
|
|
terminal.join(socket);
|
|
|
|
terminal.start();
|
|
|
|
|
|
|
|
callback({
|
|
|
|
ok: true,
|
|
|
|
});
|
|
|
|
} catch (e) {
|
|
|
|
callbackError(e, callback);
|
|
|
|
}
|
|
|
|
});
|
|
|
|
|
|
|
|
// Interactive Terminal for containers
|
|
|
|
socket.on("interactiveTerminal", async (stackName : unknown, serviceName : unknown, shell : unknown, callback) => {
|
|
|
|
try {
|
|
|
|
checkLogin(socket);
|
|
|
|
|
|
|
|
if (typeof(stackName) !== "string") {
|
|
|
|
throw new ValidationError("Stack name must be a string.");
|
|
|
|
}
|
|
|
|
|
|
|
|
if (typeof(serviceName) !== "string") {
|
|
|
|
throw new ValidationError("Service name must be a string.");
|
|
|
|
}
|
|
|
|
|
|
|
|
if (typeof(shell) !== "string") {
|
|
|
|
throw new ValidationError("Shell must be a string.");
|
|
|
|
}
|
|
|
|
|
|
|
|
log.debug("interactiveTerminal", "Stack name: " + stackName);
|
|
|
|
log.debug("interactiveTerminal", "Service name: " + serviceName);
|
|
|
|
|
|
|
|
// Get stack
|
2023-12-03 14:30:50 +01:00
|
|
|
const stack = await Stack.getStack(server, stackName);
|
2023-11-11 15:18:37 +01:00
|
|
|
stack.joinContainerTerminal(socket, serviceName, shell);
|
|
|
|
|
|
|
|
callback({
|
|
|
|
ok: true,
|
|
|
|
});
|
|
|
|
} catch (e) {
|
|
|
|
callbackError(e, callback);
|
|
|
|
}
|
|
|
|
});
|
|
|
|
|
|
|
|
// Join Output Terminal
|
|
|
|
socket.on("terminalJoin", async (terminalName : unknown, callback) => {
|
|
|
|
if (typeof(callback) !== "function") {
|
|
|
|
log.debug("console", "Callback is not a function.");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
try {
|
|
|
|
checkLogin(socket);
|
|
|
|
if (typeof(terminalName) !== "string") {
|
|
|
|
throw new ValidationError("Terminal name must be a string.");
|
|
|
|
}
|
|
|
|
|
|
|
|
let buffer : string = Terminal.getTerminal(terminalName)?.getBuffer() ?? "";
|
|
|
|
|
|
|
|
if (!buffer) {
|
|
|
|
log.debug("console", "No buffer found.");
|
|
|
|
}
|
|
|
|
|
|
|
|
callback({
|
|
|
|
ok: true,
|
|
|
|
buffer,
|
|
|
|
});
|
|
|
|
} catch (e) {
|
|
|
|
callbackError(e, callback);
|
|
|
|
}
|
|
|
|
});
|
|
|
|
|
2023-11-24 19:04:16 +01:00
|
|
|
// Leave Combined Terminal
|
|
|
|
socket.on("leaveCombinedTerminal", async (stackName : unknown, callback) => {
|
|
|
|
try {
|
|
|
|
checkLogin(socket);
|
|
|
|
|
|
|
|
log.debug("leaveCombinedTerminal", "Stack name: " + stackName);
|
|
|
|
|
|
|
|
if (typeof(stackName) !== "string") {
|
|
|
|
throw new ValidationError("Stack name must be a string.");
|
|
|
|
}
|
2023-11-11 15:18:37 +01:00
|
|
|
|
2023-12-03 14:30:50 +01:00
|
|
|
const stack = await Stack.getStack(server, stackName);
|
2023-11-24 19:04:16 +01:00
|
|
|
await stack.leaveCombinedTerminal(socket);
|
|
|
|
|
|
|
|
callback({
|
|
|
|
ok: true,
|
|
|
|
});
|
|
|
|
} catch (e) {
|
|
|
|
callbackError(e, callback);
|
|
|
|
}
|
2023-11-11 15:18:37 +01:00
|
|
|
});
|
|
|
|
|
2023-12-16 10:57:21 +01:00
|
|
|
// Resize Terminal
|
|
|
|
socket.on(
|
|
|
|
"terminalResize",
|
|
|
|
async (terminalName: unknown, rows: unknown, cols: unknown) => {
|
|
|
|
log.info("terminalResize", `Terminal: ${terminalName}`);
|
|
|
|
try {
|
|
|
|
checkLogin(socket);
|
|
|
|
if (typeof terminalName !== "string") {
|
|
|
|
throw new Error("Terminal name must be a string.");
|
|
|
|
}
|
|
|
|
|
|
|
|
if (typeof rows !== "number") {
|
|
|
|
throw new Error("Command must be a number.");
|
|
|
|
}
|
|
|
|
if (typeof cols !== "number") {
|
|
|
|
throw new Error("Command must be a number.");
|
|
|
|
}
|
|
|
|
|
|
|
|
let terminal = Terminal.getTerminal(terminalName);
|
|
|
|
|
|
|
|
// log.info("terminal", terminal);
|
|
|
|
if (terminal instanceof Terminal) {
|
|
|
|
//log.debug("terminalInput", "Terminal found, writing to terminal.");
|
|
|
|
terminal.rows = rows;
|
|
|
|
terminal.cols = cols;
|
|
|
|
} else {
|
|
|
|
throw new Error(`${terminalName} Terminal not found.`);
|
|
|
|
}
|
|
|
|
} catch (e) {
|
|
|
|
log.debug(
|
|
|
|
"terminalResize",
|
|
|
|
// Added to prevent the lint error when adding the type
|
|
|
|
// and ts type checker saying type is unknown.
|
|
|
|
// @ts-ignore
|
|
|
|
`Error on ${terminalName}: ${e.message}`
|
|
|
|
);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
);
|
2023-11-11 15:18:37 +01:00
|
|
|
}
|
|
|
|
}
|