Skip to content

Added a workflow to parallelise the E2E tests #6

Added a workflow to parallelise the E2E tests

Added a workflow to parallelise the E2E tests #6

name: Parallel Code Coverage
permissions:
contents: read
on: [pull_request, workflow_dispatch]
jobs:
discover-tests:
runs-on: ubuntu-latest
outputs:
test-files: ${{ steps.discover.outputs.test-files }}
steps:
- name: Check out repository
uses: actions/checkout@v4
with:
fetch-depth: 0
ref: ${{ github.event.pull_request.head.ref || github.ref_name }}
repository: ${{ github.event.pull_request.head.repo.full_name || github.repository }}
- name: Discover test files
id: discover
run: |
# Find all test files in e2e directory and create JSON array (excluding test_driver.py for now)
TEST_FILES=$(find tests/e2e -name "test_*.py" -type f | grep -v test_driver.py | sort | jq -R -s -c 'split("\n")[:-1]')
echo "test-files=$TEST_FILES" >> $GITHUB_OUTPUT
echo "Discovered test files: $TEST_FILES"
e2e-tests:
runs-on: ubuntu-latest
environment: azure-prod
needs: discover-tests
strategy:
matrix:
test_file: ${{ fromJson(needs.discover-tests.outputs.test-files) }}
mode: ["thrift", "sea"]
env:
DATABRICKS_SERVER_HOSTNAME: ${{ secrets.DATABRICKS_HOST }}
DATABRICKS_HTTP_PATH: ${{ secrets.TEST_PECO_WAREHOUSE_HTTP_PATH }}
DATABRICKS_TOKEN: ${{ secrets.DATABRICKS_TOKEN }}
DATABRICKS_CATALOG: peco
DATABRICKS_USER: ${{ secrets.TEST_PECO_SP_ID }}
steps:
- name: Check out repository
uses: actions/checkout@v4
with:
fetch-depth: 0
ref: ${{ github.event.pull_request.head.ref || github.ref_name }}
repository: ${{ github.event.pull_request.head.repo.full_name || github.repository }}
- name: Set up python
id: setup-python
uses: actions/setup-python@v5
with:
python-version: "3.10"
- name: Install Poetry
uses: snok/install-poetry@v1
with:
virtualenvs-create: true
virtualenvs-in-project: true
installer-parallel: true
- name: Load cached venv
id: cached-poetry-dependencies
uses: actions/cache@v4
with:
path: .venv
key: venv-${{ runner.os }}-${{ steps.setup-python.outputs.python-version }}-${{ github.event.repository.name }}-${{ hashFiles('**/poetry.lock') }}
- name: Install dependencies
if: steps.cached-poetry-dependencies.outputs.cache-hit != 'true'
run: poetry install --no-interaction --no-root
- name: Install library
run: poetry install --no-interaction --all-extras
- name: Run ${{ matrix.mode }} tests for ${{ matrix.test_file }}
run: |
echo "Running ${{ matrix.mode }} tests for ${{ matrix.test_file }}"
# Set test filter based on mode
if [ "${{ matrix.mode }}" = "sea" ]; then
TEST_FILTER="-k"
TEST_EXPRESSION="extra_params1 or extra_params2"
else
TEST_FILTER="-k"
TEST_EXPRESSION="extra_params0 or not extra_params"
fi
TEST_NAME=$(basename "${{ matrix.test_file }}" .py)
COVERAGE_FILE="coverage-${TEST_NAME}-${{ matrix.mode }}.xml"
COVERAGE_DATA=".coverage-${TEST_NAME}-${{ matrix.mode }}"
echo "TEST_NAME=$TEST_NAME" >> $GITHUB_ENV
poetry run pytest "${{ matrix.test_file }}" "$TEST_FILTER" "$TEST_EXPRESSION" \
--cov=src --cov-append --cov-report=xml:$COVERAGE_FILE --cov-report=term \
-v || [ $? -eq 5 ]
# Ensure .coverage file exists for merging (even if empty)
if [ -f ".coverage" ]; then
mv .coverage "$COVERAGE_DATA"
echo "✅ Saved coverage data as $COVERAGE_DATA"
else
echo "⚠️ No .coverage generated, creating empty file"
touch "$COVERAGE_DATA"
fi
# Debug: Show what files we have for upload
echo "📁 Files available for upload:"
ls -la .coverage* 2>/dev/null || echo "No coverage files found"
- name: Upload coverage artifact
uses: actions/upload-artifact@v4
if: always()
with:
name: coverage-${{ env.TEST_NAME }}-${{ matrix.mode }}
path: |
.coverage-*
coverage-*-${{ matrix.mode }}.xml
if-no-files-found: warn
merge-coverage:
runs-on: ubuntu-latest
needs: [e2e-tests]
steps:
- name: Check out repository
uses: actions/checkout@v4
- name: Set up python
id: setup-python
uses: actions/setup-python@v5
with:
python-version: "3.10"
- name: Install Poetry
uses: snok/install-poetry@v1
with:
virtualenvs-create: true
virtualenvs-in-project: true
installer-parallel: true
- name: Load cached venv
id: cached-poetry-dependencies
uses: actions/cache@v4
with:
path: .venv
key: venv-${{ runner.os }}-${{ steps.setup-python.outputs.python-version }}-${{ github.event.repository.name }}-${{ hashFiles('**/poetry.lock') }}
- name: Install dependencies
if: steps.cached-poetry-dependencies.outputs.cache-hit != 'true'
run: poetry install --no-interaction --no-root
- name: Install library
run: poetry install --no-interaction --all-extras
- name: Download all coverage artifacts
uses: actions/download-artifact@v4
with:
path: coverage_files
- name: Merge coverage
run: |
# Install xmllint if not available
if ! command -v xmllint &> /dev/null; then
sudo apt-get update && sudo apt-get install -y libxml2-utils
fi
# Copy all coverage files with unique names
echo "📁 Checking for coverage files in artifacts..."
for artifact_dir in coverage_files/*/; do
if [ -d "$artifact_dir" ]; then
echo "Artifact: $(basename "$artifact_dir")"
ls -la "$artifact_dir" || echo " (empty or inaccessible)"
for cov_file in "$artifact_dir"/.coverage-*; do
if [ -f "$cov_file" ]; then
cp "$cov_file" .
echo " ✅ Copied $(basename "$cov_file")"
fi
done
fi
done
echo "Available .coverage files for merging:"
ls -la .coverage-* 2>/dev/null || echo "No .coverage-* files found"
poetry run coverage combine .coverage-* || true
poetry run coverage xml || echo '<coverage lines-covered="0" lines-valid="1"></coverage>' > coverage.xml
poetry run coverage report || true
- name: Report coverage percentage
run: |
COVERAGE_FILE="coverage.xml"
if [ ! -f "$COVERAGE_FILE" ]; then
echo "ERROR: Coverage file not found at $COVERAGE_FILE"
exit 1
fi
COVERED=$(xmllint --xpath "string(//coverage/@lines-covered)" "$COVERAGE_FILE")
TOTAL=$(xmllint --xpath "string(//coverage/@lines-valid)" "$COVERAGE_FILE")
# Calculate percentage using Python for precision
PERCENTAGE=$(python3 -c "covered=${COVERED}; total=${TOTAL}; print(round((covered/total)*100, 2))")
echo "📊 Combined Coverage: ${PERCENTAGE}%"