mirror of
https://github.com/hellodigua/ChatLab.git
synced 2026-04-23 18:19:01 +08:00
309 lines
11 KiB
YAML
309 lines
11 KiB
YAML
name: Build and Release
|
||
|
||
on:
|
||
# 手动触发
|
||
workflow_dispatch:
|
||
inputs:
|
||
version:
|
||
description: '版本号 (如 0.1.0)'
|
||
required: true
|
||
type: string
|
||
# 推送 tag 时自动触发
|
||
push:
|
||
tags:
|
||
- 'v*'
|
||
|
||
jobs:
|
||
# macOS 构建需要分架构
|
||
# better-sqlite3 等原生模块会通过 prebuild 下载对应架构的预编译二进制
|
||
build-mac:
|
||
strategy:
|
||
matrix:
|
||
include:
|
||
- os: macos-14 # 使用 ARM runner 交叉编译 x64
|
||
arch: x64
|
||
- os: macos-14 # Apple Silicon (arm64) 原生构建
|
||
arch: arm64
|
||
runs-on: ${{ matrix.os }}
|
||
steps:
|
||
- name: Checkout code
|
||
uses: actions/checkout@v4
|
||
|
||
- name: Setup Node.js
|
||
uses: actions/setup-node@v4
|
||
with:
|
||
node-version: '20'
|
||
|
||
- name: Setup pnpm
|
||
uses: pnpm/action-setup@v4
|
||
with:
|
||
version: 9
|
||
|
||
- name: Get pnpm store directory
|
||
shell: bash
|
||
run: |
|
||
echo "STORE_PATH=$(pnpm store path --silent)" >> $GITHUB_ENV
|
||
|
||
- name: Setup pnpm cache
|
||
uses: actions/cache@v4
|
||
with:
|
||
path: ${{ env.STORE_PATH }}
|
||
key: ${{ runner.os }}-${{ matrix.arch }}-pnpm-store-${{ hashFiles('**/pnpm-lock.yaml') }}
|
||
restore-keys: |
|
||
${{ runner.os }}-${{ matrix.arch }}-pnpm-store-
|
||
|
||
- name: Install dependencies
|
||
run: |
|
||
echo "node-linker=hoisted" >> .npmrc
|
||
pnpm install
|
||
|
||
# macOS 签名和公证需要的 API Key 文件
|
||
- name: Create Apple API Key File
|
||
run: |
|
||
mkdir -p ~/private_keys
|
||
echo "${{ secrets.APPLE_API_KEY }}" > ~/private_keys/AuthKey_${{ secrets.APPLE_API_KEY_ID }}.p8
|
||
|
||
- name: Build Electron app for macOS (${{ matrix.arch }})
|
||
env:
|
||
GH_TOKEN: ${{ secrets.GH_TOKEN }}
|
||
# 代码签名
|
||
CSC_LINK: ${{ secrets.CSC_LINK }}
|
||
CSC_KEY_PASSWORD: ${{ secrets.CSC_KEY_PASSWORD }}
|
||
# 公证
|
||
APPLE_API_KEY: ~/private_keys/AuthKey_${{ secrets.APPLE_API_KEY_ID }}.p8
|
||
APPLE_API_KEY_ID: ${{ secrets.APPLE_API_KEY_ID }}
|
||
APPLE_API_ISSUER: ${{ secrets.APPLE_API_ISSUER }}
|
||
# 分析服务
|
||
APTABASE_APP_KEY: ${{ secrets.APTABASE_APP_KEY }}
|
||
run: pnpm build && pnpm exec electron-builder --mac --${{ matrix.arch }} --config electron-builder.yml -p never
|
||
|
||
- name: Upload macOS artifacts (${{ matrix.arch }})
|
||
uses: actions/upload-artifact@v4
|
||
with:
|
||
name: ChatLab-mac-${{ matrix.arch }}
|
||
path: |
|
||
dist/*.dmg
|
||
dist/*.zip
|
||
dist/*.yml
|
||
dist/*.json
|
||
dist/*.blockmap
|
||
if-no-files-found: warn
|
||
retention-days: 1 # 只保留1天,Release后会上传到GitHub Releases
|
||
|
||
build-win:
|
||
runs-on: windows-latest
|
||
steps:
|
||
- name: Checkout code
|
||
uses: actions/checkout@v4
|
||
|
||
- name: Setup Node.js
|
||
uses: actions/setup-node@v4
|
||
with:
|
||
node-version: '20'
|
||
|
||
- name: Setup pnpm
|
||
uses: pnpm/action-setup@v4
|
||
with:
|
||
version: 9
|
||
|
||
- name: Get pnpm store directory
|
||
shell: bash
|
||
run: |
|
||
echo "STORE_PATH=$(pnpm store path --silent)" >> $GITHUB_ENV
|
||
|
||
- name: Setup pnpm cache
|
||
uses: actions/cache@v4
|
||
with:
|
||
path: ${{ env.STORE_PATH }}
|
||
key: ${{ runner.os }}-pnpm-store-${{ hashFiles('**/pnpm-lock.yaml') }}
|
||
restore-keys: |
|
||
${{ runner.os }}-pnpm-store-
|
||
|
||
- name: Install dependencies
|
||
run: |
|
||
echo "node-linker=hoisted" >> .npmrc
|
||
pnpm install
|
||
|
||
- name: Build Electron app for Windows
|
||
env:
|
||
GH_TOKEN: ${{ secrets.GH_TOKEN }}
|
||
# 分析服务
|
||
APTABASE_APP_KEY: ${{ secrets.APTABASE_APP_KEY }}
|
||
run: pnpm build:win
|
||
|
||
- name: Upload Windows artifacts
|
||
uses: actions/upload-artifact@v4
|
||
with:
|
||
name: ChatLab-win
|
||
path: |
|
||
dist/*.exe
|
||
dist/*.yml
|
||
dist/*.json
|
||
dist/*.blockmap
|
||
if-no-files-found: warn
|
||
retention-days: 1 # 只保留1天,Release后会上传到GitHub Releases
|
||
|
||
release:
|
||
needs: [build-mac, build-win]
|
||
runs-on: ubuntu-latest
|
||
if: startsWith(github.ref, 'refs/tags/') || github.event_name == 'workflow_dispatch'
|
||
|
||
steps:
|
||
- name: Checkout code
|
||
uses: actions/checkout@v4
|
||
|
||
- name: Download macOS artifacts (x64)
|
||
uses: actions/download-artifact@v4
|
||
with:
|
||
name: ChatLab-mac-x64
|
||
path: dist
|
||
|
||
- name: Download macOS artifacts (arm64)
|
||
uses: actions/download-artifact@v4
|
||
with:
|
||
name: ChatLab-mac-arm64
|
||
path: dist
|
||
|
||
- name: Download Windows artifacts
|
||
uses: actions/download-artifact@v4
|
||
with:
|
||
name: ChatLab-win
|
||
path: dist
|
||
|
||
- name: List files
|
||
run: ls -la dist/
|
||
|
||
- name: Generate Release Notes
|
||
id: release_notes
|
||
run: |
|
||
# 获取当前版本号
|
||
if [ "${{ github.event_name }}" == "workflow_dispatch" ]; then
|
||
CURRENT_VERSION="v${{ inputs.version }}"
|
||
else
|
||
CURRENT_VERSION="${{ github.ref_name }}"
|
||
fi
|
||
VERSION_NUMBER="${CURRENT_VERSION#v}"
|
||
|
||
echo "Current version: $CURRENT_VERSION"
|
||
|
||
# 从 changelog JSON 中提取最新版本的 summary
|
||
EN_SUMMARY=$(jq -r '.[0].summary' docs/changelogs_en.json)
|
||
CN_SUMMARY=$(jq -r '.[0].summary' docs/changelogs_cn.json)
|
||
|
||
echo "EN summary: $EN_SUMMARY"
|
||
echo "CN summary: $CN_SUMMARY"
|
||
|
||
# 生成 release notes
|
||
{
|
||
echo "## What's New"
|
||
echo ""
|
||
echo "$EN_SUMMARY"
|
||
echo ""
|
||
echo "## 更新内容"
|
||
echo ""
|
||
echo "$CN_SUMMARY"
|
||
echo ""
|
||
echo "---"
|
||
echo ""
|
||
echo "Full Changelog: [English](https://github.com/hellodigua/ChatLab/blob/main/docs/changelogs_en.json) | [中文](https://github.com/hellodigua/ChatLab/blob/main/docs/changelogs_cn.json) | [日本語](https://github.com/hellodigua/ChatLab/blob/main/docs/changelogs_ja.json)"
|
||
echo ""
|
||
echo "---"
|
||
echo ""
|
||
echo "## Download"
|
||
echo ""
|
||
echo "| Platform | File |"
|
||
echo "|-----------------|-------------|"
|
||
echo "| Mac (Apple Silicon) | [ChatLab-${VERSION_NUMBER}-arm64.dmg](https://github.com/hellodigua/ChatLab/releases/download/v${VERSION_NUMBER}/ChatLab-${VERSION_NUMBER}-arm64.dmg) |"
|
||
echo "| Mac (Intel) | [ChatLab-${VERSION_NUMBER}-x64.dmg](https://github.com/hellodigua/ChatLab/releases/download/v${VERSION_NUMBER}/ChatLab-${VERSION_NUMBER}-x64.dmg) |"
|
||
echo "| Windows | [ChatLab-${VERSION_NUMBER}-setup.exe](https://github.com/hellodigua/ChatLab/releases/download/v${VERSION_NUMBER}/ChatLab-${VERSION_NUMBER}-setup.exe) |"
|
||
} > release_notes.md
|
||
|
||
echo "Generated release notes:"
|
||
cat release_notes.md
|
||
|
||
- name: Create Release
|
||
uses: softprops/action-gh-release@v2
|
||
with:
|
||
tag_name: ${{ github.event_name == 'workflow_dispatch' && format('v{0}', inputs.version) || github.ref_name }}
|
||
name: ChatLab ${{ github.event_name == 'workflow_dispatch' && inputs.version || github.ref_name }}
|
||
body_path: release_notes.md
|
||
files: |
|
||
dist/*.exe
|
||
dist/*.dmg
|
||
dist/*.zip
|
||
dist/*.yml
|
||
dist/*.blockmap
|
||
env:
|
||
GITHUB_TOKEN: ${{ secrets.GITHUB_TOKEN }}
|
||
|
||
# 同步到 Cloudflare R2(国内镜像)
|
||
# 注意:beta/alpha/rc 等预发布版本不上传到 R2
|
||
- name: Get version for R2 upload
|
||
id: version
|
||
run: |
|
||
if [ "${{ github.event_name }}" == "workflow_dispatch" ]; then
|
||
VERSION_TAG="v${{ inputs.version }}"
|
||
else
|
||
VERSION_TAG="${{ github.ref_name }}"
|
||
fi
|
||
echo "tag=$VERSION_TAG" >> $GITHUB_OUTPUT
|
||
|
||
# 检查是否为预发布版本(包含 -beta、-alpha、-rc 等)
|
||
if [[ "$VERSION_TAG" =~ -(beta|alpha|rc)\. ]]; then
|
||
echo "is_prerelease=true" >> $GITHUB_OUTPUT
|
||
echo "Detected prerelease version: $VERSION_TAG, will skip R2 upload"
|
||
else
|
||
echo "is_prerelease=false" >> $GITHUB_OUTPUT
|
||
echo "Detected stable version: $VERSION_TAG, will upload to R2"
|
||
fi
|
||
|
||
# 上传所有文件到版本目录
|
||
- name: Upload to Cloudflare R2 (version directory)
|
||
if: steps.version.outputs.is_prerelease == 'false'
|
||
uses: ryand56/r2-upload-action@latest
|
||
with:
|
||
r2-account-id: ${{ secrets.R2_ACCOUNT_ID }}
|
||
r2-access-key-id: ${{ secrets.R2_ACCESS_KEY_ID }}
|
||
r2-secret-access-key: ${{ secrets.R2_SECRET_ACCESS_KEY }}
|
||
r2-bucket: ${{ secrets.R2_BUCKET_NAME }}
|
||
source-dir: dist
|
||
destination-dir: releases/download/${{ steps.version.outputs.tag }}
|
||
|
||
# 准备 yml 文件用于根目录上传
|
||
# 需要修改 yml 中的 path/url 为包含版本号的完整路径
|
||
- name: Prepare yml files for root upload
|
||
if: steps.version.outputs.is_prerelease == 'false'
|
||
run: |
|
||
mkdir -p dist-yml
|
||
VERSION_TAG="${{ steps.version.outputs.tag }}"
|
||
|
||
# 处理所有 yml 文件
|
||
for yml_file in dist/*.yml; do
|
||
filename=$(basename "$yml_file")
|
||
echo "Processing $filename..."
|
||
|
||
# 使用 sed 修改 path 和 url 字段,添加版本目录前缀
|
||
# path: ChatLab-x.x.x-arm64.dmg → path: vx.x.x/ChatLab-x.x.x-arm64.dmg
|
||
# url: ChatLab-x.x.x-arm64.dmg → url: vx.x.x/ChatLab-x.x.x-arm64.dmg
|
||
sed -E "s|^(path: )(.+)|\1${VERSION_TAG}/\2|g; s|^( - url: )(.+)|\1${VERSION_TAG}/\2|g" "$yml_file" > "dist-yml/$filename"
|
||
|
||
echo "Original:"
|
||
cat "$yml_file"
|
||
echo ""
|
||
echo "Modified:"
|
||
cat "dist-yml/$filename"
|
||
echo "---"
|
||
done
|
||
|
||
# 上传 yml 文件到根目录(覆盖旧版本,electron-updater 从这里检测更新)
|
||
- name: Upload yml files to Cloudflare R2 (root directory)
|
||
if: steps.version.outputs.is_prerelease == 'false'
|
||
uses: ryand56/r2-upload-action@latest
|
||
with:
|
||
r2-account-id: ${{ secrets.R2_ACCOUNT_ID }}
|
||
r2-access-key-id: ${{ secrets.R2_ACCESS_KEY_ID }}
|
||
r2-secret-access-key: ${{ secrets.R2_SECRET_ACCESS_KEY }}
|
||
r2-bucket: ${{ secrets.R2_BUCKET_NAME }}
|
||
source-dir: dist-yml
|
||
destination-dir: releases/download
|