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 -->
199 lines
6.6 KiB
YAML
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 }}"
|
|
}
|