mirror of
https://github.com/Dokploy/dokploy.git
synced 2026-06-20 14:45:42 +02:00
Refactor the database restore process to consistently use a single container ID for the PostgreSQL container. This change enhances reliability by ensuring that commands are executed against the correct container, preventing potential errors from multiple matches. Co-authored-by: Merloss 54235902+Merloss@users.noreply.github.com
58 lines
2.0 KiB
TypeScript
58 lines
2.0 KiB
TypeScript
import type { BackupSchedule } from "@dokploy/server/services/backup";
|
|
import { execAsync } from "../process/execAsync";
|
|
import { getS3Credentials, normalizeS3Path } from "./utils";
|
|
import { findDestinationById } from "@dokploy/server/services/destination";
|
|
import { IS_CLOUD, paths } from "@dokploy/server/constants";
|
|
import { mkdtemp } from "node:fs/promises";
|
|
import { join } from "node:path";
|
|
import { tmpdir } from "node:os";
|
|
|
|
export const runWebServerBackup = async (backup: BackupSchedule) => {
|
|
try {
|
|
if (IS_CLOUD) {
|
|
return;
|
|
}
|
|
const destination = await findDestinationById(backup.destinationId);
|
|
const rcloneFlags = getS3Credentials(destination);
|
|
const timestamp = new Date().toISOString().replace(/[:.]/g, "-");
|
|
const { BASE_PATH } = paths();
|
|
const tempDir = await mkdtemp(join(tmpdir(), "dokploy-backup-"));
|
|
const backupFileName = `webserver-backup-${timestamp}.zip`;
|
|
const s3Path = `:s3:${destination.bucket}/${normalizeS3Path(backup.prefix)}${backupFileName}`;
|
|
|
|
try {
|
|
await execAsync(`mkdir -p ${tempDir}/filesystem`);
|
|
|
|
// First get the container ID
|
|
const { stdout: containerId } = await execAsync(
|
|
`docker ps --filter "name=dokploy-postgres" --filter "status=running" -q | head -n 1`,
|
|
);
|
|
|
|
if (!containerId) {
|
|
throw new Error("PostgreSQL container not found");
|
|
}
|
|
|
|
const postgresContainerId = containerId.trim();
|
|
|
|
const postgresCommand = `docker exec ${postgresContainerId} pg_dump -v -Fc -U dokploy -d dokploy > '${tempDir}/database.sql'`;
|
|
await execAsync(postgresCommand);
|
|
|
|
await execAsync(`cp -r ${BASE_PATH}/* ${tempDir}/filesystem/`);
|
|
|
|
await execAsync(
|
|
// Zip all .sql files since we created more than one
|
|
`cd ${tempDir} && zip -r ${backupFileName} *.sql filesystem/ > /dev/null 2>&1`,
|
|
);
|
|
|
|
const uploadCommand = `rclone copyto ${rcloneFlags.join(" ")} "${tempDir}/${backupFileName}" "${s3Path}"`;
|
|
await execAsync(uploadCommand);
|
|
return true;
|
|
} finally {
|
|
await execAsync(`rm -rf ${tempDir}`);
|
|
}
|
|
} catch (error) {
|
|
console.error("Backup error:", error);
|
|
throw error;
|
|
}
|
|
};
|