2026-01-14 06:33:58 +00:00
|
|
|
|
pipeline {
|
|
|
|
|
|
agent any
|
|
|
|
|
|
|
|
|
|
|
|
environment {
|
|
|
|
|
|
// 远程服务器配置
|
|
|
|
|
|
DEPLOY_HOST = '180.76.180.105'
|
|
|
|
|
|
DEPLOY_USER = 'root' // 根据实际情况修改用户名
|
|
|
|
|
|
DEPLOY_PATH = '/opt/nginx/html/ai'
|
|
|
|
|
|
// 如果需要 SSH 密钥,可以在 Jenkins 中配置 SSH credentials
|
|
|
|
|
|
// SSH_CREDENTIALS = credentials('deploy-ssh-key')
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
stages {
|
|
|
|
|
|
stage('Checkout') {
|
|
|
|
|
|
steps {
|
|
|
|
|
|
echo '检出代码...'
|
|
|
|
|
|
checkout scm
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2026-01-14 07:21:22 +00:00
|
|
|
|
stage('Build') {
|
2026-01-14 06:33:58 +00:00
|
|
|
|
steps {
|
2026-01-16 03:56:51 +00:00
|
|
|
|
echo '构建项目...'
|
|
|
|
|
|
sh '''
|
2026-01-16 05:37:50 +00:00
|
|
|
|
# 加载 nvm(支持多个可能的安装路径)
|
|
|
|
|
|
export NVM_DIR="$HOME/.nvm"
|
|
|
|
|
|
if [ -s "$NVM_DIR/nvm.sh" ]; then
|
2026-01-16 05:41:33 +00:00
|
|
|
|
. "$NVM_DIR/nvm.sh"
|
2026-01-16 05:37:50 +00:00
|
|
|
|
elif [ -s "/usr/local/opt/nvm/nvm.sh" ]; then
|
2026-01-16 05:41:33 +00:00
|
|
|
|
. "/usr/local/opt/nvm/nvm.sh"
|
2026-01-16 05:37:50 +00:00
|
|
|
|
elif [ -s "/opt/homebrew/opt/nvm/nvm.sh" ]; then
|
2026-01-16 05:41:33 +00:00
|
|
|
|
. "/opt/homebrew/opt/nvm/nvm.sh"
|
2026-01-16 05:37:50 +00:00
|
|
|
|
fi
|
|
|
|
|
|
|
|
|
|
|
|
# 使用 Node.js 22
|
|
|
|
|
|
nvm use 22 2>/dev/null || nvm use default || true
|
|
|
|
|
|
|
2026-01-16 05:45:09 +00:00
|
|
|
|
# 确保 PATH 包含 nvm 的 node 路径
|
|
|
|
|
|
NVM_NODE_VERSION=$(nvm version 2>/dev/null || echo "v22.22.0")
|
|
|
|
|
|
export PATH="$NVM_DIR/versions/node/$NVM_NODE_VERSION/bin:$PATH"
|
|
|
|
|
|
|
|
|
|
|
|
# 检查 Node.js(直接检查文件是否存在)
|
|
|
|
|
|
if [ ! -f "$NVM_DIR/versions/node/$NVM_NODE_VERSION/bin/node" ]; then
|
2026-01-16 03:56:51 +00:00
|
|
|
|
echo "错误: Node.js 未安装,请安装 Node.js 22+"
|
2026-01-16 05:37:50 +00:00
|
|
|
|
echo "提示: 如果使用 nvm,请确保已安装: nvm install 22"
|
2026-01-16 03:56:51 +00:00
|
|
|
|
exit 1
|
|
|
|
|
|
fi
|
|
|
|
|
|
|
|
|
|
|
|
echo "Node 版本: $(node --version)"
|
|
|
|
|
|
echo "NPM 版本: $(npm --version)"
|
2026-01-16 03:43:45 +00:00
|
|
|
|
|
2026-01-16 03:56:51 +00:00
|
|
|
|
# 安装依赖
|
|
|
|
|
|
echo "安装依赖..."
|
|
|
|
|
|
npm install
|
|
|
|
|
|
npm run install:all
|
2026-01-14 07:21:22 +00:00
|
|
|
|
|
2026-01-16 03:56:51 +00:00
|
|
|
|
# 构建前端
|
|
|
|
|
|
echo "构建前端..."
|
|
|
|
|
|
npm run build --workspace=frontend
|
2026-01-14 07:21:22 +00:00
|
|
|
|
|
2026-01-16 03:56:51 +00:00
|
|
|
|
# 构建后端
|
|
|
|
|
|
echo "构建后端..."
|
|
|
|
|
|
cd backend
|
2026-01-16 05:56:25 +00:00
|
|
|
|
# 先生成 Prisma Client(构建时需要类型)
|
2026-01-16 03:56:51 +00:00
|
|
|
|
npm run prisma:generate
|
2026-01-16 05:56:25 +00:00
|
|
|
|
# 然后构建 TypeScript
|
|
|
|
|
|
npm run build
|
2026-01-16 03:56:51 +00:00
|
|
|
|
cd ..
|
|
|
|
|
|
'''
|
2026-01-14 06:33:58 +00:00
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
stage('Prepare Deployment Package') {
|
|
|
|
|
|
steps {
|
|
|
|
|
|
echo '准备部署包...'
|
|
|
|
|
|
sh '''
|
|
|
|
|
|
# 创建临时部署目录
|
|
|
|
|
|
mkdir -p deploy-package
|
|
|
|
|
|
|
|
|
|
|
|
# 复制前端构建产物
|
|
|
|
|
|
cp -r frontend/dist deploy-package/frontend-dist
|
|
|
|
|
|
|
2026-01-16 03:43:45 +00:00
|
|
|
|
# 复制后端文件
|
2026-01-14 06:33:58 +00:00
|
|
|
|
mkdir -p deploy-package/backend
|
2026-01-16 06:00:57 +00:00
|
|
|
|
cp -r backend/dist deploy-package/backend/ 2>/dev/null || true
|
|
|
|
|
|
cp -r backend/prisma deploy-package/backend/ 2>/dev/null || true
|
|
|
|
|
|
cp backend/package.json deploy-package/backend/ 2>/dev/null || true
|
|
|
|
|
|
cp backend/Dockerfile deploy-package/backend/ 2>/dev/null || true
|
|
|
|
|
|
cp backend/entrypoint.sh deploy-package/backend/ 2>/dev/null || true
|
2026-01-14 06:33:58 +00:00
|
|
|
|
cp backend/package-lock.json deploy-package/backend/ 2>/dev/null || true
|
2026-01-16 03:43:45 +00:00
|
|
|
|
chmod +x deploy-package/backend/entrypoint.sh 2>/dev/null || true
|
2026-01-14 06:33:58 +00:00
|
|
|
|
|
2026-01-16 03:43:45 +00:00
|
|
|
|
# 复制 shared 包和 Docker 配置
|
2026-01-16 06:00:57 +00:00
|
|
|
|
if [ -d shared ]; then
|
|
|
|
|
|
cp -r shared deploy-package/
|
|
|
|
|
|
fi
|
2026-01-14 06:33:58 +00:00
|
|
|
|
mkdir -p deploy-package/docker
|
2026-01-16 06:00:57 +00:00
|
|
|
|
cp nginx/docker-compose.production.yml deploy-package/docker/ 2>/dev/null || true
|
|
|
|
|
|
cp nginx/nginx.conf.docker deploy-package/docker/ 2>/dev/null || true
|
|
|
|
|
|
cp scripts/deploy-docker.sh deploy-package/ 2>/dev/null || true
|
|
|
|
|
|
chmod +x deploy-package/deploy-docker.sh 2>/dev/null || true
|
2026-01-14 06:33:58 +00:00
|
|
|
|
|
|
|
|
|
|
# 创建部署包
|
|
|
|
|
|
tar -czf deploy-package.tar.gz deploy-package/
|
|
|
|
|
|
'''
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2026-01-16 03:43:45 +00:00
|
|
|
|
stage('Deploy') {
|
2026-01-14 06:33:58 +00:00
|
|
|
|
steps {
|
|
|
|
|
|
echo '部署到远程服务器...'
|
|
|
|
|
|
sh '''
|
2026-01-16 03:43:45 +00:00
|
|
|
|
# 传输部署包
|
|
|
|
|
|
scp -o StrictHostKeyChecking=no deploy-package.tar.gz ${DEPLOY_USER}@${DEPLOY_HOST}:/tmp/
|
|
|
|
|
|
|
|
|
|
|
|
# 在远程服务器上执行部署(deploy-docker.sh 已包含重启服务逻辑)
|
2026-01-16 06:44:52 +00:00
|
|
|
|
ssh -o StrictHostKeyChecking=no ${DEPLOY_USER}@${DEPLOY_HOST} bash << 'ENDSSH'
|
2026-01-16 06:40:38 +00:00
|
|
|
|
# 移除 set -e,改为手动检查关键命令
|
2026-01-16 03:43:45 +00:00
|
|
|
|
mkdir -p /opt/nginx/html/ai
|
|
|
|
|
|
cd /tmp && tar -xzf deploy-package.tar.gz
|
2026-01-14 06:33:58 +00:00
|
|
|
|
|
2026-01-16 03:43:45 +00:00
|
|
|
|
# 备份旧版本
|
2026-01-16 06:40:38 +00:00
|
|
|
|
if [ -d /opt/nginx/html/ai/current ]; then
|
2026-01-16 03:43:45 +00:00
|
|
|
|
mv /opt/nginx/html/ai/current /opt/nginx/html/ai/backup-$(date +%Y%m%d-%H%M%S)
|
2026-01-16 06:40:38 +00:00
|
|
|
|
fi
|
2026-01-14 06:33:58 +00:00
|
|
|
|
|
2026-01-16 03:43:45 +00:00
|
|
|
|
# 创建新版本目录并复制文件
|
|
|
|
|
|
mkdir -p /opt/nginx/html/ai/current/{backend,docker/logs}
|
|
|
|
|
|
cp -r deploy-package/frontend-dist/* /opt/nginx/html/ai/current/
|
|
|
|
|
|
cp -r deploy-package/backend/* /opt/nginx/html/ai/current/backend/
|
2026-01-16 06:40:38 +00:00
|
|
|
|
if [ -d deploy-package/shared ]; then
|
|
|
|
|
|
cp -r deploy-package/shared /opt/nginx/html/ai/current/
|
|
|
|
|
|
fi
|
2026-01-16 06:13:44 +00:00
|
|
|
|
# 复制 Docker 配置文件,确保文件名为 docker-compose.yml(podman-compose 需要)
|
2026-01-16 06:40:38 +00:00
|
|
|
|
if [ -f deploy-package/docker/docker-compose.production.yml ]; then
|
|
|
|
|
|
cp deploy-package/docker/docker-compose.production.yml /opt/nginx/html/ai/current/docker/docker-compose.yml
|
|
|
|
|
|
else
|
|
|
|
|
|
cp deploy-package/docker/* /opt/nginx/html/ai/current/docker/
|
|
|
|
|
|
fi
|
|
|
|
|
|
if [ -f deploy-package/docker/nginx.conf.docker ]; then
|
|
|
|
|
|
cp deploy-package/docker/nginx.conf.docker /opt/nginx/html/ai/current/docker/
|
|
|
|
|
|
fi
|
2026-01-14 06:33:58 +00:00
|
|
|
|
|
2026-01-16 03:43:45 +00:00
|
|
|
|
# 执行部署脚本(包含停止、构建、启动和健康检查)
|
|
|
|
|
|
chmod +x deploy-package/deploy-docker.sh
|
2026-01-16 06:44:52 +00:00
|
|
|
|
DEPLOY_RESULT=0
|
|
|
|
|
|
bash deploy-package/deploy-docker.sh /opt/nginx/html/ai/current || DEPLOY_RESULT=$?
|
|
|
|
|
|
|
|
|
|
|
|
if [ $DEPLOY_RESULT -eq 0 ]; then
|
2026-01-16 06:40:38 +00:00
|
|
|
|
echo "部署脚本执行成功"
|
|
|
|
|
|
else
|
2026-01-16 06:44:52 +00:00
|
|
|
|
echo "部署脚本执行失败,退出码: $DEPLOY_RESULT"
|
|
|
|
|
|
exit $DEPLOY_RESULT
|
2026-01-16 06:40:38 +00:00
|
|
|
|
fi
|
2026-01-14 06:33:58 +00:00
|
|
|
|
|
2026-01-16 03:43:45 +00:00
|
|
|
|
# 清理临时文件
|
2026-01-16 06:44:52 +00:00
|
|
|
|
rm -rf /tmp/deploy-package /tmp/deploy-package.tar.gz || true
|
|
|
|
|
|
ENDSSH
|
2026-01-14 06:33:58 +00:00
|
|
|
|
'''
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
post {
|
|
|
|
|
|
success {
|
|
|
|
|
|
echo '部署成功!'
|
|
|
|
|
|
}
|
|
|
|
|
|
failure {
|
|
|
|
|
|
echo '部署失败!'
|
|
|
|
|
|
}
|
|
|
|
|
|
always {
|
|
|
|
|
|
// 清理本地临时文件
|
|
|
|
|
|
sh 'rm -rf deploy-package deploy-package.tar.gz'
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|