You've already forked wc-bootstrap
Add test infrastructure for isolated unit testing without WordPress/WooCommerce: - 27 tests (54 assertions) covering TemplateOverride and WooCommerceExtension - Brain\Monkey for WordPress function mocking, class stubs for TwigService and WC_Product - PHPUnit test job added to Gitea CI pipeline between lint and build-release - Test artifacts excluded from release packages Co-Authored-By: Claude Opus 4.6 <noreply@anthropic.com>
232 lines
7.9 KiB
YAML
232 lines
7.9 KiB
YAML
name: Create Release Package
|
|
|
|
on:
|
|
push:
|
|
tags:
|
|
- 'v*'
|
|
|
|
jobs:
|
|
lint:
|
|
name: PHP Lint
|
|
runs-on: ubuntu-latest
|
|
steps:
|
|
- name: Checkout code
|
|
uses: actions/checkout@v4
|
|
|
|
- name: Setup PHP
|
|
uses: shivammathur/setup-php@v2
|
|
with:
|
|
php-version: '8.3'
|
|
extensions: mbstring, xml, zip
|
|
tools: composer:v2
|
|
|
|
- name: PHP Syntax Check
|
|
run: |
|
|
find . -name "*.php" -not -path "./vendor/*" -not -path "./node_modules/*" -print0 | xargs -0 -n1 php -l
|
|
|
|
test:
|
|
name: PHPUnit Tests
|
|
runs-on: ubuntu-latest
|
|
needs: [lint]
|
|
steps:
|
|
- name: Checkout code
|
|
uses: actions/checkout@v4
|
|
|
|
- name: Setup PHP
|
|
uses: shivammathur/setup-php@v2
|
|
with:
|
|
php-version: '8.3'
|
|
extensions: mbstring, xml, dom
|
|
tools: composer:v2
|
|
|
|
- name: Install Composer dependencies
|
|
run: composer install --no-interaction
|
|
|
|
- name: Run PHPUnit
|
|
run: composer exec -- phpunit
|
|
|
|
build-release:
|
|
name: Build Release
|
|
runs-on: ubuntu-latest
|
|
needs: [test]
|
|
steps:
|
|
- name: Checkout code
|
|
uses: actions/checkout@v4
|
|
|
|
- name: Setup PHP
|
|
uses: shivammathur/setup-php@v2
|
|
with:
|
|
php-version: '8.3'
|
|
extensions: mbstring, xml, zip, intl, gettext
|
|
tools: composer:v2
|
|
|
|
- name: Get version from tag
|
|
id: version
|
|
run: |
|
|
VERSION=${GITHUB_REF_NAME#v}
|
|
echo "version=$VERSION" >> $GITHUB_OUTPUT
|
|
echo "Building version: $VERSION"
|
|
|
|
- name: Validate composer.json
|
|
run: composer validate --no-check-lock --no-check-all
|
|
|
|
- name: Install Composer dependencies (production)
|
|
run: |
|
|
composer config platform.php 8.3.0
|
|
composer install --no-dev --optimize-autoloader --no-interaction
|
|
|
|
- name: Install gettext
|
|
run: apt-get update && apt-get install -y gettext
|
|
|
|
- name: Compile translations
|
|
run: |
|
|
for po in languages/*.po; do
|
|
if [ -f "$po" ]; then
|
|
mo="${po%.po}.mo"
|
|
echo "Compiling $po to $mo"
|
|
msgfmt -o "$mo" "$po"
|
|
fi
|
|
done
|
|
|
|
- name: Verify theme version matches tag
|
|
run: |
|
|
THEME_VERSION=$(grep -oP "Version:\s*\K[0-9]+\.[0-9]+\.[0-9]+" style.css | head -1)
|
|
TAG_VERSION=${{ steps.version.outputs.version }}
|
|
if [ "$THEME_VERSION" != "$TAG_VERSION" ]; then
|
|
echo "Error: Theme version ($THEME_VERSION) does not match tag version ($TAG_VERSION)"
|
|
exit 1
|
|
fi
|
|
echo "Version verified: $THEME_VERSION"
|
|
|
|
- name: Build release package
|
|
run: |
|
|
VERSION=${{ steps.version.outputs.version }}
|
|
THEME_NAME="wc-bootstrap"
|
|
|
|
# Stage files into a properly named directory for the zip archive.
|
|
WORKSPACE_DIR="$(pwd)"
|
|
STAGING_DIR=$(mktemp -d)
|
|
cp -a . "${STAGING_DIR}/${THEME_NAME}"
|
|
|
|
cd "${STAGING_DIR}/${THEME_NAME}"
|
|
rm -rf .git .gitea .github .vscode .claude releases node_modules tests .phpunit.cache
|
|
rm -f CLAUDE.md PLAN.md composer.json composer.lock .gitignore .editorconfig phpunit.xml.dist
|
|
find . -name '*.log' -o -name '*.po~' -o -name '*.bak' -o -name '.DS_Store' | xargs rm -f 2>/dev/null || true
|
|
|
|
cd "${STAGING_DIR}"
|
|
mkdir -p "${WORKSPACE_DIR}/releases"
|
|
zip -r "${WORKSPACE_DIR}/releases/${THEME_NAME}-${VERSION}.zip" "${THEME_NAME}"
|
|
|
|
cd "${WORKSPACE_DIR}"
|
|
rm -rf "${STAGING_DIR}"
|
|
echo "Created: releases/${THEME_NAME}-${VERSION}.zip"
|
|
ls -lh "releases/${THEME_NAME}-${VERSION}.zip"
|
|
|
|
- name: Generate checksums
|
|
run: |
|
|
VERSION=${{ steps.version.outputs.version }}
|
|
THEME_NAME="wc-bootstrap"
|
|
|
|
cd releases
|
|
sha256sum "${THEME_NAME}-${VERSION}.zip" > "${THEME_NAME}-${VERSION}.zip.sha256"
|
|
echo "SHA256:"
|
|
cat "${THEME_NAME}-${VERSION}.zip.sha256"
|
|
|
|
- name: Verify package structure
|
|
run: |
|
|
set +o pipefail
|
|
VERSION=${{ steps.version.outputs.version }}
|
|
THEME_NAME="wc-bootstrap"
|
|
|
|
echo "Package contents (first 50 entries):"
|
|
unzip -l "releases/${THEME_NAME}-${VERSION}.zip" | head -50 || true
|
|
|
|
# Verify main style.css file exists
|
|
if unzip -l "releases/${THEME_NAME}-${VERSION}.zip" | grep -q "${THEME_NAME}/style.css"; then
|
|
echo "Main style.css file: OK"
|
|
else
|
|
echo "ERROR: Main style.css file not found!"
|
|
exit 1
|
|
fi
|
|
|
|
# Verify functions.php exists
|
|
if unzip -l "releases/${THEME_NAME}-${VERSION}.zip" | grep -q "${THEME_NAME}/functions.php"; then
|
|
echo "functions.php file: OK"
|
|
else
|
|
echo "ERROR: functions.php file not found!"
|
|
exit 1
|
|
fi
|
|
|
|
- name: Extract changelog for release notes
|
|
id: changelog
|
|
run: |
|
|
VERSION=${{ steps.version.outputs.version }}
|
|
NOTES=$(sed -n "/^## \[${VERSION}\]/,/^## \[/p" CHANGELOG.md | sed '$ d' | tail -n +2)
|
|
if [ -z "$NOTES" ]; then
|
|
NOTES="Release version ${VERSION}"
|
|
fi
|
|
echo "$NOTES" > release_notes.txt
|
|
echo "Release notes extracted"
|
|
|
|
- name: Create Gitea Release
|
|
env:
|
|
GITEA_TOKEN: ${{ secrets.SRC_GITEA_TOKEN }}
|
|
run: |
|
|
VERSION=${{ steps.version.outputs.version }}
|
|
TAG_NAME=${{ github.ref_name }}
|
|
THEME_NAME="wc-bootstrap"
|
|
|
|
PRERELEASE="false"
|
|
if [[ "$TAG_NAME" == *-* ]] || [[ "$VERSION" == 0.* ]]; then
|
|
PRERELEASE="true"
|
|
fi
|
|
|
|
BODY=$(cat release_notes.txt)
|
|
|
|
# Check if release already exists and delete it
|
|
EXISTING_RELEASE=$(curl -s \
|
|
-H "Authorization: token ${GITEA_TOKEN}" \
|
|
"${GITHUB_SERVER_URL}/api/v1/repos/${GITHUB_REPOSITORY}/releases/tags/${TAG_NAME}")
|
|
|
|
EXISTING_ID=$(echo "$EXISTING_RELEASE" | jq -r '.id // empty')
|
|
if [ -n "$EXISTING_ID" ] && [ "$EXISTING_ID" != "null" ]; then
|
|
echo "Deleting existing release ID: $EXISTING_ID"
|
|
curl -s -X DELETE \
|
|
-H "Authorization: token ${GITEA_TOKEN}" \
|
|
"${GITHUB_SERVER_URL}/api/v1/repos/${GITHUB_REPOSITORY}/releases/${EXISTING_ID}"
|
|
echo "Existing release deleted"
|
|
fi
|
|
|
|
# Create release via Gitea API
|
|
RELEASE_RESPONSE=$(curl -s -X POST \
|
|
-H "Authorization: token ${GITEA_TOKEN}" \
|
|
-H "Content-Type: application/json" \
|
|
-d "{\"tag_name\": \"${TAG_NAME}\", \"name\": \"WooCommerce Bootstrap ${VERSION}\", \"body\": $(echo "$BODY" | jq -Rs .), \"draft\": false, \"prerelease\": ${PRERELEASE}}" \
|
|
"${GITHUB_SERVER_URL}/api/v1/repos/${GITHUB_REPOSITORY}/releases")
|
|
|
|
RELEASE_ID=$(echo "$RELEASE_RESPONSE" | jq -r '.id')
|
|
|
|
if [ "$RELEASE_ID" == "null" ] || [ -z "$RELEASE_ID" ]; then
|
|
echo "Failed to create release:"
|
|
echo "$RELEASE_RESPONSE"
|
|
exit 1
|
|
fi
|
|
|
|
echo "Created release ID: $RELEASE_ID"
|
|
|
|
# Upload release assets
|
|
for file in "releases/${THEME_NAME}-${VERSION}.zip" "releases/${THEME_NAME}-${VERSION}.zip.sha256"; do
|
|
if [ -f "$file" ]; then
|
|
FILENAME=$(basename "$file")
|
|
echo "Uploading $FILENAME..."
|
|
curl -s -X POST \
|
|
-H "Authorization: token ${GITEA_TOKEN}" \
|
|
-H "Content-Type: application/octet-stream" \
|
|
--data-binary "@$file" \
|
|
"${GITHUB_SERVER_URL}/api/v1/repos/${GITHUB_REPOSITORY}/releases/${RELEASE_ID}/assets?name=${FILENAME}"
|
|
echo "Uploaded $FILENAME"
|
|
fi
|
|
done
|
|
|
|
echo "Release created: ${GITHUB_SERVER_URL}/${GITHUB_REPOSITORY}/releases/tag/${TAG_NAME}"
|