在开发名为 Textpod(用 Rust 编写的笔记应用)时,我需要自动化地在 GitHub 上完成构建和发布。 以下内容(以及相应的 YAML 配置文件)演示了整套自动化流程的配置步骤,包括:
- 为 Windows、Linux、macOS (Intel + ARM) 构建二进制
- 将这些构建产物及校验文件上传到最新的 GitHub Release
- 发布到 crates.io
- 构建面向 amd64 和 arm64 的精简 Docker 镜像,并推送到 Docker Hub
一、触发条件:GitHub Release
首先,我们以 release 事件作为执行触发器。YAML 配置大致如下:
on:
release:
types:
- created
当我们在 GitHub 中创建一个新的 Release 时,这些后续的构建及发布任务就会自动执行。
二、在 Linux 环境中构建 Linux 与 Windows 二进制
可以在同一个 Linux 环境中通过 rustup
对不同的目标进行交叉编译(例如 Windows 和 Linux)。
jobs:
linux_windows:
runs-on: ubuntu-latest
steps:
- name: Checkout the repository
uses: actions/checkout@v2
- name: Install Linux and Windows Cross Compilers
run: sudo apt-get install --yes --no-install-recommends musl-tools gcc-mingw-w64-x86-64-win32
- name: Install rustup targets
run: rustup target add x86_64-unknown-linux-musl x86_64-pc-windows-gnu
- name: Build the executable
run: cargo build --release --target x86_64-unknown-linux-musl --target x86_64-pc-windows-gnu
- name: Tar x86_64 binary
run: tar -czvf textpod-gnu-linux-x86_64.tar.gz -C target/x86_64-unknown-linux-musl/release textpod
- name: Zip windows binary
run: zip -j textpod-windows.zip target/x86_64-pc-windows-gnu/release/textpod.exe
- name: Generate SHA256 checksums
run: |
shasum -a 256 textpod-gnu-linux-x86_64.tar.gz > textpod-gnu-linux-x86_64.tar.gz.sha256
shasum -a 256 textpod-windows.zip > textpod-windows.zip.sha256
- name: Upload release binaries
uses: alexellis/upload-assets@0.4.0
env:
GITHUB_TOKEN: ${{ github.token }}
with:
asset_paths: '["textpod-gnu-linux-x86_64.tar.gz", "textpod-windows.zip", "textpod-gnu-linux-x86_64.tar.gz.sha256", "textpod-windows.zip.sha256"]'
构建说明
- Step 4 (
cargo build
) 一次性构建两个目标:x86_64-unknown-linux-musl
x86_64-pc-windows-gnu
- 后续分别将 Linux 产物打包为
tar.gz
文件、Windows 产物打包为zip
文件。 - 生成各自的 SHA256 校验文件:
textpod-gnu-linux-x86_64.tar.gz.sha256
与textpod-windows.zip.sha256
。 - 使用
alexellis/upload-assets@0.4.0
Action 将这四个文件一并上传到 GitHub Release。
注意:
${{ github.token }}
是 GitHub 自动提供的内置 Token,不需要你额外创建或配置。
三、在 macOS 环境中构建 x86 和 ARM 二进制
对 macOS 的构建与上述类似,但需要指定在 macOS 平台下(runs-on: macos-latest