Skip to content

E2E Tests

E2E Tests #1080

Workflow file for this run

name: E2E Tests
on:
# See: https://crontab.guru/examples.html
# Run at 6 A.M and 6 P.M. (CST)
schedule:
- cron: '0 12,0 * * *'
workflow_dispatch:
workflow_call:
jobs:
versions:
name: Get Versions 📜
uses: CrisisCleanup/crisiscleanup-4-web/.github/workflows/get-versions.yml@master
e2e:
name: 🧪 E2E Tests 💻 | ${{ matrix.stage }} | ${{ matrix.project }}
timeout-minutes: 60
runs-on: ${{ matrix.os }}
needs: [versions]
strategy:
fail-fast: false
matrix:
stage: [development, staging, production]
project: [chromium, firefox, webkit]
include:
- project: chromium
os: ubuntu-latest
cache_dir: ~/.cache/ms-playwright
- project: firefox
os: ubuntu-latest
cache_dir: ~/.cache/ms-playwright
- project: webkit
os: macos-12
cache_dir: ~/Library/Caches/ms-playwright
steps:
- uses: actions/checkout@v4
- uses: pnpm/action-setup@v2
with:
version: ${{ needs.versions.outputs.PNPM_VERSION }}
- uses: actions/setup-node@v4
with:
node-version: ${{ needs.versions.outputs.NODE_VERSION }}
cache: pnpm
# NOTE: Using single quotes when setting $GITHUB_ENV in-case secret has special characters
- name: 🔧 Setup Common Env
run: |
echo 'PW_SKIP_WEBSERVER=1' >> $GITHUB_ENV
- name: 🔧 Setup Development Test Env
if: matrix.stage == 'development'
run: |
echo 'PW_TEST_TITLE_PREFIX=DEV: ' >> $GITHUB_ENV
echo 'VITE_APP_BASE_URL=https://app.dev.crisiscleanup.io' >> $GITHUB_ENV
echo 'TEST_APP_EMAIL=${{ secrets.TEST_APP_EMAIL_DEV }}' >> $GITHUB_ENV
echo 'TEST_APP_PASSWORD=${{ secrets.TEST_APP_PASSWORD_DEV }}' >> $GITHUB_ENV
echo 'TEST_APP_ADMIN_EMAIL=${{ secrets.TEST_APP_ADMIN_EMAIL_DEV }}' >> $GITHUB_ENV
echo 'TEST_APP_ADMIN_PASSWORD=${{ secrets.TEST_APP_ADMIN_PASSWORD_DEV }}' >> $GITHUB_ENV
- name: 🔧 Setup Staging Test Env
if: matrix.stage == 'staging'
run: |
echo 'PW_TEST_TITLE_PREFIX=STAGING: ' >> $GITHUB_ENV
echo 'VITE_APP_BASE_URL=https://app.staging.crisiscleanup.io' >> $GITHUB_ENV
echo 'TEST_APP_EMAIL=${{ secrets.TEST_APP_EMAIL_PROD }}' >> $GITHUB_ENV
echo 'TEST_APP_PASSWORD=${{ secrets.TEST_APP_PASSWORD_PROD }}' >> $GITHUB_ENV
echo 'TEST_APP_ADMIN_EMAIL=${{ secrets.TEST_APP_ADMIN_EMAIL_PROD }}' >> $GITHUB_ENV
echo 'TEST_APP_ADMIN_PASSWORD=${{ secrets.TEST_APP_ADMIN_PASSWORD_PROD }}' >> $GITHUB_ENV
- name: 🔧 Setup Production Test Env
if: matrix.stage == 'production'
run: |
# add 'PROD: ' prefix in test titles. ex: 'PROD: should do this - ( @production )'
echo 'PW_TEST_TITLE_PREFIX=PROD: ' >> $GITHUB_ENV
echo 'VITE_APP_BASE_URL=https://crisiscleanup.org' >> $GITHUB_ENV
echo 'TEST_APP_EMAIL=${{ secrets.TEST_APP_EMAIL_PROD }}' >> $GITHUB_ENV
echo 'TEST_APP_PASSWORD=${{ secrets.TEST_APP_PASSWORD_PROD }}' >> $GITHUB_ENV
echo 'TEST_APP_ADMIN_EMAIL=${{ secrets.TEST_APP_ADMIN_EMAIL_PROD }}' >> $GITHUB_ENV
echo 'TEST_APP_ADMIN_PASSWORD=${{ secrets.TEST_APP_ADMIN_PASSWORD_PROD }}' >> $GITHUB_ENV
- name: Install dependencies
run: pnpm install
- name: 🔧 Set Playwright Version
run: pnpm exec playwright --version > .github/.playwright-version
- name: ⚡️ Cache playwright binaries
uses: actions/cache@v4
id: playwright-cache
with:
path: ${{ matrix.cache_dir }}
key: ${{ runner.os }}-${{ matrix.project }}-pw-${{ hashFiles('**/.playwright-version') }}
- name: 📥 Install ${{ matrix.project }} with Playwright
if: steps.playwright-cache.outputs.cache-hit != 'true'
run: pnpm exec playwright install --with-deps ${{ matrix.project }}
- name: 🎭 Playwright tests
run: pnpm run test:e2e:${{ matrix.stage }} --project=${{ matrix.project }}
- name: 📤 Upload blob report to GitHub Actions Artifacts
if: always()
uses: actions/upload-artifact@v3
with:
name: all-blob-reports-${{ matrix.stage }}
path: blob-report
retention-days: 1
- name: Save Test Result
if: always()
run: |
echo ${{ job.status }} > test-result-${{ matrix.stage }}-${{ matrix.project }}.txt
- name: Upload Test Result
if: always()
uses: actions/upload-artifact@v3
with:
name: matrix-outputs
path: test-result-${{ matrix.stage }}-${{ matrix.project }}.txt
aggregate-results:
name: Aggregate Test Results
needs: [e2e]
runs-on: ubuntu-latest
if: always()
outputs:
overall_status: ${{ steps.aggregate.outputs.overall_status }}
steps:
- uses: actions/checkout@v4
- name: Download all artifacts
uses: actions/download-artifact@v3
with:
name: matrix-outputs
path: test-results
- name: Aggregate Test Results
id: aggregate
run: |
#!/bin/bash
overall_status="success"
# Loop through each test result file
for file in test-results/*.txt; do
if [[ -f "$file" ]]; then
status=$(cat "$file")
echo "Test result in $file: $status"
if [[ "$status" == "failure" ]]; then
overall_status="failure"
break
fi
fi
done
# Output the overall status
echo "Overall status: $overall_status"
echo "overall_status=$overall_status" >> $GITHUB_OUTPUT
notify-failed:
name: 🔴 Notify Failure 🔴
needs: [e2e, aggregate-results]
runs-on: ubuntu-latest
if: always()
steps:
- name: Notify Slack
uses: rtCamp/action-slack-notify@v2
if: ${{ needs.aggregate-results.outputs.overall_status == 'failure' }}
env:
SLACK_WEBHOOK: ${{ secrets.SLACK_WEBHOOK }}
SLACK_ICON: ${{ secrets.SLACK_ICON }}
SLACK_USERNAME: 'CrisisCleanup Bot'
SLACK_CHANNEL: 'qa'
SLACK_COLOR: '#D94F37'
SLACK_TITLE: 'E2e Tests Failed!'
SLACK_MESSAGE: '🧪 E2e Tests Failed... Someone broke something! 😦'
notify-success:
name: 🟢 Notify Success 🟢
needs: [e2e, aggregate-results]
runs-on: ubuntu-latest
if: always()
steps:
- name: Notify Slack
uses: rtCamp/action-slack-notify@v2
if: ${{ needs.aggregate-results.outputs.overall_status == 'success' }}
env:
SLACK_WEBHOOK: ${{ secrets.SLACK_WEBHOOK }}
SLACK_ICON: ${{ secrets.SLACK_ICON }}
SLACK_USERNAME: 'CrisisCleanup Bot'
SLACK_CHANNEL: 'qa'
SLACK_COLOR: '#13E768'
SLACK_TITLE: 'E2e Tests Passed!'
SLACK_MESSAGE: '🧪 E2e Tests Passed!!'
merge-e2e-reports:
# Merge reports after playwright-tests, even if some shards have failed
name: 🔄 Merge Playwright Reports | ${{ matrix.stage }}
if: always()
needs: [versions, e2e]
runs-on: ubuntu-latest
strategy:
matrix:
stage: [development, staging, production]
steps:
- uses: actions/checkout@v4
- uses: pnpm/action-setup@v2
with:
version: ${{ needs.versions.outputs.PNPM_VERSION }}
- uses: actions/setup-node@v4
with:
node-version: ${{ needs.versions.outputs.NODE_VERSION }}
cache: pnpm
- name: 📥 Install dependencies
run: pnpm install
- name: 📥 Download blob reports from GitHub Actions Artifacts
uses: actions/download-artifact@v3
with:
name: all-blob-reports-${{ matrix.stage }}
path: all-blob-reports
- name: 🔄 Merge all blob reports
env:
PLAYWRIGHT_JSON_OUTPUT_NAME: e2e-results.json
run: pnpm exec playwright merge-reports --reporter html,json,github,markdown ./all-blob-reports
- name: 📨 Send report to slack
env:
SLACK_BOT_USER_OAUTH_TOKEN: ${{ secrets.SLACK_BOT_USER_OAUTH_TOKEN }}
REPORT_NAME: Playwright Report | ${{ matrix.stage }}
BUILD_ID: ${{ github.run_id }}
BRANCH_NAME: ${{ github.ref_name }}
COMMIT_ID: ${{ github.sha }}
# Todo: replace with html report url when we upload html reports somewhere
RESULT_URL: ${{ github.server_url }}/${{ github.repository }}/actions/runs/${{ github.run_id }}
run: |
cat > cli_config.json <<EOF
{
"sendResults": "always",
"slackLogLevel": "debug",
"sendUsingBot": {
"channels": ["qa"]
},
"showInThread": true,
"meta": [
{ "key": "name", "value" : "__ENV_REPORT_NAME"},
{ "key": "build", "value" : "__ENV_BUILD_ID"},
{ "key": "branch", "value" : "__ENV_BRANCH_NAME"},
{ "key": "commit", "value" : "__ENV_COMMIT_ID"},
{ "key": "results", "value" : "__ENV_RESULT_URL"}
],
"maxNumberOfFailures": 10,
"disableUnfurl": true
}
EOF
echo "Created cli_config.json for playwright-slack-report..."
cat cli_config.json
echo "Playwright JSON report:"
cat e2e-results.json
echo "Sending report to slack..."
pnpm exec playwright-slack-report --config=cli_config.json --json-results=e2e-results.json
- name: 🛠️ Generate Job Summary
run: |
echo "# 🧪 E2E Summary 🧪" >> $GITHUB_STEP_SUMMARY
echo "" >> $GITHUB_STEP_SUMMARY
cat report.md >> $GITHUB_STEP_SUMMARY # report.md is generated from markdown reporter
- name: 📤 Upload HTML report (${{ matrix.stage }})
uses: actions/upload-artifact@v3
with:
name: e2e-report-${{ matrix.stage }}-${{ github.run_id }}-attempt_${{ github.run_attempt }}
path: playwright-report
retention-days: 7