import type http from "node:http"; import { findServerById } from "@dokploy/server/services/server"; import { spawn } from "node-pty"; import { Client } from "ssh2"; import { WebSocketServer } from "ws"; import { validateWebSocketRequest } from "../auth/auth"; import { getShell } from "./utils"; export const setupDockerContainerLogsWebSocketServer = ( server: http.Server, ) => { const wssTerm = new WebSocketServer({ noServer: true, path: "/docker-container-logs", }); server.on("upgrade", (req, socket, head) => { const { pathname } = new URL(req.url || "", `http://${req.headers.host}`); if (pathname === "/_next/webpack-hmr") { return; } if (pathname === "/docker-container-logs") { wssTerm.handleUpgrade(req, socket, head, function done(ws) { wssTerm.emit("connection", ws, req); }); } }); // eslint-disable-next-line @typescript-eslint/no-misused-promises wssTerm.on("connection", async (ws, req) => { const url = new URL(req.url || "", `http://${req.headers.host}`); const containerId = url.searchParams.get("containerId"); const tail = url.searchParams.get("tail"); const serverId = url.searchParams.get("serverId"); const { user, session } = await validateWebSocketRequest(req); if (!containerId) { ws.close(4000, "containerId no provided"); return; } if (!user || !session) { ws.close(); return; } try { if (serverId) { const server = await findServerById(serverId); if (!server.sshKeyId) return; const client = new Client(); new Promise((resolve, reject) => { client .once("ready", () => { const command = ` bash -c "docker container logs --tail ${tail} --follow ${containerId}" `; client.exec(command, (err, stream) => { if (err) { console.error("Execution error:", err); reject(err); return; } stream .on("close", () => { client.end(); resolve(); }) .on("data", (data: string) => { ws.send(data.toString()); }) .stderr.on("data", (data) => { ws.send(data.toString()); }); }); }) .connect({ host: server.ipAddress, port: server.port, username: server.username, privateKey: server.sshKey?.privateKey, timeout: 99999, }); }); } else { const shell = getShell(); const ptyProcess = spawn( shell, [ "-c", `docker container logs --tail ${tail} --follow ${containerId}`, ], { name: "xterm-256color", cwd: process.env.HOME, env: process.env, encoding: "utf8", cols: 80, rows: 30, }, ); ptyProcess.onData((data) => { ws.send(data); }); ws.on("close", () => { ptyProcess.kill(); }); ws.on("message", (message) => { try { let command: string | Buffer[] | Buffer | ArrayBuffer; if (Buffer.isBuffer(message)) { command = message.toString("utf8"); } else { command = message; } ptyProcess.write(command.toString()); } catch (error) { // @ts-ignore const errorMessage = error?.message as unknown as string; ws.send(errorMessage); } }); } } catch (error) { // @ts-ignore const errorMessage = error?.message as unknown as string; ws.send(errorMessage); } }); };