arcade-mcp/.github/workflows/release-on-version-change.yml
Nate Barbettini 592c3f73c0
fix: Don't double-include template files (#722)
Fixes broken publishing action:
https://github.com/ArcadeAI/arcade-mcp/actions/runs/20147239181

<!-- CURSOR_SUMMARY -->
---

> [!NOTE]
> Removes template force-include to avoid duplicate files and adds CI
wheel-duplicate validation; bumps version to 1.6.1.
> 
> - **Packaging**:
> - Bump `arcade-mcp` version from `1.6.0` to `1.6.1` in
`pyproject.toml`.
> - Remove `[tool.hatch.build.targets.wheel.force-include]` for
`arcade_cli/templates` to prevent double-including template files.
> - **CI/CD**:
> - In `.github/workflows/release-on-version-change.yml`, add a
post-build Python step to validate built wheels for duplicate filenames
before publishing.
> 
> <sup>Written by [Cursor
Bugbot](https://cursor.com/dashboard?tab=bugbot) for commit
3a15e08772b2b4851b185b04c763f3f5898bdbd5. This will update automatically
on new commits. Configure
[here](https://cursor.com/dashboard?tab=bugbot).</sup>
<!-- /CURSOR_SUMMARY -->
2025-12-11 13:55:31 -08:00

199 lines
6.6 KiB
YAML

# This workflow is used to release packages to PyPI when its
# pyproject.toml version is changed or a new package is added.
name: Release on Version Change
on:
push:
branches:
- main
jobs:
detect-version-changes:
runs-on: ubuntu-latest
outputs:
matrix: ${{ steps.set-matrix.outputs.matrix }}
has-changes: ${{ steps.set-matrix.outputs.has-changes }}
steps:
- name: Checkout code
uses: actions/checkout@v4
with:
fetch-depth: 0
- name: Get changed files
id: changed-files
uses: tj-actions/changed-files@ed68ef82c095e0d48ec87eccea555d944a631a4c # v46
with:
files: |
**/pyproject.toml
files_ignore: |
examples/**
libs/tests/**
- name: Check for version changes or new packages
id: check-versions
if: steps.changed-files.outputs.any_changed == 'true'
run: |
./.github/scripts/check-version-changes.sh "${{ steps.changed-files.outputs.all_changed_files }}"
- name: Set matrix
id: set-matrix
run: |
packages='${{ steps.check-versions.outputs.packages }}'
if [ -z "$packages" ] || [ "$packages" = "[]" ]; then
echo "has-changes=false" >> $GITHUB_OUTPUT
echo "matrix={\"include\":[]}" >> $GITHUB_OUTPUT
else
echo "has-changes=true" >> $GITHUB_OUTPUT
# Create matrix with package directories
matrix=$(echo "$packages" | jq -c '{include: [.[] | {package: .}]}')
echo "matrix=$matrix" >> $GITHUB_OUTPUT
echo "Matrix: $matrix"
fi
build-and-test:
needs: detect-version-changes
if: needs.detect-version-changes.outputs.has-changes == 'true'
runs-on: ubuntu-latest
strategy:
matrix: ${{ fromJson(needs.detect-version-changes.outputs.matrix) }}
permissions:
contents: write
packages: write
steps:
- name: Checkout code
uses: actions/checkout@v4
with:
fetch-depth: 0
- name: Extract package name and version
working-directory: ${{ matrix.package }}
run: |
# Extract package name
PACKAGE_NAME=$(grep -m1 '^name = ' pyproject.toml | cut -d'"' -f2)
echo "PACKAGE_NAME=$PACKAGE_NAME" >> $GITHUB_ENV
# Extract version
VERSION=$(grep -m1 '^version = ' pyproject.toml | cut -d'"' -f2)
echo "VERSION=$VERSION" >> $GITHUB_ENV
echo "Building $PACKAGE_NAME version $VERSION"
- name: Set up the environment
uses: ./.github/actions/setup-uv-env
with:
python-version: "3.10"
is-toolkit: ${{ startsWith(matrix.package, 'toolkits/') }}
is-contrib: ${{ startsWith(matrix.package, 'contrib/') }}
is-lib: ${{ startsWith(matrix.package, 'libs/') }}
working-directory: ${{ matrix.package }}
- name: Run tests
# Skip tests for toolkits - tests are run on every PR commit for toolkits
if: ${{ !startsWith(matrix.package, 'toolkits/') }}
working-directory: ${{ matrix.package }}
run: |
# Run tests if they exist
if [ -f "Makefile" ] && grep -q "^test:" Makefile; then
make test
elif [ -f "../Makefile" ] && grep -q "^test:" ../Makefile; then
cd .. && make test
else
echo "No tests found, skipping test step"
fi
- name: Build release distributions
working-directory: ${{ matrix.package }}
run: |
uv build --out-dir dist | tee build.log
# Validate wheel archives: PyPI rejects duplicate filenames in wheels
python - <<'PY'
from __future__ import annotations
from pathlib import Path
import sys
import zipfile
dist = Path("dist")
wheels = sorted(dist.glob("*.whl"))
if not wheels:
print("No wheels found in dist/, skipping wheel validation.")
raise SystemExit(0)
for whl in wheels:
with zipfile.ZipFile(whl) as zf:
names = zf.namelist()
seen: set[str] = set()
dupes: set[str] = set()
for name in names:
if name in seen:
dupes.add(name)
else:
seen.add(name)
if dupes:
print(f"ERROR: {whl} contains duplicate paths (PyPI will reject this wheel):", file=sys.stderr)
for name in sorted(dupes):
print(f" - {name}", file=sys.stderr)
raise SystemExit(1)
print("Wheel validation OK (no duplicate filenames).")
PY
# Verify build artifacts
ls -la dist/
echo "Built artifacts for ${{ env.PACKAGE_NAME }} v${{ env.VERSION }}"
- name: Upload release distributions
uses: actions/upload-artifact@v4
with:
name: release-dists-${{ env.PACKAGE_NAME }}-${{ env.VERSION }}
path: ${{ matrix.package }}/dist/
pypi-publish:
needs: [detect-version-changes, build-and-test]
if: needs.detect-version-changes.outputs.has-changes == 'true'
runs-on: ubuntu-latest
strategy:
matrix: ${{ fromJson(needs.detect-version-changes.outputs.matrix) }}
permissions:
id-token: write
steps:
- name: Checkout code
uses: actions/checkout@v4
with:
fetch-depth: 0
- name: Extract package name and version
working-directory: ${{ matrix.package }}
run: |
PACKAGE_NAME=$(grep -m1 '^name = ' pyproject.toml | cut -d'"' -f2)
echo "PACKAGE_NAME=$PACKAGE_NAME" >> $GITHUB_ENV
VERSION=$(grep -m1 '^version = ' pyproject.toml | cut -d'"' -f2)
echo "VERSION=$VERSION" >> $GITHUB_ENV
- name: Retrieve release distributions
uses: actions/download-artifact@v4
with:
name: release-dists-${{ env.PACKAGE_NAME }}-${{ env.VERSION }}
path: dist/
- name: Publish release distributions to PyPI
uses: pypa/gh-action-pypi-publish@release/v1
- name: Send status to Slack
if: always()
uses: slackapi/slack-github-action@v2.0.0
with:
webhook: ${{ secrets.PACKAGE_RELEASE_SLACK_WEBHOOK_URL }}
webhook-type: webhook-trigger
payload: |
{
"status": "${{ (job.status == 'failure' || job.status == 'cancelled') && 'Failed' || 'Success' }}",
"package": "${{ env.PACKAGE_NAME }}",
"version": "${{ env.VERSION }}",
"url": "${{ github.server_url }}/${{ github.repository }}/actions/runs/${{ github.run_id }}"
}