ai_learn_node/Jenkinsfile

196 lines
8.5 KiB
Plaintext
Raw Normal View History

pipeline {
agent any
2026-01-20 05:21:43 +00:00
tools {
2026-01-20 05:24:34 +00:00
nodejs 'NodeJS-22'
}
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-20 05:35:05 +00:00
stage('Test Node.js') {
steps {
2026-01-20 05:35:05 +00:00
echo '测试 Node.js 环境...'
2026-01-16 03:56:51 +00:00
sh '''
2026-01-20 05:35:05 +00:00
# 检查 Node.js 是否安装
2026-01-20 05:13:38 +00:00
if ! command -v node &> /dev/null; then
2026-01-20 05:35:05 +00:00
echo "❌ 错误: Node.js 未安装,请安装 Node.js 22+"
exit 1
fi
# 获取版本信息
NODE_VERSION=$(node --version)
NPM_VERSION=$(npm --version)
NODE_PATH=$(which node)
NPM_PATH=$(which npm)
# 显示版本信息
echo "=========================================="
echo "Node.js 环境信息"
echo "=========================================="
echo "Node 版本: $NODE_VERSION"
echo "NPM 版本: $NPM_VERSION"
echo "Node 路径: $NODE_PATH"
echo "NPM 路径: $NPM_PATH"
echo "=========================================="
# 检查 Node.js 版本是否符合要求22+
NODE_MAJOR_VERSION=$(echo $NODE_VERSION | sed 's/v//' | cut -d. -f1)
if [ "$NODE_MAJOR_VERSION" -lt 22 ]; then
echo "❌ 错误: Node.js 版本过低,需要 22+,当前版本: $NODE_VERSION"
2026-01-16 03:56:51 +00:00
exit 1
fi
2026-01-20 05:35:05 +00:00
echo "✅ Node.js 版本检查通过: $NODE_VERSION"
2026-01-16 03:43:45 +00:00
2026-01-20 05:35:05 +00:00
# 测试基本命令
echo "测试 Node.js 和 NPM 命令..."
node --version > /dev/null && echo "✅ node 命令正常" || (echo "❌ node 命令失败" && exit 1)
npm --version > /dev/null && echo "✅ npm 命令正常" || (echo "❌ npm 命令失败" && exit 1)
'''
}
}
stage('Build') {
steps {
echo '构建项目...'
sh '''
2026-01-16 03:56:51 +00:00
# 安装依赖
echo "安装依赖..."
npm install
npm run install:all
2026-01-16 03:56:51 +00:00
# 构建前端
echo "构建前端..."
npm run build --workspace=frontend
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 ..
'''
}
}
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 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
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 配置
2026-01-16 06:00:57 +00:00
if [ -d shared ]; then
cp -r shared deploy-package/
fi
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
# 创建部署包
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 已包含重启服务逻辑)
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-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-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.ymlpodman-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-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-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
'''
}
}
}
post {
success {
echo '部署成功!'
}
failure {
echo '部署失败!'
}
always {
// 清理本地临时文件
sh 'rm -rf deploy-package deploy-package.tar.gz'
}
}
}