321 lines
11 KiB
Groovy
321 lines
11 KiB
Groovy
pipeline {
|
||
agent any
|
||
|
||
options {
|
||
buildDiscarder(logRotator(numToKeepStr: '10'))
|
||
timeout(time: 30, unit: 'MINUTES')
|
||
timestamps()
|
||
}
|
||
|
||
environment {
|
||
JAVA_HOME = '/usr/lib/jvm/java-17-openjdk'
|
||
MAVEN_HOME = '/opt/maven'
|
||
PATH = "${MAVEN_HOME}/bin:${JAVA_HOME}/bin:${env.PATH}"
|
||
|
||
// Docker相关环境变量
|
||
DOCKER_REGISTRY = 'your-registry.com'
|
||
IMAGE_NAME = 'jenkins-demo'
|
||
IMAGE_TAG = "${BUILD_NUMBER}"
|
||
|
||
// SonarQube配置
|
||
SONAR_HOST_URL = 'http://your-sonar-server:9000'
|
||
SONAR_PROJECT_KEY = 'jenkins-demo'
|
||
}
|
||
|
||
tools {
|
||
maven 'Maven-3.9.3'
|
||
jdk 'JDK-17'
|
||
}
|
||
|
||
stages {
|
||
stage('Checkout') {
|
||
steps {
|
||
echo '🔄 开始检出代码...'
|
||
checkout scm
|
||
|
||
script {
|
||
env.GIT_COMMIT_SHORT = sh(
|
||
script: "git rev-parse --short HEAD",
|
||
returnStdout: true
|
||
).trim()
|
||
}
|
||
|
||
echo "📋 Git提交ID: ${env.GIT_COMMIT_SHORT}"
|
||
}
|
||
}
|
||
|
||
stage('环境检查') {
|
||
steps {
|
||
echo '🔍 检查构建环境...'
|
||
sh '''
|
||
echo "Java版本:"
|
||
java -version
|
||
echo "Maven版本:"
|
||
mvn -version
|
||
echo "Git版本:"
|
||
git --version
|
||
'''
|
||
}
|
||
}
|
||
|
||
stage('编译') {
|
||
steps {
|
||
echo '🔨 开始编译项目...'
|
||
sh 'mvn clean compile -DskipTests=true'
|
||
}
|
||
}
|
||
|
||
stage('单元测试') {
|
||
steps {
|
||
echo '🧪 运行单元测试...'
|
||
sh 'mvn test'
|
||
}
|
||
post {
|
||
always {
|
||
// 发布测试结果
|
||
publishTestResults testResultsPattern: 'target/surefire-reports/*.xml'
|
||
|
||
// 发布代码覆盖率报告
|
||
step([$class: 'JacocoPublisher',
|
||
execPattern: 'target/jacoco.exec',
|
||
classPattern: 'target/classes',
|
||
sourcePattern: 'src/main/java',
|
||
exclusionPattern: '**/*Test*.class'
|
||
])
|
||
}
|
||
}
|
||
}
|
||
|
||
stage('代码质量扫描') {
|
||
steps {
|
||
echo '🔍 运行SonarQube代码扫描...'
|
||
script {
|
||
try {
|
||
withSonarQubeEnv('SonarQube') {
|
||
sh '''
|
||
mvn sonar:sonar \
|
||
-Dsonar.projectKey=${SONAR_PROJECT_KEY} \
|
||
-Dsonar.host.url=${SONAR_HOST_URL} \
|
||
-Dsonar.coverage.jacoco.xmlReportPaths=target/site/jacoco/jacoco.xml
|
||
'''
|
||
}
|
||
|
||
// 等待质量门检查结果
|
||
timeout(time: 5, unit: 'MINUTES') {
|
||
def qg = waitForQualityGate()
|
||
if (qg.status != 'OK') {
|
||
error "质量门检查失败: ${qg.status}"
|
||
}
|
||
}
|
||
} catch (Exception e) {
|
||
echo "⚠️ SonarQube扫描失败,继续构建流程: ${e.getMessage()}"
|
||
}
|
||
}
|
||
}
|
||
}
|
||
|
||
stage('打包') {
|
||
steps {
|
||
echo '📦 开始打包应用程序...'
|
||
sh 'mvn package -DskipTests=true'
|
||
}
|
||
post {
|
||
success {
|
||
// 归档构建产物
|
||
archiveArtifacts artifacts: 'target/*.jar', fingerprint: true
|
||
}
|
||
}
|
||
}
|
||
|
||
stage('构建Docker镜像') {
|
||
steps {
|
||
echo '🐳 构建Docker镜像...'
|
||
script {
|
||
def image = docker.build("${IMAGE_NAME}:${IMAGE_TAG}")
|
||
|
||
// 也创建latest标签
|
||
sh "docker tag ${IMAGE_NAME}:${IMAGE_TAG} ${IMAGE_NAME}:latest"
|
||
|
||
echo "✅ Docker镜像构建完成: ${IMAGE_NAME}:${IMAGE_TAG}"
|
||
}
|
||
}
|
||
}
|
||
|
||
stage('推送Docker镜像') {
|
||
when {
|
||
anyOf {
|
||
branch 'main'
|
||
branch 'master'
|
||
branch 'develop'
|
||
}
|
||
}
|
||
steps {
|
||
echo '📤 推送Docker镜像到仓库...'
|
||
script {
|
||
docker.withRegistry("https://${DOCKER_REGISTRY}", 'docker-registry-credentials') {
|
||
def image = docker.image("${IMAGE_NAME}:${IMAGE_TAG}")
|
||
image.push()
|
||
image.push("latest")
|
||
}
|
||
echo "✅ Docker镜像推送完成"
|
||
}
|
||
}
|
||
}
|
||
|
||
stage('部署到测试环境') {
|
||
when {
|
||
anyOf {
|
||
branch 'develop'
|
||
branch 'feature/*'
|
||
}
|
||
}
|
||
steps {
|
||
echo '🚀 部署到测试环境...'
|
||
script {
|
||
// 部署到测试服务器
|
||
sshagent(['test-server-ssh']) {
|
||
sh '''
|
||
ssh -o StrictHostKeyChecking=no user@test-server << EOF
|
||
# 停止现有容器
|
||
docker stop jenkins-demo-test || true
|
||
docker rm jenkins-demo-test || true
|
||
|
||
# 运行新容器
|
||
docker run -d --name jenkins-demo-test \\
|
||
-p 8080:8080 \\
|
||
--restart unless-stopped \\
|
||
${IMAGE_NAME}:${IMAGE_TAG}
|
||
|
||
echo "✅ 测试环境部署完成"
|
||
EOF
|
||
'''
|
||
}
|
||
}
|
||
}
|
||
}
|
||
|
||
stage('部署到生产环境') {
|
||
when {
|
||
anyOf {
|
||
branch 'main'
|
||
branch 'master'
|
||
}
|
||
}
|
||
steps {
|
||
echo '🎯 部署到生产环境...'
|
||
script {
|
||
// 需要手动确认
|
||
input message: '确认部署到生产环境?', ok: '部署',
|
||
parameters: [choice(name: 'DEPLOY_ENV', choices: ['prod'], description: '选择部署环境')]
|
||
|
||
// 部署到生产服务器
|
||
sshagent(['prod-server-ssh']) {
|
||
sh '''
|
||
ssh -o StrictHostKeyChecking=no user@prod-server << EOF
|
||
# 备份当前版本
|
||
docker tag ${IMAGE_NAME}:latest ${IMAGE_NAME}:backup-$(date +%Y%m%d-%H%M%S) || true
|
||
|
||
# 停止现有容器
|
||
docker stop jenkins-demo-prod || true
|
||
docker rm jenkins-demo-prod || true
|
||
|
||
# 运行新容器
|
||
docker run -d --name jenkins-demo-prod \\
|
||
-p 80:8080 \\
|
||
--restart unless-stopped \\
|
||
-e SPRING_PROFILES_ACTIVE=prod \\
|
||
${IMAGE_NAME}:${IMAGE_TAG}
|
||
|
||
echo "✅ 生产环境部署完成"
|
||
EOF
|
||
'''
|
||
}
|
||
}
|
||
}
|
||
}
|
||
|
||
stage('健康检查') {
|
||
steps {
|
||
echo '🏥 执行应用健康检查...'
|
||
script {
|
||
// 等待应用启动
|
||
sleep(time: 30, unit: 'SECONDS')
|
||
|
||
// 检查应用健康状态
|
||
def healthCheckUrl = "http://localhost:8080/api/health"
|
||
def response = sh(
|
||
script: "curl -s -o /dev/null -w '%{http_code}' ${healthCheckUrl}",
|
||
returnStdout: true
|
||
).trim()
|
||
|
||
if (response == "200") {
|
||
echo "✅ 应用健康检查通过"
|
||
} else {
|
||
error "❌ 应用健康检查失败,HTTP状态码: ${response}"
|
||
}
|
||
}
|
||
}
|
||
}
|
||
}
|
||
|
||
post {
|
||
always {
|
||
echo '🧹 清理工作空间...'
|
||
// 清理Docker镜像
|
||
sh '''
|
||
docker image prune -f
|
||
docker system prune -f
|
||
'''
|
||
}
|
||
|
||
success {
|
||
echo '✅ 流水线执行成功!'
|
||
// 发送成功通知
|
||
script {
|
||
def message = """
|
||
🎉 Jenkins构建成功!
|
||
|
||
📋 项目: ${env.JOB_NAME}
|
||
🔢 构建号: ${env.BUILD_NUMBER}
|
||
🌿 分支: ${env.BRANCH_NAME}
|
||
📝 提交: ${env.GIT_COMMIT_SHORT}
|
||
⏱️ 持续时间: ${currentBuild.durationString}
|
||
🔗 构建链接: ${env.BUILD_URL}
|
||
"""
|
||
|
||
// 可以集成钉钉、企业微信、邮件等通知
|
||
echo message
|
||
}
|
||
}
|
||
|
||
failure {
|
||
echo '❌ 流水线执行失败!'
|
||
// 发送失败通知
|
||
script {
|
||
def message = """
|
||
💥 Jenkins构建失败!
|
||
|
||
📋 项目: ${env.JOB_NAME}
|
||
🔢 构建号: ${env.BUILD_NUMBER}
|
||
🌿 分支: ${env.BRANCH_NAME}
|
||
📝 提交: ${env.GIT_COMMIT_SHORT}
|
||
⏱️ 持续时间: ${currentBuild.durationString}
|
||
🔗 构建链接: ${env.BUILD_URL}
|
||
📄 查看日志: ${env.BUILD_URL}console
|
||
"""
|
||
|
||
echo message
|
||
}
|
||
}
|
||
|
||
unstable {
|
||
echo '⚠️ 构建不稳定,可能存在测试失败'
|
||
}
|
||
|
||
cleanup {
|
||
// 清理工作空间
|
||
cleanWs()
|
||
}
|
||
}
|
||
}
|