Github Actions 实现 Node.js 项目的 CICD 环境搭建

本文涉及的产品
日志服务 SLS,月写入数据量 50GB 1个月
简介: 本文我们介绍 CI / CD 的概念, 主要是帮助完成软件开发的自动化构建、测试和发布环节。同时会使用 Gtihub Actions 实现一个 Node.js 项目的自动化部署环境的搭建。

Github Actions 实现 Node.js 项目的 CICD 环境搭建

前言

上篇文章《一文掌握 Node.js 的多进程模型和项目的部署》介绍了 Node.js 的多进程模型,以及如何以手动的方式将 Node.js 项目部署到服务器上。但是这种手动部署的方式实在是太落伍了。如今敏捷开发和 DevOps 的理念已深入人心,解放生产力、专注开发、自动化构建、测试、发布成了我们的追求。

本文将以 Github Actions 为例,介绍如何自动化部署一个简单的 Node 项目,主要内容有:

  • CI / CD 简介
  • Github Actions 是什么
  • 工作流配置文件的编写
  • 自动化部署 Node.js 项目

CICD

DevOps

DevOps 的理念并不复杂,简单说就是通过一套工具链,高效可靠的持续向用户交付产品。

DevOps 并没有标准化的规定,每个团队在落地实践时也不尽相同,但大体上它是这样一套流程:

它将软件产品开发的完整生命周期划分为几个不同的阶段,按照图里所描述的有:

  • Plan:计划
  • Code:开发
  • Build:构建
  • Test:测试
  • Release:发布
  • Deploy:部署
  • Operate:运维
  • Monitor:监控,并持续反馈,形成一个闭环

按照本文的写作背景,我们已经完成了代码的编写工作,接下来要进入的就是 Code 之后的阶段了。其中我们重点关注 Build、Test、Release 和 Deloy 这四个阶段,它们其实就是本文的重头戏 CI / CD。

CICD

CI,Continuous Integration持续集成,是指在开发过程中,持续的将开发者编写的代码多次的合并到主干上开发人员每次提交代码之后,都会触发自动构建和测试,以尽快发现问题。如果测试通过,则将新代码合并到主干上。如果测试失败,则反馈给开发人员,进行bug的修复。

上面 DevOps 流程图中,Build 和 Test 阶段,加起来就是 CI。

CDContinuous Delivery 持续交付Continuous Deploy 持续部署的简称。

交付和部署是两个容易混淆的概念。其实他们的含义并不十分相同。持续交付是指在完成 CI 中的自动化构建和测试之后,自动将代码发布到存储库。也就是说,持续交付的目的是生成一套可随时部署到生产环境的代码。持续部署则是持续交付的延伸,是指自动化的将代码部署到生产环境中。上图 DevOps 中的 Release 阶段就属于持续交付,Deploy 阶段就是持续部署。

CICD 工具

通过上面简单的介绍,我们知道 CICD 虽然美丽,但是要完成这样一套流程,具体要做的事情还是挺多的。市面上有很多优秀的 CICD 工具,可供我们使用:

  • Jenkins:基于 Java、跨平台、开源、独立的 CI/CD 工具,有大量的插件帮助完成软件开发过程的自动化构建、测试和部署。

  • Travis CI:基于云平台的、开源的 CICD 工具,可用于构建托管在 Github、Bitbucket、GitLab上的项目。

  • Circle CI:与 Travis CI 同样是云平台构建,支持 Github、Bitbucket、GitLab。

202208051743559.webp

  • Github Actions:使用云平台构建托管在 Github 之上的项目。

对于 Jenkins,需要先准备一台服务器安装好环境。而后三者全部是基于云平台的工具,我们只需要将代码托管到对应的平台,在让它们介入即可,使用起来会更加简单。

所以接下来我们以 Github Actions 为例进行演示。

Github Actions

简介

GitHub Actions 是微软推出的一款世界级(官网是这么说的)的 CI / CD 工具,可以帮助轻松实现软件开发工作流的自动化。如果代码托管在 Github中,可以直接使用它来完成构建、测试和部署。

主要特点

  • 多平台:支持主流的 Linux、Windows、 macOS 虚拟机和容器,可以方便的项目的构建、测试和部署,也支持自定义运行器。
  • 多语言: 支持 Node.js、Python、Java、Ruby、PHP、Go、Rust、.NET 等语言,以您选择的语言构建、测试和部署应用程序。
  • 矩阵构建:使用矩阵工作流可同时跨多个操作系统和运行时版本进行测试,节省大量时间。
  • 实时日志:在仓库 Actions 面板中可以查看实时日志,以丰富的颜色和表情符号组织的日志可以一眼看出问题所在
  • 多容器测试:在工作流配置中添加 docker-compose 即可完成对 web 服务和数据库的测试
  • 社区支持:官方和社区有丰富的 Actions 可供选用,只要你能想到的需求,基本都能找到现成的方案

开源仓库免费使用

在 Github 上托管的开源仓库,可以免费使用该功能。

私有仓库,根据账号类型的不同,每月可享有 2000分钟到50000分钟的免费时长,超出后会根据运行器(即构建我们的代码的虚拟机)的操作系统按时长收费。

image-20220805195755295

核心概念

workflow 工作流

工作流是一种可配置的自动化过程,用于完成一项或多项工作。可以将其理解为完成一次 CI / CD 的 过程

我们可以在仓库中 .github/workflows 目录下创建一个 YAML 格式的配置文件,在配置文件中定义一个工作流。

配置文件的名字可以自定义,没有特殊要求。同时,还可以创建多个 YAML 文件来执行不同的任务集。

event 事件

配置文件中通过 on 属性来指定工作流执行的时机,可以在仓库中的事件(比如 push)触发时运行,也可以手动触发,或使用定时器触发。

有关可使用的事件的更多介绍,可参阅“触发工作流程的事件”。

job 任务

一个工作流由一个或多个 jobs (任务集)组成,默认情况下任务集并行运行。

一个任务集可包含多个 job (任务),每个任务都需要指定一个 id。

runner 运行器

在一个任务中,需要通过 runs-on 指定任务运行的虚拟机环境。Github 提供了常用的 Ubuntu,Windows Server 和 MacOS 虚拟机环境可供选择。这个虚拟机在文档中叫作 “runner” 运行器。更多可供选择的运行器可访问“选择 GitHub 托管的运行器”。

Runner 对于不同的操作系统有不同的配置。Windows 和 Linux 虚拟机的硬件规格是:

  • 2 核 CPU (x86_64)
  • 7 GB RAM
  • 14 GB SSD 空间

MacOS 虚拟机的硬件规格:

  • 3 核 CPU (x86_64)
  • 14 GB RAM
  • 14 GB SSD 空间

除了使用 Github 提供的运行器,也可以使用自己的服务器作为一个运行器。

setp 步骤

任务的具体工作由 steps 指定,它是由多个 step 分步组成的一组步骤集合。一个步骤可能是一个 shell 脚本,可能是一个动作(action)。

action 动作

Action 是 GitHub Actions 平台的一个自定义的应用,是最好用的特性之一。Action 就是将一些需要频繁多次执行的任务进行了封装。比如有一个最常用的一个 checkout action ,它可以 GitHub 中拉取你的仓库中的代码,被任务中后续的 step 使用。

Actions 市场有非常丰富的 action 可供使用,也可以实现符合自己需求的 action。

了解了 CI CD 的概念以及 Github Actions 的基本用法之后,我们来实践一下,加深理解。

开始实践

1. 创建 Github 仓库

使用 Gtihub Actions,项目必须托管在 Github 上。所以先建一个仓库。

2. 本地开发

本文的演示项目,依然是上文中的 express 项目,点击访问仓库。它主要提供了一个 /api/user 接口,返回10条 mock 的数据,不涉及到数据库操作。

初始目录结构如下:

image-20220806150939415

初始化 git 环境:

git init

在根目录新建 .gitignore 文件,或者从其他项目中拷贝一份使用。作用是将一些不需要上传至远程仓库的文件过滤掉:

node_modules
pnpm-debug.log*

然后将本地仓库关联到 Github 远程仓库:

# Github 默认主分支为 main,git 工具默认为 master,修改以统一
git branch -M main  

git remote add origin git@github.com:hsyq/github-actions-express.git

然后完成一次推送:

git add .
git commit -m 'init'
git push origin main

3. 部署服务器的环境准备

Github Actions 为我们提供了自动化构建、测试的虚拟机环境,叫作 runner 运行器;项目最终要部署到我们自己的服务器上,所以需要先准备好服务器的环境,主要是:

  • 安装 node 环境
  • 安装 pm2 进程管理工具
  • nginx 端口转发

这三个步骤在上文中演示过了,所以不再过多介绍。不熟悉的朋友可以看下《一文掌握 Node.js 的多进程模型和项目的部署》

主要作用就是,node 环境提供 express.js 项目运行的环境,pm2 来管理应用,保证应用的稳健运行,nginx 将请求从80端口转发到 express 项目的3000端口。

在服务器中我们同样使用了 pnpm 作为包管理工具,所以在安装好 node 之后,还需要再装一下 pnpm :

npm install -g pnpm

其实这个环节还是有些繁琐,后面的文章我们会引入 docker 容器,在 Github Actions 中将应用打包成一个镜像,直接发布到服务器上创建容器,就不再需要这个环节了。

4. 配置仓库的 secrets

在工作流配置文件中会用到服务器地址,用户名密码等敏感信息。直接将它们暴露出去是很危险的操作。所以 Github 仓库提供了 Secrets,将一些敏感信息保存为加密后的环境变量,可供在工作流中通过 ${{ expression }} 的表达式使用。

进入仓库设置面板,来到 Security - Secrets -Actions,创建 repository secret:

image-20220806000151832

演示项目的工作流配置文件需要三个 secret:

  • SERVER_HOST:部署服务器 IP
  • SERVER_USER:登录服务器的用户
  • SERVER_PASS:登录服务器的密码

本文案例中使用了传统的用户名密码的方式来登录服务器。 其实更推荐使用 SSH 密钥对的方式,会更加安全。后面的文章会进行演示。

分别创建这三个 secret:

image-20220806003048318

image-20220806003302475

5. 创建工作流配置文件

到目前为止,已经准备好了所需要的环境,开始创建一份工作流配置文件。

有两种方式,一是可以到仓库的 Actions 面板中,在线创建一个配置。二是直接在本地项目中编写,然后提交上去。

方式1

选择一个 node.js 项目工作流的模板:

image-20220806152527033

image-20220806153125903

根据 action 的文档,完成我们自己的工作流配置文件的编写:

image-20220806153228438

编辑好配置文件后,点击右上角的提交按钮:

image-20220806153603735

方式2

在项目根目录创建 .github/workflows/express.yml ,然后推送到仓库中。

本文使用第二种进行操作,每一步配置的具体含义已经写好了注释,方便理解:

# workflow 工作流的名字,自定义
name: Express App

# 配置触发条件
on:
  push: # 监听到 main 分支的 push 动作
    branches:
      - main

# 工作流的任务集配置
jobs:
  # 定义一个 job,id 为 build
  build:
    # 指定任务执行的运行器。latest 表示是 GitHub 提供的最新稳定映像,但可能不是操作系统供应商提供的最新版本。
    runs-on: ubuntu-latest

    # 定义 job 的具体步骤
    steps: # 每一个 step 或者是执行一个 action,或者是执行一个命令
      - name: Checkout # step 的名字,方便作日志排查
        uses: actions/checkout@v3  # uses 表示该步骤使用一个 action 。斜线前面的 'actions' 表示这是官方的action

      - name: Deploy to Server # 执行部署任务
        uses: cross-the-world/ssh-scp-ssh-pipelines@latest 
        with:
          host: ${{ secrets.SERVER_HOST }}
          user: ${{ secrets.SERVER_USER }}
          pass: ${{ secrets.SERVER_PASS }}

          # 由于网路情况,很容易超时,设置为60s
          connect_timeout: 60s

          # 将工作目录下的文件全部拷贝到部署服务器的工作目录
          scp: |
            ./* => /www/github-actions-express
          # 完成拷贝后在部署服务器执行的命令:进入项目目录,安装生产依赖,并使用 pm2 管理
          last_ssh: |
            cd /www/github-actions-express
            pnpm i --prod
            pm2 reload app.config.js

一些说明:

  • checkout action:从 Github 仓库中 checkout 代码到 Github Actions 的 运行器中使用
  • ssh-scp-ssh-pipelines:一个第三方的 action,可以使用 scp 命令向其他服务器传输文件,配置中的作用就是将 Github Actions 运行器的项目文件拷贝到我们自己的服务器中,并执行一些命令

配至好工作流之后,只要向 main 分支执行push 操作,Github Actions 就会开始工作。

6. 测试

将本地编写好的工作流配置文件,推送到远程仓库:

git add .
git commit -m 'add workflow'
git push origin main

然后来到 Github 仓库中,切换到 Actions 面板,可以看到有一个工作流正在运行中:

image-20220806165931528

根据工作流配置文件中事件的配置,Github Actions 监听到了 main 分支上的 push 动作,于是开始执行这个工作流。

这个过程持续的时间可能会比较长,甚至会出现超时出错,需要耐心等待一下。

等待工作流执行成功后,工作流状态变为绿色:

可以点开看一下,里面有整个构建任务运行的日志信息:image-20220806170131701

image-20220806170257663

登录到云服务器中,切换到项目目录,发现项目已经从 Github 仓库传输到自己的服务器中了:

cd /www/github-actions-express
ls

image-20220806135420672

由于在工作流中的配置,使用 action 传输完成后,会执行一些命令:

cd /www/github-actions-express
pnpm i --prod
pm2 reload app.config.js

所以现在项目应该已经由 pm2 托管了,打开 pm2 的管理面板:

pm2 list

image-20220806140006884

项目已经在正常运行中了。

本地浏览器访问接口 ironfan.site/api/users

image-20220806140156165

小结

到这里,一个基于 Github Actions 实现的 Node.js 项目自动化部署的环境就搭建好了。

通过一张流程图来总结当前的这个流程:

image-20220806174233091

在项目后续开发中,当本地的项目代码更新以后,只需要推送到 Github 仓库的 main 分支上,就能触发一次工作流的执行。Gtihub Actions 会帮我们把项目代码传输到服务器上,并在服务器上执行相关的命令,比如安装依赖,使用 pm2 重载应用。

总结

本文中,我们介绍了 CI / CD 的概念, 主要是帮助完成软件开发的自动化构建、测试和发布环节。

同时介绍了 Github Actions,它是由微软推出的功能非常强大的CI / CD 工具。我们介绍了它的核心概念,以及常用的配置。

最后演示了使用 Github Actions 完成一个 Node.js 项目的自动化部署。

本次演示也有一些遗憾,虽然前文提到了 CI 中的自动化构建和测试。但是演示项目非常简单,无需构建,也没有准备测试环境,仅仅是演示了持续部署的环节。下次会准备一个 vue 项目,结合 docker 容器,完成一次完整 CI CD 的流程构建。

CI / CD 的内涵非常丰富,没办法通过一篇简单的文章就能介绍完整。我也是半路出家,文中如有描述不准确和错误的地方,请朋友们在评论区留言指出,感谢大家的指导!

目录
相关文章
|
3月前
|
编解码 Oracle Java
java9到java17的新特性学习--github新项目
本文宣布了一个名为"JavaLearnNote"的新GitHub项目,该项目旨在帮助Java开发者深入理解和掌握从Java 9到Java 17的每个版本的关键新特性,并通过实战演示、社区支持和持续更新来促进学习。
99 3
|
2月前
|
JavaScript 前端开发 安全
JavaScript与TypeScript的对比,分析了两者的特性及在实际项目中的应用选择
本文深入探讨了JavaScript与TypeScript的对比,分析了两者的特性及在实际项目中的应用选择。JavaScript以其灵活性和广泛的生态支持著称,而TypeScript通过引入静态类型系统,提高了代码的可靠性和可维护性,特别适合大型项目。文章还讨论了结合使用两种语言的优势,以及如何根据项目需求和技术背景做出最佳选择。
65 4
|
2月前
|
CDN
如何在项目中使用Moment.js库?
如何在项目中使用Moment.js库?
|
3月前
|
JavaScript 测试技术 API
跟随通义灵码一步步升级vue2(js)项目到vue3版本
Vue 3 相较于 Vue 2 在性能、特性和开发体验上都有显著提升。本文介绍了如何利用通义灵码逐步将 Vue 2 项目升级到 Vue 3,包括备份项目、了解新特性、选择升级方式、升级依赖、迁移组件和全局 API、调整测试代码等步骤,并提供了注意事项和常见问题的解决方案。
108 4
|
4月前
|
算法 JavaScript 前端开发
第一个算法项目 | JS实现并查集迷宫算法Demo学习
本文是关于使用JavaScript实现并查集迷宫算法的中国象棋demo的学习记录,包括项目运行方法、知识点梳理、代码赏析以及相关CSS样式表文件的介绍。
第一个算法项目 | JS实现并查集迷宫算法Demo学习
|
3月前
|
SQL JavaScript 关系型数据库
node博客小项目:接口开发、连接mysql数据库
【10月更文挑战第14天】node博客小项目:接口开发、连接mysql数据库
|
3月前
|
JavaScript 前端开发 测试技术
JavaScript与TypeScript:为何TypeScript成为大型项目的首选
JavaScript与TypeScript:为何TypeScript成为大型项目的首选
37 1
|
3月前
|
人工智能 JavaScript 网络安全
ToB项目身份认证AD集成(三完):利用ldap.js实现与windows AD对接实现用户搜索、认证、密码修改等功能 - 以及针对中文转义问题的补丁方法
本文详细介绍了如何使用 `ldapjs` 库在 Node.js 中实现与 Windows AD 的交互,包括用户搜索、身份验证、密码修改和重置等功能。通过创建 `LdapService` 类,提供了与 AD 服务器通信的完整解决方案,同时解决了中文字段在 LDAP 操作中被转义的问题。
|
3月前
|
JavaScript API 开发工具
使用GitHub Actions自动发布electron多端安装程序
使用GitHub Actions自动发布electron多端安装程序
59 8
|
4月前
vite.config.js中vite.defineConfig is not defined以及创建最新版本的vite项目
本文讨论了在配置Vite项目时遇到的`vite.defineConfig is not defined`错误,这通常是由于缺少必要的导入语句导致的。文章还涉及了如何创建最新版本的Vite项目以及如何处理`configEnv is not defined`的问题。
241 3
vite.config.js中vite.defineConfig is not defined以及创建最新版本的vite项目