首页 > 基础资料 博客日记
Jenkins-批量自动化构建指定目录或者视图下所有Job或者指定Job
2026-04-24 11:00:02基础资料围观2次
本篇文章分享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 "===== 全部构建完成 =====" } } } } }
定时或者手动执行效果:

文章来源:https://www.cnblogs.com/Jame-mei/p/19921295
本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若内容造成侵权/违法违规/事实不符,请联系邮箱:jacktools123@163.com进行投诉反馈,一经查实,立即删除!
本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若内容造成侵权/违法违规/事实不符,请联系邮箱:jacktools123@163.com进行投诉反馈,一经查实,立即删除!
标签:

