Skip to content

Merge branch 'main' of https://github.com/codebuilderinc/codebuilder-app #59

Merge branch 'main' of https://github.com/codebuilderinc/codebuilder-app

Merge branch 'main' of https://github.com/codebuilderinc/codebuilder-app #59

name: πŸš€ EAS Android Build & Smart Release
on:
push:
branches: [main]
workflow_dispatch:
env:
EXPO_TOKEN: ${{ secrets.EXPO_TOKEN }}
GITHUB_TOKEN: ${{ secrets.GITHUB_TOKEN }}
jobs:
build-android:
name: πŸ”¨ Build Android - Version Manager
runs-on: ubuntu-latest
outputs:
build_id: ${{ steps.build.outputs.BUILD_ID }}
app_version: ${{ steps.version-control.outputs.app_version }}
build_number: ${{ steps.version-control.outputs.build_number }}
build_date: ${{ steps.version-control.outputs.build_date }}
is_production: ${{ steps.version-control.outputs.is_production }}
steps:
# ========================
# πŸ› οΈ Repository Setup
# ========================
- name: "πŸ“¦ Checkout (Full History)"
uses: actions/checkout@v4
with:
fetch-depth: 0
# ========================
# βš™οΈ Environment Configuration
# ========================
- name: "πŸ“¦ Setup Node.js 20.x"
uses: actions/setup-node@v4
with:
node-version: 20.x
cache: "npm"
- name: "🧩 Install dependencies (ci)"
run: npm ci
# ========================
# πŸ”„ Version Management
# ========================
- name: "πŸ”„ Update Production Version"
if: github.ref == 'refs/heads/main'
run: node scripts/bumpVersion.cjs
- name: "πŸ”§ Configure Git for Automation"
if: github.ref == 'refs/heads/main'
run: |
git config --global user.name "GitHub Actions"
git config --global user.email "[email protected]"
- name: "πŸ’Ύ Commit Version Update"
if: github.ref == 'refs/heads/main'
run: |
git add version.json
git commit -m "chore: Auto-increment version [skip ci]"
git push
# ========================
# πŸ“Œ Version Tagging
# ========================
- name: "🏷️ Set CI/CD Versions"
id: version-control
run: |
# Use version from version.json (requires jq)
if [ "${{ github.ref }}" == "refs/heads/main" ]; then
APP_VERSION=$(jq -r '.version' version.json)
IS_PRODUCTION="true"
else
APP_VERSION="1.0.0-prerelease.${{ github.run_number }}"
IS_PRODUCTION="false"
fi
# Generate build identifiers
BUILD_NUMBER="${{ github.run_id }}"
BUILD_DATE=$(date +'%Y%m%d-%H%M%S')
# Set outputs for downstream jobs
echo "app_version=$APP_VERSION" >> $GITHUB_OUTPUT
echo "build_number=$BUILD_NUMBER" >> $GITHUB_OUTPUT
echo "build_date=$BUILD_DATE" >> $GITHUB_OUTPUT
echo "is_production=$IS_PRODUCTION" >> $GITHUB_OUTPUT
# Export environment variables
echo "APP_VERSION=$APP_VERSION" >> $GITHUB_ENV
echo "BUILD_NUMBER=$BUILD_NUMBER" >> $GITHUB_ENV
echo "BUILD_DATE=$BUILD_DATE" >> $GITHUB_ENV
# ========================
# πŸ” EAS Setup & Auth
# ========================
- name: "βš™οΈ Install EAS CLI"
run: npm install -g eas-cli@latest
- name: "πŸ” Verify Expo Credentials"
run: npx eas whoami --token $EXPO_TOKEN
# ========================
# πŸ—οΈ Build Execution
# ========================
- name: "πŸš€ Trigger EAS Build"
id: build
run: |
echo "πŸ”„ Initializing build process..."
sudo apt-get install -y jq
BUILD_JSON=$(npx eas build -p android --profile production --non-interactive --json)
BUILD_ID=$(echo "$BUILD_JSON" | jq -r '.[0].id')
echo "BUILD_ID=$BUILD_ID" >> $GITHUB_ENV
echo "BUILD_ID=$BUILD_ID" >> $GITHUB_OUTPUT
download-apk:
name: "πŸ“₯ APK Artifact Handler"
runs-on: ubuntu-latest
needs: build-android
outputs:
apk_path: ${{ steps.download.outputs.APK_PATH }}
steps:
# ========================
# πŸ› οΈ Environment Setup
# ========================
- name: "πŸ“¦ Checkout Repository"
uses: actions/checkout@v4
- name: "βš™οΈ Setup Node.js"
uses: actions/setup-node@v4
with:
node-version: 20.x
- name: "πŸ“¦ Install Dependencies"
run: npm ci
# ========================
# πŸ“¦ Dependency Management
# ========================
- name: "🧰 Install Build Tools"
run: |
npm install -g eas-cli@latest
sudo apt-get install -y jq curl dotenv
# ========================
# πŸ” Build Monitoring
# ========================
- name: "⏳ Wait for Build Completion"
run: |
echo "⏰ Monitoring build status..."
cd $GITHUB_WORKSPACE
BUILD_ID=${{ needs.build-android.outputs.build_id }}
echo "πŸ” Starting build monitoring for BUILD_ID: $BUILD_ID"
# Initial check without JSON for better error visibility
npx eas build:view $BUILD_ID || true
RETRY_COUNT=0
MAX_RETRIES=120
SLEEP_TIME=30
while [[ $RETRY_COUNT -lt $MAX_RETRIES ]]; do
echo -e "\n=== Attempt $((RETRY_COUNT+1))/$MAX_RETRIES ==="
# Fetch build status in JSON format
BUILD_STATUS_JSON=$(npx eas build:view --json $BUILD_ID)
echo "πŸ“„ Raw API response: $BUILD_STATUS_JSON"
# Validate JSON and check for empty response
if ! echo "$BUILD_STATUS_JSON" | jq empty >/dev/null 2>&1 || [[ -z "$BUILD_STATUS_JSON" ]]; then
echo "❌ Error: Invalid or empty response from EAS API! Retrying..."
RETRY_COUNT=$((RETRY_COUNT+1))
sleep $SLEEP_TIME
continue
fi
BUILD_STATUS=$(echo "$BUILD_STATUS_JSON" | jq -r '.status')
ERROR_MESSAGE=$(echo "$BUILD_STATUS_JSON" | jq -r '.error.message // empty')
echo "πŸ” Parsed status: $BUILD_STATUS"
[[ -n "$ERROR_MESSAGE" ]] && echo "❌ Error message: $ERROR_MESSAGE"
case $BUILD_STATUS in
"FINISHED")
APK_URL=$(echo "$BUILD_STATUS_JSON" | jq -r '.artifacts.buildUrl')
if [[ -z "$APK_URL" || "$APK_URL" == "null" ]]; then
echo "❌ Error: Successful build but no APK URL found!"
exit 1
fi
echo "βœ… APK_URL=$APK_URL" >> $GITHUB_ENV
exit 0
;;
"ERRORED"|"CANCELLED")
echo "❌ Build failed! Error details:"
echo "$BUILD_STATUS_JSON" | jq .
exit 1
;;
"NEW"|"IN_QUE"|"IN_PROGRESS"|"PENDING")
echo "⏳ Build is still in progress..."
;;
*)
echo "❌ Unknown build status: $BUILD_STATUS"
exit 1
;;
esac
RETRY_COUNT=$((RETRY_COUNT+1))
sleep $SLEEP_TIME
done
echo "❌ Error: Build did not complete within the expected time!"
exit 1
env:
EXPO_TOKEN: ${{ secrets.EXPO_TOKEN }}
# ========================
# πŸ“¦ Artifact Handling
# ========================
- name: "πŸ“₯ Download APK"
id: download
run: |
echo "πŸ”½ Retrieving APK URL..."
APK_URL=$(npx eas build:list --build-id ${{ needs.build-android.outputs.build_id }} --json | jq -r '.[0].artifacts.url')
echo "πŸ“₯ Downloading APK from $APK_URL..."
curl -L "$APK_URL" -o app-release.apk
echo "APK_PATH=app-release.apk" >> $GITHUB_OUTPUT
- name: "πŸ“€ Upload Artifact"
uses: actions/upload-artifact@v4
with:
name: android-apk
path: app-release.apk
generate-changelog:
name: "πŸ“œ Changelog Generator"
runs-on: ubuntu-latest
needs: build-android
outputs:
changelog: ${{ steps.changelog.outputs.CHANGELOG }}
steps:
# ========================
# πŸ› οΈ Repository Setup
# ========================
- name: "πŸ“‚ Checkout with Full History"
uses: actions/checkout@v4
with:
fetch-depth: 0
# ========================
# πŸ“„ Changelog Generation
# ========================
- name: "πŸ“‹ Create Release Notes"
id: changelog
run: |
echo "πŸ“ Generating changelog from git history..."
CHANGELOG=$(git log --pretty=format:"- %s (%h) by %an" -n 15)
echo "$CHANGELOG" > changelog.txt
echo "CHANGELOG=$(cat changelog.txt)" >> $GITHUB_OUTPUT
- name: "πŸ“€ Upload Changelog"
uses: actions/upload-artifact@v4
with:
name: changelog
path: changelog.txt
create-release:
name: "πŸš€ Smart Release Publisher"
runs-on: ubuntu-latest
needs: [build-android, download-apk, generate-changelog]
steps:
# ========================
# πŸ“₯ Artifact Retrieval
# ========================
- name: "πŸ“¦ Get APK Artifact"
uses: actions/download-artifact@v4
with:
name: android-apk
- name: "πŸ“„ Get Changelog"
uses: actions/download-artifact@v4
with:
name: changelog
# ========================
# 🏷️ Release Creation
# ========================
- name: "🎚️ Determine Release Type"
id: release-type
run: |
echo "πŸ” Detecting release type..."
if [ "${{ needs.build-android.outputs.is_production }}" = "true" ]; then
echo "🟒 Production release detected"
echo "RELEASE_TAG=v${{ needs.build-android.outputs.app_version }}" >> $GITHUB_ENV
echo "RELEASE_TITLE=Production Release v${{ needs.build-android.outputs.app_version }}" >> $GITHUB_ENV
else
echo "🟑 Nightly build detected"
echo "RELEASE_TAG=nightly-${{ needs.build-android.outputs.build_date }}" >> $GITHUB_ENV
echo "RELEASE_TITLE=Nightly Build (${{ needs.build-android.outputs.build_date }})" >> $GITHUB_ENV
fi
- name: "πŸŽ‰ Publish GitHub Release"
uses: softprops/action-gh-release@v2
with:
tag_name: ${{ env.RELEASE_TAG }}
name: ${{ env.RELEASE_TITLE }}
body_path: changelog.txt
files: app-release.apk
prerelease: ${{ needs.build-android.outputs.is_production != 'true' }}