mirror of
https://github.com/Dokploy/dokploy
synced 2025-06-26 18:27:59 +00:00
215 lines
5.0 KiB
TypeScript
215 lines
5.0 KiB
TypeScript
import { readSSHKey } from "@/server/utils/filesystem/ssh";
|
|
import { execAsync } from "@/server/utils/process/execAsync";
|
|
import { tail } from "lodash";
|
|
import { stderr, stdout } from "node:process";
|
|
import { Client } from "ssh2";
|
|
import { findServerById } from "./server";
|
|
|
|
export const getContainers = async () => {
|
|
try {
|
|
const { stdout, stderr } = await execAsync(
|
|
"docker ps -a --format 'CONTAINER ID : {{.ID}} | Name: {{.Names}} | Image: {{.Image}} | Ports: {{.Ports}} | State: {{.State}} | Status: {{.Status}}'",
|
|
);
|
|
|
|
if (stderr) {
|
|
console.error(`Error: ${stderr}`);
|
|
return;
|
|
}
|
|
|
|
const lines = stdout.trim().split("\n");
|
|
|
|
const containers = lines
|
|
.map((line) => {
|
|
const parts = line.split(" | ");
|
|
const containerId = parts[0]
|
|
? parts[0].replace("CONTAINER ID : ", "").trim()
|
|
: "No container id";
|
|
const name = parts[1]
|
|
? parts[1].replace("Name: ", "").trim()
|
|
: "No container name";
|
|
const image = parts[2]
|
|
? parts[2].replace("Image: ", "").trim()
|
|
: "No image";
|
|
const ports = parts[3]
|
|
? parts[3].replace("Ports: ", "").trim()
|
|
: "No ports";
|
|
const state = parts[4]
|
|
? parts[4].replace("State: ", "").trim()
|
|
: "No state";
|
|
const status = parts[5]
|
|
? parts[5].replace("Status: ", "").trim()
|
|
: "No status";
|
|
return {
|
|
containerId,
|
|
name,
|
|
image,
|
|
ports,
|
|
state,
|
|
status,
|
|
};
|
|
})
|
|
.filter((container) => !container.name.includes("dokploy"));
|
|
|
|
return containers;
|
|
} catch (error) {}
|
|
};
|
|
|
|
export const getConfig = async (containerId: string) => {
|
|
try {
|
|
const { stdout, stderr } = await execAsync(
|
|
`docker inspect ${containerId} --format='{{json .}}'`,
|
|
);
|
|
|
|
if (stderr) {
|
|
console.error(`Error: ${stderr}`);
|
|
return;
|
|
}
|
|
|
|
const config = JSON.parse(stdout);
|
|
|
|
return config;
|
|
} catch (error) {}
|
|
};
|
|
|
|
export const getContainersByAppNameMatch = async (
|
|
appName: string,
|
|
appType?: "stack" | "docker-compose",
|
|
serverId?: string,
|
|
) => {
|
|
try {
|
|
let result: string[] = [];
|
|
const cmd =
|
|
"docker ps -a --format 'CONTAINER ID : {{.ID}} | Name: {{.Names}} | State: {{.State}}'";
|
|
|
|
const command =
|
|
appType === "docker-compose"
|
|
? `${cmd} --filter='label=com.docker.compose.project=${appName}'`
|
|
: `${cmd} | grep ${appName}`;
|
|
if (serverId) {
|
|
const server = await findServerById(serverId);
|
|
|
|
if (!server.sshKeyId) return;
|
|
const keys = await readSSHKey(server.sshKeyId);
|
|
const client = new Client();
|
|
result = await new Promise<string[]>((resolve, reject) => {
|
|
let output = "";
|
|
client
|
|
.on("ready", () => {
|
|
client.exec(command, (err, stream) => {
|
|
if (err) {
|
|
console.error("Execution error:", err);
|
|
reject(err);
|
|
return;
|
|
}
|
|
stream
|
|
.on("close", () => {
|
|
client.end();
|
|
resolve(output.trim().split("\n"));
|
|
})
|
|
.on("data", (data: string) => {
|
|
output += data.toString();
|
|
})
|
|
.stderr.on("data", (data) => {});
|
|
});
|
|
})
|
|
.connect({
|
|
host: server.ipAddress,
|
|
port: server.port,
|
|
username: server.username,
|
|
privateKey: keys.privateKey,
|
|
timeout: 99999,
|
|
});
|
|
});
|
|
} else {
|
|
const { stdout, stderr } = await execAsync(command);
|
|
|
|
if (stderr) {
|
|
return [];
|
|
}
|
|
|
|
if (!stdout) return [];
|
|
|
|
result = stdout.trim().split("\n");
|
|
}
|
|
|
|
const containers = result.map((line) => {
|
|
const parts = line.split(" | ");
|
|
const containerId = parts[0]
|
|
? parts[0].replace("CONTAINER ID : ", "").trim()
|
|
: "No container id";
|
|
const name = parts[1]
|
|
? parts[1].replace("Name: ", "").trim()
|
|
: "No container name";
|
|
|
|
const state = parts[2]
|
|
? parts[2].replace("State: ", "").trim()
|
|
: "No state";
|
|
return {
|
|
containerId,
|
|
name,
|
|
state,
|
|
};
|
|
});
|
|
|
|
return containers || [];
|
|
} catch (error) {}
|
|
|
|
return [];
|
|
};
|
|
|
|
export const getContainersByAppLabel = async (appName: string) => {
|
|
try {
|
|
const { stdout, stderr } = await execAsync(
|
|
`docker ps --filter "label=com.docker.swarm.service.name=${appName}" --format 'CONTAINER ID : {{.ID}} | Name: {{.Names}} | State: {{.State}}'`,
|
|
);
|
|
|
|
if (stderr) {
|
|
console.error(`Error: ${stderr}`);
|
|
return;
|
|
}
|
|
|
|
if (!stdout) return [];
|
|
|
|
const lines = stdout.trim().split("\n");
|
|
|
|
const containers = lines.map((line) => {
|
|
const parts = line.split(" | ");
|
|
const containerId = parts[0]
|
|
? parts[0].replace("CONTAINER ID : ", "").trim()
|
|
: "No container id";
|
|
const name = parts[1]
|
|
? parts[1].replace("Name: ", "").trim()
|
|
: "No container name";
|
|
const state = parts[2]
|
|
? parts[2].replace("State: ", "").trim()
|
|
: "No state";
|
|
return {
|
|
containerId,
|
|
name,
|
|
state,
|
|
};
|
|
});
|
|
|
|
return containers || [];
|
|
} catch (error) {}
|
|
|
|
return [];
|
|
};
|
|
|
|
export const containerRestart = async (containerId: string) => {
|
|
try {
|
|
const { stdout, stderr } = await execAsync(
|
|
`docker container restart ${containerId}`,
|
|
);
|
|
|
|
if (stderr) {
|
|
console.error(`Error: ${stderr}`);
|
|
return;
|
|
}
|
|
|
|
const config = JSON.parse(stdout);
|
|
|
|
return config;
|
|
} catch (error) {}
|
|
};
|