mirror of
https://github.com/Dokploy/dokploy
synced 2025-06-26 18:27:59 +00:00
* feat(WIP): compose implementation * feat: add volumes, networks, services name hash generate * feat: add compose config test unique * feat: add tests for each unique config * feat: implement lodash for docker compose parsing * feat: add tests for generating compose file * refactor: implement logs docker compose * refactor: composeFile set not empty * feat: implement providers for compose deployments * feat: add Files volumes to compose * feat: add stop compose button * refactor: change strategie of building compose * feat: create .env file in composepath * refactor: simplify git and github function * chore: update deps * refactor: update migrations and add badge to recognize compose type * chore: update lock yaml * refactor: use code editor * feat: add monitoring for app types * refactor: reset stats on change appName * refactor: add option to clean monitoring folder * feat: show current command that will run * feat: add prefix * fix: add missing types * refactor: add docker provider and expose by default as false * refactor: customize error page * refactor: unified deployments to be a single one * feat: add vitest to ci/cd * revert: back to initial version * refactor: add maxconcurrency vitest * refactor: add pool forks to vitest * feat: add pocketbase template * fix: update path resolution compose * removed * feat: add template pocketbase * feat: add pocketbase template * feat: add support button * feat: add plausible template * feat: add calcom template * feat: add version to each template * feat: add code editor to enviroment variables and swarm settings json * refactor: add loader when download the image * fix: use base64 to generate keys plausible * feat: add recognized domain names by enviroment compose * refactor: show alert to redeploy in each card advanced tab * refactor: add validation to prevent create compose if not have permissions * chore: add templates section to contributing * chore: add example contributing
94 lines
2.3 KiB
TypeScript
94 lines
2.3 KiB
TypeScript
import type http from "node:http";
|
|
import { WebSocketServer } from "ws";
|
|
import { validateWebSocketRequest } from "../auth/auth";
|
|
import {
|
|
getLastAdvancedStatsFile,
|
|
recordAdvancedStats,
|
|
} from "../monitoring/utilts";
|
|
import { docker } from "../constants";
|
|
|
|
export const setupDockerStatsMonitoringSocketServer = (
|
|
server: http.Server<typeof http.IncomingMessage, typeof http.ServerResponse>,
|
|
) => {
|
|
const wssTerm = new WebSocketServer({
|
|
noServer: true,
|
|
path: "/listen-docker-stats-monitoring",
|
|
});
|
|
|
|
server.on("upgrade", (req, socket, head) => {
|
|
const { pathname } = new URL(req.url || "", `http://${req.headers.host}`);
|
|
|
|
if (pathname === "/_next/webpack-hmr") {
|
|
return;
|
|
}
|
|
if (pathname === "/listen-docker-stats-monitoring") {
|
|
wssTerm.handleUpgrade(req, socket, head, function done(ws) {
|
|
wssTerm.emit("connection", ws, req);
|
|
});
|
|
}
|
|
});
|
|
|
|
wssTerm.on("connection", async (ws, req) => {
|
|
const url = new URL(req.url || "", `http://${req.headers.host}`);
|
|
const appName = url.searchParams.get("appName");
|
|
const appType = (url.searchParams.get("appType") || "application") as
|
|
| "application"
|
|
| "stack"
|
|
| "docker-compose";
|
|
const { user, session } = await validateWebSocketRequest(req);
|
|
|
|
if (!appName) {
|
|
ws.close(4000, "appName no provided");
|
|
return;
|
|
}
|
|
|
|
if (!user || !session) {
|
|
ws.close();
|
|
return;
|
|
}
|
|
|
|
const intervalId = setInterval(async () => {
|
|
try {
|
|
const filter = {
|
|
status: ["running"],
|
|
...(appType === "application" && {
|
|
label: [`com.docker.swarm.service.name=${appName}`],
|
|
}),
|
|
...(appType === "stack" && {
|
|
label: [`com.docker.swarm.task.name=${appName}`],
|
|
}),
|
|
...(appType === "docker-compose" && {
|
|
name: [appName],
|
|
}),
|
|
};
|
|
|
|
const containers = await docker.listContainers({
|
|
filters: JSON.stringify(filter),
|
|
});
|
|
|
|
const container = containers[0];
|
|
if (!container || container?.State !== "running") {
|
|
ws.close(4000, "Container not running");
|
|
return;
|
|
}
|
|
|
|
await recordAdvancedStats(appName, container?.Id);
|
|
const data = await getLastAdvancedStatsFile(appName);
|
|
|
|
ws.send(
|
|
JSON.stringify({
|
|
data,
|
|
}),
|
|
);
|
|
} catch (error) {
|
|
// @ts-ignore
|
|
ws.close(4000, `Error: ${error.message}`);
|
|
}
|
|
}, 1300);
|
|
|
|
ws.on("close", () => {
|
|
clearInterval(intervalId);
|
|
});
|
|
});
|
|
};
|