358 lines
12 KiB
YAML
358 lines
12 KiB
YAML
# ClawX Release Workflow
|
|
# Builds and publishes releases for macOS, Windows, and Linux
|
|
|
|
name: Release
|
|
|
|
on:
|
|
push:
|
|
tags:
|
|
- 'v*'
|
|
workflow_dispatch:
|
|
inputs:
|
|
version:
|
|
description: 'Version to release (e.g., 1.0.0)'
|
|
required: true
|
|
|
|
permissions:
|
|
contents: write
|
|
|
|
jobs:
|
|
release:
|
|
strategy:
|
|
matrix:
|
|
include:
|
|
- os: macos-latest
|
|
platform: mac
|
|
- os: windows-latest
|
|
platform: win
|
|
- os: ubuntu-latest
|
|
platform: linux
|
|
|
|
runs-on: ${{ matrix.os }}
|
|
|
|
steps:
|
|
- name: Checkout code
|
|
uses: actions/checkout@v4
|
|
with:
|
|
fetch-depth: 0
|
|
|
|
- name: Setup Node.js
|
|
uses: actions/setup-node@v4
|
|
with:
|
|
node-version: '20'
|
|
|
|
- name: Setup pnpm
|
|
uses: pnpm/action-setup@v4
|
|
|
|
- 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: pnpm install
|
|
|
|
- name: Download uv binaries for macOS
|
|
if: matrix.platform == 'mac'
|
|
run: pnpm run uv:download:mac
|
|
|
|
- name: Download uv binaries for Windows
|
|
if: matrix.platform == 'win'
|
|
run: pnpm run uv:download:win
|
|
|
|
- name: Download uv binaries for Linux
|
|
if: matrix.platform == 'linux'
|
|
run: pnpm run uv:download:linux
|
|
|
|
# macOS specific steps
|
|
- name: Build macOS
|
|
if: matrix.platform == 'mac'
|
|
env:
|
|
GH_TOKEN: ${{ secrets.GITHUB_TOKEN }}
|
|
# Code signing
|
|
CSC_LINK: ${{ secrets.MAC_CERTS }}
|
|
CSC_KEY_PASSWORD: ${{ secrets.MAC_CERTS_PASSWORD }}
|
|
# Notarization
|
|
APPLE_ID: ${{ secrets.APPLE_ID }}
|
|
APPLE_APP_SPECIFIC_PASSWORD: ${{ secrets.APPLE_APP_SPECIFIC_PASSWORD }}
|
|
APPLE_TEAM_ID: ${{ secrets.APPLE_TEAM_ID }}
|
|
run: |
|
|
# Increase file descriptor limit to handle large number of files during code signing
|
|
ulimit -n 65536
|
|
echo "File descriptor limit: $(ulimit -n)"
|
|
|
|
pnpm run package:mac
|
|
|
|
# Windows specific steps
|
|
- name: Build Windows
|
|
if: matrix.platform == 'win'
|
|
env:
|
|
GH_TOKEN: ${{ secrets.GITHUB_TOKEN }}
|
|
# For code signing (optional)
|
|
# CSC_LINK: ${{ secrets.WIN_CERTS }}
|
|
# CSC_KEY_PASSWORD: ${{ secrets.WIN_CERTS_PASSWORD }}
|
|
run: pnpm run package:win
|
|
|
|
# Linux specific steps
|
|
- name: Build Linux
|
|
if: matrix.platform == 'linux'
|
|
env:
|
|
GH_TOKEN: ${{ secrets.GITHUB_TOKEN }}
|
|
run: pnpm run package:linux
|
|
|
|
- name: Upload artifacts
|
|
uses: actions/upload-artifact@v4
|
|
with:
|
|
name: release-${{ matrix.platform }}
|
|
path: |
|
|
release/*.dmg
|
|
release/*.zip
|
|
release/*.exe
|
|
release/*.AppImage
|
|
release/*.deb
|
|
release/*.rpm
|
|
release/latest*.yml
|
|
retention-days: 7
|
|
|
|
# ──────────────────────────────────────────────────────────────
|
|
# Job: Publish to GitHub Releases
|
|
# ──────────────────────────────────────────────────────────────
|
|
publish:
|
|
needs: release
|
|
runs-on: ubuntu-latest
|
|
|
|
steps:
|
|
- name: Checkout code
|
|
uses: actions/checkout@v4
|
|
|
|
- name: Download all artifacts
|
|
uses: actions/download-artifact@v4
|
|
with:
|
|
path: release-artifacts
|
|
|
|
- name: List all downloaded artifacts
|
|
run: |
|
|
echo "=== All artifacts downloaded ==="
|
|
find release-artifacts/ -type f -exec ls -lh {} \;
|
|
echo ""
|
|
echo "=== File tree ==="
|
|
tree release-artifacts/ || find release-artifacts/ -print
|
|
|
|
- name: Create GitHub Release
|
|
uses: softprops/action-gh-release@v2
|
|
if: startsWith(github.ref, 'refs/tags/')
|
|
with:
|
|
files: |
|
|
release-artifacts/**/*.dmg
|
|
release-artifacts/**/*.zip
|
|
release-artifacts/**/*.exe
|
|
release-artifacts/**/*.AppImage
|
|
release-artifacts/**/*.deb
|
|
release-artifacts/**/*.rpm
|
|
release-artifacts/**/latest*.yml
|
|
draft: false
|
|
prerelease: ${{ contains(github.ref, 'alpha') || contains(github.ref, 'beta') }}
|
|
make_latest: ${{ !(contains(github.ref, 'alpha') || contains(github.ref, 'beta')) }}
|
|
generate_release_notes: true
|
|
body: |
|
|
## 🚀 ClawX ${{ github.ref_name }}
|
|
|
|
ClawX - Graphical AI Assistant based on OpenClaw
|
|
|
|
### 📦 Downloads
|
|
|
|
Please select the appropriate installer for your operating system and architecture:
|
|
|
|
#### macOS
|
|
- **Apple Silicon (M1/M2/M3/M4)**: `ClawX-*-mac-arm64.dmg`
|
|
- **Intel (x64)**: `ClawX-*-mac-x64.dmg`
|
|
|
|
#### Windows
|
|
- **Installer (x64)**: `ClawX-*-win-x64.exe`
|
|
- **Installer (ARM64)**: `ClawX-*-win-arm64.exe`
|
|
|
|
#### Linux
|
|
- **AppImage (x64)**: `ClawX-*-linux-x86_64.AppImage` (Universal format, recommended)
|
|
- **AppImage (ARM64)**: `ClawX-*-linux-arm64.AppImage`
|
|
- **Debian/Ubuntu (x64)**: `ClawX-*-linux-amd64.deb`
|
|
- **Debian/Ubuntu (ARM64)**: `ClawX-*-linux-arm64.deb`
|
|
- **RPM (x64)**: `ClawX-*-linux-x86_64.rpm`
|
|
|
|
### 📝 Release Notes
|
|
|
|
See the auto-generated release notes below for detailed changes.
|
|
|
|
### ⚠️ Installation Notes
|
|
|
|
- **macOS**: On first launch, you may see "cannot verify developer". Go to System Preferences → Security & Privacy to allow the app to run
|
|
- **Windows**: SmartScreen may block the app. Click "More info" → "Run anyway" to proceed
|
|
- **Linux**: AppImage requires executable permission: `chmod +x ClawX-*.AppImage`
|
|
|
|
---
|
|
|
|
💬 Found an issue? Please submit an [Issue](https://github.com/${{ github.repository }}/issues)
|
|
env:
|
|
GITHUB_TOKEN: ${{ secrets.GITHUB_TOKEN }}
|
|
|
|
# ──────────────────────────────────────────────────────────────
|
|
# Job: Upload to Alibaba Cloud OSS
|
|
# Uploads all release artifacts to OSS for:
|
|
# - Official website downloads (via release-info.json)
|
|
# - electron-updater auto-update (via latest-*.yml)
|
|
#
|
|
# Directory structure on OSS:
|
|
# latest/ → always overwritten with the newest version
|
|
# releases/vX.Y.Z/ → permanent archive, never deleted
|
|
# ──────────────────────────────────────────────────────────────
|
|
upload-oss:
|
|
needs: release
|
|
runs-on: ubuntu-latest
|
|
|
|
steps:
|
|
- name: Checkout code
|
|
uses: actions/checkout@v4
|
|
|
|
- name: Download all artifacts
|
|
uses: actions/download-artifact@v4
|
|
with:
|
|
path: release-artifacts
|
|
|
|
- name: Extract version
|
|
id: version
|
|
run: |
|
|
if [[ "${{ github.ref }}" == refs/tags/v* ]]; then
|
|
VERSION="${GITHUB_REF#refs/tags/v}"
|
|
else
|
|
VERSION="${{ github.event.inputs.version }}"
|
|
fi
|
|
echo "version=${VERSION}" >> $GITHUB_OUTPUT
|
|
echo "tag=v${VERSION}" >> $GITHUB_OUTPUT
|
|
echo "Detected version: ${VERSION}"
|
|
|
|
- name: Prepare upload directories
|
|
run: |
|
|
VERSION="${{ steps.version.outputs.version }}"
|
|
TAG="${{ steps.version.outputs.tag }}"
|
|
|
|
mkdir -p staging/latest
|
|
mkdir -p staging/releases/${TAG}
|
|
|
|
# Flatten all platform artifacts into staging directories
|
|
find release-artifacts/ -type f | while read file; do
|
|
filename=$(basename "$file")
|
|
cp "$file" "staging/latest/${filename}"
|
|
cp "$file" "staging/releases/${TAG}/${filename}"
|
|
done
|
|
|
|
echo "=== staging/latest/ ==="
|
|
ls -lh staging/latest/
|
|
echo ""
|
|
echo "=== staging/releases/${TAG}/ ==="
|
|
ls -lh staging/releases/${TAG}/
|
|
|
|
- name: Generate release-info.json
|
|
run: |
|
|
VERSION="${{ steps.version.outputs.version }}"
|
|
BASE_URL="https://oss.intelli-spectrum.com/latest"
|
|
|
|
jq -n \
|
|
--arg version "$VERSION" \
|
|
--arg date "$(date -u +%Y-%m-%dT%H:%M:%SZ)" \
|
|
--arg base "$BASE_URL" \
|
|
--arg changelog "https://github.com/${{ github.repository }}/releases/tag/v${VERSION}" \
|
|
'{
|
|
version: $version,
|
|
releaseDate: $date,
|
|
downloads: {
|
|
mac: {
|
|
x64: ($base + "/ClawX-" + $version + "-mac-x64.dmg"),
|
|
arm64: ($base + "/ClawX-" + $version + "-mac-arm64.dmg")
|
|
},
|
|
win: {
|
|
x64: ($base + "/ClawX-" + $version + "-win-x64.exe"),
|
|
arm64: ($base + "/ClawX-" + $version + "-win-arm64.exe")
|
|
},
|
|
linux: {
|
|
deb_amd64: ($base + "/ClawX-" + $version + "-linux-amd64.deb"),
|
|
deb_arm64: ($base + "/ClawX-" + $version + "-linux-arm64.deb"),
|
|
appimage_x64: ($base + "/ClawX-" + $version + "-linux-x86_64.AppImage"),
|
|
appimage_arm64: ($base + "/ClawX-" + $version + "-linux-arm64.AppImage"),
|
|
rpm_x64: ($base + "/ClawX-" + $version + "-linux-x86_64.rpm")
|
|
}
|
|
},
|
|
changelog: $changelog
|
|
}' > staging/latest/release-info.json
|
|
|
|
echo "=== release-info.json ==="
|
|
cat staging/latest/release-info.json
|
|
|
|
- name: Install and configure ossutil
|
|
env:
|
|
OSS_ACCESS_KEY_ID: ${{ secrets.OSS_ACCESS_KEY_ID }}
|
|
OSS_ACCESS_KEY_SECRET: ${{ secrets.OSS_ACCESS_KEY_SECRET }}
|
|
run: |
|
|
curl -sL https://gosspublic.alicdn.com/ossutil/install.sh | sudo bash
|
|
|
|
# Write config file for non-interactive use
|
|
cat > $HOME/.ossutilconfig << EOF
|
|
[Credentials]
|
|
language=EN
|
|
endpoint=oss-cn-hangzhou.aliyuncs.com
|
|
accessKeyID=${OSS_ACCESS_KEY_ID}
|
|
accessKeySecret=${OSS_ACCESS_KEY_SECRET}
|
|
EOF
|
|
|
|
ossutil --version
|
|
|
|
- name: "Upload to OSS: latest/ (overwrite)"
|
|
run: |
|
|
# Clean old latest/ to remove stale version files
|
|
ossutil rm -r -f oss://valuecell-clawx/latest/ || true
|
|
|
|
# Upload all files with no-cache so clients always get the freshest version
|
|
ossutil cp -r -f \
|
|
--meta="Cache-Control:no-cache,no-store,must-revalidate" \
|
|
staging/latest/ \
|
|
oss://valuecell-clawx/latest/
|
|
|
|
echo "Uploaded to latest/"
|
|
|
|
- name: "Upload to OSS: releases/vX.Y.Z/ (archive)"
|
|
run: |
|
|
TAG="${{ steps.version.outputs.tag }}"
|
|
|
|
# Upload to permanent archive (long cache, immutable)
|
|
ossutil cp -r \
|
|
staging/releases/${TAG}/ \
|
|
oss://valuecell-clawx/releases/${TAG}/ \
|
|
--meta "Cache-Control:public,max-age=31536000,immutable"
|
|
|
|
echo "Uploaded to releases/${TAG}/"
|
|
|
|
- name: Verify OSS upload
|
|
run: |
|
|
TAG="${{ steps.version.outputs.tag }}"
|
|
|
|
echo "=== latest/ ==="
|
|
ossutil ls oss://valuecell-clawx/latest/ --short
|
|
|
|
echo ""
|
|
echo "=== releases/${TAG}/ ==="
|
|
ossutil ls oss://valuecell-clawx/releases/${TAG}/ --short
|
|
|
|
echo ""
|
|
echo "=== Verify release-info.json ==="
|
|
ossutil cp oss://valuecell-clawx/latest/release-info.json /tmp/release-info.json -f
|
|
jq . /tmp/release-info.json
|
|
|
|
echo ""
|
|
echo "✅ All files uploaded and verified successfully!"
|