mirror of
https://github.com/Dokploy/dokploy
synced 2025-06-26 18:27:59 +00:00
- Added support for handling commit normalization across GitHub, GitLab, and Gitea in the deployment API. - Implemented a new utility function to determine the provider based on request headers. - Improved deployment path validation to ensure consistency across different source types. - Cleaned up the code by removing redundant checks and enhancing readability.
201 lines
5.4 KiB
TypeScript
201 lines
5.4 KiB
TypeScript
import { db } from "@/server/db";
|
|
import { compose } from "@/server/db/schema";
|
|
import type { DeploymentJob } from "@/server/queues/queue-types";
|
|
import { myQueue } from "@/server/queues/queueSetup";
|
|
import { deploy } from "@/server/utils/deploy";
|
|
import { IS_CLOUD, shouldDeploy } from "@dokploy/server";
|
|
import { eq } from "drizzle-orm";
|
|
import type { NextApiRequest, NextApiResponse } from "next";
|
|
import {
|
|
extractBranchName,
|
|
extractCommitMessage,
|
|
extractCommitedPaths,
|
|
extractHash,
|
|
getProviderByHeader,
|
|
} from "../[refreshToken]";
|
|
|
|
export default async function handler(
|
|
req: NextApiRequest,
|
|
res: NextApiResponse,
|
|
) {
|
|
const { refreshToken } = req.query;
|
|
try {
|
|
if (req.headers["x-github-event"] === "ping") {
|
|
res.status(200).json({ message: "Ping received, webhook is active" });
|
|
return;
|
|
}
|
|
const composeResult = await db.query.compose.findFirst({
|
|
where: eq(compose.refreshToken, refreshToken as string),
|
|
with: {
|
|
project: true,
|
|
bitbucket: true,
|
|
},
|
|
});
|
|
|
|
if (!composeResult) {
|
|
res.status(404).json({ message: "Compose Not Found" });
|
|
return;
|
|
}
|
|
if (!composeResult?.autoDeploy) {
|
|
res.status(400).json({
|
|
message: "Automatic deployments are disabled for this compose",
|
|
});
|
|
return;
|
|
}
|
|
|
|
const deploymentTitle = extractCommitMessage(req.headers, req.body);
|
|
const deploymentHash = extractHash(req.headers, req.body);
|
|
const sourceType = composeResult.sourceType;
|
|
|
|
if (sourceType === "github") {
|
|
const branchName = extractBranchName(req.headers, req.body);
|
|
const normalizedCommits = req.body?.commits?.flatMap(
|
|
(commit: any) => commit.modified,
|
|
);
|
|
|
|
const shouldDeployPaths = shouldDeploy(
|
|
composeResult.watchPaths,
|
|
normalizedCommits,
|
|
);
|
|
|
|
if (!shouldDeployPaths) {
|
|
res.status(301).json({ message: "Watch Paths Not Match" });
|
|
return;
|
|
}
|
|
|
|
if (!branchName || branchName !== composeResult.branch) {
|
|
res.status(301).json({ message: "Branch Not Match" });
|
|
return;
|
|
}
|
|
} else if (sourceType === "gitlab") {
|
|
const branchName = extractBranchName(req.headers, req.body);
|
|
const normalizedCommits = req.body?.commits?.flatMap(
|
|
(commit: any) => commit.modified,
|
|
);
|
|
|
|
const shouldDeployPaths = shouldDeploy(
|
|
composeResult.watchPaths,
|
|
normalizedCommits,
|
|
);
|
|
|
|
if (!shouldDeployPaths) {
|
|
res.status(301).json({ message: "Watch Paths Not Match" });
|
|
return;
|
|
}
|
|
if (!branchName || branchName !== composeResult.gitlabBranch) {
|
|
res.status(301).json({ message: "Branch Not Match" });
|
|
return;
|
|
}
|
|
} else if (sourceType === "bitbucket") {
|
|
const branchName = extractBranchName(req.headers, req.body);
|
|
if (!branchName || branchName !== composeResult.bitbucketBranch) {
|
|
res.status(301).json({ message: "Branch Not Match" });
|
|
return;
|
|
}
|
|
|
|
const commitedPaths = await extractCommitedPaths(
|
|
req.body,
|
|
composeResult.bitbucketOwner,
|
|
composeResult.bitbucket?.appPassword || "",
|
|
composeResult.bitbucketRepository || "",
|
|
);
|
|
|
|
const shouldDeployPaths = shouldDeploy(
|
|
composeResult.watchPaths,
|
|
commitedPaths,
|
|
);
|
|
|
|
if (!shouldDeployPaths) {
|
|
res.status(301).json({ message: "Watch Paths Not Match" });
|
|
return;
|
|
}
|
|
} else if (sourceType === "git") {
|
|
const branchName = extractBranchName(req.headers, req.body);
|
|
if (!branchName || branchName !== composeResult.customGitBranch) {
|
|
res.status(301).json({ message: "Branch Not Match" });
|
|
return;
|
|
}
|
|
const provider = getProviderByHeader(req.headers);
|
|
let normalizedCommits: string[] = [];
|
|
|
|
if (provider === "github") {
|
|
normalizedCommits = req.body?.commits?.flatMap(
|
|
(commit: any) => commit.modified,
|
|
);
|
|
} else if (provider === "gitlab") {
|
|
normalizedCommits = req.body?.commits?.flatMap(
|
|
(commit: any) => commit.modified,
|
|
);
|
|
} else if (provider === "gitea") {
|
|
normalizedCommits = req.body?.commits?.flatMap(
|
|
(commit: any) => commit.modified,
|
|
);
|
|
}
|
|
|
|
const shouldDeployPaths = shouldDeploy(
|
|
composeResult.watchPaths,
|
|
normalizedCommits,
|
|
);
|
|
|
|
if (!shouldDeployPaths) {
|
|
res.status(301).json({ message: "Watch Paths Not Match" });
|
|
return;
|
|
}
|
|
} else if (sourceType === "gitea") {
|
|
const branchName = extractBranchName(req.headers, req.body);
|
|
|
|
const normalizedCommits = req.body?.commits?.flatMap(
|
|
(commit: any) => commit.modified,
|
|
);
|
|
|
|
const shouldDeployPaths = shouldDeploy(
|
|
composeResult.watchPaths,
|
|
normalizedCommits,
|
|
);
|
|
|
|
if (!shouldDeployPaths) {
|
|
res.status(301).json({ message: "Watch Paths Not Match" });
|
|
return;
|
|
}
|
|
|
|
if (!branchName || branchName !== composeResult.giteaBranch) {
|
|
res.status(301).json({ message: "Branch Not Match" });
|
|
return;
|
|
}
|
|
}
|
|
|
|
try {
|
|
const jobData: DeploymentJob = {
|
|
composeId: composeResult.composeId as string,
|
|
titleLog: deploymentTitle,
|
|
type: "deploy",
|
|
applicationType: "compose",
|
|
descriptionLog: `Hash: ${deploymentHash}`,
|
|
server: !!composeResult.serverId,
|
|
};
|
|
|
|
if (IS_CLOUD && composeResult.serverId) {
|
|
jobData.serverId = composeResult.serverId;
|
|
await deploy(jobData);
|
|
return true;
|
|
}
|
|
await myQueue.add(
|
|
"deployments",
|
|
{ ...jobData },
|
|
{
|
|
removeOnComplete: true,
|
|
removeOnFail: true,
|
|
},
|
|
);
|
|
} catch (error) {
|
|
res.status(400).json({ message: "Error deploying Compose", error });
|
|
return;
|
|
}
|
|
|
|
res.status(200).json({ message: "Compose deployed successfully" });
|
|
} catch (error) {
|
|
console.log(error);
|
|
res.status(400).json({ message: "Error deploying Compose", error });
|
|
}
|
|
}
|