Jenkins-批量自动化构建指定目录或者视图下所有Job或者指定Job

复制代码
## Jenkins Pipeline 功能总结

### 概述

这是一个 **Jenkins 批量自动化构建脚本**,用于定时或手动触发指定目录下的所有 Job,支持并发分批构建和串行构建两种模式。

---

### 触发方式

| 方式 | 说明 |
|------|------|
| 定时触发 | 每周六 UTC 17:00(北京时间周日凌晨 1:00)自动执行 |
| 手动触发 | 支持自定义参数手动触发 |

---

### 参数说明

| 参数 | 默认值 | 说明 |
|------|--------|------|
| `FOLDER_NAME` | `UAT` | 扫描的 Jenkins 目录 |
| `BRANCH` | `release` | 构建分支 |
| `BATCH_SIZE` | `5` | 并发批次大小 |
| `SPECIFIC_JOBS` | 空 | 手动指定 Job 列表(含参数覆盖) |

---

### 核心流程

```
开始
 ├─ 1. 扫描 MAVEN_API_MODE 标记
 │      检测上次构建参数,自动识别需要 JAR-Only 模式的 Job
 │
 ├─ 2. 生成任务列表
 │      ├─ 手动模式:解析 SPECIFIC_JOBS(支持 :: 分隔传参)
 │      └─ 自动模式:扫描目录,按 MODULE × Deploy_Env 笛卡尔积展开任务
 │
 ├─ 3. 任务分类
 │      ├─ 普通任务     → 并发分批执行
 │      └─ rapidtrade-* → 串行逐个执行(CPU 密集型保护)
 │
 ├─ 4. 第一阶段:普通任务并发构建
 │      按 BATCH_SIZE 分批,批内并发,批间顺序
 │      自动注入 BRANCH、MODULE、Deploy_Env、MAVEN_API_MODE 等参数
 │
 └─ 5. 第二阶段:rapidtrade-* 串行构建
        全部走默认参数,一次只跑 1 个,等待完成后再触发下一个
```

---

### 关键设计点

**自动参数展开**:Job 若同时有 `MODULE` 和 `Deploy_Env` 两个选项参数,会自动进行笛卡尔积展开,为每个组合各触发一次构建。

**MAVEN_API_MODE 自动识别**:不需要手动标记,脚本读取该 Job 上次构建的实际参数,自动继承 `MAVEN_API_MODE=true`。

**CPU 密集型保护**:`rapidtrade-` 开头的 Job 被识别为 CPU 密集型,强制串行执行,避免并发抢占资源。

**构建异常不中断**:每个 Job 用 `try/catch` + `propagate: false` 包裹,单个 Job 失败不影响后续任务继续执行。
复制代码
pipeline {
  agent any

  triggers {
    cron('0 17 * * 6')
  }

  parameters {
    string(name: 'FOLDER_NAME', defaultValue: 'UAT', description: '目录名称')
    string(name: 'BRANCH', defaultValue: 'release', description: '构建分支(不需要加 origin/ 前缀)')
    string(name: 'BATCH_SIZE', defaultValue: '5', description: '每批并发数量')
    text(
      name: 'SPECIFIC_JOBS',
      defaultValue: '',
      description: '''手动指定任务(每行一个),不填则自动扫描目录下所有 Job:
UAT/job-name
UAT/job-name::MODULE=pigeon-gw
UAT/job-name::Deploy_Env=uat-shard-30
UAT/job-name::MODULE=pigeon-gw::Deploy_Env=uat-shard-30'''
    )
  }

  stages {
    stage('扫描并生成任务列表') {
      steps {
        script {
          def allTasks = []

          // ============================================================
          // 自动检测哪些 Job 需要 MAVEN_API_MODE = true (Build JAR Only)
          // 通过读取最近一次成功构建的实际参数值来判断
          // ============================================================
          def jarOnlyJobs = [] as Set
          def scanFolder = Jenkins.instance.getItemByFullName(params.FOLDER_NAME)
          if (scanFolder) {
            scanFolder.getItems().each { job ->
              def paramsDef = job.getProperty(hudson.model.ParametersDefinitionProperty.class)
              if (!paramsDef) return
              def mavenParam = paramsDef.getParameterDefinitions().find { it.name == 'MAVEN_API_MODE' }
              if (!mavenParam) return
              // 查最近一次构建的实际值
              def lastBuild = job.getLastBuild()
              if (lastBuild) {
                def action = lastBuild.getAction(hudson.model.ParametersAction.class)
                if (action) {
                  def p = action.getParameter('MAVEN_API_MODE')
                  if (p && p.getValue() == true) {
                    jarOnlyJobs << job.fullName
                  }
                }
              }
            }
          }
          if (jarOnlyJobs) {
            echo "===== 以下 Job 将自动勾选 MAVEN_API_MODE=true (Build JAR Only) ====="
            jarOnlyJobs.sort().each { echo "  ☑️ ${it}" }
          }

          if (params.SPECIFIC_JOBS?.trim()) {
            params.SPECIFIC_JOBS.trim().split('\n').each { line ->
              line = line.trim()
              if (!line) return
              def parts = line.split('::')
              def jobName = parts[0].trim()
              def extraParams = [:]
              for (int i = 1; i < parts.size(); i++) {
                def kv = parts[i].trim().split('=')
                if (kv.size() == 2) extraParams[kv[0].trim()] = kv[1].trim()
              }
              allTasks << [job: jobName, extra: extraParams]
            }
            echo "手动模式,共 ${allTasks.size()} 个任务"

          } else {
            echo "自动扫描 [${params.FOLDER_NAME}] 目录..."

            def folder = Jenkins.instance.getItemByFullName(params.FOLDER_NAME)
            if (!folder) error("目录 [${params.FOLDER_NAME}] 不存在")

            folder.getItems().each { job ->
              def paramsDef = job.getProperty(hudson.model.ParametersDefinitionProperty.class)
              if (!paramsDef) {
                allTasks << [job: job.fullName, extra: [:]]
                return
              }

              def moduleChoices    = []
              def deployEnvChoices = []

              paramsDef.getParameterDefinitions().each { p ->
                if (p.name == 'MODULE' && p instanceof hudson.model.ChoiceParameterDefinition) {
                  moduleChoices = p.choices.collect { it.trim() }.findAll { it }
                }
                if (p.name == 'Deploy_Env' && p instanceof hudson.model.ChoiceParameterDefinition) {
                  deployEnvChoices = p.choices.collect { it.trim() }.findAll { it }
                }
              }

              def buildTask = { extra -> [job: job.fullName, extra: extra] }

              if (moduleChoices && deployEnvChoices) {
                moduleChoices.each { m ->
                  deployEnvChoices.each { d ->
                    allTasks << buildTask([MODULE: m, Deploy_Env: d])
                  }
                }
              } else if (moduleChoices) {
                moduleChoices.each { m ->
                  allTasks << buildTask([MODULE: m])
                }
              } else if (deployEnvChoices) {
                deployEnvChoices.each { d ->
                  allTasks << buildTask([Deploy_Env: d])
                }
              } else {
                allTasks << buildTask([:])
              }
            }

            echo "自动扫描完成,共 ${allTasks.size()} 个任务"
          }

          if (allTasks.isEmpty()) error("没有找到任何任务")

          // ============================================================
          // 分离 rapidtrade- 开头的 CPU 密集型任务(串行执行)
          // 和普通任务(按批次并发执行)
          // ============================================================
          def rapidtradeTasks = []
          def normalTasks = []
          allTasks.each { task ->
            // 取 Job 名称最后一段(去掉目录前缀)做匹配
            def shortName = task.job.contains('/') ? task.job.split('/').last() : task.job
            if (shortName.startsWith('rapidtrade-')) {
              rapidtradeTasks << task
            } else {
              normalTasks << task
            }
          }

          if (rapidtradeTasks) {
            echo "===== rapidtrade-* 任务 (串行执行, CPU密集型) ====="
            rapidtradeTasks.each {
              def extraStr = it.extra.collect { k, v -> "$k=$v" }.join(', ')
              echo "  - ${it.job}" + (extraStr ? "  [$extraStr]" : "") + "  [默认参数]"
            }
          }

          echo "===== 普通任务列表 ====="
          normalTasks.each {
            def extraStr = it.extra.collect { k, v -> "$k=$v" }.join(', ')
            def jarTag = jarOnlyJobs.contains(it.job) ? '  [JAR_ONLY ☑️]' : ''
            echo "  - ${it.job}" + (extraStr ? "  [$extraStr]" : "") + "  [BRANCH=${params.BRANCH}]" + jarTag
          }

          // ============================================================
          // 第一阶段:普通任务按批次并发执行
          // ============================================================
          if (normalTasks) {
            def batchSize = params.BATCH_SIZE.toInteger()
            def batches = []
            for (int i = 0; i < normalTasks.size(); i += batchSize) {
              batches << new ArrayList(normalTasks.subList(i, Math.min(i + batchSize, normalTasks.size())))
            }

            echo "===== 普通任务共 ${normalTasks.size()} 个,分 ${batches.size()} 批,每批最多 ${batchSize} 个 ====="

            for (int b = 0; b < batches.size(); b++) {
              def batch = batches[b]
              echo "===== 第 ${b + 1} 批,共 ${batch.size()} 个任务 ====="

              def parallelJobs = [:]
              for (int j = 0; j < batch.size(); j++) {
                def task    = batch[j]
                def taskKey = task.extra ? "${task.job}[${task.extra.collect { k,v -> "$k=$v" }.join('_')}]" : task.job

                parallelJobs[taskKey] = {
                  def extraInfo = task.extra.collect { k, v -> "$k=$v" }.join(', ')
                  def isJarOnly = jarOnlyJobs.contains(task.job)
                  echo "触发构建: ${task.job}" + (extraInfo ? " [$extraInfo]" : "") + " (BRANCH=${params.BRANCH})" + (isJarOnly ? " [MAVEN_API_MODE=true]" : "")
                  try {
                    def buildParams = []
                    buildParams << string(name: 'BRANCH', value: params.BRANCH)
                    task.extra.each { k, v -> buildParams << string(name: k, value: v) }

                    // 自动为需要的 Job 勾选 MAVEN_API_MODE
                    if (isJarOnly) {
                      buildParams << booleanParam(name: 'MAVEN_API_MODE', value: true)
                    }

                    build job: task.job,
                          parameters: buildParams,
                          wait: true,
                          propagate: false
                    echo "✅ ${taskKey} 构建完成" + (isJarOnly ? " (JAR Only)" : "")
                  } catch (e) {
                    echo "❌ ${taskKey} 构建异常: ${e.message}"
                  }
                }
              }

              parallel parallelJobs
              echo "第 ${b + 1} 批完成,继续下一批..."
            }
          }

          // ============================================================
          // 第二阶段:rapidtrade-* 任务逐个串行执行(CPU密集型,一次只跑1个)
          // 所有参数走默认,直接触发构建
          // ============================================================
          if (rapidtradeTasks) {
            echo "===== 开始 rapidtrade-* 串行构建(共 ${rapidtradeTasks.size()} 个)====="
            for (int r = 0; r < rapidtradeTasks.size(); r++) {
              def task = rapidtradeTasks[r]
              def taskKey = task.extra ? "${task.job}[${task.extra.collect { k,v -> "$k=$v" }.join('_')}]" : task.job
              echo "串行构建 [${r + 1}/${rapidtradeTasks.size()}]: ${task.job} (默认参数)"
              try {
                // rapidtrade-* 走默认参数,不传 BRANCH 等自定义参数
                build job: task.job,
                      wait: true,
                      propagate: false
                echo "✅ ${taskKey} 构建完成 (串行)"
              } catch (e) {
                echo "❌ ${taskKey} 构建异常: ${e.message}"
              }
            }
          }

          echo "===== 全部构建完成 ====="
        }
      }
    }
  }
}

定时或者手动执行效果:

相关推荐
搬砖天才、2 小时前
es数据备份
大数据·elasticsearch·jenkins
终端行者3 小时前
Jenkins Pipeline在不同阶段指定不同的 agent 或 Docker 容器进行执行任务和固定一个节点分段执行任务,一文带你搞定
java·docker·jenkins·cicd
@土豆3 小时前
Jenkins CI_CD流水线案例
运维·ci/cd·jenkins
Cyber4K4 小时前
【DevOps专项】GitLab 与 Jenkins 介绍及部署持续集成环境
运维·ci/cd·gitlab·jenkins·devops
终端行者4 小时前
Jenkins流水线添加企业微信或者钉钉通知 pipeline 如何通过企微/钉钉通知
ci/cd·jenkins·钉钉·企业微信
郝开4 小时前
Docker Compose 本地环境搭建:elasticsearch
elasticsearch·docker·jenkins
Elivs.Xiang1 天前
centos9中安装Jenkins
linux·运维·centos·jenkins
我叫唧唧波1 天前
【自动化部署】CI/CD 实战(三):让 Argo CD 接管 CD,Jenkins 镜像自动同步到集群
运维·前端·ci/cd·docker·自动化·jenkins·argocd
Elivs.Xiang1 天前
ubuntu20中安装Jenkins
linux·运维·ubuntu·jenkins