mirror of
https://github.com/KwiTsukasa/kt-template-online-api.git
synced 2026-05-27 15:44:54 +08:00
236 lines
8.2 KiB
Groovy
236 lines
8.2 KiB
Groovy
def runCmd(String unixScript, String windowsScript = null) {
|
||
if (isUnix()) {
|
||
sh unixScript
|
||
} else {
|
||
bat(windowsScript ?: unixScript)
|
||
}
|
||
}
|
||
|
||
def normalizeDockerTag(String value) {
|
||
return value.replaceAll(/[^A-Za-z0-9_.-]/, '-')
|
||
}
|
||
|
||
def resolveSourceName(String branchName, String changeBranch, String changeId, String tagName) {
|
||
if (changeId) {
|
||
return changeBranch ?: "PR-${changeId}"
|
||
}
|
||
return tagName ?: branchName ?: 'local'
|
||
}
|
||
|
||
def isPublishBranch(String branchName, String pattern) {
|
||
return branchName ==~ pattern
|
||
}
|
||
|
||
pipeline {
|
||
agent { label 'kt-node-agent' }
|
||
|
||
options {
|
||
skipDefaultCheckout(true)
|
||
timestamps()
|
||
disableConcurrentBuilds()
|
||
buildDiscarder(logRotator(numToKeepStr: '20', artifactNumToKeepStr: '10'))
|
||
}
|
||
|
||
parameters {
|
||
booleanParam(name: 'BUILD_DOCKER_IMAGE', defaultValue: true, description: '是否在非 PR 分支使用项目现有 dockerfile 构建镜像')
|
||
booleanParam(name: 'PUSH_DOCKER_IMAGE', defaultValue: false, description: '是否执行 docker push;仅发布分支生效,需要 Agent 已提前完成 docker login')
|
||
booleanParam(name: 'RUN_DOCKER_CONTAINER', defaultValue: true, description: 'Docker 镜像构建成功后是否重启业务容器;仅发布分支生效')
|
||
string(name: 'PUBLISH_BRANCH_PATTERN', defaultValue: '^(main|master|release/.+)$', description: '允许推送镜像的分支正则')
|
||
string(name: 'DOCKER_REGISTRY', defaultValue: '', description: '镜像仓库地址,为空时只生成本地镜像')
|
||
string(name: 'IMAGE_NAME', defaultValue: 'kt-template-online-api', description: 'Docker 镜像名称')
|
||
string(name: 'IMAGE_TAG', defaultValue: '', description: '镜像标签,为空时使用 分支名-BUILD_NUMBER;PR 使用源分支名')
|
||
string(name: 'CONTAINER_NAME', defaultValue: 'kt-template-online-api', description: '业务容器名称')
|
||
string(name: 'CONTAINER_PORT', defaultValue: '48085', description: '宿主机映射端口,容器内固定使用 48085')
|
||
string(name: 'CONTAINER_ENV_FILE', defaultValue: '/home/jenkins/agent/env/kt-template-online-api/.env.production', description: 'Agent workdir 内可读取的业务 env 文件路径')
|
||
string(name: 'CONTAINER_NETWORK', defaultValue: '', description: '业务容器加入的 Docker 网络,为空则使用 Docker 默认网络')
|
||
string(name: 'CONTAINER_EXTRA_ARGS', defaultValue: '', description: 'docker run 额外参数,例如 -v /host/data:/app/data')
|
||
}
|
||
|
||
environment {
|
||
APP_PORT = '48085'
|
||
CI = 'true'
|
||
NODE_ENV = 'development'
|
||
PNPM_VERSION = '9'
|
||
}
|
||
|
||
stages {
|
||
stage('Checkout') {
|
||
steps {
|
||
checkout scm
|
||
}
|
||
}
|
||
|
||
stage('Prepare') {
|
||
steps {
|
||
script {
|
||
def sourceName = resolveSourceName(env.BRANCH_NAME, env.CHANGE_BRANCH, env.CHANGE_ID, env.TAG_NAME)
|
||
def branchTag = normalizeDockerTag(sourceName)
|
||
def imageTagParam = params.IMAGE_TAG?.trim()
|
||
env.IMAGE_TAG_FINAL = imageTagParam ? normalizeDockerTag(imageTagParam) : "${branchTag}-${env.BUILD_NUMBER}"
|
||
env.IS_CHANGE_REQUEST = env.CHANGE_ID ? 'true' : 'false'
|
||
def publishPattern = params.PUBLISH_BRANCH_PATTERN?.trim() ?: '^(main|master|release/.+)$'
|
||
env.IS_PUBLISH_BRANCH = (!env.CHANGE_ID && isPublishBranch(env.BRANCH_NAME ?: '', publishPattern)) ? 'true' : 'false'
|
||
def registry = params.DOCKER_REGISTRY?.trim()
|
||
env.DOCKER_IMAGE = registry ? "${registry}/${params.IMAGE_NAME}:${env.IMAGE_TAG_FINAL}" : "${params.IMAGE_NAME}:${env.IMAGE_TAG_FINAL}"
|
||
|
||
// Agent 由 NAS 侧预先创建;这里仅确认 CI 所需的 Node/pnpm 环境可用。
|
||
if (isUnix()) {
|
||
runCmd("""
|
||
node --version
|
||
if ! command -v pnpm >/dev/null 2>&1; then
|
||
if command -v corepack >/dev/null 2>&1; then
|
||
corepack enable
|
||
corepack prepare pnpm@${env.PNPM_VERSION} --activate
|
||
else
|
||
echo "pnpm or corepack is required on the Jenkins Agent."
|
||
exit 1
|
||
fi
|
||
fi
|
||
pnpm --version
|
||
""".stripIndent())
|
||
} else {
|
||
runCmd('', """
|
||
node --version
|
||
where pnpm >nul 2>nul
|
||
if errorlevel 1 (
|
||
where corepack >nul 2>nul
|
||
if errorlevel 1 exit /b 1
|
||
corepack enable
|
||
corepack prepare pnpm@${env.PNPM_VERSION} --activate
|
||
)
|
||
pnpm --version
|
||
""".stripIndent())
|
||
}
|
||
|
||
echo """
|
||
Branch: ${env.BRANCH_NAME ?: '-'}
|
||
Change request: ${env.CHANGE_ID ?: '-'}
|
||
Tag: ${env.TAG_NAME ?: '-'}
|
||
Docker image: ${env.DOCKER_IMAGE}
|
||
Publish branch: ${env.IS_PUBLISH_BRANCH}
|
||
Run container: ${params.RUN_DOCKER_CONTAINER}
|
||
""".stripIndent()
|
||
}
|
||
}
|
||
}
|
||
|
||
stage('Install') {
|
||
steps {
|
||
script {
|
||
runCmd('pnpm install --frozen-lockfile')
|
||
}
|
||
}
|
||
}
|
||
|
||
stage('Lint') {
|
||
steps {
|
||
script {
|
||
runCmd('pnpm run lint')
|
||
}
|
||
}
|
||
}
|
||
|
||
stage('Test') {
|
||
steps {
|
||
script {
|
||
// 当前单测配置查找 src/**/*.spec.ts,允许空测试集,后续补齐 spec 后仍会正常执行。
|
||
runCmd('pnpm test -- --passWithNoTests')
|
||
}
|
||
}
|
||
}
|
||
|
||
stage('Build') {
|
||
steps {
|
||
script {
|
||
runCmd('pnpm run build')
|
||
}
|
||
}
|
||
}
|
||
|
||
stage('Docker Build') {
|
||
when {
|
||
allOf {
|
||
expression { return params.BUILD_DOCKER_IMAGE }
|
||
expression { return env.IS_CHANGE_REQUEST != 'true' }
|
||
}
|
||
}
|
||
steps {
|
||
script {
|
||
runCmd("docker build -f dockerfile -t ${env.DOCKER_IMAGE} .")
|
||
}
|
||
}
|
||
}
|
||
|
||
stage('Docker Push') {
|
||
when {
|
||
allOf {
|
||
expression { return params.BUILD_DOCKER_IMAGE && params.PUSH_DOCKER_IMAGE }
|
||
expression { return env.IS_PUBLISH_BRANCH == 'true' }
|
||
}
|
||
}
|
||
steps {
|
||
script {
|
||
runCmd("docker push ${env.DOCKER_IMAGE}")
|
||
}
|
||
}
|
||
}
|
||
|
||
stage('Docker Run') {
|
||
when {
|
||
allOf {
|
||
expression { return params.BUILD_DOCKER_IMAGE && params.RUN_DOCKER_CONTAINER }
|
||
expression { return env.IS_CHANGE_REQUEST != 'true' }
|
||
expression { return env.IS_PUBLISH_BRANCH == 'true' }
|
||
}
|
||
}
|
||
steps {
|
||
script {
|
||
if (!isUnix()) {
|
||
error('Docker Run stage requires a Linux/NAS Jenkins Agent.')
|
||
}
|
||
|
||
def containerName = params.CONTAINER_NAME?.trim() ?: 'kt-template-online-api'
|
||
def containerPort = params.CONTAINER_PORT?.trim() ?: env.APP_PORT
|
||
def containerEnvFile = params.CONTAINER_ENV_FILE?.trim()
|
||
if (!containerEnvFile) {
|
||
error('CONTAINER_ENV_FILE is required when RUN_DOCKER_CONTAINER is enabled.')
|
||
}
|
||
|
||
def networkArg = params.CONTAINER_NETWORK?.trim() ? "--network ${params.CONTAINER_NETWORK.trim()}" : ''
|
||
def extraArgs = params.CONTAINER_EXTRA_ARGS?.trim() ?: ''
|
||
|
||
// 部署阶段会替换同名容器;真实 env 文件只从 NAS 挂载进 Agent,不进入 Git。
|
||
runCmd("""
|
||
set -e
|
||
if [ ! -f '${containerEnvFile}' ]; then
|
||
echo "Container env file not found: ${containerEnvFile}"
|
||
echo "Put .env.production under the existing Agent workdir volume, for example:"
|
||
echo "/home/jenkins/agent/env/kt-template-online-api/.env.production"
|
||
exit 1
|
||
fi
|
||
|
||
docker rm -f '${containerName}' >/dev/null 2>&1 || true
|
||
docker run -d \\
|
||
--name '${containerName}' \\
|
||
--restart=always \\
|
||
${networkArg} \\
|
||
--env-file '${containerEnvFile}' \\
|
||
-e NODE_ENV=production \\
|
||
-p '${containerPort}':${env.APP_PORT} \\
|
||
${extraArgs} \\
|
||
'${env.DOCKER_IMAGE}'
|
||
|
||
docker ps --filter "name=^/${containerName}\$"
|
||
""".stripIndent())
|
||
}
|
||
}
|
||
}
|
||
}
|
||
|
||
post {
|
||
success {
|
||
archiveArtifacts artifacts: 'dist/**,package.json,pnpm-lock.yaml,dockerfile', fingerprint: true, allowEmptyArchive: true
|
||
}
|
||
}
|
||
}
|