mirror of
https://github.com/langflow-ai/langflow.git
synced 2026-03-28 01:21:53 +00:00
* chore: use self hosted ephemeral runner
In November and periodically since we have ran into issue with runners hanging and not properly cleaning themselves up causing our CI/CD pipeline to fail due to timeout or self canceling.
* chore: add actionlint for selfhosted runners
* fix: skip nightly test when skip flags are set
* fix: json string format
* chore: change to use run_on string directly
${{ inputs['runs_on'] || github.event.inputs['runs_on'] || 'ubuntu-latest' }}
* chore: use self hosted ephemeral runner
* chore: use self hosted ephemeral runner
* chore: use self hosted ephemeral runner
* chore: revert ephemeral runner testing changes, restore main-only guards
---------
Co-authored-by: vijay kumar katuri <vijay.katuri@ibm.com>
256 lines
9.1 KiB
YAML
256 lines
9.1 KiB
YAML
name: Python tests
|
|
|
|
on:
|
|
workflow_call:
|
|
secrets:
|
|
OPENAI_API_KEY:
|
|
required: true
|
|
ANTHROPIC_API_KEY:
|
|
required: true
|
|
CODECOV_TOKEN:
|
|
required: false
|
|
inputs:
|
|
python-versions:
|
|
description: "(Optional) Python versions to test"
|
|
required: true
|
|
type: string
|
|
default: "['3.10', '3.11', '3.12', '3.13']"
|
|
ref:
|
|
description: "(Optional) ref to checkout"
|
|
required: false
|
|
type: string
|
|
nightly:
|
|
description: "Whether run is from the nightly build"
|
|
required: false
|
|
type: boolean
|
|
default: false
|
|
runs-on:
|
|
description: "Runner to use for the tests"
|
|
required: false
|
|
type: string
|
|
default: "ubuntu-latest"
|
|
workflow_dispatch:
|
|
inputs:
|
|
python-versions:
|
|
description: "(Optional) Python versions to test"
|
|
required: true
|
|
type: string
|
|
default: "['3.10', '3.11', '3.12', '3.13']"
|
|
runs-on:
|
|
description: "Runner to use for the tests"
|
|
required: false
|
|
type: choice
|
|
options:
|
|
- ubuntu-latest
|
|
- self-hosted
|
|
- '["self-hosted", "linux", "ARM64", "langflow-ai-arm64-40gb-ephemeral"]'
|
|
default: ubuntu-latest
|
|
env:
|
|
POETRY_VERSION: "1.8.2"
|
|
NODE_VERSION: "22"
|
|
PYTEST_RUN_PATH: "src/backend/tests"
|
|
OPENAI_API_KEY: ${{ secrets.OPENAI_API_KEY }}
|
|
ANTHROPIC_API_KEY: ${{ secrets.ANTHROPIC_API_KEY }}
|
|
|
|
jobs:
|
|
build:
|
|
name: Unit Tests - Python ${{ matrix.python-version }} - Group ${{ matrix.group }}
|
|
runs-on: ${{ (inputs['runs-on'] && startsWith(format('{0}', inputs['runs-on']), '[') && fromJSON(inputs['runs-on'])) || inputs['runs-on'] || github.event.inputs['runs-on'] || 'ubuntu-latest' }}
|
|
strategy:
|
|
matrix:
|
|
python-version: ${{ fromJson(inputs.python-versions || '["3.10", "3.11", "3.12", "3.13"]' ) }}
|
|
splitCount: [5]
|
|
group: [1, 2, 3, 4, 5]
|
|
steps:
|
|
- uses: actions/checkout@v6
|
|
with:
|
|
ref: ${{ inputs.ref || github.ref }}
|
|
|
|
- name: Setup Node.js
|
|
uses: actions/setup-node@v6
|
|
id: setup-node
|
|
with:
|
|
node-version: ${{ env.NODE_VERSION }}
|
|
|
|
- name: "Setup Environment"
|
|
uses: astral-sh/setup-uv@v6
|
|
with:
|
|
enable-cache: true
|
|
cache-dependency-glob: "uv.lock"
|
|
python-version: ${{ matrix.python-version }}
|
|
prune-cache: false
|
|
|
|
- name: Install the project
|
|
run: uv sync
|
|
|
|
- name: Generate dynamic coverage configuration
|
|
run: |
|
|
echo "Generating dynamic coverage configuration..."
|
|
python3 scripts/generate_coverage_config.py
|
|
echo "Generated .coveragerc with the following exclusions:"
|
|
echo "Bundled components and legacy files excluded from coverage"
|
|
|
|
- name: Run unit tests
|
|
uses: nick-fields/retry@v3
|
|
with:
|
|
timeout_minutes: 12
|
|
max_attempts: 2
|
|
command: make unit_tests args="-x -vv --splits ${{ matrix.splitCount }} --group ${{ matrix.group }} --reruns 5 --cov --cov-config=src/backend/.coveragerc --cov-report=xml --cov-report=html"
|
|
|
|
- name: Upload coverage to Codecov
|
|
uses: codecov/codecov-action@v5
|
|
if: matrix.python-version == '3.10'
|
|
with:
|
|
token: ${{ secrets.CODECOV_TOKEN }}
|
|
files: ./coverage.xml
|
|
flags: backend
|
|
name: backend-coverage-group-${{ matrix.group }}
|
|
fail_ci_if_error: false
|
|
directory: ./
|
|
|
|
- name: Upload coverage artifacts
|
|
uses: actions/upload-artifact@v6
|
|
if: matrix.python-version == '3.10'
|
|
with:
|
|
name: backend-coverage-report-group-${{ matrix.group }}
|
|
path: |
|
|
coverage.xml
|
|
htmlcov/
|
|
retention-days: 30
|
|
|
|
integration-tests:
|
|
name: Integration Tests - Python ${{ matrix.python-version }}
|
|
runs-on: ${{ (inputs['runs-on'] && startsWith(format('{0}', inputs['runs-on']), '[') && fromJSON(inputs['runs-on'])) || inputs['runs-on'] || github.event.inputs['runs-on'] || 'ubuntu-latest' }}
|
|
strategy:
|
|
matrix:
|
|
python-version: ${{ fromJson(inputs.python-versions || '["3.10", "3.11", "3.12", "3.13"]' ) }}
|
|
steps:
|
|
- uses: actions/checkout@v6
|
|
with:
|
|
ref: ${{ inputs.ref || github.ref }}
|
|
- name: "Setup Environment"
|
|
uses: astral-sh/setup-uv@v6
|
|
with:
|
|
enable-cache: true
|
|
cache-dependency-glob: "uv.lock"
|
|
python-version: ${{ matrix.python-version }}
|
|
prune-cache: false
|
|
- name: Install the project
|
|
run: uv sync
|
|
- name: Run integration tests
|
|
run: make integration_tests_no_api_keys
|
|
env:
|
|
PYLEAK_LOG_LEVEL: debug # enable pyleak logging
|
|
DO_NOT_TRACK: true # disable telemetry reporting
|
|
|
|
lfx-tests:
|
|
name: LFX Tests - Python ${{ matrix.python-version }}
|
|
runs-on: ${{ (inputs['runs-on'] && startsWith(format('{0}', inputs['runs-on']), '[') && fromJSON(inputs['runs-on'])) || inputs['runs-on'] || github.event.inputs['runs-on'] || 'ubuntu-latest' }}
|
|
strategy:
|
|
matrix:
|
|
python-version: ${{ fromJson(inputs.python-versions || '["3.10", "3.11", "3.12", "3.13"]' ) }}
|
|
steps:
|
|
- uses: actions/checkout@v6
|
|
with:
|
|
ref: ${{ inputs.ref || github.ref }}
|
|
- name: "Setup Environment"
|
|
uses: astral-sh/setup-uv@v6
|
|
with:
|
|
enable-cache: true
|
|
cache-dependency-glob: "uv.lock"
|
|
python-version: ${{ matrix.python-version }}
|
|
prune-cache: false
|
|
- name: Run lfx tests
|
|
run: uv run make lfx_tests
|
|
env:
|
|
DO_NOT_TRACK: true # disable telemetry reporting
|
|
|
|
- name: Upload coverage to Codecov
|
|
uses: codecov/codecov-action@v5
|
|
if: matrix.python-version == '3.10'
|
|
with:
|
|
token: ${{ secrets.CODECOV_TOKEN }}
|
|
files: ./src/lfx/coverage.xml
|
|
flags: lfx
|
|
name: lfx-coverage
|
|
fail_ci_if_error: false
|
|
directory: ./src/lfx/
|
|
|
|
- name: Upload coverage artifacts
|
|
uses: actions/upload-artifact@v6
|
|
if: matrix.python-version == '3.10'
|
|
with:
|
|
name: lfx-coverage-report
|
|
path: |
|
|
src/lfx/coverage.xml
|
|
src/lfx/htmlcov/
|
|
retention-days: 30
|
|
|
|
test-cli:
|
|
name: Test CLI - Python ${{ matrix.python-version }}
|
|
runs-on: ${{ (inputs['runs-on'] && startsWith(format('{0}', inputs['runs-on']), '[') && fromJSON(inputs['runs-on'])) || inputs['runs-on'] || github.event.inputs['runs-on'] || 'ubuntu-latest' }}
|
|
strategy:
|
|
matrix:
|
|
python-version: ${{ fromJson(inputs.python-versions || '["3.10", "3.11", "3.12", "3.13"]') }}
|
|
steps:
|
|
- name: Check out the code at a specific ref
|
|
uses: actions/checkout@v6
|
|
with:
|
|
ref: ${{ inputs.ref || github.ref }}
|
|
- name: "Setup Environment"
|
|
uses: astral-sh/setup-uv@v6
|
|
with:
|
|
enable-cache: true
|
|
cache-dependency-glob: "uv.lock"
|
|
python-version: ${{ matrix.python-version }}
|
|
prune-cache: false
|
|
|
|
- name: Check Version
|
|
id: check-version
|
|
# We need to print $3 because langflow-base is a dependency of langflow
|
|
# For langlow we'd use print $2
|
|
run: |
|
|
version=$(uv tree | grep 'langflow-base' | awk '{print $3}' | sed 's/^v//' | head -n 1)
|
|
url="https://pypi.org/pypi/langflow-base/json"
|
|
if [ ${{ inputs.nightly }} == true ]; then
|
|
url="https://pypi.org/pypi/langflow-base-nightly/json"
|
|
fi
|
|
|
|
last_released_version=$(curl -s $url | jq -r '.releases | keys | .[]' | sort -V | tail -n 1)
|
|
if [ "$version" != "$last_released_version" ]; then
|
|
echo "Version $version has not been released yet. Skipping the rest of the job."
|
|
echo skipped=true >> $GITHUB_OUTPUT
|
|
exit 0
|
|
else
|
|
echo version=$version >> $GITHUB_OUTPUT
|
|
echo skipped=false >> $GITHUB_OUTPUT
|
|
fi
|
|
- name: Build wheel
|
|
if: steps.check-version.outputs.skipped == 'false'
|
|
run: |
|
|
make build main=true
|
|
- name: Install wheel and Test CLI
|
|
if: steps.check-version.outputs.skipped == 'false'
|
|
run: |
|
|
uv venv new-venv
|
|
source new-venv/bin/activate
|
|
uv pip install dist/*.whl
|
|
- name: Test CLI
|
|
if: steps.check-version.outputs.skipped == 'false'
|
|
run: |
|
|
source new-venv/bin/activate
|
|
python -m langflow run --host localhost --port 7860 --backend-only &
|
|
SERVER_PID=$!
|
|
# Wait for the server to start
|
|
timeout 120 bash -c 'until curl -f http://localhost:7860/api/v1/auto_login; do sleep 5; done' || (echo "Server did not start in time" && kill $SERVER_PID && exit 1)
|
|
# Terminate the server
|
|
kill $SERVER_PID || (echo "Failed to terminate the server" && exit 1)
|
|
sleep 20 # give the server some time to terminate
|
|
# Check if the server is still running
|
|
if kill -0 $SERVER_PID 2>/dev/null; then
|
|
echo "Failed to terminate the server"
|
|
exit 0
|
|
else
|
|
echo "Server terminated successfully"
|
|
fi
|