2026-04-24 20:54:03 +08:00
|
|
|
|
pipeline {
|
2026-04-25 06:35:58 +08:00
|
|
|
|
agent any
|
2026-04-24 20:54:03 +08:00
|
|
|
|
|
|
|
|
|
|
parameters {
|
2026-05-09 16:22:13 +08:00
|
|
|
|
choice(name: 'APP', choices: ['tenant-platform', 'ops-platform'], description: '要构建的 Web 应用')
|
2026-04-24 20:54:03 +08:00
|
|
|
|
booleanParam(name: 'DEPLOY', defaultValue: true, description: '构建后是否自动部署')
|
2026-06-11 15:11:43 +08:00
|
|
|
|
booleanParam(name: 'NO_CACHE', defaultValue: false, description: '禁用 Docker 构建缓存(缓存错误时使用)')
|
2026-04-24 20:54:03 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
environment {
|
2026-05-16 11:31:21 +08:00
|
|
|
|
ACR_REGISTRY = 'crpi-n44qjpuucgjt8e8c.cn-beijing.personal.cr.aliyuncs.com'
|
|
|
|
|
|
ACR_NAMESPACE = 'xuqmgroup'
|
|
|
|
|
|
ACR_USERNAME = 'xuqinmin12'
|
|
|
|
|
|
PROD_HOST = '106.54.23.149'
|
|
|
|
|
|
PROD_USER = 'ubuntu'
|
|
|
|
|
|
COMPOSE_FILE = '/opt/xuqm/deploy/compose.production.yaml'
|
2026-06-16 19:33:30 +08:00
|
|
|
|
VERSIONS_FILE = '/opt/xuqm/deploy/versions.json'
|
2026-05-08 18:32:00 +08:00
|
|
|
|
DOCKER_BUILDKIT = '1'
|
2026-04-24 20:54:03 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
2026-05-16 11:31:21 +08:00
|
|
|
|
options {
|
|
|
|
|
|
timeout(time: 20, unit: 'MINUTES')
|
|
|
|
|
|
buildDiscarder(logRotator(numToKeepStr: '20'))
|
|
|
|
|
|
disableConcurrentBuilds()
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2026-04-24 20:54:03 +08:00
|
|
|
|
stages {
|
|
|
|
|
|
stage('Checkout') {
|
2026-05-15 23:56:46 +08:00
|
|
|
|
steps {
|
2026-05-16 11:31:21 +08:00
|
|
|
|
checkout([
|
|
|
|
|
|
$class: 'GitSCM',
|
|
|
|
|
|
branches: [[name: 'main']],
|
|
|
|
|
|
extensions: [[$class: 'CleanBeforeCheckout']],
|
2026-05-15 23:56:46 +08:00
|
|
|
|
userRemoteConfigs: scm.userRemoteConfigs
|
|
|
|
|
|
])
|
|
|
|
|
|
}
|
2026-04-24 20:54:03 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
2026-05-09 16:22:13 +08:00
|
|
|
|
stage('Resolve Build Plan') {
|
|
|
|
|
|
steps {
|
|
|
|
|
|
script {
|
|
|
|
|
|
switch (params.APP) {
|
|
|
|
|
|
case 'tenant-platform':
|
2026-05-16 11:31:21 +08:00
|
|
|
|
env.IMAGE_NAME = 'tenant-web'
|
|
|
|
|
|
env.DOCKERFILE = 'Dockerfile.tenant'
|
2026-05-09 16:22:13 +08:00
|
|
|
|
env.DEPLOY_SERVICE = 'tenant-web'
|
2026-06-16 19:33:30 +08:00
|
|
|
|
env.VERSION_FILE = 'VERSION.tenant-web'
|
2026-05-16 11:31:21 +08:00
|
|
|
|
env.BUILD_ARGS = '--build-arg TENANT_APP_BASE=/ --build-arg TENANT_API_BASE_URL=/api'
|
2026-05-09 16:22:13 +08:00
|
|
|
|
break
|
|
|
|
|
|
case 'ops-platform':
|
2026-05-16 11:31:21 +08:00
|
|
|
|
env.IMAGE_NAME = 'ops-web'
|
|
|
|
|
|
env.DOCKERFILE = 'Dockerfile.ops'
|
2026-05-09 16:22:13 +08:00
|
|
|
|
env.DEPLOY_SERVICE = 'ops-web'
|
2026-06-16 19:33:30 +08:00
|
|
|
|
env.VERSION_FILE = 'VERSION.ops-web'
|
2026-05-16 11:31:21 +08:00
|
|
|
|
env.BUILD_ARGS = '--build-arg OPS_APP_BASE=/ --build-arg OPS_API_BASE_URL=/api'
|
2026-05-09 16:22:13 +08:00
|
|
|
|
break
|
|
|
|
|
|
default:
|
|
|
|
|
|
error("Unsupported APP: ${params.APP}")
|
|
|
|
|
|
}
|
2026-06-16 19:33:30 +08:00
|
|
|
|
|
|
|
|
|
|
// 自动递增版本号(与后端 Jenkinsfile 保持一致)
|
|
|
|
|
|
def vf = env.VERSION_FILE
|
|
|
|
|
|
def current = fileExists(vf) ? readFile(vf).trim() : '1.0.0'
|
|
|
|
|
|
def parts = current.tokenize('.')
|
|
|
|
|
|
while (parts.size() < 3) parts.add('0')
|
|
|
|
|
|
env.SERVICE_VERSION = "${parts[0]}.${parts[1]}.${parts[2].toInteger() + 1}"
|
|
|
|
|
|
writeFile file: vf, text: env.SERVICE_VERSION
|
|
|
|
|
|
echo "${env.IMAGE_NAME}: ${current} → ${env.SERVICE_VERSION}"
|
2026-05-09 16:22:13 +08:00
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2026-04-24 20:54:03 +08:00
|
|
|
|
stage('Docker Build & Push') {
|
|
|
|
|
|
steps {
|
|
|
|
|
|
withCredentials([string(credentialsId: 'ACR_PASSWORD', variable: 'ACR_PASS')]) {
|
|
|
|
|
|
script {
|
2026-06-16 19:33:30 +08:00
|
|
|
|
def versionedImage = "${env.ACR_REGISTRY}/${env.ACR_NAMESPACE}/${env.IMAGE_NAME}:${env.SERVICE_VERSION}"
|
|
|
|
|
|
def latestImage = "${env.ACR_REGISTRY}/${env.ACR_NAMESPACE}/${env.IMAGE_NAME}:latest"
|
2026-06-11 15:11:43 +08:00
|
|
|
|
def cacheArgs = params.NO_CACHE
|
|
|
|
|
|
? '--no-cache'
|
2026-06-16 19:33:30 +08:00
|
|
|
|
: "--build-arg BUILDKIT_INLINE_CACHE=1 --cache-from ${latestImage}"
|
2026-06-11 15:11:43 +08:00
|
|
|
|
def gitCommit = env.GIT_COMMIT ?: 'unknown'
|
2026-04-25 06:50:06 +08:00
|
|
|
|
bat """
|
2026-05-09 16:22:13 +08:00
|
|
|
|
docker login ${env.ACR_REGISTRY} -u ${env.ACR_USERNAME} -p %ACR_PASS%
|
2026-06-11 15:20:16 +08:00
|
|
|
|
if %errorlevel% neq 0 exit /b 1
|
2026-06-16 19:33:30 +08:00
|
|
|
|
docker pull --platform=linux/amd64 ${latestImage} || echo Pull failed, will build fresh
|
|
|
|
|
|
docker build --platform=linux/amd64 -f ${env.DOCKERFILE} ${env.BUILD_ARGS} --build-arg GIT_COMMIT=${gitCommit} --build-arg SERVICE_VERSION=${env.SERVICE_VERSION} ${cacheArgs} -t ${versionedImage} -t ${latestImage} .
|
|
|
|
|
|
if %errorlevel% neq 0 exit /b 1
|
|
|
|
|
|
docker push ${versionedImage}
|
2026-06-11 15:20:16 +08:00
|
|
|
|
if %errorlevel% neq 0 exit /b 1
|
2026-06-16 19:33:30 +08:00
|
|
|
|
docker push ${latestImage}
|
2026-06-11 15:20:16 +08:00
|
|
|
|
if %errorlevel% neq 0 exit /b 1
|
2026-06-16 19:33:30 +08:00
|
|
|
|
docker rmi ${versionedImage} ${latestImage}
|
2026-06-11 15:20:16 +08:00
|
|
|
|
exit /b 0
|
2026-04-24 20:54:03 +08:00
|
|
|
|
"""
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
stage('Deploy to Production') {
|
|
|
|
|
|
when { expression { return params.DEPLOY } }
|
|
|
|
|
|
steps {
|
2026-06-11 15:16:37 +08:00
|
|
|
|
lock('prod-deploy') {
|
|
|
|
|
|
withCredentials([sshUserPrivateKey(credentialsId: 'PROD_SSH_KEY', keyFileVariable: 'SSH_KEY')]) {
|
|
|
|
|
|
script {
|
2026-06-16 19:33:30 +08:00
|
|
|
|
def latestImage = "${env.ACR_REGISTRY}/${env.ACR_NAMESPACE}/${env.IMAGE_NAME}:latest"
|
2026-06-11 15:16:37 +08:00
|
|
|
|
retry(2) {
|
|
|
|
|
|
bat """
|
2026-06-16 19:33:30 +08:00
|
|
|
|
ssh -i "%SSH_KEY%" -o StrictHostKeyChecking=no ${env.PROD_USER}@${env.PROD_HOST} "docker image prune -f 2>/dev/null || true; docker pull ${latestImage} || exit 1; docker compose -f ${env.COMPOSE_FILE} up -d --no-deps --force-recreate ${env.DEPLOY_SERVICE} || exit 1; docker image prune -f"
|
2026-06-11 15:16:37 +08:00
|
|
|
|
"""
|
|
|
|
|
|
}
|
2026-06-16 19:33:30 +08:00
|
|
|
|
|
|
|
|
|
|
// 更新 versions.json(与后端 Jenkinsfile 保持一致的合并逻辑)
|
|
|
|
|
|
def svcName = env.DEPLOY_SERVICE
|
|
|
|
|
|
def svcVer = env.SERVICE_VERSION
|
|
|
|
|
|
def releasedAt = new Date().format("yyyy-MM-dd'T'HH:mm:ss'Z'", TimeZone.getTimeZone('UTC'))
|
|
|
|
|
|
def updateScript = """\
|
|
|
|
|
|
import json, os
|
|
|
|
|
|
path = '${env.VERSIONS_FILE}'
|
|
|
|
|
|
d = json.load(open(path)) if os.path.exists(path) else {}
|
|
|
|
|
|
d.setdefault('services', {})
|
|
|
|
|
|
d['services'].setdefault('${svcName}', {})
|
|
|
|
|
|
d['services']['${svcName}']['version'] = '${svcVer}'
|
|
|
|
|
|
d['services']['${svcName}']['changed'] = True
|
|
|
|
|
|
d['releasedAt'] = '${releasedAt}'
|
|
|
|
|
|
json.dump(d, open(path, 'w'), indent=2, ensure_ascii=False)
|
|
|
|
|
|
print('versions.json updated: ${svcName}=${svcVer}')
|
|
|
|
|
|
""".stripIndent()
|
|
|
|
|
|
writeFile file: 'update_versions_web.py', text: updateScript
|
|
|
|
|
|
bat """
|
|
|
|
|
|
scp -i "%SSH_KEY%" -o StrictHostKeyChecking=no update_versions_web.py ${env.PROD_USER}@${env.PROD_HOST}:/tmp/update_versions_web.py
|
|
|
|
|
|
ssh -i "%SSH_KEY%" -o StrictHostKeyChecking=no ${env.PROD_USER}@${env.PROD_HOST} "python3 /tmp/update_versions_web.py && rm /tmp/update_versions_web.py"
|
|
|
|
|
|
"""
|
2026-06-11 15:16:37 +08:00
|
|
|
|
}
|
2026-04-24 20:54:03 +08:00
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
2026-06-16 19:33:30 +08:00
|
|
|
|
|
|
|
|
|
|
stage('Commit Version') {
|
|
|
|
|
|
steps {
|
|
|
|
|
|
bat """
|
|
|
|
|
|
git config user.email "jenkins@xuqm.com"
|
|
|
|
|
|
git config user.name "Jenkins CI"
|
|
|
|
|
|
git add ${env.VERSION_FILE}
|
|
|
|
|
|
git diff --cached --quiet || git commit -m "ci: bump ${env.IMAGE_NAME} to ${env.SERVICE_VERSION} [skip ci]"
|
|
|
|
|
|
git push origin HEAD:main
|
|
|
|
|
|
"""
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
2026-04-24 20:54:03 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
post {
|
2026-06-16 19:33:30 +08:00
|
|
|
|
success { echo "✅ ${env.IMAGE_NAME}:${env.SERVICE_VERSION} 构建部署成功" }
|
2026-04-24 20:54:03 +08:00
|
|
|
|
failure { echo "❌ 构建失败,请检查日志" }
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|