This commit is contained in:
Heidi
2026-05-15 00:27:08 +01:00
parent 1cac33f480
commit ca574601d1
3 changed files with 76 additions and 15 deletions
+59 -15
View File
@@ -11,6 +11,10 @@ const webhookPath = process.env.WEBHOOK_PATH || '/webhook/github';
const webhookSecret = process.env.WEBHOOK_SECRET || '';
const pm2AppName = process.env.PM2_APP_NAME || 'linkweb';
const pm2Command = process.env.PM2_BIN || (process.platform === 'win32' ? 'pm2.cmd' : 'pm2');
const npmCommand = process.env.NPM_BIN || (process.platform === 'win32' ? 'npm.cmd' : 'npm');
const gitCommand = process.env.GIT_BIN || 'git';
let deployInProgress = false;
function verifyGitHubSignature(req) {
if (!webhookSecret) {
@@ -33,27 +37,67 @@ function verifyGitHubSignature(req) {
return received.length === expected.length && crypto.timingSafeEqual(received, expected);
}
function runStep(label, command, args) {
console.log(`[deploy] ${label}`);
return new Promise((resolve, reject) => {
execFile(
command,
args,
{
cwd: __dirname,
maxBuffer: 1024 * 1024 * 10,
windowsHide: true,
},
(error, stdout, stderr) => {
if (stdout) {
console.log(stdout);
}
if (stderr) {
console.error(stderr);
}
if (error) {
reject(error);
return;
}
resolve();
},
);
});
}
async function runDeploy() {
if (deployInProgress) {
console.log('[deploy] Webhook ignored because a deploy is already running.');
return;
}
deployInProgress = true;
try {
await runStep('Pulling latest changes', gitCommand, ['pull', '--ff-only']);
await runStep('Installing dependencies', npmCommand, ['install']);
await runStep('Building site', npmCommand, ['run', 'build']);
await runStep(`Restarting PM2 app "${pm2AppName}"`, pm2Command, ['restart', pm2AppName]);
console.log('[deploy] Complete.');
} catch (error) {
console.error('[deploy] Failed:', error.message);
} finally {
deployInProgress = false;
}
}
app.post(webhookPath, express.raw({ type: '*/*' }), (req, res) => {
if (!verifyGitHubSignature(req)) {
res.status(401).json({ ok: false, error: 'Invalid webhook signature' });
return;
}
res.status(202).json({ ok: true, restarting: pm2AppName });
execFile(pm2Command, ['restart', pm2AppName], (error, stdout, stderr) => {
if (error) {
console.error('PM2 restart failed:', error.message);
if (stderr) {
console.error(stderr);
}
return;
}
if (stdout) {
console.log(stdout);
}
});
res.status(202).json({ ok: true, deploying: pm2AppName });
runDeploy();
});
app.get('/healthz', (_req, res) => {