Table of Contents generated with DocToc
In order to optimise our CI jobs, we've implemented optimisations to only run selected checks for some kind of changes. The logic implemented reflects the internal architecture of Airflow 2.0 packages, and it helps to keep down both the usage of jobs in GitHub Actions and CI feedback time to contributors in case of simpler changes.
We have the following Groups of files for CI that determine which tests are run:
Environment files
- if any of those changes, that forces 'run everything' mode, because changes there might simply change the whole environment of what is going on in CI (Container image, dependencies)Python and Javascript production files
- this area is useful in CodeQL Security scanning - if any of the python or javascript files for airflow "production" changed, this means that the security scans should runAPI tests and codegen files
- those are OpenAPI definition files that impact Open API specification and determine that we should run dedicated API tests.Helm files
- change in those files impacts helm "rendering" testsSetup files
- change in the setup files indicates that we should runupgrade to newer dependencies
DOCs files
- change in those files indicate that we should run documentation buildsUI and WWW files
- those are files for the UI and WWW part of our UI (useful to determine if UI tests should run)Kubernetes files
- determine if any of Kubernetes related tests should be runAll Python files
- if none of the Python file changed, that indicates that we should not run unit testsAll source files
- if none of the sources change, that indicates that we should probably not build an image and run any image-based static checks
We have the following unit test types that can be selectively disabled/enabled based on the content of the incoming PR:
- Always - those are tests that should be always executed (always folder)
- Core - for the core Airflow functionality (core folder)
- API - Tests for the Airflow API (api and api_connexion folders)
- CLI - Tests for the Airflow CLI (cli folder)
- WWW - Tests for the Airflow webserver (www folder)
- Providers - Tests for all Providers of Airflow (providers folder)
We also have several special kinds of tests that are not separated by packages, but they are marked with
pytest markers. They can be found in any of those packages and they can be selected by the appropriate
pytest custom command line options. See TESTING.rst <TESTING.rst>
_ for details but those are:
- Integration - tests that require external integration images running in docker-compose
- Quarantined - tests that are flaky and need to be fixed
- Postgres - tests that require Postgres database. They are only run when backend is Postgres
- MySQL - tests that require MySQL database. They are only run when backend is MySQL
Even if the types are separated, In case they share the same backend version/python version, they are
run sequentially in the same job, on the same CI machine. Each of them in a separate docker run
command
and with additional docker cleaning between the steps to not fall into the trap of exceeding resource
usage in one big test run, but also not to increase the number of jobs per each Pull Request.
The logic implements the following rules:
Full tests
mode is enabled when the event is PUSH, or SCHEDULE or when "full tests needed" label is set. That enables all matrix combinations of variables, and all possible tests- Python, Kubernetes, Backend, Kind, Helm versions are limited to "defaults" only unless
Full tests
mode is enabled. - If "Commit" to work on cannot be determined, or
Full Test
mode is enabled or some of the important environment files (setup.py, setup.cfg, Dockerfile, build scripts) changed - all unit tests are executed - this isrun everything
mode. No further checks are performed. Python scans
,Javascript scans
,API tests/codegen
,UI
,WWW
,Kubernetes
tests andDOC builds
are enabled if any of the relevant files have been changed.Helm
tests are run only if relevant files have been changed and if current branch ismain
.- If no Source files are changed - no tests are run and no further rules below are checked.
Image building
is enabled if either test are run, docs are build or kubernetes tests are run. All those needCI
orPROD
images to be built.- In case of
Providers
test in regular PRs, additional check is done in order to determine which providers are affected and the actual selection is made based on that:- if directly provider code is changed (either in the provider, test or system tests) then this provider is selected.
- if there are any providers that depend on the affected providers, they are also included in the list of affected providers (but not recursively - only direct dependencies are added)
- if there are any changes to "common" provider code not belonging to any provider (usually system tests or tests), then tests for all Providers are run
- The specific unit test type is enabled only if changed files match the expected patterns for each type
(
API
,CLI
,WWW
,Providers
). TheAlways
test type is added always if any unit tests are run.Providers
tests are removed if current branch is different thanmain
- If there are no files left in sources after matching the test types and Kubernetes files, then apparently some Core/Other files have been changed. This automatically adds all test types to execute. This is done because changes in core might impact all the other test types.
- if
Image building
is disabled, only basic pre-commits are enabled - no 'image-depending` pre-commits are enabled. - If there are some setup files changed,
upgrade to newer dependencies
is enabled.
The selective check outputs available are described below:
Output | Meaning of the output | Example value |
---|---|---|
all-python-versions | List of all python versions there are available in the form of JSON array | ['3.7', '3.8', '3.9', '3.10'] |
all-python-versions-list-as-string | List of all python versions there are available in the form of space separated string | 3.7 3.8 3.9 3.10 |
basic-checks-only | Whether to run all static checks ("false") or only basic set of static checks ("true") | false |
cache-directive | Which cache should be be used for images ("registry", "local" , "disabled") | registry |
debug-resources | Whether resources usage should be printed during parallel job execution ("true"/ "false") | false |
default-branch | Which branch is default for the the build ("main" for main branch, "v2-4-test" for 2.4 line etc.) | main |
default-constraints-branch | Which branch is default for the the build ("constraints-main" for main branch, "constraints-2-4" etc.) | constraints-main |
default-helm-version | Which Helm version to use as default | v3.9.4 |
default-kind-version | Which Kind version to use as default | v0.16.0 |
default-kubernetes-version | Which Kubernetes version to use as default | v1.25.2 |
default-mssql-version | Which MsSQL version to use as default | 2017-latest |
default-mysql-version | Which MySQL version to use as default | 5.7 |
default-postgres-version | Which Postgres version to use as default | 10 |
default-python-version | Which Python version to use as default | 3.7 |
docs-build | Whether to build documentation ("true"/"false") | true |
docs-filter | What filter to apply to docs building - used in non-main branches to skip provider and chart docs. | --package-filter apache-airflow --package-filter docker-stack |
helm-version | Which Helm version to use for tests | v3.9.4 |
image-build | Whether CI image build is needed | true |
kind-version | Which Kind version to use for tests | v0.16.0 |
kubernetes-combos | All combinations of Python version and Kubernetes version to use for tests as space-separated string | 3.7-v1.25.2 3.8-v1.26.4 |
kubernetes-versions | All Kubernetes versions to use for tests as JSON array | ['v1.25.2'] |
kubernetes-versions-list-as-string | All Kubernetes versions to use for tests as space-separated string | v1.25.2 |
mssql-exclude | Which versions of MsSQL to exclude for tests as JSON array | [] |
mssql-versions | Which versions of MsSQL to use for tests as JSON array | ['2017-latest'] |
mysql-exclude | Which versions of MySQL to exclude for tests as JSON array | [] |
mysql-versions | Which versions of MySQL to use for tests as JSON array | ['5.7'] |
needs-api-codegen | Whether "api-codegen" are needed to run ("true"/"false") | true |
needs-api-tests | Whether "api-tests" are needed to run ("true"/"false") | true |
needs-helm-tests | Whether Helm tests are needed to run ("true"/"false") | true |
needs-javascript-scans | Whether javascript CodeQL scans should be run ("true"/"false") | true |
needs-python-scans | Whether Python CodeQL scans should be run ("true"/"false") | true |
postgres-exclude | Which versions of Postgres to exclude for tests as JSON array | [] |
postgres-versions | Which versions of Postgres to use for tests as JSON array | ['10'] |
python-versions | Which versions of Python to use for tests as JSON array | ['3.7'] |
python-versions-list-as-string | Which versions of MySQL to use for tests as space-separated string | 3.7 |
run-kubernetes-tests | Whether Kubernetes tests should be run ("true"/"false") | true |
run-tests | Whether unit tests should be run ("true"/"false") | true |
run-ui-tests | Whether UI tests should be run ("true"/"false") | true |
run-www-tests | Whether WWW tests should be run ("true"/"false") | true |
skip-pre-commits | Which pre-commits should be skipped during the static-checks run | identity |
sqlite-exclude | Which versions of Sqlite to exclude for tests as JSON array | [] |
test-types | Which test types should be run for unit tests | API Always CLI Core Integration Other Providers WWW |
upgrade-to-newer-dependencies | Whether the image build should attempt to upgrade all dependencies | false |