• 如何使用 Github Action 管理 Issue


    本文作者为 360 奇舞团前端开发工程师 Daryl

    前言

    8dff5425a37a841be58635a469034ee2.png很多小伙伴打开 github 上的仓库都只使用Code查看代码,或者只是把 github 当成一个代码仓库,但是 github 还提供了很多好用的功能。

    其中,GitHub Action就是一个很好用的功能,本文将通过几个管理Issue的示例带大家了解GitHub Action:

    什么是 Github Action

    github 给所有用户都提供了临时可用的虚拟机, 我们通过创建 github action 工作流来使用这个虚拟机. 我们可以使用它来实现自动化部署、自动化测试、代码检查、管理 Issues...

    使用步骤

    在学习之前还需要准备一些资料:

    1. Github Action文档

    2. 官方仓库中有很多可以复用的 Action, 通过uses字段引用就可以直接使用了。

    3. 阮一峰的YAML 教程;

    也推荐大家使用Vscode GitHub Action插件,这个插件在登录后可以用来做语法校验,还能查看运行过的记录。

    除了这些资料之外还有些基础概念需要了解:

    1. 事件: 在工作流中可以监听 github 的一些事件, 在事件触发后执行我们定义的工作流;

    2. 上下文: github 上下文包含有关工作流运行和触发运行的事件的信息,可以读取环境变量中的大多数 github 上下文数据,并允许我们通过变量访问这些数据。

    3. 变量: 变量提供了一种存储和重用非敏感配置信息的方法。 可以将任何配置数据(如编译器标志、用户名或服务器名称)存储为变量。 变量在运行工作流的运行器计算机上插值。 在操作或工作流步骤中运行的命令可以创建、读取和修改变量。

    4. 表达式: 可以使用表达式来运算工作流程文件中的变量。

    5. 秘钥: 普通变量中存储的信息并不安全,很容易泄露,一些需要保密的信息就可以存储到秘钥中。

    如果不想去从头学习yml语法, 可以先了解一些yml的基础用法:

    • 大小写敏感

    • 使用缩进表示层级关系

    • 缩进时不允许使用 Tab 键,只允许使用空格。

    • 缩进的空格数目不重要,只要相同层级的元素左侧对齐即可


    下面开始介绍 GitHub Action 的用法

    使用 github Action ,第一步需要在项目根目录下创建.github/workflows文件夹, 所有的工作流文件都要放到这个文件夹,当事件触发时会自动执行;fd320390b995a60d40181b200cef1eed.png

    大家可以通过这个 workflow 文件示例来简单了解下各个字段的用处:

    1. name: build # workflow的名称,缺省时会使用文件名
    2. on: # workflow监听事件
    3.   push # 具体的事件
    4.     branches: # 在这些分支上才会触发
    5.       - main
    6.       - 'mona/octocat'
    7.       - 'releases/**'
    8. jobs: # 执行的工作的集合
    9.   build: # ‘build’是一个自定义的工作的id
    10.     name: rele # 当前工作的名称
    11.     runs-on: ubuntu-latest # 因为工作实际是运行在虚拟机上的,runs-on就是指定虚拟机的版本
    12.     steps: # steps是步骤的集合
    13.     - name: checkout # name是指定当前工作的名称 在workflow(工作流)文件的steps中,每个用‘-’代替缩进视为一个步骤的开始
    14.       uses: actions/checkout@v2 # uses字段是选择一个可以直接复用的action,并且在github action store中的action可以直接使用,不需要下载
    15.     - name: setting env
    16.       id: setting # id 是步骤的唯一标识符,可以使用 id 在上下文中引用该步骤
    17.       env: # 设置环境变量
    18.         NODEV: 18
    19.       run: echo "nodev=$NODEV" >> $GITHUB_OUTPUT # run字段会在命令行执行一条命令,这个命令是将"nodev=18"写入到$GITHUB_OUTPUT,这样可以为'output'添加test属性值为test1,详情参考
    20.     - name: addnode
    21.       uses: actions/setup-node@v3 # 使用node环境
    22.       with: # 为‘uses’使用的action传递参数
    23.         node-version: ${{steps.setting.output.nodev}} # 使用上面设置的变量

    在编写工作流文件之前有两件事要做:

    一. 可以在 github 或者 github action 的仓库里查找公用 action.这样可以减少很多工作量:

    常用的 action 有:

    • checkout: 帮我们自动把项目克隆到虚拟机上

    • Setup Node: 自动安装 node

    • issues-helper: 辅助处理 issues

    二. 如果没有设置 action 的读写权限,第一次运行会报错:27be5e8c4bf2d81957c5d6cefbbd7ec4.png设置权限:52bbe2c1d8090cd1dbc378dd8d5caeba.png

    再次运行:

    2346f4f49bc77397a84bf140f4c61a7a.png
    Alt text

    下面通过一些具体示例给大家介绍一些Github Action的用法:

    • 检查 issues 格式是否规范, 并关闭不规范 issue

      在开源项目中,很多人都会提一些 issue。为了方便开发者查看,可以上传 issue 模版,并且根据模版校验 issue 格式,然后关闭不符合模板格式的 issue。

      在这个示例主要介绍脚本执行怎么向github上下文注入变量,我们可以参考github 设置环境变量的教程.

      这个脚本可以配合.github/ISSUE_TEMPLATE文件夹中的 issue 模板使用的,将.md文件放入这个文件夹中就可以作为 Issue 的默认模板;

      9c7a80f20f429e4aa7c3852d33033560.png
      Alt text

      选择 Bug 提交将使用这个模板:

      1. ---
      2. name: Bug 提交
      3. about: 使用此模板来提交一个 bug。
      4. ---
      5. # BUG 提交
      6. ## 描述该错误
      7. 简明扼要地描述一下这个错误是什么。
      8. 可以添加屏幕截图以帮助解释你的问题。
      9. ## 复现
      10. 这个 BUG 的复现步骤:
      11. 1. ...
      12. 2. ...
      13. 3. ...
      14.    ...
      15.    或者添加录屏链接
      16. ## 运行结果
      17. 预期的结果:
      18. ... ...
      19. 实际的结果:
      20. ... ...
      21. ## 运行环境信息
      22. - Device: [e.g. 设备名称]
      23. - OS: [e.g. 操作系统]
      24. - Browser [e.g. 浏览器]
      25. - Version [e.g. 浏览器版本]
      26. ## 其它
      27. 在此添加关于问题的任何其它信息。

      选择功能请求将使用这个模板:

      1. ---
      2. name: 功能请求
      3. about: 使用此模板来提交一个功能请求。
      4. ---
      5. # 功能请求
      6. ## 你期望添加什么样的功能?
      7. 你期望新增功能的描述,或者示例链接。

      检查 issue 格式的脚本:

      1. // action_script/lintIssue.js
      2. const issueText = process.env.ISSUE
      3. const textSplit = issueText
      4.   .split(
      5.     `
      6. `
      7.   )
      8.   .map((str) => str.replace('\r'''))
      9. const bugHandle = () => {
      10.   // 缺少错误描述
      11.   const desIndex = textSplit.indexOf('## 描述该错误')
      12.   if (desIndex === -1) {
      13.     console.log('ISSUE_CHECK_RESULT=unqualified')
      14.     console.log('ISSUE_CHECK_REPLY=缺少错误描述')
      15.     return
      16.   }
      17.   const repeatIndex = textSplit.indexOf('## 复现')
      18.   if (repeatIndex === -1) {
      19.     console.log('ISSUE_CHECK_RESULT=unqualified')
      20.     console.log('ISSUE_CHECK_REPLY=缺少复现步骤')
      21.     return
      22.   }
      23.   const desContent = textSplit
      24.     .slice(desIndex + 1, repeatIndex)
      25.     .join('')
      26.     .replaceAll(' ''')
      27.   if (!desContent) {
      28.     console.log('ISSUE_CHECK_RESULT=unqualified')
      29.     console.log('ISSUE_CHECK_REPLY=缺少错误描述')
      30.     return
      31.   }
      32.   // 缺少复现步骤
      33.   const runResultIndex = textSplit.indexOf('## 运行结果')
      34.   if (runResultIndex === -1) {
      35.     console.log('ISSUE_CHECK_RESULT=unqualified')
      36.     console.log('ISSUE_CHECK_REPLY=缺少运行结果')
      37.     return
      38.   }
      39.   const repeatContent = textSplit
      40.     .slice(repeatIndex + 1, runResultIndex)
      41.     .join('')
      42.     .replaceAll(' ''')
      43.   if (!repeatContent) {
      44.     console.log('ISSUE_CHECK_RESULT=unqualified')
      45.     console.log('ISSUE_CHECK_REPLY=缺少复现步骤')
      46.     return
      47.   }
      48.   // 运行结果
      49.   const envIndex = textSplit.indexOf('## 运行环境信息')
      50.   if (envIndex === -1) {
      51.     console.log('ISSUE_CHECK_RESULT=unqualified')
      52.     console.log('ISSUE_CHECK_REPLY=缺少运行环境信息')
      53.     return
      54.   }
      55.   const resContent = textSplit
      56.     .slice(runResultIndex + 1, envIndex)
      57.     .join('')
      58.     .replaceAll(' ''')
      59.   if (!resContent) {
      60.     console.log('ISSUE_CHECK_RESULT=unqualified')
      61.     console.log('ISSUE_CHECK_REPLY=缺少运行结果')
      62.     return
      63.   }
      64.   // 运行环境信息
      65.   const otherIndex = textSplit.indexOf('## 其他') || textSplit.length + 1
      66.   const envContent = textSplit.slice(envIndex + 1, otherIndex)
      67.   const envReg = /\[e\.g\..{5,}\]/
      68.   let hasDevice = false
      69.   let hasOS = false
      70.   let hasBrowser = false
      71.   let hasVersion = false
      72.   let errMsg = ''
      73.   envContent.forEach((str) => {
      74.     if (str.match(envReg)) {
      75.       if (str.includes('Device')) {
      76.         hasDevice = true
      77.       } else if (str.includes('OS')) {
      78.         hasOS = true
      79.       } else if (str.includes('Browser')) {
      80.         hasBrowser = true
      81.       } else if (str.includes('Version')) {
      82.         hasVersion = true
      83.       }
      84.     }
      85.   })
      86.   if (!hasDevice) {
      87.     errMsg += '缺少设备名称;'
      88.   } else if (!hasOS) {
      89.     errMsg += '缺少操作系统名称;'
      90.   } else if (!hasBrowser) {
      91.     errMsg += '缺少浏览器名称;'
      92.   } else if (!hasVersion) {
      93.     errMsg += '缺少浏览器版本;'
      94.   }
      95.   if (errMsg) {
      96.     console.log('ISSUE_CHECK_RESULT=unqualified')
      97.     console.log('ISSUE_CHECK_REPLY=' + errMsg)
      98.   } else {
      99.     console.log('ISSUE_CHECK_RESULT=pass')
      100.   }
      101. }
      102. const featureHandle = () => {
      103.   // 缺少错误描述
      104.   const desIndex = textSplit.indexOf('## 你期望添加什么样的功能?')
      105.   const desContent = textSplit
      106.     .slice(desIndex + 1, textSplit.length)
      107.     .join('')
      108.     .replaceAll(' ''')
      109.     .replaceAll('\n''')
      110.   if (desIndex === -1 || desIndex === textSplit.length - 1 || !desContent) {
      111.     console.log('ISSUE_CHECK_RESULT=unqualified')
      112.     console.log('ISSUE_CHECK_REPLY=缺少功能描述')
      113.     return
      114.   } else {
      115.     console.log('ISSUE_CHECK_RESULT=pass')
      116.   }
      117. }
      118. if (textSplit[0] === '# BUG 提交') {
      119.   console.log('ISSUE_CHECK_TYPE=bug')
      120.   bugHandle()
      121. else if (textSplit[0] === '# 功能请求') {
      122.   console.log('ISSUE_CHECK_TYPE=feature')
      123.   featureHandle()
      124. else {
      125.   console.log('ISSUE_CHECK_TYPE=invalid')
      126.   console.log('ISSUE_CHECK_RESULT=unqualified')
      127.   console.log('ISSUE_CHECK_REPLY=这不是一个BUG或者功能请求')
      128. }

      工作流示例文件:

      1. # .github/workflows/close-non_standard-issue.yml
      2.   name: close non-standard issues
      3. on:
      4.   issues:
      5.     types: [opened, edited] # issue 打开或者编辑后
      6. jobs:
      7.   close-issue:
      8.     runs-on: ubuntu-latest
      9.     env:
      10.       ISSUE: ${{ github.event.issue.body }}
      11.     steps:
      12.       - name: 'checkout'
      13.         uses: actions/checkout@v3
      14.       - name: Setup node
      15.         uses: actions/setup-node@v3
      16.         with:
      17.           node-version: 18
      18.           registry-url: https://registry.npmjs.com/
      19.       - name: lint sh
      20.         run: node ./action_script/lintIssue.js  >> "$GITHUB_ENV" # 设置自定义github变量
      21.       - name: add-label
      22.         uses: actions-cool/issues-helper@v3
      23.         with:
      24.           actions: 'add-labels'
      25.           token: ${{ secrets.GITHUB_TOKEN }}
      26.           issue-number: ${{ github.event.issue.number }}
      27.           labels: ${{env.ISSUE_CHECK_TYPE}}
      28.       - name: 'close-issue'
      29.         if: ${{env.ISSUE_CHECK_RESULT == 'unqualified'}}
      30.         uses: actions-cool/issues-helper@v3
      31.         with:
      32.           actions: 'close-issue'
      33.           token: ${{ secrets.GITHUB_TOKEN }}
      34.           body: |
      35.             Hello @${{ github.event.issue.user.login }}.你的Issue因为下面的原因被关闭了:
      36.             ${{env.ISSUE_CHECK_REPLY}}
    • 如果是做一些简单的校验可以使用 actions-cool/issues-helper@v3 中的 check-issue:

      这个示例主要是演示运算符的使用:

      1. name: add label of non_standard Issue
      2. on:
      3.   issues:
      4.     types: [opened, edited]
      5. jobs:
      6.   check-issue:
      7.     runs-on: ubuntu-latest
      8.     steps:
      9.       - id: check-issue # 必须使用id,不然不能访问到运行结果
      10.         uses: actions-cool/issues-helper@v3
      11.         with:
      12.           actions: "check-issue"
      13.           token: ${{ secrets.GITHUB_TOKEN }}
      14.           issue-number: ${{ github.event.issue.number }}
      15.           title-includes: "【,BUG,】" # 标题包含['【','BUG','】']
      16.           body-includes: "问题描述,问题复现步骤" # 内容包含 ['问题描述','问题复现步骤']
      17.       - name: add-label
      18.         uses: actions-cool/issues-helper@v3
      19.         if: ${{steps.check-issue.outputs.check-result == 'true'}} # 如果判断条件是 'true', 继续运行
      20.         with:
      21.           actions: "add-labels"
      22.           token: ${{ secrets.GITHUB_TOKEN }}
      23.           issue-number: ${{ github.event.issue.number }} # 当前Issue的编号
      24.           labels: "bug" # 添加标签: 'bug'
    • 关闭缺少复现步骤的 Issue

      有些 Issue 虽然通过了格式检查,但是还会缺少一些步骤,或者不能复现。我们可以手动添加 label,并且在三天不活跃的情况下自动关闭。

      这个示例主要是介绍怎么执行定时任务:

      1. # .github/workflows/close-inactive-issue.yml
      2. name: close inactive issue
      3. on:
      4.   schedule:
      5.     - cron: "00 12,00,18 * * *" # 在每天标准时间的1200/00:00/18:00 执行
      6. jobs:
      7.   check-inactive-info:
      8.     runs-on: ubuntu-latest
      9.     steps:
      10.       - name: need reproduction
      11.         uses: actions-cool/issues-helper@v3
      12.         with: # 关闭有label是'need reproduction',并且三天不活跃的Issue
      13.           actions: "close-issues"
      14.           token: ${{ secrets.GITHUB_TOKEN }}
      15.           labels: "need reproduction"
      16.           inactive-day: 3
    • 添加标签时自动评论、关闭 BUG

      有些常用的评论内容,或者用的比较多的操作,每次使用都比较繁琐。我们可以在添加标签的时候自动评论,或者执行对应的操作。比如:自动评论欢迎词,关闭手动筛选出来不符合 issue 模版的 issue 并告诉用户原因。

      这个示例是介绍在.yml文件中长文本的写法

      1. name: Issue Labeled
      2. # .github/workflows/issue-labeled.yml
      3. # 在issue添加标签后回复对应的评论
      4. on:
      5.   issues:
      6.     types: [labeled]
      7. jobs:
      8.   reply-labeled:
      9.     runs-on: ubuntu-latest
      10.     steps:
      11.       # 需要帮助
      12.       - name: contribution welcome
      13.         if: github.event.label.name == 'help wanted'
      14.         uses: actions-cool/issues-helper@v3
      15.         with:
      16.           actions: "create-comment"
      17.           issue-number: ${{ github.event.issue.number }}
      18.           body: |
      19.             你好 @${{ github.event.issue.user.login }},我们完全同意你的提议/反馈,欢迎直接在此仓库 [创建一个 Pull Request](https://github.com/NI-Web-Infra-Team/vue3-template/pulls) 来解决这个问题。请将 Pull Request 发到 `dev` 分支,提供改动所需相应的 changelog、TypeScript 定义、测试用例、文档等,并确保 CI 通过,我们会尽快进行 Review,提前感谢和期待您的贡献。
      20.         # 补充复现流程
      21.       - name: need reproduction
      22.         if: github.event.label.name == 'need reproduction'
      23.         uses: actions-cool/issues-helper@v3
      24.         with:
      25.           actions: "create-comment"
      26.           issue-number: ${{ github.event.issue.number }}
      27.           body: |
      28.             你好 @${{ github.event.issue.user.login }}, 我们需要你提供一个在线的重现实例以便于我们帮你排查问题。可以通过点击 [此处](https://codepen.io/pen/) 创建或者提供一个最小化的 GitHub 仓库。3 天内未跟进此 issue 将会被自动关闭。
      29.       - name: invalid
      30.         if: github.event.label.name == 'invalid'
      31.         uses: actions-cool/issues-helper@v3
      32.         with:
      33.           actions: "create-comment, close-issue"
      34.           issue-number: ${{ github.event.issue.number }}
      35.           body: |
      36.             你好 @${{ github.event.issue.user.login }},为了能够进行高效沟通,我们对 issue 有一定的格式要求,你的 issue 因为不符合要求而被自动关闭。你可以通过模板来创建 issue 以方便我们定位错误。谢谢配合!
    • 使用 ftp 自动部署

      如果有在公网的服务器,可以在编译完成后打包发送到自己的服务器实现自动部署。 在 github 中搜索 ftp action, 第一个 SamKirkland/FTP-Deploy-Action 就是将 github 项目部署到 ftp 服务器的:

      最后这个示例介绍的是怎么设置秘钥:

      1. name: ftp send file
      2. on: workflow_dispatch # 手动部署
      3. jobs:
      4.   web-deploy:
      5.     name: Deploy
      6.     runs-on: ubuntu-latest
      7.     steps:
      8.       - name: Get latest code
      9.         uses: actions/checkout@v3 # 拉取项目代码
      10.       - name: Setup node
      11.         uses: actions/setup-node@v3 # 安装node
      12.         with:
      13.           node-version: 18
      14.           registry-url: https://registry.npmjs.com/
      15.       - name: install package
      16.         run: npm i # 安装项目依赖
      17.       - name: build
      18.         run: npm run build # 编译
      19.       - name: pack
      20.         run: zip -q -r dist.zip ./dist # 打包成zip
      21.       - name: Sync files
      22.         uses: SamKirkland/FTP-Deploy-Action@v4.3.4 # 上传到ftp服务器
      23.         with:
      24.           server: 10.52.0.x # 服务器地址
      25.           username: testusername # 服务器用户名
      26.           password: ${{ secrets.ftp_password }} # 服务器密码
      • 设置在 action 中可以访问的秘钥

        服务器密码secrets.ftp_password,就是在 github 中设置的秘钥:b3d99db7d108040500fff6d37fa69e6e.png867d509abfe9593c34cc56843d76a50d.pngb7c069e7c56c2492b06faab1c25cedc2.png

    结语

    通过使用 Github Action 来管理 Issue,可以有效的提高生产力和效率,在自动化、协作、代码质量管理等方面都有提升,并帮助我们更好地组织和管理问题。文章篇幅有限,我们暂且介绍到这里,感兴趣的小伙伴们可以再自行探索。

    - END -

    关于奇舞团

    奇舞团是 360 集团最大的大前端团队,代表集团参与 W3C 和 ECMA 会员(TC39)工作。奇舞团非常重视人才培养,有工程师、讲师、翻译官、业务接口人、团队 Leader 等多种发展方向供员工选择,并辅以提供相应的技术力、专业力、通用力、领导力等培训课程。奇舞团以开放和求贤的心态欢迎各种优秀人才关注和加入奇舞团。

    bcac0ea2df12f76a06b72b602213b300.png

  • 相关阅读:
    tkmybatis 权威指南 官方文档
    完美PDF打印:PDFPrinting.NET Crack
    基于LEX的词法分析实验
    【小呆的热力学笔记】热力学第一定律
    李宏毅机器学习笔记-transformer
    Linearization
    python-opencv 之开运算、闭运算、形态学梯度、“礼帽”和“黑帽”
    含文档+PPT+源码等]精品基于springboot幼儿园管理系统包运行成功]Java毕业设计SpringBoot项目源码
    HCIA-HarmonyOS设备开发V2.0证书
    Jetson Agx Xavier平台V4L2 qeues是否可优化问题
  • 原文地址:https://blog.csdn.net/qiwoo_weekly/article/details/134410841