mirror of
https://github.com/lobehub/lobehub.git
synced 2026-03-26 13:19:34 +07:00
* add claude dedupe issue workflow * add triage * add auto close duplicate * improve triage * improve
357 lines
12 KiB
YAML
357 lines
12 KiB
YAML
name: Desktop PR Build
|
||
|
||
on:
|
||
pull_request:
|
||
types: [synchronize, labeled, unlabeled] # PR 更新或标签变化时触发
|
||
|
||
# 确保同一时间只运行一个相同的 workflow,取消正在进行的旧的运行
|
||
concurrency:
|
||
group: ${{ github.ref }}-${{ github.workflow }}
|
||
cancel-in-progress: true
|
||
|
||
# Add default permissions
|
||
permissions: read-all
|
||
|
||
env:
|
||
PR_TAG_PREFIX: pr- # PR 构建版本的前缀标识
|
||
|
||
jobs:
|
||
test:
|
||
name: Code quality check
|
||
# 添加 PR label 触发条件,只有添加了 trigger:build-desktop 标签的 PR 才会触发构建
|
||
if: contains(github.event.pull_request.labels.*.name, 'trigger:build-desktop')
|
||
runs-on: ubuntu-latest # 只在 ubuntu 上运行一次检查
|
||
steps:
|
||
- name: Checkout base
|
||
uses: actions/checkout@v5
|
||
with:
|
||
fetch-depth: 0
|
||
|
||
- name: Setup Node.js
|
||
uses: actions/setup-node@v5
|
||
with:
|
||
node-version: 22
|
||
package-manager-cache: false
|
||
|
||
- name: Install bun
|
||
uses: oven-sh/setup-bun@v2
|
||
with:
|
||
bun-version: 1.2.23
|
||
|
||
- name: Install deps
|
||
run: bun i
|
||
env:
|
||
NODE_OPTIONS: --max-old-space-size=6144
|
||
|
||
- name: Lint
|
||
run: bun run lint
|
||
env:
|
||
NODE_OPTIONS: --max-old-space-size=6144
|
||
|
||
version:
|
||
name: Determine version
|
||
# 与 test job 相同的触发条件
|
||
if: contains(github.event.pull_request.labels.*.name, 'trigger:build-desktop')
|
||
runs-on: ubuntu-latest
|
||
outputs:
|
||
# 输出版本信息,供后续 job 使用
|
||
version: ${{ steps.set_version.outputs.version }}
|
||
steps:
|
||
- uses: actions/checkout@v5
|
||
with:
|
||
fetch-depth: 0
|
||
|
||
- name: Setup Node.js
|
||
uses: actions/setup-node@v5
|
||
with:
|
||
node-version: 22
|
||
package-manager-cache: false
|
||
|
||
# 主要逻辑:确定构建版本号
|
||
- name: Set version
|
||
id: set_version
|
||
run: |
|
||
# 从 apps/desktop/package.json 读取基础版本号
|
||
base_version=$(node -p "require('./apps/desktop/package.json').version")
|
||
|
||
# PR 构建:在基础版本号上添加 PR 信息
|
||
pr_number="${{ github.event.pull_request.number }}"
|
||
ci_build_number="${{ github.run_number }}" # CI 构建编号
|
||
version="0.0.0-nightly.pr${pr_number}.${ci_build_number}"
|
||
echo "version=${version}" >> $GITHUB_OUTPUT
|
||
echo "📦 Release Version: ${version} (based on base version ${base_version})"
|
||
|
||
env:
|
||
NODE_OPTIONS: --max-old-space-size=6144
|
||
|
||
# 输出版本信息总结,方便在 GitHub Actions 界面查看
|
||
- name: Version Summary
|
||
run: |
|
||
echo "🚦 Release Version: ${{ steps.set_version.outputs.version }}"
|
||
|
||
build:
|
||
needs: [version, test]
|
||
name: Build Desktop App
|
||
runs-on: ${{ matrix.os }}
|
||
strategy:
|
||
matrix:
|
||
os: [macos-latest, macos-15-intel, windows-2025, ubuntu-latest]
|
||
steps:
|
||
- uses: actions/checkout@v5
|
||
with:
|
||
fetch-depth: 0
|
||
|
||
- name: Install pnpm
|
||
uses: pnpm/action-setup@v4
|
||
with:
|
||
run_install: false
|
||
|
||
- name: Setup Node.js
|
||
uses: actions/setup-node@v5
|
||
with:
|
||
node-version: 22
|
||
package-manager-cache: false
|
||
|
||
# node-linker=hoisted 模式将可以确保 asar 压缩可用
|
||
- name: Install dependencies
|
||
run: pnpm install --node-linker=hoisted
|
||
|
||
- name: Install deps on Desktop
|
||
run: npm run install-isolated --prefix=./apps/desktop
|
||
|
||
# 设置 package.json 的版本号
|
||
- name: Set package version
|
||
run: npm run workflow:set-desktop-version ${{ needs.version.outputs.version }} nightly
|
||
|
||
# macOS 构建处理
|
||
- name: Build artifact on macOS
|
||
if: runner.os == 'macOS'
|
||
run: npm run desktop:build
|
||
env:
|
||
# 设置更新通道,PR构建为nightly,否则为stable
|
||
UPDATE_CHANNEL: "nightly"
|
||
APP_URL: http://localhost:3015
|
||
DATABASE_URL: "postgresql://postgres@localhost:5432/postgres"
|
||
# 默认添加一个加密 SECRET
|
||
KEY_VAULTS_SECRET: "oLXWIiR/AKF+rWaqy9lHkrYgzpATbW3CtJp3UfkVgpE="
|
||
# macOS 签名和公证配置
|
||
CSC_LINK: ${{ secrets.APPLE_CERTIFICATE_BASE64 }}
|
||
CSC_KEY_PASSWORD: ${{ secrets.APPLE_CERTIFICATE_PASSWORD }}
|
||
NEXT_PUBLIC_DESKTOP_PROJECT_ID: ${{ secrets.UMAMI_NIGHTLY_DESKTOP_PROJECT_ID }}
|
||
NEXT_PUBLIC_DESKTOP_UMAMI_BASE_URL: ${{ secrets.UMAMI_NIGHTLY_DESKTOP_BASE_URL }}
|
||
|
||
# allow provisionally
|
||
CSC_FOR_PULL_REQUEST: true
|
||
APPLE_ID: ${{ secrets.APPLE_ID }}
|
||
APPLE_APP_SPECIFIC_PASSWORD: ${{ secrets.APPLE_APP_SPECIFIC_PASSWORD }}
|
||
APPLE_TEAM_ID: ${{ secrets.APPLE_TEAM_ID }}
|
||
|
||
# Windows 平台构建处理
|
||
- name: Build artifact on Windows
|
||
if: runner.os == 'Windows'
|
||
run: npm run desktop:build
|
||
env:
|
||
# 设置更新通道,PR构建为nightly,否则为stable
|
||
UPDATE_CHANNEL: "nightly"
|
||
APP_URL: http://localhost:3015
|
||
DATABASE_URL: "postgresql://postgres@localhost:5432/postgres"
|
||
KEY_VAULTS_SECRET: "oLXWIiR/AKF+rWaqy9lHkrYgzpATbW3CtJp3UfkVgpE="
|
||
NEXT_PUBLIC_DESKTOP_PROJECT_ID: ${{ secrets.UMAMI_NIGHTLY_DESKTOP_PROJECT_ID }}
|
||
NEXT_PUBLIC_DESKTOP_UMAMI_BASE_URL: ${{ secrets.UMAMI_NIGHTLY_DESKTOP_BASE_URL }}
|
||
# 将 TEMP 和 TMP 目录设置到 C 盘
|
||
TEMP: C:\temp
|
||
TMP: C:\temp
|
||
|
||
# Linux 平台构建处理
|
||
- name: Build artifact on Linux
|
||
if: runner.os == 'Linux'
|
||
run: npm run desktop:build
|
||
env:
|
||
# 设置更新通道,PR构建为nightly,否则为stable
|
||
UPDATE_CHANNEL: "nightly"
|
||
APP_URL: http://localhost:3015
|
||
DATABASE_URL: "postgresql://postgres@localhost:5432/postgres"
|
||
KEY_VAULTS_SECRET: "oLXWIiR/AKF+rWaqy9lHkrYgzpATbW3CtJp3UfkVgpE="
|
||
NEXT_PUBLIC_DESKTOP_PROJECT_ID: ${{ secrets.UMAMI_NIGHTLY_DESKTOP_PROJECT_ID }}
|
||
NEXT_PUBLIC_DESKTOP_UMAMI_BASE_URL: ${{ secrets.UMAMI_NIGHTLY_DESKTOP_BASE_URL }}
|
||
|
||
# 处理 macOS latest-mac.yml 重命名 (避免多架构覆盖)
|
||
- name: Rename macOS latest-mac.yml for multi-architecture support
|
||
if: runner.os == 'macOS'
|
||
run: |
|
||
cd apps/desktop/release
|
||
if [ -f "latest-mac.yml" ]; then
|
||
# 使用系统架构检测,与 electron-builder 输出保持一致
|
||
SYSTEM_ARCH=$(uname -m)
|
||
if [[ "$SYSTEM_ARCH" == "arm64" ]]; then
|
||
ARCH_SUFFIX="arm64"
|
||
else
|
||
ARCH_SUFFIX="x64"
|
||
fi
|
||
|
||
mv latest-mac.yml "latest-mac-${ARCH_SUFFIX}.yml"
|
||
echo "✅ Renamed latest-mac.yml to latest-mac-${ARCH_SUFFIX}.yml (detected: $SYSTEM_ARCH)"
|
||
ls -la latest-mac-*.yml
|
||
else
|
||
echo "⚠️ latest-mac.yml not found, skipping rename"
|
||
ls -la latest*.yml || echo "No latest*.yml files found"
|
||
fi
|
||
|
||
# 上传构建产物
|
||
- name: Upload artifact
|
||
uses: actions/upload-artifact@v4
|
||
with:
|
||
name: release-${{ matrix.os }}
|
||
path: |
|
||
apps/desktop/release/latest*
|
||
apps/desktop/release/*.dmg*
|
||
apps/desktop/release/*.zip*
|
||
apps/desktop/release/*.exe*
|
||
apps/desktop/release/*.AppImage
|
||
apps/desktop/release/*.deb*
|
||
apps/desktop/release/*.snap*
|
||
apps/desktop/release/*.rpm*
|
||
apps/desktop/release/*.tar.gz*
|
||
retention-days: 5
|
||
|
||
# 合并 macOS 多架构 latest-mac.yml 文件
|
||
merge-mac-files:
|
||
needs: [build, version]
|
||
name: Merge macOS Release Files for PR
|
||
runs-on: ubuntu-latest
|
||
permissions:
|
||
contents: write
|
||
steps:
|
||
- name: Checkout repository
|
||
uses: actions/checkout@v5
|
||
|
||
- name: Setup Node.js
|
||
uses: actions/setup-node@v5
|
||
with:
|
||
node-version: 22
|
||
package-manager-cache: false
|
||
|
||
- name: Install bun
|
||
uses: oven-sh/setup-bun@v2
|
||
with:
|
||
bun-version: 1.2.23
|
||
|
||
# 下载所有平台的构建产物
|
||
- name: Download artifacts
|
||
uses: actions/download-artifact@v4
|
||
with:
|
||
path: release
|
||
pattern: release-*
|
||
merge-multiple: true
|
||
|
||
# 列出下载的构建产物
|
||
- name: List downloaded artifacts
|
||
run: ls -R release
|
||
|
||
# 仅为该步骤在脚本目录安装 yaml 单包,避免安装整个 monorepo 依赖
|
||
- name: Install yaml only for merge step
|
||
run: |
|
||
cd scripts/electronWorkflow
|
||
# 在脚本目录创建最小 package.json,防止 bun 向上寻找根 package.json
|
||
if [ ! -f package.json ]; then
|
||
echo '{"name":"merge-mac-release","private":true}' > package.json
|
||
fi
|
||
bun add --no-save yaml@2.8.1
|
||
|
||
# 合并 macOS YAML 文件 (使用 bun 运行 JavaScript)
|
||
- name: Merge latest-mac.yml files
|
||
run: bun run scripts/electronWorkflow/mergeMacReleaseFiles.js
|
||
|
||
# 上传合并后的构建产物
|
||
- name: Upload artifacts with merged macOS files
|
||
uses: actions/upload-artifact@v4
|
||
with:
|
||
name: merged-release-pr
|
||
path: release/
|
||
retention-days: 1
|
||
|
||
publish-pr:
|
||
needs: [merge-mac-files, version]
|
||
name: Publish PR Build
|
||
runs-on: ubuntu-latest
|
||
# Grant write permissions for creating release and commenting on PR
|
||
permissions:
|
||
contents: write
|
||
pull-requests: write
|
||
outputs:
|
||
artifact_path: ${{ steps.set_path.outputs.path }}
|
||
steps:
|
||
- uses: actions/checkout@v5
|
||
with:
|
||
fetch-depth: 0
|
||
|
||
# 下载合并后的构建产物
|
||
- name: Download merged artifacts
|
||
uses: actions/download-artifact@v4
|
||
with:
|
||
name: merged-release-pr
|
||
path: release
|
||
|
||
# 列出所有构建产物
|
||
- name: List final artifacts
|
||
run: ls -R release
|
||
|
||
# 生成PR发布描述
|
||
- name: Generate PR Release Body
|
||
id: pr_release_body
|
||
uses: actions/github-script@v8
|
||
with:
|
||
result-encoding: string
|
||
script: |
|
||
const generateReleaseBody = require('${{ github.workspace }}/.github/scripts/pr-release-body.js');
|
||
|
||
const body = generateReleaseBody({
|
||
version: "${{ needs.version.outputs.version }}",
|
||
prNumber: "${{ github.event.pull_request.number }}",
|
||
branch: "${{ github.head_ref }}"
|
||
});
|
||
|
||
return body;
|
||
|
||
- name: Create Temporary Release for PR
|
||
id: create_release
|
||
uses: softprops/action-gh-release@v1
|
||
with:
|
||
name: PR Build v${{ needs.version.outputs.version }}
|
||
tag_name: v${{ needs.version.outputs.version }}
|
||
# tag_name: pr-build-${{ github.event.pull_request.number }}-${{ github.sha }}
|
||
body: ${{ steps.pr_release_body.outputs.result }}
|
||
draft: false
|
||
prerelease: true
|
||
files: |
|
||
release/latest*
|
||
release/*.dmg*
|
||
release/*.zip*
|
||
release/*.exe*
|
||
release/*.AppImage
|
||
release/*.deb*
|
||
release/*.snap*
|
||
release/*.rpm*
|
||
release/*.tar.gz*
|
||
env:
|
||
GITHUB_TOKEN: ${{ secrets.GITHUB_TOKEN }}
|
||
|
||
# 在 PR 上添加评论,包含构建信息和下载链接
|
||
- name: Comment on PR
|
||
uses: actions/github-script@v8
|
||
with:
|
||
github-token: ${{ secrets.GITHUB_TOKEN }}
|
||
script: |
|
||
const releaseUrl = "${{ steps.create_release.outputs.url }}";
|
||
const prCommentGenerator = require('${{ github.workspace }}/.github/scripts/pr-comment.js');
|
||
|
||
const result = await prCommentGenerator({
|
||
github,
|
||
context,
|
||
releaseUrl,
|
||
version: "${{ needs.version.outputs.version }}",
|
||
tag: "v${{ needs.version.outputs.version }}"
|
||
});
|
||
|
||
console.log(`评论状态: ${result.updated ? '已更新' : '已创建'}, ID: ${result.id}`);
|