流水线

流水线(Pipeline)是平台自研的工作流引擎,覆盖平台所有的流水线相关任务,为平台提供强大及灵活的任务编排执行能力和动态配置能力等,具体包含 DevOps 平台的 CI/CD、快数据平台的流式计算、运维 Ops 和监控报表等。

功能清单

  • 配置即代码,通过 YAML 文件(pipeline.yml) 描述流水线定义,基于 Stage(阶段) 语法简化编排复杂度。
  • 可视化编辑,通过图形界面交互快速配置流水线。
  • 支持任务串行或并行执行策略。
  • 定时流水线,同时提供强大的定时补偿功能。
  • 动态配置,通过动态配置满足实现同一流水线满足不同场景需求,配置类型包含值和文件,配置内容均支持加密存储,确保数据安全性。
  • 多维度重试机制和中止处理。支持从失败节点开始重试;支持全流程重试;运行中可以取消中止。
  • 超时配置(任务级别)。
  • 归档策略可配置(流水线级别)。
  • 任务分为短任务和长任务,任务之间可传递上下文(可传递内容包含:值和文件)。
  • 扩展市场,平台预置丰富的 Action 满足大部分场景的同时,能够帮助用户沉淀自己的 Action。
  • 内置丰富的制品仓库,满足主流框架的编译构建需要,主要包含:Docker 镜像、Maven JARs、NPM Packages 等。
  • 对外开放 OpenAPI 接口,方便第三方系统快速接入使用。
  • Event 事件订阅,提供流水线级别和任务级别事件消息订阅。

基本元素

Action

Action 是流水线的最小执行单元,表示一个 任务动作

一条流水线包含若干个 Action。

Stage

Stage 表示一个 阶段

一条流水线由若干个 Stage 构成,多个 Stage 之间串行顺序执行; 一个 Stage 由若干个 Action 构成,多个 Action 之间并行执行。

格式规范

pipeline.yml 是一个 YAML 格式的文件,描述了任务编排的全流程。

pipeline.yml 的字段详细说明如下:

version

version 表示 pipeline.yml 的版本号。目前最新版本为 1.1。

只需要配置为:version: "1.1" 即可。

envs

envs 表示全局环境变量,会注入到所有 Action 中。

envs 的典型场景为:

  • 用于多个 自定义脚本 (custom-script) Action 共享环境变量

envs 格式为 map[string]string,比如:

  1. envs:
  2. PLATFORM: PIPELINE
  3. DEBUG: true

cron

定时配置。配置该字段后,在流水线界面可以操作 定时按钮 来启用和禁用 定时功能。

目前支持两种格式:

分钟级 (Linux Crontab)

* * * * *

示例:

  • 0 3 * * * 每天凌晨 3 点执行
  • */2 * * * * 每 2 分钟执行一次

秒级 (Spring Schedule)

* * * * * *

示例:

  • 15 * * * * * 在每分钟的第 15 秒执行
  • 0 0 3 * * * 每天凌晨 3 点执行

cron_compensator

enable

未执行补偿策略开关,系统会根据下面的策略调度执行

  • false (默认) 代表补偿是关闭的
  • true 代表补偿开启

tip

补偿的2种类型

  • 中断补偿 平台原因导致短暂中断,在中断时间内的定时任务无法触发,流水线记录没创建, 中断补偿无法关闭
  • 未执行补偿 流水线记录已经创建,但未开始执行,这时候就需要补偿策略去调度这些记录了

latest_first

当有多个未执行的记录的时候,从最新的记录往老记录依次执行,或者从老的记录往新的记录依次执行

  • false (默认) 代表从老记录往新记录依次执行
  • true 代表从新记录往老记录依次执行

stop_if_latter_executed

依赖 latest_first = true, 2个策略结合就是只调度最新的记录

  • false (默认) 跟随 latest_first 进行策略
  • true 在 latest_first 策略之上就只跑一次最新的记录

策略调度说明

pipeline.yml - 图1 如图,策略目前只关心几个点

  1. 是否是定时的记录(非定时任务不会被策略调度)
  2. 状态是,分析完成,成功,进行中

pipeline.yml - 图2

调度策略过程, 途中画的 1,2,3,4 其实就是执行记录中的版本

支持策略配置如下

  • 执行补偿,只跑最新的记录
  1. cron_compensator:
  2. enable: true
  3. latest_first: true
  4. stop_if_latter_executed: true
  • 执行补偿,从老记录往新的记录依次执行
  1. cron_compensator:
  2. enable: true
  3. latest_first: false
  4. # ----- 上面和下面配置相同
  5. cron_compensator:
  6. enable: true
  • 执行补偿,从新记录往老记录依次执行
  1. cron_compensator:
  2. enable: true
  3. latest_first: true

stages

stages 是 pipeline.yml 的核心字段,定义了具体的一组 stage 集合,它们组成了整个流水线的执行过程。

依赖关系为:

  • stages 由 stage 列表组成
  • stage 由 Action 列表组成

details YAML 示例

  1. # stage 和 stage 之间是串行的,即前面一个 stage 执行完毕才会开始执行下一个 stage.
  2. # 该例子中一共有 2 个 stage.
  3. stages:
  4. # 在一个 stage 中,多个 Actions 是并行的,不会有依赖关系;
  5. # 该例子中,该 stage 包含 2 个并行 Action.
  6. - stage:
  7. # Action1-1 是 Action 的类型,不能任意填写,需要能在扩展市场中找到该类型.
  8. - Action1-1:
  9. params:
  10. ...
  11. # Action1-2 是 Action 的类型,不能任意填写,需要能在扩展市场中找到该类型.
  12. - Action1-2:
  13. params:
  14. ...
  15. # 这里定义了第二个 stage,它在第一个 stage 执行结束后才会开始执行.
  16. # 该 stage 只有一个 Action.
  17. - stage:
  18. # Action2-1 是 Action 的类型,不能任意填写,需要能在扩展市场中找到该类型.
  19. - Action2-1:
  20. params:
  21. ...

Action 配置项

Action 配置项是用来描述一个具体的 Action。

alias

alias 是 Action 的别名,Action 之间可以通过 alias 来进行引用,为非必填项,

未设置的时候,系统默认把 alias 设置为 Aciton 类型,

也正是因为默认值的原因,当一个类型的 Action 多次出现的时候,若不设置 alias,会导致无法正确引用 Action。

params

Action 的参数,主要用来定义 Action 的行为。

每个 Action 的参数都不一样,具体 Action 的参数信息请参考 扩展市场 里具体内容。

version

Action 的版本。

一般来说,用户不需要填写。默认会使用扩展市场里该 Action 的最新可用版本。

当用户需要使用特定版本的 Action 的时候,才需要填写该字段。

image

tip

该字段仅在 Action 类型为 custom-script 时有效。

custom-script Action 运行时的镜像。

默认使用的镜像可以在 扩展市场-自定义脚本 Action 里看到。

commands

自定义命令可以是多个命令组成的列表, 也可以是多行命令组成的一个段落。

tip 在2.3版本往后,该字段在所有的action都会生效。 且自定义命令将在action的主逻辑结束后被执行

用户声明的命令列表。

YAML 示例

  1. commands:
  2. - echo "hello Action!"
  3. - cat /etc/hosts
  4. - sleep 10

用户声明的命令段落。

YAML 示例

  1. commands: |-
  2. import time # 引入time模块
  3. ticks = time.time()
  4. print ("当前时间戳为:", ticks)

shell

该字段可以指定执行commands的执行器,默认是/bin/sh。 可以根据自定义命令的内容来修改执行器,例如如果是python代码,则可以指定为python

timeout

Action 执行的超时时间。超过该时间,平台会强制停止 Action 的执行。

timeout 单位为秒, 系统默认的 timeout 为 3600(1小时)

details YAML 示例

  1. version: "1.1"
  2. stages:
  3. - stage:
  4. - custom-script:
  5. alias: 自定义shell脚本
  6. version: "1.0"
  7. commands:
  8. - sleep 60s
  9. timeout: 50

resources

Action 运行资源。

可配置项包括:

  • cpu
  • mem (单位为 MB)

流水线引擎在执行的时候会根据三个维度的cpu大小来设定cpu的limits值

  • 设定的 CPU 核数较小

    这时 pipeline 自动将用户设定的值乘以一个超卖比(默认超卖比为2),如果得到值仍然小于 action 本身默认的值,pipeline 会将 CPU 的值设定为 action 本身的值。

  • 设定的 CPU 核数适中

    仍然将设定的 CPU 大小乘以超卖比,如果得到的值大于 action 默认的值且没有超过 pipeline 默认的最大 CPU 值,将以该值作为 action 执行的时候的 CPU limits 值,因为通常情况下资源的 CPU 都是空闲的。

  • 设定的 CPU 过大

    如果请求的 CPU 资源过大,pipeline 会自动将 action 的 CPU limits 值设定为默认的最大值(目前默认最大值为2)。

mem 的 limits 值不会进行超卖的处理,因为内存通常在服务器上是一种比较紧俏的资源,其大小将自动取用户设定的值与默认值中两个较小的值。

details YAML 示例

  1. - git-checkout:
  2. params:
  3. ...
  4. resources:
  5. cpu: 0.5
  6. mem: 2048

loop

Action 的循环执行策略。

可配置项包括:

  • break: 退出条件
  • strategy: 循环策略
    • max_times: 最大重试次数,-1 表示不限制
    • decline_ratio: 循环衰退速率
    • decline_limit_sec: 循环间隔最大值(秒)
    • interval_sec: 循环起始间隔(秒)

用公式表示循环策略:

  1. loop_interval = min(interval_sec * (decline_ratio)^N, decline_limit_sec)
  2. 其中 N 为循环次数,从 0 开始计数。

例子

循环 10000 次,每次循环间隔 10 秒

  1. loop:
  2. break: 'false' == 'true'
  3. strategy:
  4. max_times: 10000
  5. interval_sec: 10

当前任务如果执行失败重试

  1. loop:
  2. break: task_status == 'Success'
  3. strategy:
  4. max_times: 10000

某个任务或当前任务的出参 xxx 不为 401 时候重试

  1. loop:
  2. break: '${{ outputs.任务的alias.xxx }}' == '401'
  3. strategy:
  4. max_times: 10000

多条件组合判断

  1. loop:
  2. break: '1' == '2' && '1' == '3' || '1' == '3' || '${{ outputs.任务的alias.xxx }}' == '401'
  3. strategy:
  4. max_times: 10000

caches

caches 缓存

if

Action 条件执行

  1. - stage:
  2. - custom-script:
  3. alias: script1
  4. version: "1.0"
  5. commands:
  6. - echo 1
  7. - echo "image=123" >> $METAFILE # 输出出参给下个任务使用
  8. if: ${{ 1==1 }}
  9. - stage:
  10. - custom-script:
  11. alias: script2
  12. version: "1.0"
  13. commands:
  14. - echo 1
  15. if: ${{ ${{ outputs.script1.image }}==123 && ${{ outputs.script1.image }}==123 }}

使用${{ xxx }},中间 xxx 输入数学表达式(注意xxx前后有空格),可以使用前置任务出参作为执行条件

目前 pipeline 执行的时候假如没有加入条件执行,那么当一个任务失败,下面的任务就会自动失败,而当下面的任务加上了条件执行,条件成立的话还是会继续执行

内置表达式

  • task_status: 可选值 Success / Failed (注意大小写)

配置失败重试

  1. loop:
  2. break: task_status == 'Success' #
  3. strategy:
  4. max_times: 5 # 最多循环5次
  5. decline_limit_sec: 60 # 衰退最大值,60s
  6. interval_sec: 5 # 起始间隔,5秒
  7. decline_ratio: 2 # 衰退比例为 2,即 5 -> 10(5*2) -> 20(10*2) -> 40(20*2) -> 60(40*2>60)

该配置同样适用于在 extension 的 spec.yml 中顶层配置。

上下文引用

在一条流水线中,Action 独立运行的场景很少,大多数 Action 需要拿上一个 Action 的输出作为输入。

例如:

用于编译和制作镜像的 buildpack Action 需要代码作为输入,而代码一般由 git-checkout Action 拉取。

因此在 buildpack Action 中可以使用类似 ${git-chekcout} 的语法来引用指定 git-checkout Action 命名空间里的代码仓库。

YAML 示例

  1. ...
  2. stages:
  3. - stage:
  4. - git-checkout: # 代码仓库1
  5. alias: repo1
  6. params:
  7. uri: xxx
  8. - git-checkout: # 代码仓库2
  9. alias: repo2
  10. params:
  11. uri: yyy
  12. - stage:
  13. - buildpack:
  14. params:
  15. code_dirs:
  16. - ${repo1} # 引用代码仓库1的代码
  17. - ${repo2} # 引用代码仓库2的代码

示例

YAML 示例

  1. version: "1.1"
  2. # 定时配置
  3. cron: 0 */10 * * * ?
  4. # 环境变量, map[string]string 格式
  5. envs:
  6. PLATFORM: pipeline
  7. ENV: production
  8. # 关键字,表示流水线流程定义
  9. # stage 表示 阶段,多个 stage 串行成为 stages
  10. stages:
  11. # 关键字,表示流水线的一个阶段
  12. # 一个 stage 内包含多个 并行 的 Action
  13. - stage: # 该 stage 只有一个 Action
  14. - git-checkout: # Action 类型
  15. params: # Action 参数,key/value pairs,key 为 string,value 为简单类型或任意可 JSON 序列化的结构体
  16. depth: 1
  17. - stage: # 该 stage 拥有三个并行执行的 Action
  18. - buildpack: # Action 类型
  19. # 同一个类型的 Action 可以被声明多次,但是每个 Action 需要能被唯一标识,
  20. # 因此需要用 别名 来标识
  21. alias: backend
  22. params:
  23. context: ${git-checkout}/services/showcase # 使用 ${xxx} 语法来引用其他 Action 的目录地址
  24. modules: # value 支持复杂结构
  25. - name: blog-web
  26. - name: blog-service
  27. path: blog-service/blog-service-impl
  28. - name: user-service
  29. path: user-service/user-service-impl
  30. resources: # 预计该 Action 运行时需要分配的资源
  31. cpu: 0.5 # 0.5 核
  32. mem: 2048 # 2G 内存
  33. - buildpack:
  34. # 上面已经声明了一个 buildpack 类型的 Action,为了唯一标识,需要 alias 区分
  35. alias: frontend
  36. params:
  37. context: ${git-checkout}/endpoints/showcase-front
  38. modules:
  39. - name: showcase-front
  40. - custom-script: # custom-script 为自定义脚本类型
  41. image: centos:7 # 声明运行时镜像
  42. commands: # 声明 Commands
  43. - sleep 5
  44. - echo hello world
  45. - cat ${git-checkout}/dice.yml # 这里通过 ${git} 语法来 cat git Action 目录下的 dice.yml 文件
  46. - stage: # 该 stage 只有一个 Action
  47. - release:
  48. # 指定 Action 版本,Action 可以拥有多个历史版本
  49. # 若不指定,则使用平台当前的稳定版
  50. version: "1.0"
  51. params:
  52. dice_yml: ${git-checkout}/dice.yml # dice.yml 文件路径
  53. replacement_images: # 通过 ${backend}/${frontend} 引用打包结果
  54. - ${backend}/pack-result
  55. - ${frontend}/pack-result
  56. - stage: # 该 stage 只有一个 Action
  57. - dice:
  58. params:
  59. release_id_path: ${release}

定时流水线

定时流水线是指按照预先配置的执行计划,定时触发并执行的流水线。

定时流水线 = 流水线 + 定时配置。因此,只需在原来流水线定义上增加定时配置,即可将流水线升级为定时流水线。

定时配置 (Cron) 支持以下格式:

  • 分钟级 (Linux Crontab)
  • 秒级 (Spring Schedule)

具体配置见规范 Cron 部分

由于 cron 语法无法描述从某个时间点开始的周期任务,因此在 REST API POST /api/pipelines 的请求参数中增加了 cronStartFrom 参数,用于指定周期任务的开始时间点。

定时流水线的补偿

具体策略配置见规范 cron_compensator 部分

tip 常见原因

需要补偿的原因很多,比如:

  • 定时间隔不合理,导致新的流水线定时触发时,仍有正在运行中的流水线实例(未执行补偿)
  • 平台升级导致的短暂中断,定时触发时间恰好在中断时间内,导致流水线实例未能按时执行(中断补偿)

代码更新触发流水线

当配置对应规则的分支代码被更新时,会自动创建当前分支的流水线并执行

配置参数作为顶级pipeline参数

details YAML 示例

  1. version: "1.1"
  2. on:
  3. push:
  4. branches:
  5. - develop
  6. - feature/*
  7. stages:
  8. ...

动态配置

流水线拥有独立的配置管理体系。

流水线配置从类型上分为 文件 两类。

两种类型均支持 明文存储加密存储,保证数据安全。

动态配置使得 pipeline.yml 模板化成为现实,同一份流程定义文件,通过不同配置可以实现截然不同的行为。

同时,加密配置 使得用户在 pipeline.yml 中无需存储明文,更加安全。

DevOps 平台中流水线配置入口位于:

DevOps 平台 -> 我的应用 -> 应用 A -> 应用设置 -> 流水线 -> 变量配置

某个参数的值需要动态设置

示例:buildpack-Action 里打包需要感知环境,不同的环境使用不同的 profile 进行构建。

details YAML 示例

  1. ...
  2. stages:
  3. - stage:
  4. - buildpack:
  5. params:
  6. bp_args:
  7. MAVEN_EXTRA_ARGS: ((maven.profile)) # maven 编译时可以动态设置 profile
  8. ...

可以在 默认设置 里设置默认值,并且针对不同分支设置不同的值。

例如配置:

  1. default:
  2. maven.profile = dev
  3. master:
  4. maven.profile = prod

pipeline.yml - 图3

某个参数的值不应该明文写在文件中

例如:git-checkout Action 拉取一个需要鉴权的 git repo 时,需要用到 username 和 password。

details YAML 示例

  1. ...
  2. stages:
  3. - stage:
  4. - git-checkout:
  5. params:
  6. uri: http://git.terminus.io/xxx/yyy.git
  7. username: ((your.username))
  8. password: ((your.password))
  9. ...

pipeline.yml - 图4

流水线内置环境变量

流水线中在运行中含有以下环境变量

流水线环境变量 pipeline 相关的环境变量

  1. PIPELINE_CRON_EXPR
  2. PIPELINE_TYPE
  3. PIPELINE_TIME_BEGIN_TIMESTAMP
  4. PIPELINE_TRIGGER_MODE
  5. PIPELINE_STORAGE_URL
  6. PIPELINE_TASK_LOG_ID
  7. PIPELINE_TASK_NAME
  8. PIPELINE_ID
  9. PIPELINE_LIMITED_DISK
  10. PIPELINE_CRON_TRIGGER_TIME
  11. PIPELINE_DEBUG_MODE
  12. PIPELINE_LIMITED_MEM
  13. PIPELINE_TASK_ID
  14. PIPELINE_LIMITED_CPU

dice 相关的环境变量

  1. DICE_VERSION
  2. DICE_INTERNAL_CLIENT
  3. DICE_MEM_REQUEST
  4. DICE_PROJECT_APPLICATION
  5. DICE_OPENAPI_ADDR
  6. DICE_ORG_NAME
  7. DICE_ENV=DEV
  8. DICE_APPLICATION_ID
  9. DICE_CPU_ORIGIN
  10. DICE_INSIDE
  11. DICE_OPENAPI_PUBLIC_URL
  12. DICE_CPU_REQUEST
  13. DICE_NAMESPACE
  14. DICE_IS_EDGE
  15. DICE_PROJECT_ID
  16. DICE_STORAGE_MOUNTPOINT
  17. DICE_OPERATOR_ID
  18. DICE_OPERATOR_NAME
  19. DICE_PROTOCOL
  20. DICE_PROJECT_NAME
  21. DICE_MEM_ORIGIN
  22. DICE_ROOT_DOMAIN
  23. DICE_APPLICATION_NAME
  24. DICE_WORKSPACE
  25. DICE_CPU_LIMIT
  26. DICE_ORG_ID
  27. DICE_HTTPS_PORT
  28. DICE_CLUSTER_TYPE
  29. DICE_SIZE
  30. DICE_CLUSTER_NAME
  31. DICE_HTTP_PORT
  32. DICE_USER_ID
  33. DICE_MEM_LIMIT

gittar 相关的环境变量

  1. GITTAR_AUTHOR
  2. GITTAR_BRANCH
  3. GITTAR_COMMIT
  4. GITTAR_MESSAGE
  5. GITTAR_REPO

流水线表达式语法

${{ random.key }}

随机表达式占位符,用户填入对应的随机表达式,流水线运行的时候会随机生成数据来替换 ${{ random.integer }} 占位符

随机参数支持的类型如下:

integer:整型,例如 100
string:所有字符串,例如 Abc
float:浮点型,例如 13.14
boolean:布尔型,例如 true/false
upper:大写字母,例如 ABC
lower:小写字母,例如 abc
mobile:11位手机号,例如 18888888888
digital_letters:数字和大小写字母,例如 Abc123
letters:大小写字母,例如 Abc
character:单个字符,例如 a
timestamp:当前时间戳格式:1586917254,单位是 s
timestamp_hour:1小时前时间戳格式:1583317290,单位是 s
timestamp_ns:当前时间戳ns格式:1586917325230422166,单位是 ns
timestamp_ns_hour:1小时前时间戳格式:1586913750801408626,单位是 ns
date:当前日期、格式:2020-01-01
date_day:1天前日期、格式:2020-01-01
datetime:当前时间、格式:2020-01-01 15:04:05
datetime_hour:1小时前时间、格式:2020-01-01 14:04:05

除此之外还支持还见的日期格式化:

  • datetime_custom_ANSIC: Tue Mar 8 16:33:06 2022
  • datetime_custom_UnixDate: Tue Mar 8 16:33:06 CST 2022
  • datetime_custom_RubyDate: Tue Mar 08 16:33:06 +0800 2022
  • datetime_custom_RFC822: 08 Mar 22 16:33 CST
  • datetime_custom_RFC822Z: 08 Mar 22 16:33 +0800
  • datetime_custom_RFC850: Tuesday, 08-Mar-22 16:33:06 CST
  • datetime_custom_RFC1123: Tue, 08 Mar 2022 16:33:06 CST
  • datetime_custom_RFC1123Z: Tue, 08 Mar 2022 16:33:06 +0800
  • datetime_custom_RFC3339: 2022-03-08T16:33:06+08:00
  • datetime_custom_RFC3339Nano: 2022-03-08T16:33:06.629310528+08:00
  • datetime_custom_Kitchen: 4:33PM
  • datetime_custom_Stamp: Mar 8 16:33:06
  • datetime_custom_StampMilli: Mar 8 16:33:06.629
  • datetime_custom_StampMicro: Mar 8 16:33:06.629326
  • datetime_custom_StampNano: Mar 8 16:33:06.629330204

以及自定义日期格式化:

  • datetime_custom_20060102150405: 20220308163306

表达式是 “datetimecustom“ 拼接 Go 语言风格的日期格式模板。

  1. version: "1.1"
  2. stages:
  3. - stage:
  4. - custom-script:
  5. alias: custom-script
  6. description: 运行自定义命令
  7. version: "1.0"
  8. commands:
  9. - echo ${{ random.integer }}

${{ outputs.alias.val }}

前置任务输出值,后面的任务使用占位符获取该值

  1. - stage:
  2. - custom-script:
  3. alias: action1
  4. version: "1.0"
  5. commands:
  6. - echo "key=value" >> $METAFILE # 输出 key 值为 value 给下个任务使用
  7. - stage:
  8. - custom-script:
  9. alias: action2
  10. version: "1.0"
  11. commands:
  12. - echo ${{ outputs.action1.key }} # 使用上面任务的输出值(key对应的value)

说明:

$METAFILE 是一个文件, 写入到这个文件是固定写法

${{ params.val }}

流水线运行入参,在 yaml 文件中在 params 字段下填写多个输入框的定义,用户在运行流水线的时候就会弹出定义的输入框,输入框填入的值就会替换掉 ${{ params.key }} 占位符

  1. version: "1.1"
  2. params:
  3. - name: key1 # 定义输入框的 key
  4. required: true # 是否必填
  5. default: 111 # 没有输入值给的默认值
  6. type: int # 定义输入框的类型
  7. - name: key2
  8. required: true
  9. default: 111
  10. type: string
  11. stages:
  12. - stage:
  13. - custom-script:
  14. alias: action1
  15. description: 运行自定义命令
  16. version: "1.0"
  17. commands:
  18. - echo ${{ params.key1 }} # 替换 key1 输入框填的值
  19. - echo ${{ params.key2 }} # 替换 key2 输入框填的值

${{ dirs.alias }}

每个 action 都会有个工作目录, 可以使用 ${{ dirs.xxx }} 占位符进入和获取 action 的工作目录文件和地址

  1. version: "1.1"
  2. stages:
  3. - stage:
  4. - git-checkout:
  5. alias: git-checkout
  6. description: 代码仓库克隆
  7. version: "1.0"
  8. - stage:
  9. - custom-script:
  10. alias: custom-script
  11. description: 运行自定义命令
  12. version: "1.0"
  13. commands:
  14. - echo ${{ dirs.git-checkout }} # 打印任务工作目录的地址
  15. - cd ${{ dirs.git-checkout }} # 进入任务的工作目录
  16. - ls # 打印目录下的文件

action 之间文件传递,每个任务只有在自己的工作目录下创建的文件才能被其他任务使用,否则产生的文件将会被丢弃,$WORKDIR 或者 ${{ dirs.xxx }} 都可以获取当前任务的工作目录

  1. version: "1.1"
  2. stages:
  3. - stage:
  4. - git-checkout:
  5. alias: git
  6. description: 代码仓库克隆
  7. version: "1.0"
  8. - stage:
  9. - custom-script:
  10. alias: script1
  11. description: 运行自定义命令
  12. version: "1.0"
  13. commands:
  14. - cd ${{ dirs.git }}
  15. - touch test.txt # 在 action git 的工作目录下创建 test.txt 文件
  16. - touch test1.txt # 在 action git 的工作目录下创建 test1.txt 文件
  17. - cp test1.txt $WORKDIR # 将 action git 工作目录下的 test1.txt 文件拷贝到当前 action(script1) 的工作目录
  18. - stage:
  19. - custom-script:
  20. alias: script2
  21. description: 运行自定义命令
  22. version: "1.0"
  23. commands:
  24. - cd ${{ dirs.git }} # 进入 action git 的工作目录
  25. - ls # action script1 创建的 test.txt,test1.txt 文件不会存在
  26. - cd ${{ dirs.script1 }} # 进入 action script1 的工作目录
  27. - ls # test1.txt 文件可以看到,test.txt 文件丢失, 因为只有 test1.txt 文件拷贝到了 script1 的工作目录

tip 如果使用共享目录的方案,并行任务由于执行完成顺序的不确定性,当 Stage 完成时,共享目录里最终看到的是当前 Stage 里最后一个完成的任务的改动。

${{ configs.val }}

获取平台流水线配置

进入 DevOps > 我的应用 > 选择应用 > 应用设置 > 流水线 > 变量配置 > 选择环境

点击 增加变量,变量名称定义为 TEST_CONFIG_KEY 值为 xxx

  1. version: "1.1"
  2. stages:
  3. - stage:
  4. - custom-script:
  5. alias: custom-script
  6. description: 运行自定义命令
  7. version: "1.0"
  8. commands:
  9. - echo ${{ configs.TEST_CONFIG_KEY }}

${{ base64-decode.xxx }}

base64 解码参数

如果参数本身是一个 base64 加密过的一段内容,可以直接用base64-decode表达式将内容解码。

  1. version: "1.1"
  2. stages:
  3. - stage:
  4. - custom-script:
  5. alias: custom-script
  6. description: 运行自定义命令
  7. version: "1.0"
  8. commands:
  9. - echo ${{ base64-decode.aGVsbG8gd29ybGQh }}

如果参数的值本身包含占位符,但是在执行流水线时并不想被替换掉,也可以先将内容进行 base64 加密一下,再用base64-decode进行解码。

Action 输出 Meta

每个 Action 可以输出一组 Meta 表示执行过程中的一些信息,比如拉取的代码版本、制作好的镜像名等。 通过这些 Meta 可以很方便地在界面上看到 Action 的关键信息,当鼠标移动到节点上时,会展示如下图所示的界面:

pipeline.yml - 图5

后续 Action 可以通过 ${{ outputs.alias.val }} 获取并使用 Meta 信息。

Action 如何生成 Meta

Action 目前有以下几种生成方式,Action 开发者和使用者均可以使用。

输出日志到 STDOUT/STDERR

这是最简单的一种方式。用户只需要将 Meta 按指定格式输出到标准输出(STDOUT) 或标准错误(STDERR)。 平台会拦截这些日志,解析出 Meta 信息。

格式:

  1. action meta: key=value

解析规则

  1. 获取日志行 rawLine
  2. 删除 rawLine 前后的空格,得到 polishedLine
  3. polishedLine 必须包含前缀 action meta:;否则忽略该日志行
  4. polishedLine 删除前缀 action meta:,得到 KV
  5. 以第一个 = 作为分隔符,将 KV 分隔为两部分;如果没有 =,则忽略该日志行
  6. 得到 KV
  7. 删除 KV 前后的空格,得到 polishedKpolishedV,即为最终的一个 Meta 信息

注意

  • 输出日志时注意 value 不要换行,否则只能拿到换行前的 value;因为日志行是按行处理的

示例

  1. # key: image
  2. # value: alpine:3.12
  3. $ echo action meta: image=alpine:3.12
  4. # key: tag
  5. # value: 3.12
  6. $ echo action meta: tag=3.12
  7. # key: expr
  8. # value: a=b=c=d
  9. $ echo action meta: expr=a=b=c=d
  10. # key: c
  11. # value: dd d
  12. $ echo action meta:c = dd d

结果如图所示:

pipeline.yml - 图6

输出日志到 METAFILE 文件

使用者可以通过环境变量 METAFILE 获取到 Meta 文件的完整路径,将内容写入到该文件中即可。

文件内容支持以下两种格式:

日志行

每行格式为 k=v,每行解析规则见 日志行解析规则 5-7条。

JSON 格式

这种格式的优势是支持 value 换行

  1. {
  2. "metadata":[
  3. {
  4. "name":"k1",
  5. "value":"v1"
  6. },
  7. {
  8. "name":"k2",
  9. "value":"line1\nline2\nline3"
  10. }
  11. ]
  12. }

结果如图所示:

pipeline.yml - 图7