0.1.8-alpha.0 #14
Workflow file for this run
This file contains hidden or bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
| # 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!" |