ai_learn_node/Jenkinsfile

152 lines
6.3 KiB
Plaintext
Raw Normal View History

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
}
}
stage('Build') {
steps {
script {
2026-01-16 03:43:45 +00:00
// 定义构建命令(公共部分)
def buildCommands = '''
echo "Node 版本: $(node --version)"
echo "NPM 版本: $(npm --version)"
echo "安装依赖..."
npm install
npm run install:all
echo "构建前端..."
npm run build --workspace=frontend
echo "构建后端..."
cd backend && npm run build && npm run prisma:generate && cd ..
'''
// 尝试使用 Docker 构建,失败则回退到主机构建
def useDocker = false
try {
def dockerCheck = sh(
2026-01-16 03:43:45 +00:00
script: 'command -v docker >/dev/null 2>&1 && docker --version',
returnStatus: true
)
if (dockerCheck == 0) {
2026-01-16 03:43:45 +00:00
useDocker = true
}
} catch (Exception e) {
2026-01-16 03:43:45 +00:00
echo "Docker 不可用,使用主机构建"
}
2026-01-16 03:43:45 +00:00
if (useDocker) {
try {
2026-01-16 03:43:45 +00:00
docker.image("node:18").inside('-v /var/run/docker.sock:/var/run/docker.sock') {
sh buildCommands
}
} catch (Exception e) {
2026-01-16 03:43:45 +00:00
echo "Docker 构建失败,回退到主机构建: ${e.getMessage()}"
useDocker = false
}
}
2026-01-16 03:43:45 +00:00
if (!useDocker) {
sh '''
if ! command -v node &> /dev/null; then
echo "错误: Node.js 未安装,请安装 Node.js 18+"
exit 1
fi
2026-01-16 03:43:45 +00:00
''' + buildCommands
}
}
}
}
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
# 复制后端文件
mkdir -p deploy-package/backend
2026-01-16 03:43:45 +00:00
cp -r backend/{dist,prisma} deploy-package/backend/
cp backend/{package.json,Dockerfile,entrypoint.sh} deploy-package/backend/ 2>/dev/null || true
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-16 03:43:45 +00:00
# 复制 shared 包和 Docker 配置
[ -d shared ] && cp -r shared deploy-package/
mkdir -p deploy-package/docker
2026-01-16 03:43:45 +00:00
cp nginx/{docker-compose.production.yml,nginx.conf.docker} deploy-package/docker/
cp scripts/deploy-docker.sh deploy-package/ && chmod +x deploy-package/deploy-docker.sh
# 创建部署包
tar -czf deploy-package.tar.gz deploy-package/
'''
}
}
2026-01-16 03:43:45 +00:00
stage('Deploy') {
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 已包含重启服务逻辑)
ssh -o StrictHostKeyChecking=no ${DEPLOY_USER}@${DEPLOY_HOST} << 'ENDSSH'
2026-01-16 03:43:45 +00:00
set -e
mkdir -p /opt/nginx/html/ai
cd /tmp && tar -xzf deploy-package.tar.gz
2026-01-16 03:43:45 +00:00
# 备份旧版本
[ -d /opt/nginx/html/ai/current ] && \
mv /opt/nginx/html/ai/current /opt/nginx/html/ai/backup-$(date +%Y%m%d-%H%M%S)
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/
[ -d deploy-package/shared ] && cp -r deploy-package/shared /opt/nginx/html/ai/current/
cp deploy-package/docker/* /opt/nginx/html/ai/current/docker/
2026-01-16 03:43:45 +00:00
# 执行部署脚本(包含停止、构建、启动和健康检查)
chmod +x deploy-package/deploy-docker.sh
bash deploy-package/deploy-docker.sh /opt/nginx/html/ai/current
2026-01-16 03:43:45 +00:00
# 清理临时文件
rm -rf /tmp/deploy-package /tmp/deploy-package.tar.gz
ENDSSH
'''
}
}
}
post {
success {
echo '部署成功!'
}
failure {
echo '部署失败!'
}
always {
// 清理本地临时文件
sh 'rm -rf deploy-package deploy-package.tar.gz'
}
}
}