From 374cc5d361535f1d776c517de6c9c09330484d85 Mon Sep 17 00:00:00 2001 From: Jarek Potiuk Date: Fri, 20 Dec 2024 14:08:35 +0100 Subject: [PATCH] Simplify caching mechanisms for CI and PROD images For a long time we had used a sophisticated mechanism to speed up our CI jobs by building the images in "pull_request_target" workflow and pushing them to GitHub registry. That however had several drawbacks: * CI image was complex when it comes to layer setup (we had to pre- cache installed dependencies by installing them from branch tip * The pull_request_target is a very dangerous workflow, we had a number of security problems with it (and it's difficult to debug) * Caching of `pip` and `uv` was not used because it increased size of the image significantly This PR significantly improves the caching mechanisms for the images building of several advacements that were not possible before: * The upload-artifacts@v4 action and improved stash action developed by @assignUser and published in "apache/infrastructure-actions" allows us to store all images (8GB per run) in artifacts rather than in registry - so we can do the image build once and share it with all the jobs. * The uv speed is "enough" to allow occasional installation of Airlfow locally. This allows to utilize cache-mount and locally build uv cache, rather than rely on "remote" cache when we are building local images for breeze. The first time you build local breeze image it will take 2-5 more minutes (depending on your network speed, but because we can utilise cache mounts, every subsequent build should be very fast - even if all dependencies change. Using uv also allows to "always" reinstall airflow when you build the image even if single source file changed, because with cache it takes sub-seconds to reinstall airflow and all dependencies. * the cache mounts are not included in the image size, and since we can export and import images in CI in artifacts and we do not need to rebuild them, the images shared as compressed artifacts are relatively small (2GB) - cache of `uv` is around 4GB on top of that so sharing image built in the "build image" job with other jobs in the same workflow is fast. * we are still using registry cache for the "non-python" parts of the image - both CI and breeze image build speed benefit from using the image cache for system dependencies, database clients etc. this helps with faster rebuilds of the images for local development environment * documentation has been updated to reflect the new CI setup. The diagrams showing the workflows of ours are no longer needed as the workflows are quite straightforward when they are looked at. Fixes: #42999 Fixes: #43268 --- .../actions/checkout_target_commit/action.yml | 81 --- .../actions/prepare_all_ci_images/action.yml | 71 +++ .../prepare_breeze_and_image/action.yml | 46 +- .../prepare_single_ci_image/action.yml | 47 ++ .../workflows/additional-ci-image-checks.yml | 9 +- .../workflows/additional-prod-image-tests.yml | 47 +- .github/workflows/basic-tests.yml | 1 - .github/workflows/build-images.yml | 264 ---------- .github/workflows/ci-image-build.yml | 87 ++-- .github/workflows/ci-image-checks.yml | 40 +- .github/workflows/ci.yml | 164 +----- .github/workflows/finalize-tests.yml | 9 - .github/workflows/generate-constraints.yml | 25 +- .github/workflows/helm-tests.yml | 12 +- .../workflows/integration-system-tests.yml | 28 +- .github/workflows/k8s-tests.yml | 73 ++- .github/workflows/prod-image-build.yml | 122 ++--- .github/workflows/prod-image-extra-checks.yml | 5 - .github/workflows/push-image-cache.yml | 64 ++- .github/workflows/release_dockerhub_image.yml | 2 +- .github/workflows/run-unit-tests.yml | 12 +- .github/workflows/special-tests.yml | 10 - .github/workflows/task-sdk-tests.yml | 12 +- .github/workflows/test-provider-packages.yml | 23 +- Dockerfile | 129 +---- Dockerfile.ci | 202 ++------ dev/breeze/doc/06_managing_docker_images.rst | 64 ++- dev/breeze/doc/ci/01_ci_environment.md | 89 ++-- dev/breeze/doc/ci/02_images.md | 66 +-- dev/breeze/doc/ci/04_selective_checks.md | 18 +- dev/breeze/doc/ci/05_workflows.md | 201 +++----- .../ci/{07_debugging.md => 06_debugging.md} | 7 +- dev/breeze/doc/ci/06_diagrams.md | 466 ------------------ dev/breeze/doc/ci/07_running_ci_locally.md | 129 +++++ dev/breeze/doc/ci/08_running_ci_locally.md | 141 ------ dev/breeze/doc/ci/README.md | 5 +- dev/breeze/doc/images/image_artifacts.png | Bin 0 -> 47666 bytes dev/breeze/doc/images/output_ci-image.svg | 16 +- dev/breeze/doc/images/output_ci-image.txt | 2 +- .../doc/images/output_ci-image_build.svg | 180 ++++--- .../doc/images/output_ci-image_build.txt | 2 +- .../doc/images/output_ci-image_load.svg | 144 ++++++ .../doc/images/output_ci-image_load.txt | 1 + .../doc/images/output_ci-image_pull.svg | 68 ++- .../doc/images/output_ci-image_pull.txt | 2 +- .../doc/images/output_ci-image_save.svg | 128 +++++ .../doc/images/output_ci-image_save.txt | 1 + .../doc/images/output_ci-image_verify.svg | 56 +-- .../doc/images/output_ci-image_verify.txt | 2 +- .../doc/images/output_k8s_build-k8s-image.svg | 54 +- .../doc/images/output_k8s_build-k8s-image.txt | 2 +- .../images/output_k8s_run-complete-tests.svg | 82 ++- .../images/output_k8s_run-complete-tests.txt | 2 +- dev/breeze/doc/images/output_prod-image.svg | 16 +- dev/breeze/doc/images/output_prod-image.txt | 2 +- .../doc/images/output_prod-image_build.svg | 198 ++++---- .../doc/images/output_prod-image_build.txt | 2 +- .../doc/images/output_prod-image_load.svg | 132 +++++ .../doc/images/output_prod-image_load.txt | 1 + .../doc/images/output_prod-image_pull.svg | 68 ++- .../doc/images/output_prod-image_pull.txt | 2 +- .../doc/images/output_prod-image_save.svg | 128 +++++ .../doc/images/output_prod-image_save.txt | 1 + .../doc/images/output_prod-image_verify.svg | 56 +-- .../doc/images/output_prod-image_verify.txt | 2 +- ...elease-management_generate-constraints.svg | 58 +-- ...elease-management_generate-constraints.txt | 2 +- ...utput_setup_check-all-params-in-groups.svg | 74 +-- ...utput_setup_check-all-params-in-groups.txt | 2 +- ...output_setup_regenerate-command-images.svg | 20 +- ...output_setup_regenerate-command-images.txt | 2 +- dev/breeze/doc/images/output_shell.svg | 204 ++++---- dev/breeze/doc/images/output_shell.txt | 2 +- .../doc/images/output_start-airflow.svg | 130 +++-- .../doc/images/output_start-airflow.txt | 2 +- .../doc/images/output_static-checks.svg | 36 +- .../doc/images/output_static-checks.txt | 2 +- .../output_testing_core-integration-tests.svg | 42 +- .../output_testing_core-integration-tests.txt | 2 +- .../doc/images/output_testing_core-tests.svg | 90 ++-- .../doc/images/output_testing_core-tests.txt | 2 +- .../output_testing_docker-compose-tests.svg | 44 +- .../output_testing_docker-compose-tests.txt | 2 +- .../doc/images/output_testing_helm-tests.svg | 34 +- .../doc/images/output_testing_helm-tests.txt | 2 +- ...ut_testing_providers-integration-tests.svg | 42 +- ...ut_testing_providers-integration-tests.txt | 2 +- .../images/output_testing_providers-tests.svg | 108 ++-- .../images/output_testing_providers-tests.txt | 2 +- ...output_testing_python-api-client-tests.svg | 86 ++-- ...output_testing_python-api-client-tests.txt | 2 +- .../images/output_testing_system-tests.svg | 42 +- .../images/output_testing_system-tests.txt | 2 +- .../images/output_testing_task-sdk-tests.svg | 42 +- .../images/output_testing_task-sdk-tests.txt | 2 +- .../airflow_breeze/commands/ci_commands.py | 12 - .../commands/ci_image_commands.py | 152 ++++-- .../commands/ci_image_commands_config.py | 30 +- .../commands/common_image_options.py | 36 +- .../airflow_breeze/commands/common_options.py | 18 +- .../commands/developer_commands.py | 24 +- .../commands/developer_commands_config.py | 3 - .../commands/kubernetes_commands.py | 74 ++- .../commands/kubernetes_commands_config.py | 2 - .../commands/production_image_commands.py | 126 ++++- .../production_image_commands_config.py | 29 +- .../commands/release_management_commands.py | 6 - .../release_management_commands_config.py | 1 - .../commands/testing_commands.py | 29 +- .../commands/testing_commands_config.py | 4 - .../src/airflow_breeze/global_constants.py | 1 - .../airflow_breeze/params/build_ci_params.py | 1 - .../params/build_prod_params.py | 1 - .../params/common_build_params.py | 22 - .../src/airflow_breeze/params/shell_params.py | 9 +- .../utils/docker_command_utils.py | 5 +- dev/breeze/src/airflow_breeze/utils/image.py | 55 +-- .../src/airflow_breeze/utils/platforms.py | 4 +- .../airflow_breeze/utils/selective_checks.py | 8 +- dev/breeze/tests/test_shell_params.py | 12 - docs/docker-stack/build-arg-ref.rst | 37 +- docs/docker-stack/build.rst | 9 +- .../restricted/restricted_environments.sh | 1 - hatch_build.py | 4 - scripts/ci/cleanup_docker.sh | 5 +- .../ci/constraints/ci_commit_constraints.sh | 3 - scripts/ci/docker-compose/base.yml | 2 +- ...tart_arm_instance_and_connect_to_docker.sh | 91 ---- scripts/ci/images/ci_stop_arm_instance.sh | 30 -- scripts/docker/common.sh | 6 +- scripts/docker/entrypoint_ci.sh | 7 +- scripts/docker/install_airflow.sh | 16 +- ...ll_airflow_dependencies_from_branch_tip.sh | 103 ---- scripts/in_container/_in_container_utils.sh | 10 +- 134 files changed, 2556 insertions(+), 3635 deletions(-) delete mode 100644 .github/actions/checkout_target_commit/action.yml create mode 100644 .github/actions/prepare_all_ci_images/action.yml create mode 100644 .github/actions/prepare_single_ci_image/action.yml delete mode 100644 .github/workflows/build-images.yml rename dev/breeze/doc/ci/{07_debugging.md => 06_debugging.md} (93%) delete mode 100644 dev/breeze/doc/ci/06_diagrams.md create mode 100644 dev/breeze/doc/ci/07_running_ci_locally.md delete mode 100644 dev/breeze/doc/ci/08_running_ci_locally.md create mode 100644 dev/breeze/doc/images/image_artifacts.png create mode 100644 dev/breeze/doc/images/output_ci-image_load.svg create mode 100644 dev/breeze/doc/images/output_ci-image_load.txt create mode 100644 dev/breeze/doc/images/output_ci-image_save.svg create mode 100644 dev/breeze/doc/images/output_ci-image_save.txt create mode 100644 dev/breeze/doc/images/output_prod-image_load.svg create mode 100644 dev/breeze/doc/images/output_prod-image_load.txt create mode 100644 dev/breeze/doc/images/output_prod-image_save.svg create mode 100644 dev/breeze/doc/images/output_prod-image_save.txt delete mode 100755 scripts/ci/images/ci_start_arm_instance_and_connect_to_docker.sh delete mode 100755 scripts/ci/images/ci_stop_arm_instance.sh delete mode 100644 scripts/docker/install_airflow_dependencies_from_branch_tip.sh diff --git a/.github/actions/checkout_target_commit/action.yml b/.github/actions/checkout_target_commit/action.yml deleted file mode 100644 index e95e8b86254a0..0000000000000 --- a/.github/actions/checkout_target_commit/action.yml +++ /dev/null @@ -1,81 +0,0 @@ -# Licensed to the Apache Software Foundation (ASF) under one -# or more contributor license agreements. See the NOTICE file -# distributed with this work for additional information -# regarding copyright ownership. The ASF licenses this file -# to you under the Apache License, Version 2.0 (the -# "License"); you may not use this file except in compliance -# with the License. You may obtain a copy of the License at -# -# http://www.apache.org/licenses/LICENSE-2.0 -# -# Unless required by applicable law or agreed to in writing, -# software distributed under the License is distributed on an -# "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY -# KIND, either express or implied. See the License for the -# specific language governing permissions and limitations -# under the License. -# ---- -name: 'Checkout target commit' -description: > - Checks out target commit with the exception of .github scripts directories that come from the target branch -inputs: - target-commit-sha: - description: 'SHA of the target commit to checkout' - required: true - pull-request-target: - description: 'Whether the workflow is a pull request target workflow' - required: true - is-committer-build: - description: 'Whether the build is done by a committer' - required: true -runs: - using: "composite" - steps: - - name: "Checkout target commit" - uses: actions/checkout@v4 - with: - ref: ${{ inputs.target-commit-sha }} - persist-credentials: false - #################################################################################################### - # BE VERY CAREFUL HERE! THIS LINE AND THE END OF THE WARNING. IN PULL REQUEST TARGET WORKFLOW - # WE CHECK OUT THE TARGET COMMIT ABOVE TO BE ABLE TO BUILD THE IMAGE FROM SOURCES FROM THE - # INCOMING PR, RATHER THAN FROM TARGET BRANCH. THIS IS A SECURITY RISK, BECAUSE THE PR - # CAN CONTAIN ANY CODE AND WE EXECUTE IT HERE. THEREFORE, WE NEED TO BE VERY CAREFUL WHAT WE - # DO HERE. WE SHOULD NOT EXECUTE ANY CODE THAT COMES FROM THE PR. WE SHOULD NOT RUN ANY BREEZE - # COMMAND NOR SCRIPTS NOR COMPOSITE ACTIONS. WE SHOULD ONLY RUN CODE THAT IS EMBEDDED DIRECTLY IN - # THIS WORKFLOW - BECAUSE THIS IS THE ONLY CODE THAT WE CAN TRUST. - #################################################################################################### - - name: Checkout target branch to 'target-airflow' folder to use ci/scripts and breeze from there. - uses: actions/checkout@v4 - with: - path: "target-airflow" - ref: ${{ github.base_ref }} - persist-credentials: false - if: inputs.pull-request-target == 'true' && inputs.is-committer-build != 'true' - - name: > - Replace "scripts/ci", "dev", ".github/actions" and ".github/workflows" with the target branch - so that the those directories are not coming from the PR - shell: bash - run: | - echo - echo -e "\033[33m Replace scripts, dev, actions with target branch for non-committer builds!\033[0m" - echo - rm -rfv "scripts/ci" - rm -rfv "dev" - rm -rfv ".github/actions" - rm -rfv ".github/workflows" - rm -v ".dockerignore" || true - mv -v "target-airflow/scripts/ci" "scripts" - mv -v "target-airflow/dev" "." - mv -v "target-airflow/.github/actions" "target-airflow/.github/workflows" ".github" - mv -v "target-airflow/.dockerignore" ".dockerignore" || true - if: inputs.pull-request-target == 'true' && inputs.is-committer-build != 'true' - #################################################################################################### - # AFTER IT'S SAFE. THE `dev`, `scripts/ci` AND `.github/actions` and `.dockerignore` ARE NOW COMING - # FROM THE BASE_REF - WHICH IS THE TARGET BRANCH OF THE PR. WE CAN TRUST THAT THOSE SCRIPTS ARE - # SAFE TO RUN AND CODE AVAILABLE IN THE DOCKER BUILD PHASE IS CONTROLLED BY THE `.dockerignore`. - # ALL THE REST OF THE CODE COMES FROM THE PR, AND FOR EXAMPLE THE CODE IN THE `Dockerfile.ci` CAN - # BE RUN SAFELY AS PART OF DOCKER BUILD. BECAUSE IT RUNS INSIDE THE DOCKER CONTAINER AND IT IS - # ISOLATED FROM THE RUNNER. - #################################################################################################### diff --git a/.github/actions/prepare_all_ci_images/action.yml b/.github/actions/prepare_all_ci_images/action.yml new file mode 100644 index 0000000000000..fdbf51288d704 --- /dev/null +++ b/.github/actions/prepare_all_ci_images/action.yml @@ -0,0 +1,71 @@ +# Licensed to the Apache Software Foundation (ASF) under one +# or more contributor license agreements. See the NOTICE file +# distributed with this work for additional information +# regarding copyright ownership. The ASF licenses this file +# to you under the Apache License, Version 2.0 (the +# "License"); you may not use this file except in compliance +# with the License. You may obtain a copy of the License at +# +# http://www.apache.org/licenses/LICENSE-2.0 +# +# Unless required by applicable law or agreed to in writing, +# software distributed under the License is distributed on an +# "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY +# KIND, either express or implied. See the License for the +# specific language governing permissions and limitations +# under the License. +# +--- +name: 'Prepare all CI images' +description: 'Recreates current python CI images from artifacts for all python versions' +inputs: + python-versions-list-as-string: + description: 'Stringified array of all Python versions to test - separated by spaces.' + required: true + platform: + description: 'Platform for the build - linux/amd64 or linux/arm64' + required: true +outputs: + host-python-version: + description: Python version used in host + value: ${{ steps.breeze.outputs.host-python-version }} +runs: + using: "composite" + steps: + - name: "Cleanup docker" + run: ./scripts/ci/cleanup_docker.sh + shell: bash + # TODO: Currently we cannot loop through the list of python versions and have dynamic list of + # tasks. Instead we hardcode all possible python versions and they - but + # this should be implemented in stash action as list of keys to download. + # That includes 3.8 - 3.12 as we are backporting it to v2-10-test branch + - name: "Restore CI docker image ${{ inputs.platform }}:3.8" + uses: ./.github/actions/prepare_single_ci_image + with: + platform: ${{ inputs.platform }} + python: "3.8" + python-versions-list-as-string: ${{ inputs.python-versions-list-as-string }} + - name: "Restore CI docker image ${{ inputs.platform }}:3.9" + uses: ./.github/actions/prepare_single_ci_image + with: + platform: ${{ inputs.platform }} + python: "3.9" + python-versions-list-as-string: ${{ inputs.python-versions-list-as-string }} + - name: "Restore CI docker image ${{ inputs.platform }}:3.10" + uses: ./.github/actions/prepare_single_ci_image + with: + platform: ${{ inputs.platform }} + python: "3.10" + python-versions-list-as-string: ${{ inputs.python-versions-list-as-string }} + - name: "Restore CI docker image ${{ inputs.platform }}:3.11" + uses: ./.github/actions/prepare_single_ci_image + with: + platform: ${{ inputs.platform }} + python: "3.11" + python-versions-list-as-string: ${{ inputs.python-versions-list-as-string }} + - name: "Restore CI docker image ${{ inputs.platform }}:3.12" + uses: ./.github/actions/prepare_single_ci_image + with: + platform: ${{ inputs.platform }} + python: "3.12" + python-versions-list-as-string: ${{ inputs.python-versions-list-as-string }} diff --git a/.github/actions/prepare_breeze_and_image/action.yml b/.github/actions/prepare_breeze_and_image/action.yml index 41aa17092d589..5c65ecbdd1138 100644 --- a/.github/actions/prepare_breeze_and_image/action.yml +++ b/.github/actions/prepare_breeze_and_image/action.yml @@ -16,12 +16,18 @@ # under the License. # --- -name: 'Prepare breeze && current python image' -description: 'Installs breeze and pulls current python image' +name: 'Prepare breeze && current image (CI or PROD)' +description: 'Installs breeze and recreates current python image from artifact' inputs: - pull-image-type: - description: 'Which image to pull' - default: CI + python: + description: 'Python version for image to prepare' + required: true + image-type: + description: 'Which image type to prepare (CI/PROD)' + default: "CI" + platform: + description: 'Platform for the build - linux/amd64 or linux/arm64' + required: true outputs: host-python-version: description: Python version used in host @@ -29,17 +35,29 @@ outputs: runs: using: "composite" steps: + - name: "Cleanup docker" + run: ./scripts/ci/cleanup_docker.sh + shell: bash - name: "Install Breeze" uses: ./.github/actions/breeze id: breeze - - name: Login to ghcr.io - shell: bash - run: echo "${{ env.GITHUB_TOKEN }}" | docker login ghcr.io -u ${{ github.actor }} --password-stdin - - name: Pull CI image ${{ env.PYTHON_MAJOR_MINOR_VERSION }}:${{ env.IMAGE_TAG }} + - name: "Restore CI docker image ${{ inputs.platform }}:${{ inputs.python }}" + uses: apache/infrastructure-actions/stash/restore@c94b890bbedc2fc61466d28e6bd9966bc6c6643c + with: + key: "ci-image-save-${{ inputs.platform }}-${{ inputs.python }}" + path: "/tmp/" + if: inputs.image-type == 'CI' + - name: "Load CI image ${{ inputs.platform }}:${{ inputs.python }}" + run: breeze ci-image load --platform ${{ inputs.platform }} --python ${{ inputs.python }} shell: bash - run: breeze ci-image pull --tag-as-latest - if: inputs.pull-image-type == 'CI' - - name: Pull PROD image ${{ env.PYTHON_MAJOR_MINOR_VERSION }}:${{ env.IMAGE_TAG }} + if: inputs.image-type == 'CI' + - name: "Restore PROD docker image ${{ inputs.platform }}:${{ inputs.python }}" + uses: apache/infrastructure-actions/stash/restore@c94b890bbedc2fc61466d28e6bd9966bc6c6643c + with: + key: "prod-image-save-${{ inputs.platform }}-${{ inputs.python }}" + path: "/tmp/" + if: inputs.image-type == 'PROD' + - name: "Load PROD image ${{ inputs.platform }}:${{ inputs.python }}" + run: breeze prod-image load --platform ${{ inputs.platform }} --python ${{ inputs.python }} shell: bash - run: breeze prod-image pull --tag-as-latest - if: inputs.pull-image-type == 'PROD' + if: inputs.image-type == 'PROD' diff --git a/.github/actions/prepare_single_ci_image/action.yml b/.github/actions/prepare_single_ci_image/action.yml new file mode 100644 index 0000000000000..bc122596158f2 --- /dev/null +++ b/.github/actions/prepare_single_ci_image/action.yml @@ -0,0 +1,47 @@ +# Licensed to the Apache Software Foundation (ASF) under one +# or more contributor license agreements. See the NOTICE file +# distributed with this work for additional information +# regarding copyright ownership. The ASF licenses this file +# to you under the Apache License, Version 2.0 (the +# "License"); you may not use this file except in compliance +# with the License. You may obtain a copy of the License at +# +# http://www.apache.org/licenses/LICENSE-2.0 +# +# Unless required by applicable law or agreed to in writing, +# software distributed under the License is distributed on an +# "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY +# KIND, either express or implied. See the License for the +# specific language governing permissions and limitations +# under the License. +# +--- +name: 'Prepare single images' +description: 'Recreates current python image from artifacts' +inputs: + python: + description: 'Python version for image to prepare' + required: true + python-versions-list-as-string: + description: 'Stringified array of all Python versions to prepare - separated by spaces.' + required: true + platform: + description: 'Platform for the build - linux/amd64 or linux/arm64' + required: true +outputs: + host-python-version: + description: Python version used in host + value: ${{ steps.breeze.outputs.host-python-version }} +runs: + using: "composite" + steps: + - name: "Restore CI docker images ${{ inputs.platform }}:${{ inputs.python }}" + uses: apache/infrastructure-actions/stash/restore@c94b890bbedc2fc61466d28e6bd9966bc6c6643c + with: + key: "ci-image-save-${{ inputs.platform }}-${{ inputs.python }}" + path: "/tmp/" + if: contains(inputs.python-versions-list-as-string, inputs.python) + - name: "Load CI image ${{ inputs.platform }}:${{ inputs.python }}" + run: breeze ci-image load --platform "${{ inputs.platform }}" --python "${{ inputs.python }}" + shell: bash + if: contains(inputs.python-versions-list-as-string, inputs.python) diff --git a/.github/workflows/additional-ci-image-checks.yml b/.github/workflows/additional-ci-image-checks.yml index 8a3b46e70d37d..40196a6e04296 100644 --- a/.github/workflows/additional-ci-image-checks.yml +++ b/.github/workflows/additional-ci-image-checks.yml @@ -32,10 +32,6 @@ on: # yamllint disable-line rule:truthy description: "The array of labels (in json form) determining self-hosted runners." required: true type: string - image-tag: - description: "Tag to set for the image" - required: true - type: string python-versions: description: "The list of python versions (stringified JSON array) to run the tests on." required: true @@ -103,8 +99,6 @@ jobs: contents: read # This write is only given here for `push` events from "apache/airflow" repo. It is not given for PRs # from forks. This is to prevent malicious PRs from creating images in the "apache/airflow" repo. - # For regular build for PRS this "build-prod-images" workflow will be skipped anyway by the - # "in-workflow-build" condition packages: write secrets: inherit with: @@ -159,7 +153,7 @@ jobs: # # There is no point in running this one in "canary" run, because the above step is doing the # # same build anyway. # build-ci-arm-images: -# name: Build CI ARM images (in-workflow) +# name: Build CI ARM images # uses: ./.github/workflows/ci-image-build.yml # permissions: # contents: read @@ -169,7 +163,6 @@ jobs: # push-image: "false" # runs-on-as-json-public: ${{ inputs.runs-on-as-json-public }} # runs-on-as-json-self-hosted: ${{ inputs.runs-on-as-json-self-hosted }} -# image-tag: ${{ inputs.image-tag }} # python-versions: ${{ inputs.python-versions }} # platform: "linux/arm64" # branch: ${{ inputs.branch }} diff --git a/.github/workflows/additional-prod-image-tests.yml b/.github/workflows/additional-prod-image-tests.yml index 5ffd2001e0e26..e0a31bbf72413 100644 --- a/.github/workflows/additional-prod-image-tests.yml +++ b/.github/workflows/additional-prod-image-tests.yml @@ -32,10 +32,6 @@ on: # yamllint disable-line rule:truthy description: "Branch used to construct constraints URL from." required: true type: string - image-tag: - description: "Tag to set for the image" - required: true - type: string upgrade-to-newer-dependencies: description: "Whether to upgrade to newer dependencies (true/false)" required: true @@ -70,7 +66,6 @@ jobs: default-python-version: ${{ inputs.default-python-version }} branch: ${{ inputs.default-branch }} use-uv: "false" - image-tag: ${{ inputs.image-tag }} build-provider-packages: ${{ inputs.default-branch == 'main' }} upgrade-to-newer-dependencies: ${{ inputs.upgrade-to-newer-dependencies }} chicken-egg-providers: ${{ inputs.chicken-egg-providers }} @@ -88,7 +83,6 @@ jobs: default-python-version: ${{ inputs.default-python-version }} branch: ${{ inputs.default-branch }} use-uv: "false" - image-tag: ${{ inputs.image-tag }} build-provider-packages: ${{ inputs.default-branch == 'main' }} upgrade-to-newer-dependencies: ${{ inputs.upgrade-to-newer-dependencies }} chicken-egg-providers: ${{ inputs.chicken-egg-providers }} @@ -117,36 +111,25 @@ jobs: persist-credentials: false - name: "Cleanup docker" run: ./scripts/ci/cleanup_docker.sh - - name: "Install Breeze" - uses: ./.github/actions/breeze - - name: Login to ghcr.io - shell: bash - run: echo "${{ env.GITHUB_TOKEN }}" | docker login ghcr.io -u ${{ github.actor }} --password-stdin - - name: Pull PROD image ${{ inputs.default-python-version}}:${{ inputs.image-tag }} - run: breeze prod-image pull --tag-as-latest - env: - PYTHON_MAJOR_MINOR_VERSION: "${{ inputs.default-python-version }}" - IMAGE_TAG: "${{ inputs.image-tag }}" - - name: "Setup python" - uses: actions/setup-python@v5 + - name: "Prepare breeze & PROD image: ${{ inputs.default-python-version }}" + uses: ./.github/actions/prepare_breeze_and_image with: - python-version: ${{ inputs.default-python-version }} - cache: 'pip' - cache-dependency-path: ./dev/requirements.txt + platform: "linux/amd64" + image-type: "PROD" + python: ${{ inputs.default-python-version }} - name: "Test examples of PROD image building" run: " cd ./docker_tests && \ python -m pip install -r requirements.txt && \ TEST_IMAGE=\"ghcr.io/${{ github.repository }}/${{ inputs.default-branch }}\ - /prod/python${{ inputs.default-python-version }}:${{ inputs.image-tag }}\" \ + /prod/python${{ inputs.default-python-version }}\" \ python -m pytest test_examples_of_prod_image_building.py -n auto --color=yes" test-docker-compose-quick-start: timeout-minutes: 60 - name: "Docker-compose quick start with PROD image verifying" + name: "Docker Compose quick start with PROD image verifying" runs-on: ${{ fromJSON(inputs.runs-on-as-json-public) }} env: - IMAGE_TAG: "${{ inputs.image-tag }}" PYTHON_MAJOR_MINOR_VERSION: "${{ inputs.default-python-version }}" GITHUB_REPOSITORY: ${{ github.repository }} GITHUB_TOKEN: ${{ secrets.GITHUB_TOKEN }} @@ -161,14 +144,12 @@ jobs: with: fetch-depth: 2 persist-credentials: false - - name: "Cleanup docker" - run: ./scripts/ci/cleanup_docker.sh - - name: "Install Breeze" - uses: ./.github/actions/breeze - - name: Login to ghcr.io - shell: bash - run: echo "${{ env.GITHUB_TOKEN }}" | docker login ghcr.io -u ${{ github.actor }} --password-stdin - - name: "Pull image ${{ inputs.default-python-version}}:${{ inputs.image-tag }}" - run: breeze prod-image pull --tag-as-latest + - name: "Prepare breeze & PROD image: ${{ env.PYTHON_MAJOR_MINOR_VERSION }}" + uses: ./.github/actions/prepare_breeze_and_image + with: + platform: "linux/amd64" + image-type: "PROD" + python: ${{ env.PYTHON_MAJOR_MINOR_VERSION }} + id: breeze - name: "Test docker-compose quick start" run: breeze testing docker-compose-tests diff --git a/.github/workflows/basic-tests.yml b/.github/workflows/basic-tests.yml index c8ba85969f5e3..47f80f05b7ac7 100644 --- a/.github/workflows/basic-tests.yml +++ b/.github/workflows/basic-tests.yml @@ -288,7 +288,6 @@ jobs: runs-on: ${{ fromJSON(inputs.runs-on-as-json-public) }} env: PYTHON_MAJOR_MINOR_VERSION: "${{ inputs.default-python-version }}" - IMAGE_TAG: ${{ inputs.image-tag }} GITHUB_REPOSITORY: ${{ github.repository }} GITHUB_TOKEN: ${{ secrets.GITHUB_TOKEN }} GITHUB_USERNAME: ${{ github.actor }} diff --git a/.github/workflows/build-images.yml b/.github/workflows/build-images.yml deleted file mode 100644 index 9135dcb9d9e94..0000000000000 --- a/.github/workflows/build-images.yml +++ /dev/null @@ -1,264 +0,0 @@ -# Licensed to the Apache Software Foundation (ASF) under one -# or more contributor license agreements. See the NOTICE file -# distributed with this work for additional information -# regarding copyright ownership. The ASF licenses this file -# to you under the Apache License, Version 2.0 (the -# "License"); you may not use this file except in compliance -# with the License. You may obtain a copy of the License at -# -# http://www.apache.org/licenses/LICENSE-2.0 -# -# Unless required by applicable law or agreed to in writing, -# software distributed under the License is distributed on an -# "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY -# KIND, either express or implied. See the License for the -# specific language governing permissions and limitations -# under the License. -# ---- -name: Build Images -run-name: > - Build images for ${{ github.event.pull_request.title }} ${{ github.event.pull_request._links.html.href }} -on: # yamllint disable-line rule:truthy - pull_request_target: - branches: - - main - - v2-10-stable - - v2-10-test - - providers-[a-z]+-?[a-z]*/v[0-9]+-[0-9]+ -permissions: - # all other permissions are set to none - contents: read - pull-requests: read - packages: read -env: - ANSWER: "yes" - # You can override CONSTRAINTS_GITHUB_REPOSITORY by setting secret in your repo but by default the - # Airflow one is going to be used - CONSTRAINTS_GITHUB_REPOSITORY: >- - ${{ secrets.CONSTRAINTS_GITHUB_REPOSITORY != '' && - secrets.CONSTRAINTS_GITHUB_REPOSITORY || 'apache/airflow' }} - # This token is WRITE one - pull_request_target type of events always have the WRITE token - DB_RESET: "true" - GITHUB_REPOSITORY: ${{ github.repository }} - GITHUB_TOKEN: ${{ secrets.GITHUB_TOKEN }} - GITHUB_USERNAME: ${{ github.actor }} - IMAGE_TAG: "${{ github.event.pull_request.head.sha || github.sha }}" - INCLUDE_SUCCESS_OUTPUTS: "true" - USE_SUDO: "true" - VERBOSE: "true" - -concurrency: - group: build-${{ github.event.pull_request.number || github.ref }} - cancel-in-progress: true - -jobs: - build-info: - timeout-minutes: 10 - name: Build Info - # At build-info stage we do not yet have outputs so we need to hard-code the runs-on to public runners - runs-on: ["ubuntu-22.04"] - env: - TARGET_BRANCH: ${{ github.event.pull_request.base.ref }} - outputs: - image-tag: ${{ github.event.pull_request.head.sha || github.sha }} - python-versions: ${{ steps.selective-checks.outputs.python-versions }} - python-versions-list-as-string: ${{ steps.selective-checks.outputs.python-versions-list-as-string }} - default-python-version: ${{ steps.selective-checks.outputs.default-python-version }} - upgrade-to-newer-dependencies: ${{ steps.selective-checks.outputs.upgrade-to-newer-dependencies }} - run-tests: ${{ steps.selective-checks.outputs.run-tests }} - run-kubernetes-tests: ${{ steps.selective-checks.outputs.run-kubernetes-tests }} - ci-image-build: ${{ steps.selective-checks.outputs.ci-image-build }} - prod-image-build: ${{ steps.selective-checks.outputs.prod-image-build }} - docker-cache: ${{ steps.selective-checks.outputs.docker-cache }} - default-branch: ${{ steps.selective-checks.outputs.default-branch }} - disable-airflow-repo-cache: ${{ steps.selective-checks.outputs.disable-airflow-repo-cache }} - force-pip: ${{ steps.selective-checks.outputs.force-pip }} - constraints-branch: ${{ steps.selective-checks.outputs.default-constraints-branch }} - runs-on-as-json-default: ${{ steps.selective-checks.outputs.runs-on-as-json-default }} - runs-on-as-json-public: ${{ steps.selective-checks.outputs.runs-on-as-json-public }} - runs-on-as-json-self-hosted: ${{ steps.selective-checks.outputs.runs-on-as-json-self-hosted }} - is-self-hosted-runner: ${{ steps.selective-checks.outputs.is-self-hosted-runner }} - is-committer-build: ${{ steps.selective-checks.outputs.is-committer-build }} - is-airflow-runner: ${{ steps.selective-checks.outputs.is-airflow-runner }} - is-amd-runner: ${{ steps.selective-checks.outputs.is-amd-runner }} - is-arm-runner: ${{ steps.selective-checks.outputs.is-arm-runner }} - is-vm-runner: ${{ steps.selective-checks.outputs.is-vm-runner }} - is-k8s-runner: ${{ steps.selective-checks.outputs.is-k8s-runner }} - chicken-egg-providers: ${{ steps.selective-checks.outputs.chicken-egg-providers }} - target-commit-sha: "${{steps.discover-pr-merge-commit.outputs.target-commit-sha || - github.event.pull_request.head.sha || - github.sha - }}" - if: github.repository == 'apache/airflow' - steps: - - name: Cleanup repo - shell: bash - run: docker run -v "${GITHUB_WORKSPACE}:/workspace" -u 0:0 bash -c "rm -rf /workspace/*" - - name: Discover PR merge commit - id: discover-pr-merge-commit - run: | - # Sometimes target-commit-sha cannot be - TARGET_COMMIT_SHA="$(gh api '${{ github.event.pull_request.url }}' --jq .merge_commit_sha)" - if [[ ${TARGET_COMMIT_SHA} == "" ]]; then - # Sometimes retrieving the merge commit SHA from PR fails. We retry it once. Otherwise we - # fall-back to github.event.pull_request.head.sha - echo - echo "Could not retrieve merge commit SHA from PR, waiting for 3 seconds and retrying." - echo - sleep 3 - TARGET_COMMIT_SHA="$(gh api '${{ github.event.pull_request.url }}' --jq .merge_commit_sha)" - if [[ ${TARGET_COMMIT_SHA} == "" ]]; then - echo - echo "Could not retrieve merge commit SHA from PR, falling back to PR head SHA." - echo - TARGET_COMMIT_SHA="${{ github.event.pull_request.head.sha }}" - fi - fi - echo "TARGET_COMMIT_SHA=${TARGET_COMMIT_SHA}" - echo "TARGET_COMMIT_SHA=${TARGET_COMMIT_SHA}" >> ${GITHUB_ENV} - echo "target-commit-sha=${TARGET_COMMIT_SHA}" >> ${GITHUB_OUTPUT} - if: github.event_name == 'pull_request_target' - # The labels in the event aren't updated when re-triggering the job, So lets hit the API to get - # up-to-date values - - name: Get latest PR labels - id: get-latest-pr-labels - run: | - echo -n "pull-request-labels=" >> ${GITHUB_OUTPUT} - gh api graphql --paginate -F node_id=${{github.event.pull_request.node_id}} -f query=' - query($node_id: ID!, $endCursor: String) { - node(id:$node_id) { - ... on PullRequest { - labels(first: 100, after: $endCursor) { - nodes { name } - pageInfo { hasNextPage endCursor } - } - } - } - }' --jq '.data.node.labels.nodes[]' | jq --slurp -c '[.[].name]' >> ${GITHUB_OUTPUT} - if: github.event_name == 'pull_request_target' - - uses: actions/checkout@v4 - with: - ref: ${{ env.TARGET_COMMIT_SHA }} - persist-credentials: false - fetch-depth: 2 - #################################################################################################### - # WE ONLY DO THAT CHECKOUT ABOVE TO RETRIEVE THE TARGET COMMIT AND IT'S PARENT. DO NOT RUN ANY CODE - # RIGHT AFTER THAT AS WE ARE GOING TO RESTORE THE TARGET BRANCH CODE IN THE NEXT STEP. - #################################################################################################### - - name: Checkout target branch to use ci/scripts and breeze from there. - uses: actions/checkout@v4 - with: - ref: ${{ github.base_ref }} - persist-credentials: false - #################################################################################################### - # HERE EVERYTHING IS PERFECTLY SAFE TO RUN. AT THIS POINT WE HAVE THE TARGET BRANCH CHECKED OUT - # AND WE CAN RUN ANY CODE FROM IT. WE CAN RUN BREEZE COMMANDS, WE CAN RUN SCRIPTS, WE CAN RUN - # COMPOSITE ACTIONS. WE CAN RUN ANYTHING THAT IS IN THE TARGET BRANCH AND THERE IS NO RISK THAT - # CODE WILL BE RUN FROM THE PR. - #################################################################################################### - - name: Cleanup docker - run: ./scripts/ci/cleanup_docker.sh - - name: Setup python - uses: actions/setup-python@v5 - with: - python-version: "3.9" - - name: Install Breeze - uses: ./.github/actions/breeze - #################################################################################################### - # WE RUN SELECTIVE CHECKS HERE USING THE TARGET COMMIT AND ITS PARENT TO BE ABLE TO COMPARE THEM - # AND SEE WHAT HAS CHANGED IN THE PR. THE CODE IS STILL RUN FROM THE TARGET BRANCH, SO IT IS SAFE - # TO RUN IT, WE ONLY PASS TARGET_COMMIT_SHA SO THAT SELECTIVE CHECKS CAN SEE WHAT'S COMING IN THE PR - #################################################################################################### - - name: Selective checks - id: selective-checks - env: - PR_LABELS: "${{ steps.get-latest-pr-labels.outputs.pull-request-labels }}" - COMMIT_REF: "${{ env.TARGET_COMMIT_SHA }}" - VERBOSE: "false" - AIRFLOW_SOURCES_ROOT: "${{ github.workspace }}" - run: breeze ci selective-check 2>> ${GITHUB_OUTPUT} - - name: env - run: printenv - env: - PR_LABELS: ${{ steps.get-latest-pr-labels.outputs.pull-request-labels }} - GITHUB_CONTEXT: ${{ toJson(github) }} - - - build-ci-images: - name: Build CI images - permissions: - contents: read - packages: write - secrets: inherit - needs: [build-info] - uses: ./.github/workflows/ci-image-build.yml - # Only run this it if the PR comes from fork, otherwise build will be done "in-PR-workflow" - if: | - needs.build-info.outputs.ci-image-build == 'true' && - github.event.pull_request.head.repo.full_name != 'apache/airflow' - with: - runs-on-as-json-public: ${{ needs.build-info.outputs.runs-on-as-json-public }} - runs-on-as-json-self-hosted: ${{ needs.build-info.outputs.runs-on-as-json-self-hosted }} - do-build: ${{ needs.build-info.outputs.ci-image-build }} - target-commit-sha: ${{ needs.build-info.outputs.target-commit-sha }} - pull-request-target: "true" - is-committer-build: ${{ needs.build-info.outputs.is-committer-build }} - push-image: "true" - use-uv: ${{ needs.build-info.outputs.force-pip == 'true' && 'false' || 'true' }} - image-tag: ${{ needs.build-info.outputs.image-tag }} - platform: "linux/amd64" - python-versions: ${{ needs.build-info.outputs.python-versions }} - branch: ${{ needs.build-info.outputs.default-branch }} - constraints-branch: ${{ needs.build-info.outputs.constraints-branch }} - upgrade-to-newer-dependencies: ${{ needs.build-info.outputs.upgrade-to-newer-dependencies }} - docker-cache: ${{ needs.build-info.outputs.docker-cache }} - disable-airflow-repo-cache: ${{ needs.build-info.outputs.disable-airflow-repo-cache }} - - - generate-constraints: - name: Generate constraints - needs: [build-info, build-ci-images] - uses: ./.github/workflows/generate-constraints.yml - with: - runs-on-as-json-public: ${{ needs.build-info.outputs.runs-on-as-json-public }} - python-versions-list-as-string: ${{ needs.build-info.outputs.python-versions-list-as-string }} - # For regular PRs we do not need "no providers" constraints - they are only needed in canary builds - generate-no-providers-constraints: "false" - image-tag: ${{ needs.build-info.outputs.image-tag }} - chicken-egg-providers: ${{ needs.build-info.outputs.chicken-egg-providers }} - debug-resources: ${{ needs.build-info.outputs.debug-resources }} - - build-prod-images: - name: Build PROD images - permissions: - contents: read - packages: write - secrets: inherit - needs: [build-info, generate-constraints] - uses: ./.github/workflows/prod-image-build.yml - # Only run this it if the PR comes from fork, otherwise build will be done "in-PR-workflow" - if: | - needs.build-info.outputs.prod-image-build == 'true' && - github.event.pull_request.head.repo.full_name != 'apache/airflow' - with: - runs-on-as-json-public: ${{ needs.build-info.outputs.runs-on-as-json-public }} - build-type: "Regular" - do-build: ${{ needs.build-info.outputs.ci-image-build }} - upload-package-artifact: "true" - target-commit-sha: ${{ needs.build-info.outputs.target-commit-sha }} - pull-request-target: "true" - is-committer-build: ${{ needs.build-info.outputs.is-committer-build }} - push-image: "true" - use-uv: ${{ needs.build-info.outputs.force-pip == 'true' && 'false' || 'true' }} - image-tag: ${{ needs.build-info.outputs.image-tag }} - platform: linux/amd64 - python-versions: ${{ needs.build-info.outputs.python-versions }} - default-python-version: ${{ needs.build-info.outputs.default-python-version }} - branch: ${{ needs.build-info.outputs.default-branch }} - constraints-branch: ${{ needs.build-info.outputs.constraints-branch }} - build-provider-packages: ${{ needs.build-info.outputs.default-branch == 'main' }} - upgrade-to-newer-dependencies: ${{ needs.build-info.outputs.upgrade-to-newer-dependencies }} - chicken-egg-providers: ${{ needs.build-info.outputs.chicken-egg-providers }} - docker-cache: ${{ needs.build-info.outputs.docker-cache }} - disable-airflow-repo-cache: ${{ needs.build-info.outputs.disable-airflow-repo-cache }} diff --git a/.github/workflows/ci-image-build.yml b/.github/workflows/ci-image-build.yml index b8e2feac1755f..2497c1b43fdaa 100644 --- a/.github/workflows/ci-image-build.yml +++ b/.github/workflows/ci-image-build.yml @@ -28,13 +28,6 @@ on: # yamllint disable-line rule:truthy description: "The array of labels (in json form) determining self-hosted runners." required: true type: string - do-build: - description: > - Whether to actually do the build (true/false). If set to false, the build is done - already in pull-request-target workflow, so we skip it here. - required: false - default: "true" - type: string target-commit-sha: description: "The commit SHA to checkout for the build" required: false @@ -59,6 +52,11 @@ on: # yamllint disable-line rule:truthy required: false default: "true" type: string + upload-image-artifact: + description: "Whether to upload docker image artifact" + required: false + default: "false" + type: string debian-version: description: "Base Debian distribution to use for the build (bookworm)" type: string @@ -71,10 +69,6 @@ on: # yamllint disable-line rule:truthy description: "Whether to use uv to build the image (true/false)" required: true type: string - image-tag: - description: "Tag to set for the image" - required: true - type: string python-versions: description: "JSON-formatted array of Python versions to build images from" required: true @@ -104,27 +98,17 @@ jobs: strategy: fail-fast: true matrix: - # yamllint disable-line rule:line-length - python-version: ${{ inputs.do-build == 'true' && fromJSON(inputs.python-versions) || fromJSON('[""]') }} + python-version: ${{ fromJSON(inputs.python-versions) || fromJSON('[""]') }} timeout-minutes: 110 - name: "\ -${{ inputs.do-build == 'true' && 'Build' || 'Skip building' }} \ -CI ${{ inputs.platform }} image\ -${{ matrix.python-version }}${{ inputs.do-build == 'true' && ':' || '' }}\ -${{ inputs.do-build == 'true' && inputs.image-tag || '' }}" - # The ARM images need to be built using self-hosted runners as ARM macos public runners - # do not yet allow us to run docker effectively and fast. - # https://github.com/actions/runner-images/issues/9254#issuecomment-1917916016 - # https://github.com/abiosoft/colima/issues/970 - # https://github.com/actions/runner/issues/1456 - # See https://github.com/apache/airflow/pull/38640 + name: "Build CI ${{ inputs.platform }} image ${{ matrix.python-version }}" # NOTE!!!!! This has to be put in one line for runs-on to recognize the "fromJSON" properly !!!! # adding space before (with >) apparently turns the `runs-on` processed line into a string "Array" # instead of an array of strings. # yamllint disable-line rule:line-length - runs-on: ${{ (inputs.platform == 'linux/amd64') && fromJSON(inputs.runs-on-as-json-public) || fromJSON(inputs.runs-on-as-json-self-hosted) }} + runs-on: ${{ (inputs.platform == 'linuz/amd64') && fromJSON(inputs.runs-on-as-json-public) || fromJSON(inputs.runs-on-as-json-self-hosted) }} env: BACKEND: sqlite + PYTHON_MAJOR_MINOR_VERSION: ${{ matrix.python-version }} DEFAULT_BRANCH: ${{ inputs.branch }} DEFAULT_CONSTRAINTS_BRANCH: ${{ inputs.constraints-branch }} VERSION_SUFFIX_FOR_PYPI: "dev0" @@ -137,42 +121,32 @@ ${{ inputs.do-build == 'true' && inputs.image-tag || '' }}" - name: "Cleanup repo" shell: bash run: docker run -v "${GITHUB_WORKSPACE}:/workspace" -u 0:0 bash -c "rm -rf /workspace/*" - if: inputs.do-build == 'true' - name: "Checkout target branch" uses: actions/checkout@v4 with: persist-credentials: false - - name: "Checkout target commit" - uses: ./.github/actions/checkout_target_commit - if: inputs.do-build == 'true' - with: - target-commit-sha: ${{ inputs.target-commit-sha }} - pull-request-target: ${{ inputs.pull-request-target }} - is-committer-build: ${{ inputs.is-committer-build }} - name: "Cleanup docker" run: ./scripts/ci/cleanup_docker.sh - if: inputs.do-build == 'true' - name: "Install Breeze" uses: ./.github/actions/breeze - if: inputs.do-build == 'true' - - name: "Regenerate dependencies in case they were modified manually so that we can build an image" + - name: "Restore CI docker image ${{ inputs.platform }}:${{ env.PYTHON_MAJOR_MINOR_VERSION }}" + uses: apache/infrastructure-actions/stash/restore@c94b890bbedc2fc61466d28e6bd9966bc6c6643c + with: + key: "ci-image-save-${{ inputs.platform }}-${{ env.PYTHON_MAJOR_MINOR_VERSION }}" + path: "/tmp/" + id: restore-ci-image + - name: "Load CI image ${{ inputs.platform }}:${{ env.PYTHON_MAJOR_MINOR_VERSION }}" + run: breeze ci-image load --platform ${{ inputs.platform }} shell: bash - run: | - pip install rich>=12.4.4 pyyaml - python scripts/ci/pre_commit/update_providers_dependencies.py - if: inputs.do-build == 'true' && inputs.upgrade-to-newer-dependencies != 'false' - - name: "Start ARM instance" - run: ./scripts/ci/images/ci_start_arm_instance_and_connect_to_docker.sh - if: inputs.do-build == 'true' && inputs.platform == 'linux/arm64' - - name: Login to ghcr.io - run: echo "${{ env.GITHUB_TOKEN }}" | docker login ghcr.io -u ${{ github.actor }} --password-stdin - if: inputs.do-build == 'true' + if: steps.restore-ci-image.stash-hit == 'true' - name: > Build ${{ inputs.push-image == 'true' && ' & push ' || '' }} - ${{ inputs.platform }}:${{ matrix.python-version }}:${{ inputs.image-tag }} + ${{ inputs.platform }}:${{ env.PYTHON_MAJOR_MINOR_VERSION }} image run: > - breeze ci-image build --builder airflow_cache --tag-as-latest --image-tag "${{ inputs.image-tag }}" - --python "${{ matrix.python-version }}" --platform "${{ inputs.platform }}" + breeze ci-image build + --docker-cache local + --builder airflow_cache + --platform "${{ inputs.platform }}" env: DOCKER_CACHE: ${{ inputs.docker-cache }} DISABLE_AIRFLOW_REPO_CACHE: ${{ inputs.disable-airflow-repo-cache }} @@ -189,7 +163,14 @@ ${{ inputs.do-build == 'true' && inputs.image-tag || '' }}" GITHUB_USERNAME: ${{ github.actor }} PUSH: ${{ inputs.push-image }} VERBOSE: "true" - if: inputs.do-build == 'true' - - name: "Stop ARM instance" - run: ./scripts/ci/images/ci_stop_arm_instance.sh - if: always() && inputs.do-build == 'true' && inputs.platform == 'linux/arm64' + - name: "Export CI docker image ${{ env.PYTHON_MAJOR_MINOR_VERSION }}" + run: breeze ci-image save --platform "${{ inputs.platform }}" + if: inputs.upload-image-artifact == 'true' + - name: "Stash CI docker image ${{ env.PYTHON_MAJOR_MINOR_VERSION }}" + uses: apache/infrastructure-actions/stash/save@c94b890bbedc2fc61466d28e6bd9966bc6c6643c + with: + key: "ci-image-save-${{ inputs.platform }}-${{ env.PYTHON_MAJOR_MINOR_VERSION }}" + path: "/tmp/ci-image-save-*-${{ env.PYTHON_MAJOR_MINOR_VERSION }}.tar" + if-no-files-found: 'error' + retention-days: 2 + if: inputs.upload-image-artifact == 'true' diff --git a/.github/workflows/ci-image-checks.yml b/.github/workflows/ci-image-checks.yml index 63598755c32d0..8e0e5e71dd00a 100644 --- a/.github/workflows/ci-image-checks.yml +++ b/.github/workflows/ci-image-checks.yml @@ -28,10 +28,6 @@ on: # yamllint disable-line rule:truthy description: "The array of labels (in json form) determining the labels used for docs build." required: true type: string - image-tag: - description: "Tag to set for the image" - required: true - type: string needs-mypy: description: "Whether to run mypy checks (true/false)" required: true @@ -117,7 +113,6 @@ jobs: env: PYTHON_MAJOR_MINOR_VERSION: "${{ inputs.default-python-version }}" UPGRADE_TO_NEWER_DEPENDENCIES: "${{ inputs.upgrade-to-newer-dependencies }}" - IMAGE_TAG: ${{ inputs.image-tag }} GITHUB_TOKEN: ${{ secrets.GITHUB_TOKEN }} if: inputs.basic-checks-only == 'false' && inputs.latest-versions-only != 'true' steps: @@ -134,10 +129,11 @@ jobs: python-version: ${{ inputs.default-python-version }} cache: 'pip' cache-dependency-path: ./dev/breeze/pyproject.toml - - name: "Cleanup docker" - run: ./scripts/ci/cleanup_docker.sh - - name: "Prepare breeze & CI image: ${{ inputs.default-python-version}}:${{ inputs.image-tag }}" + - name: "Prepare breeze & CI image: ${{ inputs.default-python-version }}" uses: ./.github/actions/prepare_breeze_and_image + with: + platform: "linux/amd64" + python: ${{ inputs.default-python-version }} id: breeze - name: "Install pre-commit" uses: ./.github/actions/install-pre-commit @@ -165,7 +161,6 @@ jobs: mypy-check: ${{ fromJSON(inputs.mypy-checks) }} env: PYTHON_MAJOR_MINOR_VERSION: "${{inputs.default-python-version}}" - IMAGE_TAG: "${{ inputs.image-tag }}" GITHUB_TOKEN: ${{ secrets.GITHUB_TOKEN }} steps: - name: "Cleanup repo" @@ -175,10 +170,11 @@ jobs: uses: actions/checkout@v4 with: persist-credentials: false - - name: "Cleanup docker" - run: ./scripts/ci/cleanup_docker.sh - - name: "Prepare breeze & CI image: ${{ inputs.default-python-version }}:${{ inputs.image-tag }}" + - name: "Prepare breeze & CI image: ${{ inputs.default-python-version }}" uses: ./.github/actions/prepare_breeze_and_image + with: + platform: "linux/amd64" + python: ${{ inputs.default-python-version }} id: breeze - name: "Install pre-commit" uses: ./.github/actions/install-pre-commit @@ -208,7 +204,6 @@ jobs: GITHUB_REPOSITORY: ${{ github.repository }} GITHUB_TOKEN: ${{ secrets.GITHUB_TOKEN }} GITHUB_USERNAME: ${{ github.actor }} - IMAGE_TAG: "${{ inputs.image-tag }}" INCLUDE_NOT_READY_PROVIDERS: "true" INCLUDE_SUCCESS_OUTPUTS: "${{ inputs.include-success-outputs }}" PYTHON_MAJOR_MINOR_VERSION: "${{ inputs.default-python-version }}" @@ -221,10 +216,11 @@ jobs: uses: actions/checkout@v4 with: persist-credentials: false - - name: "Cleanup docker" - run: ./scripts/ci/cleanup_docker.sh - - name: "Prepare breeze & CI image: ${{ inputs.default-python-version }}:${{ inputs.image-tag }}" + - name: "Prepare breeze & CI image: ${{ inputs.default-python-version }}" uses: ./.github/actions/prepare_breeze_and_image + with: + platform: "linux/amd64" + python: ${{ inputs.default-python-version }} - uses: actions/cache@v4 id: cache-doc-inventories with: @@ -254,7 +250,6 @@ jobs: GITHUB_REPOSITORY: ${{ github.repository }} GITHUB_TOKEN: ${{ secrets.GITHUB_TOKEN }} GITHUB_USERNAME: ${{ github.actor }} - IMAGE_TAG: "${{ inputs.image-tag }}" INCLUDE_NOT_READY_PROVIDERS: "true" INCLUDE_SUCCESS_OUTPUTS: "${{ inputs.include-success-outputs }}" PYTHON_MAJOR_MINOR_VERSION: "${{ inputs.default-python-version }}" @@ -283,8 +278,11 @@ jobs: run: > git clone https://github.com/apache/airflow-site.git /mnt/airflow-site/airflow-site && echo "AIRFLOW_SITE_DIRECTORY=/mnt/airflow-site/airflow-site" >> "$GITHUB_ENV" - - name: "Prepare breeze & CI image: ${{ inputs.default-python-version }}:${{ inputs.image-tag }}" + - name: "Prepare breeze & CI image: ${{ inputs.default-python-version }}" uses: ./.github/actions/prepare_breeze_and_image + with: + platform: "linux/amd64" + python: ${{ inputs.default-python-version }} - name: "Publish docs" run: > breeze release-management publish-docs --override-versioned --run-in-parallel @@ -331,7 +329,6 @@ jobs: GITHUB_REPOSITORY: ${{ github.repository }} GITHUB_TOKEN: ${{ secrets.GITHUB_TOKEN }} GITHUB_USERNAME: ${{ github.actor }} - IMAGE_TAG: "${{ inputs.image-tag }}" JOB_ID: "python-api-client-tests" PYTHON_MAJOR_MINOR_VERSION: "${{ inputs.default-python-version }}" VERBOSE: "true" @@ -353,8 +350,11 @@ jobs: fetch-depth: 1 persist-credentials: false path: ./airflow-client-python - - name: "Prepare breeze & CI image: ${{inputs.default-python-version}}:${{inputs.image-tag}}" + - name: "Prepare breeze & CI image: ${{ inputs.default-python-version }}" uses: ./.github/actions/prepare_breeze_and_image + with: + platform: "linux/amd64" + python: ${{ inputs.default-python-version }} - name: "Generate airflow python client" run: > breeze release-management prepare-python-client --package-format both diff --git a/.github/workflows/ci.yml b/.github/workflows/ci.yml index 09cc3328dd8a7..3ee18ec1a6ece 100644 --- a/.github/workflows/ci.yml +++ b/.github/workflows/ci.yml @@ -32,7 +32,7 @@ on: # yamllint disable-line rule:truthy - providers-[a-z]+-?[a-z]*/v[0-9]+-[0-9]+ workflow_dispatch: permissions: - # All other permissions are set to none + # All other permissions are set to none by default contents: read # Technically read access while waiting for images should be more than enough. However, # there is a bug in GitHub Actions/Packages and in case private repositories are used, you get a permission @@ -44,7 +44,6 @@ env: GITHUB_REPOSITORY: ${{ github.repository }} GITHUB_TOKEN: ${{ secrets.GITHUB_TOKEN }} GITHUB_USERNAME: ${{ github.actor }} - IMAGE_TAG: "${{ github.event.pull_request.head.sha || github.sha }}" SLACK_BOT_TOKEN: ${{ secrets.SLACK_BOT_TOKEN }} VERBOSE: "true" @@ -64,7 +63,6 @@ jobs: all-python-versions-list-as-string: >- ${{ steps.selective-checks.outputs.all-python-versions-list-as-string }} basic-checks-only: ${{ steps.selective-checks.outputs.basic-checks-only }} - build-job-description: ${{ steps.source-run-info.outputs.build-job-description }} canary-run: ${{ steps.source-run-info.outputs.canary-run }} chicken-egg-providers: ${{ steps.selective-checks.outputs.chicken-egg-providers }} ci-image-build: ${{ steps.selective-checks.outputs.ci-image-build }} @@ -88,8 +86,6 @@ jobs: full-tests-needed: ${{ steps.selective-checks.outputs.full-tests-needed }} has-migrations: ${{ steps.selective-checks.outputs.has-migrations }} helm-test-packages: ${{ steps.selective-checks.outputs.helm-test-packages }} - image-tag: ${{ github.event.pull_request.head.sha || github.sha }} - in-workflow-build: ${{ steps.source-run-info.outputs.in-workflow-build }} include-success-outputs: ${{ steps.selective-checks.outputs.include-success-outputs }} individual-providers-test-types-list-as-string: >- ${{ steps.selective-checks.outputs.individual-providers-test-types-list-as-string }} @@ -99,6 +95,7 @@ jobs: is-k8s-runner: ${{ steps.selective-checks.outputs.is-k8s-runner }} is-self-hosted-runner: ${{ steps.selective-checks.outputs.is-self-hosted-runner }} is-vm-runner: ${{ steps.selective-checks.outputs.is-vm-runner }} + kubernetes-combos: ${{ steps.selective-checks.outputs.kubernetes-combos }} kubernetes-combos-list-as-string: >- ${{ steps.selective-checks.outputs.kubernetes-combos-list-as-string }} kubernetes-versions-list-as-string: >- @@ -197,25 +194,21 @@ jobs: canary-run: ${{needs.build-info.outputs.canary-run}} latest-versions-only: ${{needs.build-info.outputs.latest-versions-only}} build-ci-images: - name: > - ${{ needs.build-info.outputs.in-workflow-build == 'true' && 'Build' || 'Skip building' }} - CI images in-workflow + name: Build CI images needs: [build-info] uses: ./.github/workflows/ci-image-build.yml permissions: contents: read # This write is only given here for `push` events from "apache/airflow" repo. It is not given for PRs # from forks. This is to prevent malicious PRs from creating images in the "apache/airflow" repo. - # For regular build for PRS this "build-prod-images" workflow will be skipped anyway by the - # "in-workflow-build" condition packages: write secrets: inherit with: runs-on-as-json-public: ${{ needs.build-info.outputs.runs-on-as-json-public }} runs-on-as-json-self-hosted: ${{ needs.build-info.outputs.runs-on-as-json-self-hosted }} - do-build: ${{ needs.build-info.outputs.in-workflow-build }} - image-tag: ${{ needs.build-info.outputs.image-tag }} platform: "linux/amd64" + push-image: "false" + upload-image-artifact: "true" python-versions: ${{ needs.build-info.outputs.python-versions }} branch: ${{ needs.build-info.outputs.default-branch }} use-uv: ${{ needs.build-info.outputs.force-pip == 'true' && 'false' || 'true' }} @@ -224,54 +217,15 @@ jobs: docker-cache: ${{ needs.build-info.outputs.docker-cache }} disable-airflow-repo-cache: ${{ needs.build-info.outputs.disable-airflow-repo-cache }} - wait-for-ci-images: - timeout-minutes: 120 - name: "Wait for CI images" - runs-on: ${{ fromJSON(needs.build-info.outputs.runs-on-as-json-public) }} - needs: [build-info, build-ci-images] - if: needs.build-info.outputs.ci-image-build == 'true' - env: - BACKEND: sqlite - # Force more parallelism for pull even on public images - PARALLELISM: 6 - INCLUDE_SUCCESS_OUTPUTS: "${{needs.build-info.outputs.include-success-outputs}}" - steps: - - name: "Cleanup repo" - shell: bash - run: docker run -v "${GITHUB_WORKSPACE}:/workspace" -u 0:0 bash -c "rm -rf /workspace/*" - if: needs.build-info.outputs.in-workflow-build == 'false' - - name: "Checkout ${{ github.ref }} ( ${{ github.sha }} )" - uses: actions/checkout@v4 - with: - persist-credentials: false - if: needs.build-info.outputs.in-workflow-build == 'false' - - name: "Cleanup docker" - run: ./scripts/ci/cleanup_docker.sh - if: needs.build-info.outputs.in-workflow-build == 'false' - - name: "Install Breeze" - uses: ./.github/actions/breeze - if: needs.build-info.outputs.in-workflow-build == 'false' - - name: Login to ghcr.io - run: echo "${{ env.GITHUB_TOKEN }}" | docker login ghcr.io -u ${{ github.actor }} --password-stdin - if: needs.build-info.outputs.in-workflow-build == 'false' - - name: Wait for CI images ${{ env.PYTHON_VERSIONS }}:${{ needs.build-info.outputs.image-tag }} - id: wait-for-images - run: breeze ci-image pull --run-in-parallel --wait-for-image --tag-as-latest - env: - PYTHON_VERSIONS: ${{ needs.build-info.outputs.python-versions-list-as-string }} - DEBUG_RESOURCES: ${{needs.build-info.outputs.debug-resources}} - if: needs.build-info.outputs.in-workflow-build == 'false' - additional-ci-image-checks: name: "Additional CI image checks" - needs: [build-info, wait-for-ci-images] + needs: [build-info, build-ci-images] uses: ./.github/workflows/additional-ci-image-checks.yml if: needs.build-info.outputs.canary-run == 'true' with: runs-on-as-json-default: ${{ needs.build-info.outputs.runs-on-as-json-default }} runs-on-as-json-public: ${{ needs.build-info.outputs.runs-on-as-json-public }} runs-on-as-json-self-hosted: ${{ needs.build-info.outputs.runs-on-as-json-self-hosted }} - image-tag: ${{ needs.build-info.outputs.image-tag }} python-versions: ${{ needs.build-info.outputs.python-versions }} branch: ${{ needs.build-info.outputs.default-branch }} constraints-branch: ${{ needs.build-info.outputs.default-constraints-branch }} @@ -289,7 +243,7 @@ jobs: generate-constraints: name: "Generate constraints" - needs: [build-info, wait-for-ci-images] + needs: [build-info, build-ci-images] uses: ./.github/workflows/generate-constraints.yml if: > needs.build-info.outputs.ci-image-build == 'true' && @@ -300,19 +254,17 @@ jobs: # generate no providers constraints only in canary builds - they take quite some time to generate # they are not needed for regular builds, they are only needed to update constraints in canaries generate-no-providers-constraints: ${{ needs.build-info.outputs.canary-run }} - image-tag: ${{ needs.build-info.outputs.image-tag }} chicken-egg-providers: ${{ needs.build-info.outputs.chicken-egg-providers }} debug-resources: ${{ needs.build-info.outputs.debug-resources }} ci-image-checks: name: "CI image checks" - needs: [build-info, wait-for-ci-images] + needs: [build-info, build-ci-images] uses: ./.github/workflows/ci-image-checks.yml secrets: inherit with: runs-on-as-json-default: ${{ needs.build-info.outputs.runs-on-as-json-default }} runs-on-as-json-docs-build: ${{ needs.build-info.outputs.runs-on-as-json-docs-build }} - image-tag: ${{ needs.build-info.outputs.image-tag }} needs-mypy: ${{ needs.build-info.outputs.needs-mypy }} mypy-checks: ${{ needs.build-info.outputs.mypy-checks }} python-versions-list-as-string: ${{ needs.build-info.outputs.python-versions-list-as-string }} @@ -336,7 +288,7 @@ jobs: providers: name: "Provider packages tests" uses: ./.github/workflows/test-provider-packages.yml - needs: [build-info, wait-for-ci-images] + needs: [build-info, build-ci-images] permissions: contents: read packages: read @@ -346,7 +298,6 @@ jobs: needs.build-info.outputs.latest-versions-only != 'true' with: runs-on-as-json-default: ${{ needs.build-info.outputs.runs-on-as-json-default }} - image-tag: ${{ needs.build-info.outputs.image-tag }} canary-run: ${{ needs.build-info.outputs.canary-run }} default-python-version: ${{ needs.build-info.outputs.default-python-version }} upgrade-to-newer-dependencies: ${{ needs.build-info.outputs.upgrade-to-newer-dependencies }} @@ -360,7 +311,7 @@ jobs: tests-helm: name: "Helm tests" uses: ./.github/workflows/helm-tests.yml - needs: [build-info, wait-for-ci-images] + needs: [build-info, build-ci-images] permissions: contents: read packages: read @@ -369,7 +320,6 @@ jobs: runs-on-as-json-default: ${{ needs.build-info.outputs.runs-on-as-json-default }} runs-on-as-json-public: ${{ needs.build-info.outputs.runs-on-as-json-public }} helm-test-packages: ${{ needs.build-info.outputs.helm-test-packages }} - image-tag: ${{ needs.build-info.outputs.image-tag }} default-python-version: ${{ needs.build-info.outputs.default-python-version }} if: > needs.build-info.outputs.needs-helm-tests == 'true' && @@ -379,7 +329,7 @@ jobs: tests-postgres: name: "Postgres tests" uses: ./.github/workflows/run-unit-tests.yml - needs: [build-info, wait-for-ci-images] + needs: [build-info, build-ci-images] permissions: contents: read packages: read @@ -390,7 +340,6 @@ jobs: test-name: "Postgres" test-scope: "DB" test-groups: ${{ needs.build-info.outputs.test-groups }} - image-tag: ${{ needs.build-info.outputs.image-tag }} python-versions: ${{ needs.build-info.outputs.python-versions }} backend-versions: ${{ needs.build-info.outputs.postgres-versions }} excluded-providers-as-string: ${{ needs.build-info.outputs.excluded-providers-as-string }} @@ -406,7 +355,7 @@ jobs: tests-mysql: name: "MySQL tests" uses: ./.github/workflows/run-unit-tests.yml - needs: [build-info, wait-for-ci-images] + needs: [build-info, build-ci-images] permissions: contents: read packages: read @@ -417,7 +366,6 @@ jobs: test-name: "MySQL" test-scope: "DB" test-groups: ${{ needs.build-info.outputs.test-groups }} - image-tag: ${{ needs.build-info.outputs.image-tag }} python-versions: ${{ needs.build-info.outputs.python-versions }} backend-versions: ${{ needs.build-info.outputs.mysql-versions }} excluded-providers-as-string: ${{ needs.build-info.outputs.excluded-providers-as-string }} @@ -433,7 +381,7 @@ jobs: tests-sqlite: name: "Sqlite tests" uses: ./.github/workflows/run-unit-tests.yml - needs: [build-info, wait-for-ci-images] + needs: [build-info, build-ci-images] permissions: contents: read packages: read @@ -445,7 +393,6 @@ jobs: test-name-separator: "" test-scope: "DB" test-groups: ${{ needs.build-info.outputs.test-groups }} - image-tag: ${{ needs.build-info.outputs.image-tag }} python-versions: ${{ needs.build-info.outputs.python-versions }} # No versions for sqlite backend-versions: "['']" @@ -462,7 +409,7 @@ jobs: tests-non-db: name: "Non-DB tests" uses: ./.github/workflows/run-unit-tests.yml - needs: [build-info, wait-for-ci-images] + needs: [build-info, build-ci-images] permissions: contents: read packages: read @@ -474,7 +421,6 @@ jobs: test-name-separator: "" test-scope: "Non-DB" test-groups: ${{ needs.build-info.outputs.test-groups }} - image-tag: ${{ needs.build-info.outputs.image-tag }} python-versions: ${{ needs.build-info.outputs.python-versions }} # No versions for non-db backend-versions: "['']" @@ -490,7 +436,7 @@ jobs: tests-special: name: "Special tests" uses: ./.github/workflows/special-tests.yml - needs: [build-info, wait-for-ci-images] + needs: [build-info, build-ci-images] permissions: contents: read packages: read @@ -504,7 +450,6 @@ jobs: test-groups: ${{ needs.build-info.outputs.test-groups }} default-branch: ${{ needs.build-info.outputs.default-branch }} runs-on-as-json-default: ${{ needs.build-info.outputs.runs-on-as-json-default }} - image-tag: ${{ needs.build-info.outputs.image-tag }} core-test-types-list-as-string: ${{ needs.build-info.outputs.core-test-types-list-as-string }} providers-test-types-list-as-string: ${{ needs.build-info.outputs.providers-test-types-list-as-string }} run-coverage: ${{ needs.build-info.outputs.run-coverage }} @@ -519,7 +464,7 @@ jobs: tests-integration-system: name: Integration and System Tests - needs: [build-info, wait-for-ci-images] + needs: [build-info, build-ci-images] uses: ./.github/workflows/integration-system-tests.yml permissions: contents: read @@ -527,7 +472,6 @@ jobs: secrets: inherit with: runs-on-as-json-public: ${{ needs.build-info.outputs.runs-on-as-json-public }} - image-tag: ${{ needs.build-info.outputs.image-tag }} testable-core-integrations: ${{ needs.build-info.outputs.testable-core-integrations }} testable-providers-integrations: ${{ needs.build-info.outputs.testable-providers-integrations }} run-system-tests: ${{ needs.build-info.outputs.run-tests }} @@ -541,7 +485,7 @@ jobs: tests-with-lowest-direct-resolution: name: "Lowest direct dependency providers tests" - needs: [build-info, wait-for-ci-images] + needs: [build-info, build-ci-images] uses: ./.github/workflows/run-unit-tests.yml permissions: contents: read @@ -556,7 +500,6 @@ jobs: test-scope: "All" test-groups: ${{ needs.build-info.outputs.test-groups }} backend: "postgres" - image-tag: ${{ needs.build-info.outputs.image-tag }} python-versions: ${{ needs.build-info.outputs.python-versions }} backend-versions: "['${{ needs.build-info.outputs.default-postgres-version }}']" excluded-providers-as-string: ${{ needs.build-info.outputs.excluded-providers-as-string }} @@ -570,30 +513,25 @@ jobs: monitor-delay-time-in-seconds: 120 build-prod-images: - name: > - ${{ needs.build-info.outputs.in-workflow-build == 'true' && 'Build' || 'Skip building' }} - PROD images in-workflow + name: Build PROD images needs: [build-info, build-ci-images, generate-constraints] uses: ./.github/workflows/prod-image-build.yml permissions: contents: read # This write is only given here for `push` events from "apache/airflow" repo. It is not given for PRs # from forks. This is to prevent malicious PRs from creating images in the "apache/airflow" repo. - # For regular build for PRS this "build-prod-images" workflow will be skipped anyway by the - # "in-workflow-build" condition packages: write secrets: inherit with: runs-on-as-json-public: ${{ needs.build-info.outputs.runs-on-as-json-public }} build-type: "Regular" - do-build: ${{ needs.build-info.outputs.in-workflow-build }} - upload-package-artifact: "true" - image-tag: ${{ needs.build-info.outputs.image-tag }} platform: "linux/amd64" + push-image: "false" + upload-image-artifact: "true" + upload-package-artifact: "true" python-versions: ${{ needs.build-info.outputs.python-versions }} default-python-version: ${{ needs.build-info.outputs.default-python-version }} branch: ${{ needs.build-info.outputs.default-branch }} - push-image: "true" use-uv: ${{ needs.build-info.outputs.force-pip == 'true' && 'false' || 'true' }} build-provider-packages: ${{ needs.build-info.outputs.default-branch == 'main' }} upgrade-to-newer-dependencies: ${{ needs.build-info.outputs.upgrade-to-newer-dependencies }} @@ -602,58 +540,14 @@ jobs: docker-cache: ${{ needs.build-info.outputs.docker-cache }} disable-airflow-repo-cache: ${{ needs.build-info.outputs.disable-airflow-repo-cache }} - wait-for-prod-images: - timeout-minutes: 80 - name: "Wait for PROD images" - runs-on: ${{ fromJSON(needs.build-info.outputs.runs-on-as-json-public) }} - needs: [build-info, wait-for-ci-images, build-prod-images] - if: needs.build-info.outputs.prod-image-build == 'true' - env: - BACKEND: sqlite - PYTHON_MAJOR_MINOR_VERSION: "${{needs.build-info.outputs.default-python-version}}" - # Force more parallelism for pull on public images - PARALLELISM: 6 - INCLUDE_SUCCESS_OUTPUTS: "${{needs.build-info.outputs.include-success-outputs}}" - IMAGE_TAG: ${{ needs.build-info.outputs.image-tag }} - steps: - - name: "Cleanup repo" - shell: bash - run: docker run -v "${GITHUB_WORKSPACE}:/workspace" -u 0:0 bash -c "rm -rf /workspace/*" - if: needs.build-info.outputs.in-workflow-build == 'false' - - name: "Checkout ${{ github.ref }} ( ${{ github.sha }} )" - uses: actions/checkout@v4 - with: - persist-credentials: false - if: needs.build-info.outputs.in-workflow-build == 'false' - - name: "Cleanup docker" - run: ./scripts/ci/cleanup_docker.sh - if: needs.build-info.outputs.in-workflow-build == 'false' - - name: "Install Breeze" - uses: ./.github/actions/breeze - if: needs.build-info.outputs.in-workflow-build == 'false' - - name: Login to ghcr.io - run: echo "${{ env.GITHUB_TOKEN }}" | docker login ghcr.io -u ${{ github.actor }} --password-stdin - if: needs.build-info.outputs.in-workflow-build == 'false' - - name: Wait for PROD images ${{ env.PYTHON_VERSIONS }}:${{ needs.build-info.outputs.image-tag }} - # We wait for the images to be available either from "build-images.yml' run as pull_request_target - # or from build-prod-images (or build-prod-images-release-branch) above. - # We are utilising single job to wait for all images because this job merely waits - # For the images to be available. - run: breeze prod-image pull --wait-for-image --run-in-parallel - env: - PYTHON_VERSIONS: ${{ needs.build-info.outputs.python-versions-list-as-string }} - DEBUG_RESOURCES: ${{ needs.build-info.outputs.debug-resources }} - if: needs.build-info.outputs.in-workflow-build == 'false' - additional-prod-image-tests: name: "Additional PROD image tests" - needs: [build-info, wait-for-prod-images, generate-constraints] + needs: [build-info, build-prod-images, generate-constraints] uses: ./.github/workflows/additional-prod-image-tests.yml with: runs-on-as-json-public: ${{ needs.build-info.outputs.runs-on-as-json-public }} default-branch: ${{ needs.build-info.outputs.default-branch }} constraints-branch: ${{ needs.build-info.outputs.default-constraints-branch }} - image-tag: ${{ needs.build-info.outputs.image-tag }} upgrade-to-newer-dependencies: ${{ needs.build-info.outputs.upgrade-to-newer-dependencies }} chicken-egg-providers: ${{ needs.build-info.outputs.chicken-egg-providers }} docker-cache: ${{ needs.build-info.outputs.docker-cache }} @@ -665,20 +559,19 @@ jobs: tests-kubernetes: name: "Kubernetes tests" uses: ./.github/workflows/k8s-tests.yml - needs: [build-info, wait-for-prod-images] + needs: [build-info, build-prod-images] permissions: contents: read packages: read secrets: inherit with: + platform: "linux/amd64" runs-on-as-json-default: ${{ needs.build-info.outputs.runs-on-as-json-default }} - image-tag: ${{ needs.build-info.outputs.image-tag }} python-versions-list-as-string: ${{ needs.build-info.outputs.python-versions-list-as-string }} - kubernetes-versions-list-as-string: ${{ needs.build-info.outputs.kubernetes-versions-list-as-string }} - kubernetes-combos-list-as-string: ${{ needs.build-info.outputs.kubernetes-combos-list-as-string }} include-success-outputs: ${{ needs.build-info.outputs.include-success-outputs }} use-uv: ${{ needs.build-info.outputs.force-pip == 'true' && 'false' || 'true' }} debug-resources: ${{ needs.build-info.outputs.debug-resources }} + kubernetes-combos: ${{ needs.build-info.outputs.kubernetes-combos }} if: > ( needs.build-info.outputs.run-kubernetes-tests == 'true' || needs.build-info.outputs.needs-helm-tests == 'true') @@ -686,14 +579,13 @@ jobs: tests-task-sdk: name: "Task SDK tests" uses: ./.github/workflows/task-sdk-tests.yml - needs: [build-info, wait-for-ci-images] + needs: [build-info, build-ci-images] permissions: contents: read packages: read secrets: inherit with: runs-on-as-json-default: ${{ needs.build-info.outputs.runs-on-as-json-default }} - image-tag: ${{ needs.build-info.outputs.image-tag }} default-python-version: ${{ needs.build-info.outputs.default-python-version }} python-versions: ${{ needs.build-info.outputs.python-versions }} run-task-sdk-tests: ${{ needs.build-info.outputs.run-task-sdk-tests }} @@ -711,8 +603,6 @@ jobs: needs: - build-info - generate-constraints - - wait-for-ci-images - - wait-for-prod-images - ci-image-checks - tests-sqlite - tests-mysql @@ -723,13 +613,11 @@ jobs: with: runs-on-as-json-public: ${{ needs.build-info.outputs.runs-on-as-json-public }} runs-on-as-json-self-hosted: ${{ needs.build-info.outputs.runs-on-as-json-self-hosted }} - image-tag: ${{ needs.build-info.outputs.image-tag }} python-versions: ${{ needs.build-info.outputs.python-versions }} python-versions-list-as-string: ${{ needs.build-info.outputs.python-versions-list-as-string }} branch: ${{ needs.build-info.outputs.default-branch }} constraints-branch: ${{ needs.build-info.outputs.default-constraints-branch }} default-python-version: ${{ needs.build-info.outputs.default-python-version }} - in-workflow-build: ${{ needs.build-info.outputs.in-workflow-build }} upgrade-to-newer-dependencies: ${{ needs.build-info.outputs.upgrade-to-newer-dependencies }} include-success-outputs: ${{ needs.build-info.outputs.include-success-outputs }} docker-cache: ${{ needs.build-info.outputs.docker-cache }} diff --git a/.github/workflows/finalize-tests.yml b/.github/workflows/finalize-tests.yml index 6f9bc74168b42..b8fd240235f10 100644 --- a/.github/workflows/finalize-tests.yml +++ b/.github/workflows/finalize-tests.yml @@ -28,10 +28,6 @@ on: # yamllint disable-line rule:truthy description: "The array of labels (in json form) determining self-hosted runners." required: true type: string - image-tag: - description: "Tag to set for the image" - required: true - type: string python-versions: description: "JSON-formatted array of Python versions to test" required: true @@ -52,10 +48,6 @@ on: # yamllint disable-line rule:truthy description: "Which version of python should be used by default" required: true type: string - in-workflow-build: - description: "Whether the build is executed as part of the workflow (true/false)" - required: true - type: string upgrade-to-newer-dependencies: description: "Whether to upgrade to newer dependencies (true/false)" required: true @@ -87,7 +79,6 @@ jobs: env: DEBUG_RESOURCES: ${{ inputs.debug-resources}} PYTHON_VERSIONS: ${{ inputs.python-versions-list-as-string }} - IMAGE_TAG: ${{ inputs.image-tag }} GITHUB_REPOSITORY: ${{ github.repository }} GITHUB_TOKEN: ${{ secrets.GITHUB_TOKEN }} GITHUB_USERNAME: ${{ github.actor }} diff --git a/.github/workflows/generate-constraints.yml b/.github/workflows/generate-constraints.yml index d6e536dfd091a..332d32e3160d2 100644 --- a/.github/workflows/generate-constraints.yml +++ b/.github/workflows/generate-constraints.yml @@ -32,10 +32,6 @@ on: # yamllint disable-line rule:truthy description: "Whether to generate constraints without providers (true/false)" required: true type: string - image-tag: - description: "Tag to set for the image" - required: true - type: string chicken-egg-providers: description: "Space-separated list of providers that should be installed from context files" required: true @@ -57,7 +53,6 @@ jobs: GITHUB_TOKEN: ${{ secrets.GITHUB_TOKEN }} GITHUB_USERNAME: ${{ github.actor }} INCLUDE_SUCCESS_OUTPUTS: "true" - IMAGE_TAG: ${{ inputs.image-tag }} PYTHON_VERSIONS: ${{ inputs.python-versions-list-as-string }} VERBOSE: "true" VERSION_SUFFIX_FOR_PYPI: "dev0" @@ -69,21 +64,15 @@ jobs: uses: actions/checkout@v4 with: persist-credentials: false - - name: "Cleanup docker" - run: ./scripts/ci/cleanup_docker.sh - name: "Install Breeze" uses: ./.github/actions/breeze - - name: Login to ghcr.io - run: echo "${{ env.GITHUB_TOKEN }}" | docker login ghcr.io -u ${{ github.actor }} --password-stdin - - name: "\ - Pull CI images \ - ${{ inputs.python-versions-list-as-string }}:\ - ${{ inputs.image-tag }}" - run: breeze ci-image pull --run-in-parallel --tag-as-latest - - name: " - Verify CI images \ - ${{ inputs.python-versions-list-as-string }}:\ - ${{ inputs.image-tag }}" + id: breeze + - name: "Prepare all CI images: ${{ inputs.python-versions-list-as-string}}" + uses: ./.github/actions/prepare_all_ci_images + with: + platform: "linux/amd64" + python-versions-list-as-string: ${{ inputs.python-versions-list-as-string }} + - name: "Verify all CI images ${{ inputs.python-versions-list-as-string }}" run: breeze ci-image verify --run-in-parallel - name: "Source constraints" shell: bash diff --git a/.github/workflows/helm-tests.yml b/.github/workflows/helm-tests.yml index 4c1ec1023fc90..d7f3f0b4d5bf0 100644 --- a/.github/workflows/helm-tests.yml +++ b/.github/workflows/helm-tests.yml @@ -32,10 +32,6 @@ on: # yamllint disable-line rule:truthy description: "Stringified JSON array of helm test packages to test" required: true type: string - image-tag: - description: "Tag to set for the image" - required: true - type: string default-python-version: description: "Which version of python should be used by default" required: true @@ -57,7 +53,6 @@ jobs: DB_RESET: "false" JOB_ID: "helm-tests" USE_XDIST: "true" - IMAGE_TAG: "${{ inputs.image-tag }}" GITHUB_REPOSITORY: ${{ github.repository }} GITHUB_TOKEN: ${{ secrets.GITHUB_TOKEN }} GITHUB_USERNAME: ${{ github.actor }} @@ -70,10 +65,11 @@ jobs: uses: actions/checkout@v4 with: persist-credentials: false - - name: "Cleanup docker" - run: ./scripts/ci/cleanup_docker.sh - - name: "Prepare breeze & CI image: ${{inputs.default-python-version}}:${{inputs.image-tag}}" + - name: "Prepare breeze & CI image: ${{ inputs.default-python-version }}" uses: ./.github/actions/prepare_breeze_and_image + with: + platform: "linux/amd64" + python: ${{ inputs.default-python-version }} - name: "Helm Unit Tests: ${{ matrix.helm-test-package }}" run: breeze testing helm-tests --test-type "${{ matrix.helm-test-package }}" diff --git a/.github/workflows/integration-system-tests.yml b/.github/workflows/integration-system-tests.yml index 7fde2ae968363..0a6fd1fb406c7 100644 --- a/.github/workflows/integration-system-tests.yml +++ b/.github/workflows/integration-system-tests.yml @@ -24,10 +24,6 @@ on: # yamllint disable-line rule:truthy description: "The array of labels (in json form) determining public runners." required: true type: string - image-tag: - description: "Tag to set for the image" - required: true - type: string testable-core-integrations: description: "The list of testable core integrations as JSON array." required: true @@ -75,7 +71,6 @@ jobs: matrix: integration: ${{ fromJSON(inputs.testable-core-integrations) }} env: - IMAGE_TAG: "${{ inputs.image-tag }}" BACKEND: "postgres" BACKEND_VERSION: ${{ inputs.default-postgres-version }}" PYTHON_MAJOR_MINOR_VERSION: "${{ inputs.default-python-version }}" @@ -95,10 +90,11 @@ jobs: uses: actions/checkout@v4 with: persist-credentials: false - - name: "Cleanup docker" - run: ./scripts/ci/cleanup_docker.sh - - name: "Prepare breeze & CI image: ${{ inputs.default-python-version }}:${{ inputs.image-tag }}" + - name: "Prepare breeze & CI image: ${{ inputs.default-python-version }}" uses: ./.github/actions/prepare_breeze_and_image + with: + platform: "linux/amd64" + python: ${{ inputs.default-python-version }} - name: "Integration: core ${{ matrix.integration }}" # yamllint disable rule:line-length run: ./scripts/ci/testing/run_integration_tests_with_retry.sh core "${{ matrix.integration }}" @@ -121,7 +117,6 @@ jobs: matrix: integration: ${{ fromJSON(inputs.testable-providers-integrations) }} env: - IMAGE_TAG: "${{ inputs.image-tag }}" BACKEND: "postgres" BACKEND_VERSION: ${{ inputs.default-postgres-version }}" PYTHON_MAJOR_MINOR_VERSION: "${{ inputs.default-python-version }}" @@ -141,10 +136,11 @@ jobs: uses: actions/checkout@v4 with: persist-credentials: false - - name: "Cleanup docker" - run: ./scripts/ci/cleanup_docker.sh - - name: "Prepare breeze & CI image: ${{ inputs.default-python-version }}:${{ inputs.image-tag }}" + - name: "Prepare breeze & CI image: ${{ inputs.default-python-version }}" uses: ./.github/actions/prepare_breeze_and_image + with: + platform: "linux/amd64" + python: ${{ inputs.default-python-version }} - name: "Integration: providers ${{ matrix.integration }}" run: ./scripts/ci/testing/run_integration_tests_with_retry.sh providers "${{ matrix.integration }}" - name: "Post Tests success" @@ -162,7 +158,6 @@ jobs: name: "System Tests" runs-on: ${{ fromJSON(inputs.runs-on-as-json-public) }} env: - IMAGE_TAG: "${{ inputs.image-tag }}" BACKEND: "postgres" BACKEND_VERSION: ${{ inputs.default-postgres-version }}" PYTHON_MAJOR_MINOR_VERSION: "${{ inputs.default-python-version }}" @@ -182,10 +177,11 @@ jobs: uses: actions/checkout@v4 with: persist-credentials: false - - name: "Cleanup docker" - run: ./scripts/ci/cleanup_docker.sh - - name: "Prepare breeze & CI image: ${{ inputs.default-python-version }}:${{ inputs.image-tag }}" + - name: "Prepare breeze & CI image: ${{ inputs.default-python-version }}" uses: ./.github/actions/prepare_breeze_and_image + with: + platform: "linux/amd64" + python: ${{ inputs.default-python-version }} - name: "System Tests" run: > ./scripts/ci/testing/run_system_tests.sh diff --git a/.github/workflows/k8s-tests.yml b/.github/workflows/k8s-tests.yml index 3b3e067038db9..419664ca8ec19 100644 --- a/.github/workflows/k8s-tests.yml +++ b/.github/workflows/k8s-tests.yml @@ -20,24 +20,20 @@ name: K8s tests on: # yamllint disable-line rule:truthy workflow_call: inputs: - runs-on-as-json-default: - description: "The array of labels (in json form) determining default runner used for the build." + platform: + description: "Platform for the build - 'linux/amd64' or 'linux/arm64'" required: true type: string - image-tag: - description: "Tag to set for the image" + runs-on-as-json-default: + description: "The array of labels (in json form) determining default runner used for the build." required: true type: string python-versions-list-as-string: description: "List of Python versions to test: space separated string" required: true type: string - kubernetes-versions-list-as-string: - description: "List of Kubernetes versions to test" - required: true - type: string - kubernetes-combos-list-as-string: - description: "List of combinations of Kubernetes and Python versions to test: space separated string" + kubernetes-combos: + description: "Array of combinations of Kubernetes and Python versions to test" required: true type: string include-success-outputs: @@ -55,19 +51,17 @@ on: # yamllint disable-line rule:truthy jobs: tests-kubernetes: timeout-minutes: 240 - name: "\ - K8S System:${{ matrix.executor }} - ${{ matrix.use-standard-naming }} - \ - ${{ inputs.kubernetes-versions-list-as-string }}" + name: "K8S System:${{ matrix.executor }}-${{ matrix.kubernetes-combo }}-${{ matrix.use-standard-naming }}" runs-on: ${{ fromJSON(inputs.runs-on-as-json-default) }} strategy: matrix: executor: [KubernetesExecutor, CeleryExecutor, LocalExecutor] use-standard-naming: [true, false] + kubernetes-combo: ${{ fromJSON(inputs.kubernetes-combos) }} fail-fast: false env: DEBUG_RESOURCES: ${{ inputs.debug-resources }} INCLUDE_SUCCESS_OUTPUTS: ${{ inputs.include-success-outputs }} - IMAGE_TAG: ${{ inputs.image-tag }} GITHUB_REPOSITORY: ${{ github.repository }} GITHUB_TOKEN: ${{ secrets.GITHUB_TOKEN }} GITHUB_USERNAME: ${{ github.actor }} @@ -76,23 +70,21 @@ jobs: - name: "Cleanup repo" shell: bash run: docker run -v "${GITHUB_WORKSPACE}:/workspace" -u 0:0 bash -c "rm -rf /workspace/*" + - name: "Prepare PYTHON_MAJOR_MINOR_VERSION and KUBERNETES_VERSION" + id: prepare-versions + run: | + echo "PYTHON_MAJOR_MINOR_VERSION=${{ matrix.kubernetes-combo }}" | sed 's/-.*//' >> $GITHUB_ENV + echo "KUBERNETES_VERSION=${{ matrix.kubernetes-combo }}" | sed 's/=[^-]*-/=/' >> $GITHUB_ENV - name: "Checkout ${{ github.ref }} ( ${{ github.sha }} )" uses: actions/checkout@v4 with: persist-credentials: false - - name: "Cleanup docker" - run: ./scripts/ci/cleanup_docker.sh - - name: "Install Breeze" - uses: ./.github/actions/breeze - id: breeze - - name: Login to ghcr.io - run: echo "${{ env.GITHUB_TOKEN }}" | docker login ghcr.io -u ${{ github.actor }} --password-stdin - - name: Pull PROD images ${{ inputs.python-versions-list-as-string }}:${{ inputs.image-tag }} - run: breeze prod-image pull --run-in-parallel --tag-as-latest - env: - PYTHON_VERSIONS: ${{ inputs.python-versions-list-as-string }} - # Force more parallelism for pull even on public images - PARALLELISM: 6 + - name: "Prepare breeze & PROD image: ${{ env.PYTHON_MAJOR_MINOR_VERSION }}" + uses: ./.github/actions/prepare_breeze_and_image + with: + platform: ${{ inputs.platform }} + image-type: "PROD" + python: ${{ env.PYTHON_MAJOR_MINOR_VERSION }} - name: "Cache bin folder with tools for kubernetes testing" uses: actions/cache@v4 with: @@ -100,29 +92,34 @@ jobs: key: "\ k8s-env-${{ steps.breeze.outputs.host-python-version }}-\ ${{ hashFiles('scripts/ci/kubernetes/k8s_requirements.txt','hatch_build.py') }}" - - name: "Switch breeze to use uv" - run: breeze setup config --use-uv - if: inputs.use-uv == 'true' - - name: Run complete K8S tests ${{ inputs.kubernetes-combos-list-as-string }} - run: breeze k8s run-complete-tests --run-in-parallel --upgrade --no-copy-local-sources + - name: "\ + Run complete K8S tests ${{ matrix.executor }}-${{ env.PYTHON_MAJOR_MINOR_VERSION }}-\ + ${{env.KUBERNETES_VERSION}}-${{ matrix.use-standard-naming }}" + run: breeze k8s run-complete-tests --upgrade --no-copy-local-sources env: - PYTHON_VERSIONS: ${{ inputs.python-versions-list-as-string }} - KUBERNETES_VERSIONS: ${{ inputs.kubernetes-versions-list-as-string }} EXECUTOR: ${{ matrix.executor }} USE_STANDARD_NAMING: ${{ matrix.use-standard-naming }} VERBOSE: "false" - - name: Upload KinD logs on failure ${{ inputs.kubernetes-combos-list-as-string }} + - name: "\ + Upload KinD logs on failure ${{ matrix.executor }}-${{ matrix.kubernetes-combo }}-\ + ${{ matrix.use-standard-naming }}" uses: actions/upload-artifact@v4 if: failure() || cancelled() with: - name: kind-logs-${{ matrix.executor }}-${{ matrix.use-standard-naming }} + name: "\ + kind-logs-${{ matrix.kubernetes-combo }}-${{ matrix.executor }}-\ + ${{ matrix.use-standard-naming }}" path: /tmp/kind_logs_* retention-days: 7 - - name: Upload test resource logs on failure ${{ inputs.kubernetes-combos-list-as-string }} + - name: "\ + Upload test resource logs on failure ${{ matrix.executor }}-${{ matrix.kubernetes-combo }}-\ + ${{ matrix.use-standard-naming }}" uses: actions/upload-artifact@v4 if: failure() || cancelled() with: - name: k8s-test-resources-${{ matrix.executor }}-${{ matrix.use-standard-naming }} + name: "\ + k8s-test-resources-${{ matrix.kubernetes-combo }}-${{ matrix.executor }}-\ + ${{ matrix.use-standard-naming }}" path: /tmp/k8s_test_resources_* retention-days: 7 - name: "Delete clusters just in case they are left" diff --git a/.github/workflows/prod-image-build.yml b/.github/workflows/prod-image-build.yml index df4f24981ff30..0bbe98ad6a082 100644 --- a/.github/workflows/prod-image-build.yml +++ b/.github/workflows/prod-image-build.yml @@ -30,13 +30,6 @@ on: # yamllint disable-line rule:truthy variations. required: true type: string - do-build: - description: > - Whether to actually do the build (true/false). If set to false, the build is done - already in pull-request-target workflow, so we skip it here. - required: false - default: "true" - type: string upload-package-artifact: description: > Whether to upload package artifacts (true/false). If false, the job will rely on artifacts prepared @@ -62,6 +55,11 @@ on: # yamllint disable-line rule:truthy description: "Whether to push image to the registry (true/false)" required: true type: string + upload-image-artifact: + description: "Whether to upload docker image artifact" + required: false + default: "false" + type: string debian-version: description: "Base Debian distribution to use for the build (bookworm)" type: string @@ -74,10 +72,6 @@ on: # yamllint disable-line rule:truthy description: "Whether to use uv to build the image (true/false)" required: true type: string - image-tag: - description: "Tag to set for the image" - required: true - type: string python-versions: description: "JSON-formatted array of Python versions to build images from" required: true @@ -121,7 +115,7 @@ on: # yamllint disable-line rule:truthy jobs: build-prod-packages: - name: "${{ inputs.do-build == 'true' && 'Build' || 'Skip building' }} Airflow and provider packages" + name: "Build Airflow and provider packages" timeout-minutes: 10 runs-on: ${{ fromJSON(inputs.runs-on-as-json-public) }} env: @@ -131,32 +125,25 @@ jobs: - name: "Cleanup repo" shell: bash run: docker run -v "${GITHUB_WORKSPACE}:/workspace" -u 0:0 bash -c "rm -rf /workspace/*" - if: inputs.do-build == 'true' && inputs.upload-package-artifact == 'true' + if: inputs.upload-package-artifact == 'true' - name: "Checkout target branch" uses: actions/checkout@v4 with: persist-credentials: false - - name: "Checkout target commit" - uses: ./.github/actions/checkout_target_commit - with: - target-commit-sha: ${{ inputs.target-commit-sha }} - pull-request-target: ${{ inputs.pull-request-target }} - is-committer-build: ${{ inputs.is-committer-build }} - if: inputs.do-build == 'true' && inputs.upload-package-artifact == 'true' - name: "Cleanup docker" run: ./scripts/ci/cleanup_docker.sh - if: inputs.do-build == 'true' && inputs.upload-package-artifact == 'true' + if: inputs.upload-package-artifact == 'true' - uses: actions/setup-python@v5 with: python-version: "${{ inputs.default-python-version }}" - if: inputs.do-build == 'true' && inputs.upload-package-artifact == 'true' + if: inputs.upload-package-artifact == 'true' - name: "Cleanup dist and context file" shell: bash run: rm -fv ./dist/* ./docker-context-files/* - if: inputs.do-build == 'true' && inputs.upload-package-artifact == 'true' + if: inputs.upload-package-artifact == 'true' - name: "Install Breeze" uses: ./.github/actions/breeze - if: inputs.do-build == 'true' && inputs.upload-package-artifact == 'true' + if: inputs.upload-package-artifact == 'true' - name: "Prepare providers packages" shell: bash run: > @@ -164,7 +151,6 @@ jobs: --package-list-file ./prod_image_installed_providers.txt --package-format wheel if: > - inputs.do-build == 'true' && inputs.upload-package-artifact == 'true' && inputs.build-provider-packages == 'true' - name: "Prepare chicken-eggs provider packages" @@ -173,19 +159,18 @@ jobs: breeze release-management prepare-provider-packages --package-format wheel ${{ inputs.chicken-egg-providers }} if: > - inputs.do-build == 'true' && inputs.upload-package-artifact == 'true' && inputs.chicken-egg-providers != '' - name: "Prepare airflow package" shell: bash run: > breeze release-management prepare-airflow-package --package-format wheel - if: inputs.do-build == 'true' && inputs.upload-package-artifact == 'true' + if: inputs.upload-package-artifact == 'true' - name: "Prepare task-sdk package" shell: bash run: > breeze release-management prepare-task-sdk-package --package-format wheel - if: inputs.do-build == 'true' && inputs.upload-package-artifact == 'true' + if: inputs.upload-package-artifact == 'true' - name: "Upload prepared packages as artifacts" uses: actions/upload-artifact@v4 with: @@ -193,25 +178,21 @@ jobs: path: ./dist retention-days: 7 if-no-files-found: error - if: inputs.do-build == 'true' && inputs.upload-package-artifact == 'true' + if: inputs.upload-package-artifact == 'true' build-prod-images: strategy: fail-fast: false matrix: - # yamllint disable-line rule:line-length - python-version: ${{ inputs.do-build == 'true' && fromJSON(inputs.python-versions) || fromJSON('[""]') }} + python-version: ${{ fromJSON(inputs.python-versions) || fromJSON('[""]') }} timeout-minutes: 80 - name: "\ -${{ inputs.do-build == 'true' && 'Build' || 'Skip building' }} \ -PROD ${{ inputs.build-type }} image\ -${{ matrix.python-version }}${{ inputs.do-build == 'true' && ':' || '' }}\ -${{ inputs.do-build == 'true' && inputs.image-tag || '' }}" + name: "Build PROD ${{ inputs.build-type }} image ${{ matrix.python-version }}" runs-on: ${{ fromJSON(inputs.runs-on-as-json-public) }} needs: - build-prod-packages env: BACKEND: sqlite + PYTHON_MAJOR_MINOR_VERSION: "${{ matrix.python-version }}" DEFAULT_BRANCH: ${{ inputs.branch }} DEFAULT_CONSTRAINTS_BRANCH: ${{ inputs.constraints-branch }} VERSION_SUFFIX_FOR_PYPI: ${{ inputs.branch == 'main' && 'dev0' || '' }} @@ -231,57 +212,45 @@ ${{ inputs.do-build == 'true' && inputs.image-tag || '' }}" - name: "Cleanup repo" shell: bash run: docker run -v "${GITHUB_WORKSPACE}:/workspace" -u 0:0 bash -c "rm -rf /workspace/*" - if: inputs.do-build == 'true' - name: "Checkout target branch" uses: actions/checkout@v4 with: persist-credentials: false - - name: "Checkout target commit" - uses: ./.github/actions/checkout_target_commit - with: - target-commit-sha: ${{ inputs.target-commit-sha }} - pull-request-target: ${{ inputs.pull-request-target }} - is-committer-build: ${{ inputs.is-committer-build }} - if: inputs.do-build == 'true' - name: "Cleanup docker" run: ./scripts/ci/cleanup_docker.sh - if: inputs.do-build == 'true' - name: "Install Breeze" uses: ./.github/actions/breeze - if: inputs.do-build == 'true' - - name: "Regenerate dependencies in case they was modified manually so that we can build an image" - shell: bash - run: | - pip install rich>=12.4.4 pyyaml - python scripts/ci/pre_commit/update_providers_dependencies.py - if: inputs.do-build == 'true' && inputs.upgrade-to-newer-dependencies != 'false' - name: "Cleanup dist and context file" shell: bash run: rm -fv ./dist/* ./docker-context-files/* - if: inputs.do-build == 'true' + - name: "Restore PROD docker image ${{ inputs.platform }}:${{ env.PYTHON_MAJOR_MINOR_VERSION }}" + uses: apache/infrastructure-actions/stash/restore@c94b890bbedc2fc61466d28e6bd9966bc6c6643c + with: + key: "prod-image-save-${{ inputs.platform }}-${{ env.PYTHON_MAJOR_MINOR_VERSION }}" + path: "/tmp/" + id: restore-prod-image + - name: "Load PROD image ${{ inputs.platform }}:${{ env.PYTHON_MAJOR_MINOR_VERSION }}" + run: breeze prod-image load --platform ${{ inputs.platform }} + shell: bash + if: steps.restore-prod-image.stash-hit == 'true' - name: "Download packages prepared as artifacts" uses: actions/download-artifact@v4 with: name: prod-packages path: ./docker-context-files - if: inputs.do-build == 'true' - name: "Download constraints" uses: actions/download-artifact@v4 with: name: constraints path: ./docker-context-files - if: inputs.do-build == 'true' - - name: Login to ghcr.io - shell: bash - run: echo "${{ env.GITHUB_TOKEN }}" | docker login ghcr.io -u ${{ github.actor }} --password-stdin - if: inputs.do-build == 'true' - - name: "Build PROD images w/ source providers ${{ matrix.python-version }}:${{ inputs.image-tag }}" + - name: "Build PROD images w/ source providers ${{ env.PYTHON_MAJOR_MINOR_VERSION }}" shell: bash run: > - breeze prod-image build --tag-as-latest --image-tag "${{ inputs.image-tag }}" + breeze prod-image build + --docker-cache local --commit-sha "${{ github.sha }}" --install-packages-from-context --airflow-constraints-mode constraints-source-providers - --use-constraints-for-context-packages --python "${{ matrix.python-version }}" + --use-constraints-for-context-packages env: PUSH: ${{ inputs.push-image }} DOCKER_CACHE: ${{ inputs.docker-cache }} @@ -290,14 +259,14 @@ ${{ inputs.do-build == 'true' && inputs.image-tag || '' }}" INSTALL_MYSQL_CLIENT_TYPE: ${{ inputs.install-mysql-client-type }} UPGRADE_TO_NEWER_DEPENDENCIES: ${{ inputs.upgrade-to-newer-dependencies }} INCLUDE_NOT_READY_PROVIDERS: "true" - if: inputs.do-build == 'true' && inputs.build-provider-packages == 'true' - - name: "Build PROD images with PyPi providers ${{ matrix.python-version }}:${{ inputs.image-tag }}" + if: inputs.build-provider-packages == 'true' + - name: "Build PROD images with PyPi providers ${{ env.PYTHON_MAJOR_MINOR_VERSION }}" shell: bash run: > - breeze prod-image build --builder airflow_cache --tag-as-latest - --image-tag "${{ inputs.image-tag }}" --commit-sha "${{ github.sha }}" + breeze prod-image build --builder airflow_cache + --commit-sha "${{ github.sha }}" --install-packages-from-context --airflow-constraints-mode constraints - --use-constraints-for-context-packages --python "${{ matrix.python-version }}" + --use-constraints-for-context-packages env: PUSH: ${{ inputs.push-image }} DOCKER_CACHE: ${{ inputs.docker-cache }} @@ -306,9 +275,18 @@ ${{ inputs.do-build == 'true' && inputs.image-tag || '' }}" INSTALL_MYSQL_CLIENT_TYPE: ${{ inputs.install-mysql-client-type }} UPGRADE_TO_NEWER_DEPENDENCIES: ${{ inputs.upgrade-to-newer-dependencies }} INCLUDE_NOT_READY_PROVIDERS: "true" - if: inputs.do-build == 'true' && inputs.build-provider-packages != 'true' - - name: Verify PROD image ${{ matrix.python-version }}:${{ inputs.image-tag }} + if: inputs.build-provider-packages != 'true' + - name: Verify PROD image ${{ env.PYTHON_MAJOR_MINOR_VERSION }} + run: breeze prod-image verify + - name: "Export PROD docker image ${{ env.PYTHON_MAJOR_MINOR_VERSION }}" run: > - breeze prod-image verify --image-tag "${{ inputs.image-tag }}" - --python "${{ matrix.python-version }}" - if: inputs.do-build == 'true' + breeze prod-image save --platform "${{ inputs.platform }}" + if: inputs.upload-image-artifact == 'true' + - name: "Stash PROD docker image ${{ env.PYTHON_MAJOR_MINOR_VERSION }}" + uses: apache/infrastructure-actions/stash/save@c94b890bbedc2fc61466d28e6bd9966bc6c6643c + with: + key: "prod-image-save-${{ inputs.platform }}-${{ env.PYTHON_MAJOR_MINOR_VERSION }}" + path: "/tmp/prod-image-save-*-${{ env.PYTHON_MAJOR_MINOR_VERSION }}.tar" + if-no-files-found: 'error' + retention-days: 2 + if: inputs.upload-image-artifact == 'true' diff --git a/.github/workflows/prod-image-extra-checks.yml b/.github/workflows/prod-image-extra-checks.yml index bb63faef7b243..33ebc4700dad3 100644 --- a/.github/workflows/prod-image-extra-checks.yml +++ b/.github/workflows/prod-image-extra-checks.yml @@ -40,9 +40,6 @@ on: # yamllint disable-line rule:truthy description: "Whether to use uv to build the image (true/false)" required: true type: string - image-tag: - required: true - type: string build-provider-packages: description: "Whether to build provider packages (true/false). If false providers are from PyPI" required: true @@ -74,7 +71,6 @@ jobs: runs-on-as-json-public: ${{ inputs.runs-on-as-json-public }} build-type: "MySQL Client" upload-package-artifact: "false" - image-tag: mysql-${{ inputs.image-tag }} install-mysql-client-type: "mysql" python-versions: ${{ inputs.python-versions }} default-python-version: ${{ inputs.default-python-version }} @@ -98,7 +94,6 @@ jobs: runs-on-as-json-public: ${{ inputs.runs-on-as-json-public }} build-type: "pip" upload-package-artifact: "false" - image-tag: mysql-${{ inputs.image-tag }} install-mysql-client-type: "mysql" python-versions: ${{ inputs.python-versions }} default-python-version: ${{ inputs.default-python-version }} diff --git a/.github/workflows/push-image-cache.yml b/.github/workflows/push-image-cache.yml index 10a33275ad3f3..f36ad164c3093 100644 --- a/.github/workflows/push-image-cache.yml +++ b/.github/workflows/push-image-cache.yml @@ -110,6 +110,7 @@ jobs: GITHUB_USERNAME: ${{ github.actor }} INCLUDE_SUCCESS_OUTPUTS: "${{ inputs.include-success-outputs }}" INSTALL_MYSQL_CLIENT_TYPE: ${{ inputs.install-mysql-client-type }} + PYTHON_MAJOR_MINOR_VERSION: "${{ matrix.python }}" USE_UV: ${{ inputs.use-uv }} UPGRADE_TO_NEWER_DEPENDENCIES: "false" VERBOSE: "true" @@ -124,24 +125,36 @@ jobs: persist-credentials: false - name: "Cleanup docker" run: ./scripts/ci/cleanup_docker.sh - - name: "Install Breeze" - uses: ./.github/actions/breeze + - name: "Prepare breeze & CI image: ${{ inputs.platform }}:${{ env.PYTHON_MAJOR_MINOR_VERSION }}" + uses: ./.github/actions/prepare_breeze_and_image + with: + platform: ${{ inputs.platform }} + python: ${{ env.PYTHON_MAJOR_MINOR_VERSION }} - name: "Start ARM instance" run: ./scripts/ci/images/ci_start_arm_instance_and_connect_to_docker.sh if: inputs.platform == 'linux/arm64' - name: Login to ghcr.io run: echo "${{ env.GITHUB_TOKEN }}" | docker login ghcr.io -u ${{ github.actor }} --password-stdin - - name: "Push CI ${{ inputs.cache-type }} cache: ${{ matrix.python }} ${{ inputs.platform }}" + # yamllint disable-line rule:line-length + - name: "Push CI ${{ inputs.cache-type }} cache:${{ env.PYTHON_MAJOR_MINOR_VERSION }}:${{ inputs.platform }}" run: > - breeze ci-image build --builder airflow_cache --prepare-buildx-cache - --platform "${{ inputs.platform }}" --python ${{ matrix.python }} + breeze ci-image build + --builder airflow_cache + --prepare-buildx-cache + --docker-cache local + --platform "${{ inputs.platform }}" + --push - name: "Stop ARM instance" run: ./scripts/ci/images/ci_stop_arm_instance.sh if: always() && inputs.platform == 'linux/arm64' - - name: "Push CI latest images: ${{ matrix.python }} (linux/amd64 only)" + - name: "Push CI latest images: ${{ env.PYTHON_MAJOR_MINOR_VERSION }} (linux/amd64 only)" run: > - breeze ci-image build --builder airflow_cache --push - --python "${{ matrix.python }}" --platform "${{ inputs.platform }}" + breeze + ci-image build + --builder airflow_cache + --docker-cache local + --platform "${{ inputs.platform }}" + --push if: inputs.push-latest-images == 'true' && inputs.platform == 'linux/amd64' push-prod-image-cache: @@ -172,6 +185,7 @@ jobs: GITHUB_TOKEN: ${{ secrets.GITHUB_TOKEN }} GITHUB_USERNAME: ${{ github.actor }} INSTALL_MYSQL_CLIENT_TYPE: ${{ inputs.install-mysql-client-type }} + PYTHON_MAJOR_MINOR_VERSION: "${{ matrix.python }}" UPGRADE_TO_NEWER_DEPENDENCIES: "false" USE_UV: ${{ inputs.branch == 'main' && inputs.use-uv || 'false' }} VERBOSE: "true" @@ -187,8 +201,12 @@ jobs: persist-credentials: false - name: "Cleanup docker" run: ./scripts/ci/cleanup_docker.sh - - name: "Install Breeze" - uses: ./.github/actions/breeze + - name: "Prepare breeze & PROD image: ${{ inputs.platform }}:${{ env.PYTHON_MAJOR_MINOR_VERSION }}" + uses: ./.github/actions/prepare_breeze_and_image + with: + platform: ${{ inputs.platform }} + python: ${{ env.PYTHON_MAJOR_MINOR_VERSION }} + image-type: "PROD" - name: "Cleanup dist and context file" run: rm -fv ./dist/* ./docker-context-files/* - name: "Download packages prepared as artifacts" @@ -201,20 +219,30 @@ jobs: if: inputs.platform == 'linux/arm64' - name: Login to ghcr.io run: echo "${{ env.GITHUB_TOKEN }}" | docker login ghcr.io -u ${{ github.actor }} --password-stdin - - name: "Push PROD ${{ inputs.cache-type }} cache: ${{ matrix.python-version }} ${{ inputs.platform }}" + # yamllint disable-line rule:line-length + - name: "Push PROD ${{ inputs.cache-type }} cache: ${{ env.PYTHON_MAJOR_MINOR_VERSION }} ${{ inputs.platform }}" run: > - breeze prod-image build --builder airflow_cache - --prepare-buildx-cache --platform "${{ inputs.platform }}" - --install-packages-from-context --airflow-constraints-mode constraints-source-providers - --python ${{ matrix.python }} + breeze prod-image build + --builder airflow_cache + --prepare-buildx-cache + --install-packages-from-context + --docker-cache local + --platform "${{ inputs.platform }}" + --airflow-constraints-mode constraints-source-providers + --push - name: "Stop ARM instance" run: ./scripts/ci/images/ci_stop_arm_instance.sh if: always() && inputs.platform == 'linux/arm64' # We only push "AMD" images as it is really only needed for any kind of automated builds in CI # and currently there is not an easy way to make multi-platform image from two separate builds # and we can do it after we stopped the ARM instance as it is not needed anymore - - name: "Push PROD latest image: ${{ matrix.python }} (linux/amd64 ONLY)" + - name: "Push PROD latest image: ${{ env.PYTHON_MAJOR_MINOR_VERSION }} (linux/amd64 ONLY)" run: > - breeze prod-image build --builder airflow_cache --install-packages-from-context - --push --platform "${{ inputs.platform }}" + breeze prod-image build + --builder airflow_cache + --install-packages-from-context + --docker-cache local + --platform "${{ inputs.platform }}" + --airflow-constraints-mode constraints-source-providers + --push if: inputs.push-latest-images == 'true' && inputs.platform == 'linux/amd64' diff --git a/.github/workflows/release_dockerhub_image.yml b/.github/workflows/release_dockerhub_image.yml index 5ce1585131f76..46705c6a106fa 100644 --- a/.github/workflows/release_dockerhub_image.yml +++ b/.github/workflows/release_dockerhub_image.yml @@ -47,7 +47,7 @@ jobs: defaultPythonVersion: ${{ steps.selective-checks.outputs.default-python-version }} chicken-egg-providers: ${{ steps.selective-checks.outputs.chicken-egg-providers }} skipLatest: ${{ github.event.inputs.skipLatest == '' && ' ' || '--skip-latest' }} - limitPlatform: ${{ github.repository == 'apache/airflow' && ' ' || '--limit-platform linux/amd64' }} + limitPlatform: ${{ github.repository == 'apache/airflow' && ' ' || '--limit-platform amd64' }} env: GITHUB_CONTEXT: ${{ toJson(github) }} VERBOSE: true diff --git a/.github/workflows/run-unit-tests.yml b/.github/workflows/run-unit-tests.yml index 6b491f6bff4ab..3a9440e528eaf 100644 --- a/.github/workflows/run-unit-tests.yml +++ b/.github/workflows/run-unit-tests.yml @@ -45,10 +45,6 @@ on: # yamllint disable-line rule:truthy required: false default: ":" type: string - image-tag: - description: "Tag to set for the image" - required: true - type: string python-versions: description: "The list of python versions (stringified JSON array) to run the tests on." required: true @@ -144,7 +140,6 @@ jobs: GITHUB_REPOSITORY: ${{ github.repository }} GITHUB_TOKEN: ${{ secrets.GITHUB_TOKEN }} GITHUB_USERNAME: ${{ github.actor }} - IMAGE_TAG: "${{ inputs.image-tag }}" INCLUDE_SUCCESS_OUTPUTS: ${{ inputs.include-success-outputs }} # yamllint disable rule:line-length JOB_ID: "${{ matrix.test-group }}-${{ inputs.test-scope }}-${{ inputs.test-name }}-${{inputs.backend}}-${{ matrix.backend-version }}-${{ matrix.python-version }}" @@ -163,10 +158,11 @@ jobs: uses: actions/checkout@v4 with: persist-credentials: false - - name: "Cleanup docker" - run: ./scripts/ci/cleanup_docker.sh - - name: "Prepare breeze & CI image: ${{matrix.python-version}}:${{ inputs.image-tag }}" + - name: "Prepare breeze & CI image: ${{ matrix.python-version }}" uses: ./.github/actions/prepare_breeze_and_image + with: + platform: "linux/amd64" + python: ${{ matrix.python-version }} - name: > Migration Tests: ${{ matrix.python-version }}:${{ env.PARALLEL_TEST_TYPES }} uses: ./.github/actions/migration_tests diff --git a/.github/workflows/special-tests.yml b/.github/workflows/special-tests.yml index decc7271b728b..d416d55575fb9 100644 --- a/.github/workflows/special-tests.yml +++ b/.github/workflows/special-tests.yml @@ -32,10 +32,6 @@ on: # yamllint disable-line rule:truthy description: "The json representing list of test test groups to run" required: true type: string - image-tag: - description: "Tag to set for the image" - required: true - type: string core-test-types-list-as-string: description: "The list of core test types to run separated by spaces" required: true @@ -96,7 +92,6 @@ jobs: test-scope: "DB" test-groups: ${{ inputs.test-groups }} backend: "postgres" - image-tag: ${{ inputs.image-tag }} python-versions: "['${{ inputs.default-python-version }}']" backend-versions: "['${{ inputs.default-postgres-version }}']" excluded-providers-as-string: ${{ inputs.excluded-providers-as-string }} @@ -120,7 +115,6 @@ jobs: test-scope: "All" test-groups: ${{ inputs.test-groups }} backend: "postgres" - image-tag: ${{ inputs.image-tag }} python-versions: "['${{ inputs.default-python-version }}']" backend-versions: "['${{ inputs.default-postgres-version }}']" excluded-providers-as-string: ${{ inputs.excluded-providers-as-string }} @@ -145,7 +139,6 @@ jobs: test-scope: "All" test-groups: ${{ inputs.test-groups }} backend: "postgres" - image-tag: ${{ inputs.image-tag }} python-versions: "['${{ inputs.default-python-version }}']" backend-versions: "['${{ inputs.default-postgres-version }}']" excluded-providers-as-string: ${{ inputs.excluded-providers-as-string }} @@ -169,7 +162,6 @@ jobs: test-scope: "Quarantined" test-groups: ${{ inputs.test-groups }} backend: "postgres" - image-tag: ${{ inputs.image-tag }} python-versions: "['${{ inputs.default-python-version }}']" backend-versions: "['${{ inputs.default-postgres-version }}']" excluded-providers-as-string: ${{ inputs.excluded-providers-as-string }} @@ -193,7 +185,6 @@ jobs: test-scope: "ARM collection" test-groups: ${{ inputs.test-groups }} backend: "postgres" - image-tag: ${{ inputs.image-tag }} python-versions: "['${{ inputs.default-python-version }}']" backend-versions: "['${{ inputs.default-postgres-version }}']" excluded-providers-as-string: ${{ inputs.excluded-providers-as-string }} @@ -218,7 +209,6 @@ jobs: test-scope: "System" test-groups: ${{ inputs.test-groups }} backend: "postgres" - image-tag: ${{ inputs.image-tag }} python-versions: "['${{ inputs.default-python-version }}']" backend-versions: "['${{ inputs.default-postgres-version }}']" excluded-providers-as-string: ${{ inputs.excluded-providers-as-string }} diff --git a/.github/workflows/task-sdk-tests.yml b/.github/workflows/task-sdk-tests.yml index acc9872e6ed96..756a66e546479 100644 --- a/.github/workflows/task-sdk-tests.yml +++ b/.github/workflows/task-sdk-tests.yml @@ -24,10 +24,6 @@ on: # yamllint disable-line rule:truthy description: "The array of labels (in json form) determining default runner used for the build." required: true type: string - image-tag: - description: "Tag to set for the image" - required: true - type: string default-python-version: description: "Which version of python should be used by default" required: true @@ -53,7 +49,6 @@ jobs: GITHUB_REPOSITORY: ${{ github.repository }} GITHUB_TOKEN: ${{ secrets.GITHUB_TOKEN }} GITHUB_USERNAME: ${{ github.actor }} - IMAGE_TAG: "${{ inputs.image-tag }}" INCLUDE_NOT_READY_PROVIDERS: "true" PYTHON_MAJOR_MINOR_VERSION: "${{ inputs.default-python-version }}" VERBOSE: "true" @@ -66,10 +61,11 @@ jobs: uses: actions/checkout@v4 with: persist-credentials: false - - name: "Cleanup docker" - run: ./scripts/ci/cleanup_docker.sh - - name: "Prepare breeze & CI image: ${{ matrix.python-version }}:${{ inputs.image-tag }}" + - name: "Prepare breeze & CI image: ${{ matrix.python-version }}" uses: ./.github/actions/prepare_breeze_and_image + with: + platform: "linux/amd64" + python: ${{ matrix.python-version }} - name: "Cleanup dist files" run: rm -fv ./dist/* - name: "Prepare Task SDK packages: wheel" diff --git a/.github/workflows/test-provider-packages.yml b/.github/workflows/test-provider-packages.yml index 08715af6b58ba..d9716c164f3a7 100644 --- a/.github/workflows/test-provider-packages.yml +++ b/.github/workflows/test-provider-packages.yml @@ -24,10 +24,6 @@ on: # yamllint disable-line rule:truthy description: "The array of labels (in json form) determining default runner used for the build." required: true type: string - image-tag: - description: "Tag to set for the image" - required: true - type: string canary-run: description: "Whether this is a canary run" required: true @@ -75,7 +71,6 @@ jobs: GITHUB_REPOSITORY: ${{ github.repository }} GITHUB_TOKEN: ${{ secrets.GITHUB_TOKEN }} GITHUB_USERNAME: ${{ github.actor }} - IMAGE_TAG: "${{ inputs.image-tag }}" INCLUDE_NOT_READY_PROVIDERS: "true" PYTHON_MAJOR_MINOR_VERSION: "${{ inputs.default-python-version }}" VERBOSE: "true" @@ -87,11 +82,11 @@ jobs: uses: actions/checkout@v4 with: persist-credentials: false - - name: "Cleanup docker" - run: ./scripts/ci/cleanup_docker.sh - - name: > - Prepare breeze & CI image: ${{ inputs.default-python-version }}:${{ inputs.image-tag }} + - name: "Prepare breeze & CI image: ${{ inputs.default-python-version }}" uses: ./.github/actions/prepare_breeze_and_image + with: + platform: "linux/amd64" + python: ${{ inputs.default-python-version }} - name: "Cleanup dist files" run: rm -fv ./dist/* - name: "Prepare provider documentation" @@ -161,9 +156,8 @@ jobs: GITHUB_REPOSITORY: ${{ github.repository }} GITHUB_TOKEN: ${{ secrets.GITHUB_TOKEN }} GITHUB_USERNAME: ${{ github.actor }} - IMAGE_TAG: "${{ inputs.image-tag }}" INCLUDE_NOT_READY_PROVIDERS: "true" - PYTHON_MAJOR_MINOR_VERSION: "${{ inputs.default-python-version }}" + PYTHON_MAJOR_MINOR_VERSION: "${{ matrix.python-version }}" VERSION_SUFFIX_FOR_PYPI: "dev0" VERBOSE: "true" CLEAN_AIRFLOW_INSTALLATION: "${{ inputs.canary-run }}" @@ -176,10 +170,11 @@ jobs: uses: actions/checkout@v4 with: persist-credentials: false - - name: "Cleanup docker" - run: ./scripts/ci/cleanup_docker.sh - - name: "Prepare breeze & CI image: ${{ matrix.python-version }}:${{ inputs.image-tag }}" + - name: "Prepare breeze & CI image: ${{ matrix.python-version }}" uses: ./.github/actions/prepare_breeze_and_image + with: + platform: "linux/amd64" + python: ${{ matrix.python-version }} - name: "Cleanup dist files" run: rm -fv ./dist/* - name: "Prepare provider packages: wheel" diff --git a/Dockerfile b/Dockerfile index fe49db186479d..f32fbef633bc4 100644 --- a/Dockerfile +++ b/Dockerfile @@ -422,85 +422,6 @@ common::show_packaging_tool_version_and_location common::install_packaging_tools EOF -# The content below is automatically copied from scripts/docker/install_airflow_dependencies_from_branch_tip.sh -COPY <<"EOF" /install_airflow_dependencies_from_branch_tip.sh -#!/usr/bin/env bash - -. "$( dirname "${BASH_SOURCE[0]}" )/common.sh" - -: "${AIRFLOW_REPO:?Should be set}" -: "${AIRFLOW_BRANCH:?Should be set}" -: "${INSTALL_MYSQL_CLIENT:?Should be true or false}" -: "${INSTALL_POSTGRES_CLIENT:?Should be true or false}" - -function install_airflow_dependencies_from_branch_tip() { - echo - echo "${COLOR_BLUE}Installing airflow from ${AIRFLOW_BRANCH}. It is used to cache dependencies${COLOR_RESET}" - echo - if [[ ${INSTALL_MYSQL_CLIENT} != "true" ]]; then - AIRFLOW_EXTRAS=${AIRFLOW_EXTRAS/mysql,} - fi - if [[ ${INSTALL_POSTGRES_CLIENT} != "true" ]]; then - AIRFLOW_EXTRAS=${AIRFLOW_EXTRAS/postgres,} - fi - local TEMP_AIRFLOW_DIR - TEMP_AIRFLOW_DIR=$(mktemp -d) - # Install latest set of dependencies - without constraints. This is to download a "base" set of - # dependencies that we can cache and reuse when installing airflow using constraints and latest - # pyproject.toml in the next step (when we install regular airflow). - set -x - curl -fsSL "https://github.com/${AIRFLOW_REPO}/archive/${AIRFLOW_BRANCH}.tar.gz" | \ - tar xz -C "${TEMP_AIRFLOW_DIR}" --strip 1 - # Make sure editable dependencies are calculated when devel-ci dependencies are installed - ${PACKAGING_TOOL_CMD} install ${EXTRA_INSTALL_FLAGS} ${ADDITIONAL_PIP_INSTALL_FLAGS} \ - --editable "${TEMP_AIRFLOW_DIR}[${AIRFLOW_EXTRAS}]" - set +x - common::install_packaging_tools - set -x - echo "${COLOR_BLUE}Uninstalling providers. Dependencies remain${COLOR_RESET}" - # Uninstall airflow and providers to keep only the dependencies. In the future when - # planned https://github.com/pypa/pip/issues/11440 is implemented in pip we might be able to use this - # flag and skip the remove step. - pip freeze | grep apache-airflow-providers | xargs ${PACKAGING_TOOL_CMD} uninstall ${EXTRA_UNINSTALL_FLAGS} || true - set +x - echo - echo "${COLOR_BLUE}Uninstalling just airflow. Dependencies remain. Now target airflow can be reinstalled using mostly cached dependencies${COLOR_RESET}" - echo - set +x - ${PACKAGING_TOOL_CMD} uninstall ${EXTRA_UNINSTALL_FLAGS} apache-airflow - rm -rf "${TEMP_AIRFLOW_DIR}" - set -x - # If you want to make sure dependency is removed from cache in your PR when you removed it from - # pyproject.toml - please add your dependency here as a list of strings - # for example: - # DEPENDENCIES_TO_REMOVE=("package_a" "package_b") - # Once your PR is merged, you should make a follow-up PR to remove it from this list - # and increase the AIRFLOW_CI_BUILD_EPOCH in Dockerfile.ci to make sure your cache is rebuilt. - local DEPENDENCIES_TO_REMOVE - # IMPORTANT!! Make sure to increase AIRFLOW_CI_BUILD_EPOCH in Dockerfile.ci when you remove a dependency from that list - DEPENDENCIES_TO_REMOVE=() - if [[ "${DEPENDENCIES_TO_REMOVE[*]}" != "" ]]; then - echo - echo "${COLOR_BLUE}Uninstalling just removed dependencies (temporary until cache refreshes)${COLOR_RESET}" - echo "${COLOR_BLUE}Dependencies to uninstall: ${DEPENDENCIES_TO_REMOVE[*]}${COLOR_RESET}" - echo - set +x - ${PACKAGING_TOOL_CMD} uninstall "${DEPENDENCIES_TO_REMOVE[@]}" || true - set -x - # make sure that the dependency is not needed by something else - pip check - fi -} - -common::get_colors -common::get_packaging_tool -common::get_airflow_version_specification -common::get_constraints_location -common::show_packaging_tool_version_and_location - -install_airflow_dependencies_from_branch_tip -EOF - # The content below is automatically copied from scripts/docker/common.sh COPY <<"EOF" /common.sh #!/usr/bin/env bash @@ -524,8 +445,6 @@ function common::get_packaging_tool() { ## IMPORTANT: IF YOU MODIFY THIS FUNCTION YOU SHOULD ALSO MODIFY CORRESPONDING FUNCTION IN ## `scripts/in_container/_in_container_utils.sh` - local PYTHON_BIN - PYTHON_BIN=$(which python) if [[ ${AIRFLOW_USE_UV} == "true" ]]; then echo echo "${COLOR_BLUE}Using 'uv' to install Airflow${COLOR_RESET}" @@ -533,8 +452,8 @@ function common::get_packaging_tool() { export PACKAGING_TOOL="uv" export PACKAGING_TOOL_CMD="uv pip" if [[ -z ${VIRTUAL_ENV=} ]]; then - export EXTRA_INSTALL_FLAGS="--python ${PYTHON_BIN}" - export EXTRA_UNINSTALL_FLAGS="--python ${PYTHON_BIN}" + export EXTRA_INSTALL_FLAGS="--system" + export EXTRA_UNINSTALL_FLAGS="--system" else export EXTRA_INSTALL_FLAGS="" export EXTRA_UNINSTALL_FLAGS="" @@ -900,18 +819,12 @@ function install_airflow() { # Determine the installation_command_flags based on AIRFLOW_INSTALLATION_METHOD method local installation_command_flags if [[ ${AIRFLOW_INSTALLATION_METHOD} == "." ]]; then - # We need _a_ file in there otherwise the editable install doesn't include anything in the .pth file - mkdir -p ./providers/src/airflow/providers/ - touch ./providers/src/airflow/providers/__init__.py - - # Similarly we need _a_ file for task_sdk too - mkdir -p ./task_sdk/src/airflow/sdk/ - echo '__version__ = "0.0.0dev0"' > ./task_sdk/src/airflow/sdk/__init__.py - - trap 'rm -f ./providers/src/airflow/providers/__init__.py ./task_sdk/src/airflow/__init__.py 2>/dev/null' EXIT - # When installing from sources - we always use `--editable` mode - installation_command_flags="--editable .[${AIRFLOW_EXTRAS}]${AIRFLOW_VERSION_SPECIFICATION} --editable ./providers --editable ./task_sdk" + installation_command_flags="--editable .[${AIRFLOW_EXTRAS}]${AIRFLOW_VERSION_SPECIFICATION} --editable ./task_sdk" + while IFS= read -r -d '' pyproject_toml_file; do + project_folder=$(dirname ${pyproject_toml_file}) + installation_command_flags="${installation_command_flags} --editable ${project_folder}" + done < <(find "providers" -name "pyproject.toml" -print0) elif [[ ${AIRFLOW_INSTALLATION_METHOD} == "apache-airflow" ]]; then installation_command_flags="apache-airflow[${AIRFLOW_EXTRAS}]${AIRFLOW_VERSION_SPECIFICATION}" elif [[ ${AIRFLOW_INSTALLATION_METHOD} == apache-airflow\ @\ * ]]; then @@ -1407,7 +1320,8 @@ ARG PYTHON_BASE_IMAGE ENV PYTHON_BASE_IMAGE=${PYTHON_BASE_IMAGE} \ DEBIAN_FRONTEND=noninteractive LANGUAGE=C.UTF-8 LANG=C.UTF-8 LC_ALL=C.UTF-8 \ LC_CTYPE=C.UTF-8 LC_MESSAGES=C.UTF-8 \ - PIP_CACHE_DIR=/tmp/.cache/pip + PIP_CACHE_DIR=/tmp/.cache/pip \ + UV_CACHE_DIR=/tmp/.cache/uv ARG DEV_APT_DEPS="" ARG ADDITIONAL_DEV_APT_DEPS="" @@ -1473,9 +1387,6 @@ ARG DEFAULT_CONSTRAINTS_BRANCH="constraints-main" # By default PIP has progress bar but you can disable it. ARG PIP_PROGRESS_BAR -# By default we do not use pre-cached packages, but in CI/Breeze environment we override this to speed up -# builds in case pyproject.toml changed. This is pure optimisation of CI/Breeze builds. -ARG AIRFLOW_PRE_CACHED_PIP_PACKAGES="false" # This is airflow version that is put in the label of the image build ARG AIRFLOW_VERSION # By default latest released version of airflow is installed (when empty) but this value can be overridden @@ -1513,7 +1424,6 @@ ENV AIRFLOW_PIP_VERSION=${AIRFLOW_PIP_VERSION} \ AIRFLOW_UV_VERSION=${AIRFLOW_UV_VERSION} \ UV_HTTP_TIMEOUT=${UV_HTTP_TIMEOUT} \ AIRFLOW_USE_UV=${AIRFLOW_USE_UV} \ - AIRFLOW_PRE_CACHED_PIP_PACKAGES=${AIRFLOW_PRE_CACHED_PIP_PACKAGES} \ AIRFLOW_VERSION=${AIRFLOW_VERSION} \ AIRFLOW_INSTALLATION_METHOD=${AIRFLOW_INSTALLATION_METHOD} \ AIRFLOW_VERSION_SPECIFICATION=${AIRFLOW_VERSION_SPECIFICATION} \ @@ -1538,8 +1448,7 @@ ENV AIRFLOW_PIP_VERSION=${AIRFLOW_PIP_VERSION} \ # Copy all scripts required for installation - changing any of those should lead to # rebuilding from here -COPY --from=scripts common.sh install_packaging_tools.sh \ - install_airflow_dependencies_from_branch_tip.sh create_prod_venv.sh /scripts/docker/ +COPY --from=scripts common.sh install_packaging_tools.sh create_prod_venv.sh /scripts/docker/ # We can set this value to true in case we want to install .whl/.tar.gz packages placed in the # docker-context-files folder. This can be done for both additional packages you want to install @@ -1569,13 +1478,7 @@ ENV AIRFLOW_CI_BUILD_EPOCH=${AIRFLOW_CI_BUILD_EPOCH} # By default PIP installs everything to ~/.local and it's also treated as VIRTUALENV ENV VIRTUAL_ENV="${AIRFLOW_USER_HOME_DIR}/.local" -RUN bash /scripts/docker/install_packaging_tools.sh; \ - bash /scripts/docker/create_prod_venv.sh; \ - if [[ ${AIRFLOW_PRE_CACHED_PIP_PACKAGES} == "true" && \ - ${INSTALL_PACKAGES_FROM_CONTEXT} == "false" && \ - ${UPGRADE_INVALIDATION_STRING} == "" ]]; then \ - bash /scripts/docker/install_airflow_dependencies_from_branch_tip.sh; \ - fi +RUN bash /scripts/docker/install_packaging_tools.sh; bash /scripts/docker/create_prod_venv.sh COPY --chown=airflow:0 ${AIRFLOW_SOURCES_FROM} ${AIRFLOW_SOURCES_TO} @@ -1599,10 +1502,10 @@ COPY --from=scripts install_from_docker_context_files.sh install_airflow.sh \ # an incorrect architecture. ARG TARGETARCH # Value to be able to easily change cache id and therefore use a bare new cache -ARG PIP_CACHE_EPOCH="9" +ARG DEPENDENCY_CACHE_EPOCH="9" # hadolint ignore=SC2086, SC2010, DL3042 -RUN --mount=type=cache,id=$PYTHON_BASE_IMAGE-$AIRFLOW_PIP_VERSION-$TARGETARCH-$PIP_CACHE_EPOCH,target=/tmp/.cache/pip,uid=${AIRFLOW_UID} \ +RUN --mount=type=cache,id=prod-$TARGETARCH-$DEPENDENCY_CACHE_EPOCH,target=/tmp/.cache/,uid=${AIRFLOW_UID} \ if [[ ${INSTALL_PACKAGES_FROM_CONTEXT} == "true" ]]; then \ bash /scripts/docker/install_from_docker_context_files.sh; \ fi; \ @@ -1622,7 +1525,7 @@ RUN --mount=type=cache,id=$PYTHON_BASE_IMAGE-$AIRFLOW_PIP_VERSION-$TARGETARCH-$P # during the build additionally to whatever has been installed so far. It is recommended that # the requirements.txt contains only dependencies with == version specification # hadolint ignore=DL3042 -RUN --mount=type=cache,id=additional-requirements-$PYTHON_BASE_IMAGE-$AIRFLOW_PIP_VERSION-$TARGETARCH-$PIP_CACHE_EPOCH,target=/tmp/.cache/pip,uid=${AIRFLOW_UID} \ +RUN --mount=type=cache,id=prod-$TARGETARCH-$DEPENDENCY_CACHE_EPOCH,target=/tmp/.cache/,uid=${AIRFLOW_UID} \ if [[ -f /docker-context-files/requirements.txt ]]; then \ pip install -r /docker-context-files/requirements.txt; \ fi @@ -1650,7 +1553,9 @@ ARG PYTHON_BASE_IMAGE ENV PYTHON_BASE_IMAGE=${PYTHON_BASE_IMAGE} \ # Make sure noninteractive debian install is used and language variables set DEBIAN_FRONTEND=noninteractive LANGUAGE=C.UTF-8 LANG=C.UTF-8 LC_ALL=C.UTF-8 \ - LC_CTYPE=C.UTF-8 LC_MESSAGES=C.UTF-8 LD_LIBRARY_PATH=/usr/local/lib + LC_CTYPE=C.UTF-8 LC_MESSAGES=C.UTF-8 LD_LIBRARY_PATH=/usr/local/lib \ + PIP_CACHE_DIR=/tmp/.cache/pip \ + UV_CACHE_DIR=/tmp/.cache/uv ARG RUNTIME_APT_DEPS="" ARG ADDITIONAL_RUNTIME_APT_DEPS="" diff --git a/Dockerfile.ci b/Dockerfile.ci index 7c0b529d4711f..aefeb81709b98 100644 --- a/Dockerfile.ci +++ b/Dockerfile.ci @@ -363,85 +363,6 @@ common::show_packaging_tool_version_and_location common::install_packaging_tools EOF -# The content below is automatically copied from scripts/docker/install_airflow_dependencies_from_branch_tip.sh -COPY <<"EOF" /install_airflow_dependencies_from_branch_tip.sh -#!/usr/bin/env bash - -. "$( dirname "${BASH_SOURCE[0]}" )/common.sh" - -: "${AIRFLOW_REPO:?Should be set}" -: "${AIRFLOW_BRANCH:?Should be set}" -: "${INSTALL_MYSQL_CLIENT:?Should be true or false}" -: "${INSTALL_POSTGRES_CLIENT:?Should be true or false}" - -function install_airflow_dependencies_from_branch_tip() { - echo - echo "${COLOR_BLUE}Installing airflow from ${AIRFLOW_BRANCH}. It is used to cache dependencies${COLOR_RESET}" - echo - if [[ ${INSTALL_MYSQL_CLIENT} != "true" ]]; then - AIRFLOW_EXTRAS=${AIRFLOW_EXTRAS/mysql,} - fi - if [[ ${INSTALL_POSTGRES_CLIENT} != "true" ]]; then - AIRFLOW_EXTRAS=${AIRFLOW_EXTRAS/postgres,} - fi - local TEMP_AIRFLOW_DIR - TEMP_AIRFLOW_DIR=$(mktemp -d) - # Install latest set of dependencies - without constraints. This is to download a "base" set of - # dependencies that we can cache and reuse when installing airflow using constraints and latest - # pyproject.toml in the next step (when we install regular airflow). - set -x - curl -fsSL "https://github.com/${AIRFLOW_REPO}/archive/${AIRFLOW_BRANCH}.tar.gz" | \ - tar xz -C "${TEMP_AIRFLOW_DIR}" --strip 1 - # Make sure editable dependencies are calculated when devel-ci dependencies are installed - ${PACKAGING_TOOL_CMD} install ${EXTRA_INSTALL_FLAGS} ${ADDITIONAL_PIP_INSTALL_FLAGS} \ - --editable "${TEMP_AIRFLOW_DIR}[${AIRFLOW_EXTRAS}]" - set +x - common::install_packaging_tools - set -x - echo "${COLOR_BLUE}Uninstalling providers. Dependencies remain${COLOR_RESET}" - # Uninstall airflow and providers to keep only the dependencies. In the future when - # planned https://github.com/pypa/pip/issues/11440 is implemented in pip we might be able to use this - # flag and skip the remove step. - pip freeze | grep apache-airflow-providers | xargs ${PACKAGING_TOOL_CMD} uninstall ${EXTRA_UNINSTALL_FLAGS} || true - set +x - echo - echo "${COLOR_BLUE}Uninstalling just airflow. Dependencies remain. Now target airflow can be reinstalled using mostly cached dependencies${COLOR_RESET}" - echo - set +x - ${PACKAGING_TOOL_CMD} uninstall ${EXTRA_UNINSTALL_FLAGS} apache-airflow - rm -rf "${TEMP_AIRFLOW_DIR}" - set -x - # If you want to make sure dependency is removed from cache in your PR when you removed it from - # pyproject.toml - please add your dependency here as a list of strings - # for example: - # DEPENDENCIES_TO_REMOVE=("package_a" "package_b") - # Once your PR is merged, you should make a follow-up PR to remove it from this list - # and increase the AIRFLOW_CI_BUILD_EPOCH in Dockerfile.ci to make sure your cache is rebuilt. - local DEPENDENCIES_TO_REMOVE - # IMPORTANT!! Make sure to increase AIRFLOW_CI_BUILD_EPOCH in Dockerfile.ci when you remove a dependency from that list - DEPENDENCIES_TO_REMOVE=() - if [[ "${DEPENDENCIES_TO_REMOVE[*]}" != "" ]]; then - echo - echo "${COLOR_BLUE}Uninstalling just removed dependencies (temporary until cache refreshes)${COLOR_RESET}" - echo "${COLOR_BLUE}Dependencies to uninstall: ${DEPENDENCIES_TO_REMOVE[*]}${COLOR_RESET}" - echo - set +x - ${PACKAGING_TOOL_CMD} uninstall "${DEPENDENCIES_TO_REMOVE[@]}" || true - set -x - # make sure that the dependency is not needed by something else - pip check - fi -} - -common::get_colors -common::get_packaging_tool -common::get_airflow_version_specification -common::get_constraints_location -common::show_packaging_tool_version_and_location - -install_airflow_dependencies_from_branch_tip -EOF - # The content below is automatically copied from scripts/docker/common.sh COPY <<"EOF" /common.sh #!/usr/bin/env bash @@ -465,8 +386,6 @@ function common::get_packaging_tool() { ## IMPORTANT: IF YOU MODIFY THIS FUNCTION YOU SHOULD ALSO MODIFY CORRESPONDING FUNCTION IN ## `scripts/in_container/_in_container_utils.sh` - local PYTHON_BIN - PYTHON_BIN=$(which python) if [[ ${AIRFLOW_USE_UV} == "true" ]]; then echo echo "${COLOR_BLUE}Using 'uv' to install Airflow${COLOR_RESET}" @@ -474,8 +393,8 @@ function common::get_packaging_tool() { export PACKAGING_TOOL="uv" export PACKAGING_TOOL_CMD="uv pip" if [[ -z ${VIRTUAL_ENV=} ]]; then - export EXTRA_INSTALL_FLAGS="--python ${PYTHON_BIN}" - export EXTRA_UNINSTALL_FLAGS="--python ${PYTHON_BIN}" + export EXTRA_INSTALL_FLAGS="--system" + export EXTRA_UNINSTALL_FLAGS="--system" else export EXTRA_INSTALL_FLAGS="" export EXTRA_UNINSTALL_FLAGS="" @@ -670,18 +589,12 @@ function install_airflow() { # Determine the installation_command_flags based on AIRFLOW_INSTALLATION_METHOD method local installation_command_flags if [[ ${AIRFLOW_INSTALLATION_METHOD} == "." ]]; then - # We need _a_ file in there otherwise the editable install doesn't include anything in the .pth file - mkdir -p ./providers/src/airflow/providers/ - touch ./providers/src/airflow/providers/__init__.py - - # Similarly we need _a_ file for task_sdk too - mkdir -p ./task_sdk/src/airflow/sdk/ - echo '__version__ = "0.0.0dev0"' > ./task_sdk/src/airflow/sdk/__init__.py - - trap 'rm -f ./providers/src/airflow/providers/__init__.py ./task_sdk/src/airflow/__init__.py 2>/dev/null' EXIT - # When installing from sources - we always use `--editable` mode - installation_command_flags="--editable .[${AIRFLOW_EXTRAS}]${AIRFLOW_VERSION_SPECIFICATION} --editable ./providers --editable ./task_sdk" + installation_command_flags="--editable .[${AIRFLOW_EXTRAS}]${AIRFLOW_VERSION_SPECIFICATION} --editable ./task_sdk" + while IFS= read -r -d '' pyproject_toml_file; do + project_folder=$(dirname ${pyproject_toml_file}) + installation_command_flags="${installation_command_flags} --editable ${project_folder}" + done < <(find "providers" -name "pyproject.toml" -print0) elif [[ ${AIRFLOW_INSTALLATION_METHOD} == "apache-airflow" ]]; then installation_command_flags="apache-airflow[${AIRFLOW_EXTRAS}]${AIRFLOW_VERSION_SPECIFICATION}" elif [[ ${AIRFLOW_INSTALLATION_METHOD} == apache-airflow\ @\ * ]]; then @@ -980,9 +893,10 @@ function determine_airflow_to_use() { echo echo "${COLOR_BLUE}Uninstalling all packages first${COLOR_RESET}" echo - pip freeze | grep -ve "^-e" | grep -ve "^#" | grep -ve "^uv" | xargs pip uninstall -y --root-user-action ignore + # shellcheck disable=SC2086 + ${PACKAGING_TOOL_CMD} freeze | grep -ve "^-e" | grep -ve "^#" | grep -ve "^uv" | xargs ${PACKAGING_TOOL_CMD} uninstall -y --root-user-action ignore # Now install rich ad click first to use the installation script - uv pip install rich rich-click click --python "/usr/local/bin/python" \ + ${PACKAGING_TOOL_CMD} install rich rich-click click --python "/usr/local/bin/python" \ --constraint https://raw.githubusercontent.com/apache/airflow/constraints-main/constraints-${PYTHON_MAJOR_MINOR_VERSION}.txt fi python "${IN_CONTAINER_DIR}/install_airflow_and_providers.py" @@ -992,7 +906,7 @@ function determine_airflow_to_use() { python "${IN_CONTAINER_DIR}/install_devel_deps.py" \ --constraint https://raw.githubusercontent.com/apache/airflow/constraints-main/constraints-${PYTHON_MAJOR_MINOR_VERSION}.txt # Some packages might leave legacy typing module which causes test issues - pip uninstall -y typing || true + ${PACKAGING_TOOL_CMD} uninstall -y typing || true if [[ ${LINK_PROVIDERS_TO_AIRFLOW_PACKAGE=} == "true" ]]; then echo echo "${COLOR_BLUE}Linking providers to airflow package as we are using them from mounted sources.${COLOR_RESET}" @@ -1202,7 +1116,10 @@ ENV PYTHON_BASE_IMAGE=${PYTHON_BASE_IMAGE} \ DEPENDENCIES_EPOCH_NUMBER=${DEPENDENCIES_EPOCH_NUMBER} \ INSTALL_MYSQL_CLIENT="true" \ INSTALL_MSSQL_CLIENT="true" \ - INSTALL_POSTGRES_CLIENT="true" + INSTALL_POSTGRES_CLIENT="true" \ + PIP_CACHE_DIR=/root/.cache/pip \ + UV_CACHE_DIR=/root/.cache/uv + RUN echo "Base image version: ${PYTHON_BASE_IMAGE}" @@ -1282,12 +1199,7 @@ ARG DEFAULT_CONSTRAINTS_BRANCH="constraints-main" # By changing the epoch we can force reinstalling Airflow and pip all dependencies # It can also be overwritten manually by setting the AIRFLOW_CI_BUILD_EPOCH environment variable. ARG AIRFLOW_CI_BUILD_EPOCH="10" -ARG AIRFLOW_PRE_CACHED_PIP_PACKAGES="true" # Setup PIP -# By default PIP install run without cache to make image smaller -ARG PIP_NO_CACHE_DIR="true" -# By default UV install run without cache to make image smaller -ARG UV_NO_CACHE="true" ARG UV_HTTP_TIMEOUT="300" # By default PIP has progress bar but you can disable it. ARG PIP_PROGRESS_BAR="on" @@ -1315,7 +1227,6 @@ ENV AIRFLOW_REPO=${AIRFLOW_REPO}\ AIRFLOW_CONSTRAINTS_LOCATION=${AIRFLOW_CONSTRAINTS_LOCATION} \ DEFAULT_CONSTRAINTS_BRANCH=${DEFAULT_CONSTRAINTS_BRANCH} \ AIRFLOW_CI_BUILD_EPOCH=${AIRFLOW_CI_BUILD_EPOCH} \ - AIRFLOW_PRE_CACHED_PIP_PACKAGES=${AIRFLOW_PRE_CACHED_PIP_PACKAGES} \ AIRFLOW_VERSION=${AIRFLOW_VERSION} \ AIRFLOW_PIP_VERSION=${AIRFLOW_PIP_VERSION} \ AIRFLOW_UV_VERSION=${AIRFLOW_UV_VERSION} \ @@ -1327,9 +1238,7 @@ ENV AIRFLOW_REPO=${AIRFLOW_REPO}\ INSTALL_POSTGRES_CLIENT="true" \ AIRFLOW_INSTALLATION_METHOD="." \ AIRFLOW_VERSION_SPECIFICATION="" \ - PIP_NO_CACHE_DIR=${PIP_NO_CACHE_DIR} \ PIP_PROGRESS_BAR=${PIP_PROGRESS_BAR} \ - UV_NO_CACHE=${UV_NO_CACHE} \ ADDITIONAL_PIP_INSTALL_FLAGS=${ADDITIONAL_PIP_INSTALL_FLAGS} \ CASS_DRIVER_BUILD_CONCURRENCY=${CASS_DRIVER_BUILD_CONCURRENCY} \ CASS_DRIVER_NO_CYTHON=${CASS_DRIVER_NO_CYTHON} @@ -1338,25 +1247,10 @@ RUN echo "Airflow version: ${AIRFLOW_VERSION}" # Copy all scripts required for installation - changing any of those should lead to # rebuilding from here -COPY --from=scripts install_packaging_tools.sh install_airflow_dependencies_from_branch_tip.sh \ - common.sh /scripts/docker/ +COPY --from=scripts common.sh install_packaging_tools.sh install_additional_dependencies.sh /scripts/docker/ # We are first creating a venv where all python packages and .so binaries needed by those are # installed. -# In case of CI builds we want to pre-install main version of airflow dependencies so that -# We do not have to always reinstall it from the scratch. -# And is automatically reinstalled from the scratch every time patch release of python gets released -# The Airflow and providers are uninstalled, only dependencies remain. -# the cache is only used when "upgrade to newer dependencies" is not set to automatically -# account for removed dependencies (we do not install them in the first place) -# -# We are installing from branch tip without fixing UV or PIP version - in order to avoid rebuilding the -# base cache layer every time the UV or PIP version changes. -RUN bash /scripts/docker/install_packaging_tools.sh; \ - if [[ ${AIRFLOW_PRE_CACHED_PIP_PACKAGES} == "true" ]]; then \ - bash /scripts/docker/install_airflow_dependencies_from_branch_tip.sh; \ - fi - # Here we fix the versions so all subsequent commands will use the versions # from the sources @@ -1372,31 +1266,33 @@ ARG AIRFLOW_PRE_COMMIT_UV_VERSION="4.1.4" ENV AIRFLOW_PIP_VERSION=${AIRFLOW_PIP_VERSION} \ AIRFLOW_UV_VERSION=${AIRFLOW_UV_VERSION} \ + # This is needed since we are using cache mounted from the host + UV_LINK_MODE=copy \ AIRFLOW_PRE_COMMIT_VERSION=${AIRFLOW_PRE_COMMIT_VERSION} # The PATH is needed for PIPX to find the tools installed ENV PATH="/root/.local/bin:${PATH}" +# Useful for creating a cache id based on the underlying architecture, preventing the use of cached python packages from +# an incorrect architecture. +ARG TARGETARCH +# Value to be able to easily change cache id and therefore use a bare new cache +ARG DEPENDENCY_CACHE_EPOCH="0" + # Install useful command line tools in their own virtualenv so that they do not clash with # dependencies installed in Airflow also reinstall PIP and UV to make sure they are installed # in the version specified above -RUN bash /scripts/docker/install_packaging_tools.sh - -# Airflow sources change frequently but dependency configuration won't change that often -# We copy pyproject.toml and other files needed to perform setup of dependencies -# So in case pyproject.toml changes we can install latest dependencies required. -COPY pyproject.toml ${AIRFLOW_SOURCES}/pyproject.toml -COPY providers/pyproject.toml ${AIRFLOW_SOURCES}/providers/pyproject.toml -COPY task_sdk/pyproject.toml ${AIRFLOW_SOURCES}/task_sdk/pyproject.toml -COPY task_sdk/README.md ${AIRFLOW_SOURCES}/task_sdk/README.md -COPY airflow/__init__.py ${AIRFLOW_SOURCES}/airflow/ -COPY tests_common/ ${AIRFLOW_SOURCES}/tests_common/ -COPY generated/* ${AIRFLOW_SOURCES}/generated/ -COPY constraints/* ${AIRFLOW_SOURCES}/constraints/ -COPY LICENSE ${AIRFLOW_SOURCES}/LICENSE -COPY hatch_build.py ${AIRFLOW_SOURCES}/ +RUN --mount=type=cache,id=ci-$TARGETARCH-$DEPENDENCY_CACHE_EPOCH,target=/root/.cache/ \ + bash /scripts/docker/install_packaging_tools.sh + COPY --from=scripts install_airflow.sh /scripts/docker/ +# We can copy everything here. The Context is filtered by dockerignore. This makes sure we are not +# copying over stuff that is accidentally generated or that we do not need (such as egg-info) +# if you want to add something that is missing and you expect to see it in the image you can +# add it with ! in .dockerignore next to the airflow, test etc. directories there +COPY . ${AIRFLOW_SOURCES}/ + # Those are additional constraints that are needed for some extras but we do not want to # force them on the main Airflow package. Currently we need no extra limits as PIP 23.1+ has much better # dependency resolution and we do not need to limit the versions of the dependencies @@ -1415,36 +1311,30 @@ ENV EAGER_UPGRADE_ADDITIONAL_REQUIREMENTS=${EAGER_UPGRADE_ADDITIONAL_REQUIREMENT # Usually we will install versions based on the dependencies in pyproject.toml and upgraded only if needed. # But in cron job we will install latest versions matching pyproject.toml to see if there is no breaking change # and push the constraints if everything is successful -RUN bash /scripts/docker/install_airflow.sh - -COPY --from=scripts entrypoint_ci.sh /entrypoint -COPY --from=scripts entrypoint_exec.sh /entrypoint-exec -RUN chmod a+x /entrypoint /entrypoint-exec +RUN --mount=type=cache,id=ci-$TARGETARCH-$DEPENDENCY_CACHE_EPOCH,target=/root/.cache/ bash /scripts/docker/install_airflow.sh COPY --from=scripts install_packaging_tools.sh install_additional_dependencies.sh /scripts/docker/ -# Additional python deps to install ARG ADDITIONAL_PYTHON_DEPS="" -RUN bash /scripts/docker/install_packaging_tools.sh; \ +ENV ADDITIONAL_PYTHON_DEPS=${ADDITIONAL_PYTHON_DEPS} + +RUN --mount=type=cache,id=ci-$TARGETARCH-$DEPENDENCY_CACHE_EPOCH,target=/root/.cache/ \ + bash /scripts/docker/install_packaging_tools.sh; \ if [[ -n "${ADDITIONAL_PYTHON_DEPS}" ]]; then \ bash /scripts/docker/install_additional_dependencies.sh; \ fi -# Install autocomplete for airflow -RUN if command -v airflow; then \ - register-python-argcomplete airflow >> ~/.bashrc ; \ - fi - -# Install autocomplete for Kubectl -RUN echo "source /etc/bash_completion" >> ~/.bashrc +COPY --from=scripts entrypoint_ci.sh /entrypoint +COPY --from=scripts entrypoint_exec.sh /entrypoint-exec +RUN chmod a+x /entrypoint /entrypoint-exec -# We can copy everything here. The Context is filtered by dockerignore. This makes sure we are not -# copying over stuff that is accidentally generated or that we do not need (such as egg-info) -# if you want to add something that is missing and you expect to see it in the image you can -# add it with ! in .dockerignore next to the airflow, test etc. directories there -COPY . ${AIRFLOW_SOURCES}/ +# Install autocomplete for airflow and kubectl +RUN if command -v airflow; then \ + register-python-argcomplete airflow >> ~/.bashrc ; \ + fi; \ + echo "source /etc/bash_completion" >> ~/.bashrc WORKDIR ${AIRFLOW_SOURCES} diff --git a/dev/breeze/doc/06_managing_docker_images.rst b/dev/breeze/doc/06_managing_docker_images.rst index bb4c4f9e06f62..84e4e77a010f1 100644 --- a/dev/breeze/doc/06_managing_docker_images.rst +++ b/dev/breeze/doc/06_managing_docker_images.rst @@ -76,7 +76,7 @@ These are all available flags of ``pull`` command: Verifying CI image .................. -Finally, you can verify CI image by running tests - either with the pulled/built images or +You can verify CI image by running tests - either with the pulled/built images or with an arbitrary image. These are all available flags of ``verify`` command: @@ -86,6 +86,41 @@ These are all available flags of ``verify`` command: :width: 100% :alt: Breeze ci-image verify +Loading and saving CI image +........................... + +You can load and save PROD image - for example to transfer it to another machine or to load an image +that has been built in our CI. + +These are all available flags of ``save`` command: + +.. image:: ./images/output_ci-image_save.svg + :target: https://raw.githubusercontent.com/apache/airflow/main/dev/breeze/images/output_ci-image_save.svg + :width: 100% + :alt: Breeze ci-image save + +These are all available flags of ``load`` command: + +.. image:: ./images/output_ci-image_load.svg + :target: https://raw.githubusercontent.com/apache/airflow/main/dev/breeze/images/output_ci-image_load.svg + :width: 100% + :alt: Breeze ci-image load + +Images for every build from our CI are uploaded as artifacts to the +GitHub Action run (in summary) and can be downloaded from there for 2 days in order to reproduce the complete +environment used during the tests and loaded to the local Docker registry (note that you have +to use the same platform as the CI run). + +You will find the artifacts for each image in the summary of the CI run. The artifacts are named +``ci-image-docker-export---_merge``. Those are compressed zip files that +contain the ".tar" image that should be used with ``--image-file`` flag of the load method. Make sure to +use the same ``--python`` version as the image was built with. + +.. image:: ./images/image_artifacts.png + :target: https://raw.githubusercontent.com/apache/airflow/main/dev/breeze/images/output_ci-image_load.svg + :width: 100% + :alt: Breeze image artifacts + PROD Image tasks ---------------- @@ -170,7 +205,7 @@ These are all available flags of ``pull-prod-image`` command: Verifying PROD image .................... -Finally, you can verify PROD image by running tests - either with the pulled/built images or +You can verify PROD image by running tests - either with the pulled/built images or with an arbitrary image. These are all available flags of ``verify-prod-image`` command: @@ -180,6 +215,31 @@ These are all available flags of ``verify-prod-image`` command: :width: 100% :alt: Breeze prod-image verify +Loading and saving PROD image +............................. + +You can load and save PROD image - for example to transfer it to another machine or to load an image +that has been built in our CI. + +These are all available flags of ``save`` command: + +.. image:: ./images/output_prod-image_save.svg + :target: https://raw.githubusercontent.com/apache/airflow/main/dev/breeze/images/output_prod-image_save.svg + :width: 100% + :alt: Breeze prod-image save + +These are all available flags of ``load`` command: + +.. image:: ./images/output-prod-image_load.svg + :target: https://raw.githubusercontent.com/apache/airflow/main/dev/breeze/images/output_prod-image_load.svg + :width: 100% + :alt: Breeze prod-image load + +Similarly as in case of CI images, Images for every build from our CI are uploaded as artifacts to the +GitHub Action run (in summary) and can be downloaded from there for 2 days in order to reproduce the complete +environment used during the tests and loaded to the local Docker registry (note that you have +to use the same platform as the CI run). + ------ Next step: Follow the `Breeze maintenance tasks <07_breeze_maintenance_tasks.rst>`_ to learn about tasks that diff --git a/dev/breeze/doc/ci/01_ci_environment.md b/dev/breeze/doc/ci/01_ci_environment.md index c9501a13b208a..1b375fb45ce5f 100644 --- a/dev/breeze/doc/ci/01_ci_environment.md +++ b/dev/breeze/doc/ci/01_ci_environment.md @@ -23,7 +23,7 @@ - [CI Environment](#ci-environment) - [GitHub Actions workflows](#github-actions-workflows) - - [Container Registry used as cache](#container-registry-used-as-cache) + - [GitHub artifacts used as cache](#github-artifacts-used-as-cache) - [Authentication in GitHub Registry](#authentication-in-github-registry) @@ -32,7 +32,8 @@ Continuous Integration is an important component of making Apache Airflow robust and stable. We run a lot of tests for every pull request, -for main and v2-\*-test branches and regularly as scheduled jobs. +for `canary` runs (from `main` and `v*-\*-test` branches and +regularly as scheduled jobs. Our execution environment for CI is [GitHub Actions](https://github.com/features/actions). GitHub Actions. @@ -60,57 +61,44 @@ To run the tests, we need to ensure that the images are built using the latest sources and that the build process is efficient. A full rebuild of such an image from scratch might take approximately 15 minutes. Therefore, we've implemented optimization techniques that efficiently -use the cache from the GitHub Docker registry. In most cases, this -reduces the time needed to rebuild the image to about 4 minutes. -However, when dependencies change, it can take around 6-7 minutes, and -if the base image of Python releases a new patch-level, it can take -approximately 12 minutes. +use the cache from Github Actions Artifacts. -## Container Registry used as cache +## GitHub artifacts used as cache -We are using GitHub Container Registry to store the results of the -`Build Images` workflow which is used in the `Tests` workflow. +We are using GitHub Artifacts to store the results of the +`Build Images` jobs which is used in the remaining part of the workflow. Currently in main version of Airflow we run tests in all versions of Python supported, which means that we have to build multiple images (one CI and one PROD for each Python version). Yet we run many jobs (\>15) - for each of the CI images. That is a lot of time to just build the -environment to run. Therefore we are utilising the `pull_request_target` -feature of GitHub Actions. - -This feature allows us to run a separate, independent workflow, when the -main workflow is run -this separate workflow is different than the main -one, because by default it runs using `main` version of the sources but -also - and most of all - that it has WRITE access to the GitHub -Container Image registry. - -This is especially important in our case where Pull Requests to Airflow -might come from any repository, and it would be a huge security issue if -anyone from outside could utilise the WRITE access to the Container -Image Registry via external Pull Request. - -Thanks to the WRITE access and fact that the `pull_request_target` workflow named -`Build Imaages` which - by default - uses the `main` version of the sources. -There we can safely run some code there as it has been reviewed and merged. -The workflow checks-out the incoming Pull Request, builds -the container image from the sources from the incoming PR (which happens in an -isolated Docker build step for security) and pushes such image to the -GitHub Docker Registry - so that this image can be built only once and -used by all the jobs running tests. The image is tagged with unique -`COMMIT_SHA` of the incoming Pull Request and the tests run in the `pull` workflow -can simply pull such image rather than build it from the scratch. -Pulling such image takes ~ 1 minute, thanks to that we are saving a -lot of precious time for jobs. - -We use [GitHub Container Registry](https://docs.github.com/en/packages/guides/about-github-container-registry). -A `GITHUB_TOKEN` is needed to push to the registry. We configured -scopes of the tokens in our jobs to be able to write to the registry, -but only for the jobs that need it. - -The latest cache is kept as `:cache-linux-amd64` and `:cache-linux-arm64` -tagged cache of our CI images (suitable for `--cache-from` directive of -buildx). It contains metadata and cache for all segments in the image, -and cache is kept separately for different platform. +environment to run. That's why images are built only once, they are +exported to a file and stored in the GitHub Artifacts. The export files +are then downloaded from artifacts and image is loaded from the file +in the remaining jobs. Thanks to unlimited storage of artifacts and the +speed and flexibility of "stash" Apache Software Foundation action, we +are able to both upload it very fast as artifacts and use artifacts +from `main` branch as a source of cache when we are building the images +in CI. + +In case of local Breeze image, we are using GitHub Registry to store +the cached version if the image - both as cache "layer" as well as +source to "pull" the image from. Also local image utilises uv local +pip cache to share the cache between the local builds. This means +that both local and CI builds are optimised for speed of rebuilds, +effectively utilising the three cache sources: + +* GitHub Artifacts for CI builds - including all PRs coming from the forks. +* GitHub Registry for local development builds (for first time installation). + This cache is written only in `canary` runs that are always run from the + `apache` repository, not from the forks. +* UV / PIP cache shared between the builds in local image building case) + +The latest GitHub registry cache is kept as `:cache-linux-amd64` and +`:cache-linux-arm64` tagged cache of our CI images (suitable for +`--cache-from` directive of buildx). It contains +metadata and cache for all segments in the image, +and cache is kept separately for different platforms. The `latest` images of CI and PROD are `amd64` only images for CI, because there is no easy way to push multiplatform images without @@ -118,11 +106,14 @@ merging the manifests, and it is not really needed nor used for cache. ## Authentication in GitHub Registry -We are using GitHub Container Registry as cache for our images. -Authentication uses GITHUB_TOKEN mechanism. Authentication is needed for +since we are using GitHub Container Registry as remote cache for our development +images. Authentication uses GITHUB_TOKEN mechanism. Authentication is needed for pushing the images (WRITE) only in `push`, `pull_request_target` workflows. When you are running the CI jobs in GitHub Actions, -GITHUB_TOKEN is set automatically by the actions. +GITHUB_TOKEN is set automatically by the actions. This is used only in the `canary` +runs that have "write access" to the repository. No `write` access is needed by +Pull Requests coming from the forks - since we are only using "GitHub Artifacts" for +cache source in those runes. ---- diff --git a/dev/breeze/doc/ci/02_images.md b/dev/breeze/doc/ci/02_images.md index eb3af6ae6ce87..0ce8f6eae5216 100644 --- a/dev/breeze/doc/ci/02_images.md +++ b/dev/breeze/doc/ci/02_images.md @@ -215,10 +215,11 @@ in `docker-context-files` folder. # Using docker cache during builds -Default mechanism used in Breeze for building CI images uses images -pulled from GitHub Container Registry. This is done to speed up local +Default mechanism used in Breeze for building CI images locally uses images +pulled from GitHub Container Registry combined with locally mounted cache +folders where `uv` cache is stored. This is done to speed up local builds and building images for CI runs - instead of \> 12 minutes for -rebuild of CI images, it takes usually about 1 minute when cache is +rebuild of CI images, it takes usually less than a minute when cache is used. For CI images this is usually the best strategy - to use default "pull" cache. This is default strategy when [Breeze](../README.rst) builds are performed. @@ -227,7 +228,8 @@ For Production Image - which is far smaller and faster to build, it's better to use local build cache (the standard mechanism that docker uses. This is the default strategy for production images when [Breeze](../README.rst) builds are -performed. The first time you run it, it will take considerably longer +performed. The local `uv` cache is used from mounted sources. +The first time you run it, it will take considerably longer time than if you use the pull mechanism, but then when you do small, incremental changes to local sources, Dockerfile image and scripts, further rebuilds with local build cache will be considerably faster. @@ -293,19 +295,12 @@ See Naming convention for the GitHub packages. -Images with a commit SHA (built for pull requests and pushes). Those are -images that are snapshot of the currently run build. They are built once -per each build and pulled by each test job. - ``` bash -ghcr.io/apache/airflow//ci/python: - for CI images -ghcr.io/apache/airflow//prod/python: - for production images +ghcr.io/apache/airflow//ci/python - for CI images +ghcr.io/apache/airflow//prod/python - for production images ``` -Thoe image contain inlined cache. - -You can see all the current GitHub images at - +You can see all the current GitHub images at Note that you need to be committer and have the right to refresh the images in the GitHub Registry with latest sources from main via @@ -329,22 +324,14 @@ new version of base Python is released. However, occasionally, you might need to rebuild images locally and push them directly to the registries to refresh them. -Every developer can also pull and run images being result of a specific +Every contributor can also pull and run images being result of a specific CI run in GitHub Actions. This is a powerful tool that allows to reproduce CI failures locally, enter the images and fix them much -faster. It is enough to pass `--image-tag` and the registry and Breeze -will download and execute commands using the same image that was used -during the CI tests. - -For example this command will run the same Python 3.9 image as was used -in build identified with 9a621eaa394c0a0a336f8e1b31b35eff4e4ee86e commit -SHA with enabled rabbitmq integration. - -``` bash -breeze --image-tag 9a621eaa394c0a0a336f8e1b31b35eff4e4ee86e --python 3.9 --integration rabbitmq -``` +faster. It is enough to download and uncompress the artifact that stores the +image and run ``breeze ci-image load -i `` to load the +image and mark the image as refreshed in the local cache. -You can see more details and examples in[Breeze](../README.rst) +You can see more details and examples in[Breeze](../06_managing_docker_images.rst) # Customizing the CI image @@ -427,8 +414,6 @@ can be used for CI images: | `PYTHON_MAJOR_MINOR_VERSION` | `3.9` | major/minor version of Python (should match base image) | | `DEPENDENCIES_EPOCH_NUMBER` | `2` | increasing this number will reinstall all apt dependencies | | `ADDITIONAL_PIP_INSTALL_FLAGS` | | additional `pip` flags passed to the installation commands (except when reinstalling `pip` itself) | -| `PIP_NO_CACHE_DIR` | `true` | if true, then no pip cache will be stored | -| `UV_NO_CACHE` | `true` | if true, then no uv cache will be stored | | `HOME` | `/root` | Home directory of the root user (CI image has root user as default) | | `AIRFLOW_HOME` | `/root/airflow` | Airflow's HOME (that's where logs and sqlite databases are stored) | | `AIRFLOW_SOURCES` | `/opt/airflow` | Mounted sources of Airflow | @@ -439,7 +424,6 @@ can be used for CI images: | `AIRFLOW_CONSTRAINTS_REFERENCE` | | reference (branch or tag) from GitHub repository from which constraints are used. By default it is set to `constraints-main` but can be `constraints-2-X`. | | `AIRFLOW_EXTRAS` | `all` | extras to install | | `UPGRADE_INVALIDATION_STRING` | | If set to any random value the dependencies are upgraded to newer versions. In CI it is set to build id. | -| `AIRFLOW_PRE_CACHED_PIP_PACKAGES` | `true` | Allows to pre-cache airflow PIP packages from the GitHub of Apache Airflow This allows to optimize iterations for Image builds and speeds up CI jobs. | | `ADDITIONAL_AIRFLOW_EXTRAS` | | additional extras to install | | `ADDITIONAL_PYTHON_DEPS` | | additional Python dependencies to install | | `DEV_APT_COMMAND` | | Dev apt command executed before dev deps are installed in the first part of image | @@ -553,10 +537,6 @@ The images produced during the `Build Images` workflow of CI jobs are stored in the [GitHub Container Registry](https://github.com/orgs/apache/packages?repo_name=airflow) -The images are stored with both "latest" tag (for last main push image -that passes all the tests as well with the COMMIT_SHA id for images that -were used in particular build. - The image names follow the patterns (except the Python image, all the images are stored in in `apache` organization. @@ -567,21 +547,15 @@ percent-encoded when you access them via UI (/ = %2F) `https://github.com/apache/airflow/pkgs/container/` -| Image | Name:tag (both cases latest version and per-build) | Description | -|--------------------------|----------------------------------------------------|---------------------------------------------------------------| -| Python image (DockerHub) | python:\-slim-bookworm | Base Python image used by both production and CI image. | -| CI image | airflow/\/ci/python\:\ | CI image - this is the image used for most of the tests. | -| PROD image | airflow/\/prod/python\:\ | faster to build or pull. Production image optimized for size. | +| Image | Name | Description | +|--------------------------|----------------------------------------|---------------------------------------------------------------| +| Python image (DockerHub) | python:\-slim-bookworm | Base Python image used by both production and CI image. | +| CI image | airflow/\/ci/python\ | CI image - this is the image used for most of the tests. | +| PROD image | airflow/\/prod/python\ | faster to build or pull. Production image optimized for size. | - \ might be either "main" or "v2-\*-test" - \ - Python version (Major + Minor).Should be one of \["3.9", "3.10", "3.11", "3.12" \]. -- \ - full-length SHA of commit either from the tip of the - branch (for pushes/schedule) or commit from the tip of the branch used - for the PR. -- \ - tag of the image. It is either "latest" or \ - (full-length SHA of commit either from the tip of the branch (for - pushes/schedule) or commit from the tip of the branch used for the - PR). + ---- diff --git a/dev/breeze/doc/ci/04_selective_checks.md b/dev/breeze/doc/ci/04_selective_checks.md index 08e5906745cbf..91b50bea48fe2 100644 --- a/dev/breeze/doc/ci/04_selective_checks.md +++ b/dev/breeze/doc/ci/04_selective_checks.md @@ -273,20 +273,10 @@ modified. This can be overridden by setting `full tests needed` label in the PR. There is a difference in how the CI jobs are run for committer and non-committer PRs from forks. The main reason is security; we do not want to run untrusted code on our infrastructure for self-hosted runners. -Additionally, we do not want to run unverified code during the `Build imaage` workflow, as that workflow has -access to the `GITHUB_TOKEN`, which can write to our Github Registry (used to cache -images between runs). These images are built on self-hosted runners, and we must ensure that -those runners are not misused, such as for mining cryptocurrencies on behalf of the person who opened the -pull request from their newly created fork of Airflow. - -This is why the `Build Images` workflow checks whether the actor of the PR (`GITHUB_ACTOR`) is one of the committers. -If not, the workflows and scripts used to run image building come only from the ``target`` branch -of the repository, where these scripts have been reviewed and approved by committers before being merged. This is controlled by the selective checks that set `is-committer-build` to `true` in -the build-info job of the workflow to determine if the actor is in the committers' -list. This setting can be overridden by the `non-committer build` label in the PR. - -Also, for most of the jobs, committer builds use "Self-hosted" runners by default, while non-committer -builds use "Public" runners. For committers, this can be overridden by setting the + +Currently there is no difference because we are not using `self-hosted` runners (until we implement `Action +Runner Controller` but most of the jobs, committer builds will use "Self-hosted" runners by default, +while non-committer builds will use "Public" runners. For committers, this can be overridden by setting the `use public runners` label in the PR. ## Changing behaviours of the CI runs by setting labels diff --git a/dev/breeze/doc/ci/05_workflows.md b/dev/breeze/doc/ci/05_workflows.md index 130774a730cb6..0c66505508f02 100644 --- a/dev/breeze/doc/ci/05_workflows.md +++ b/dev/breeze/doc/ci/05_workflows.md @@ -24,11 +24,8 @@ - [CI run types](#ci-run-types) - [Pull request run](#pull-request-run) - [Canary run](#canary-run) - - [Scheduled run](#scheduled-run) - [Workflows](#workflows) - - [Build Images Workflow](#build-images-workflow) - - [Differences for main and release branches](#differences-for-main-and-release-branches) - - [Committer vs. Non-committer PRs](#committer-vs-non-committer-prs) + - [Differences for `main` and `v*-*-test` branches](#differences-for-main-and-v--test-branches) - [Tests Workflow](#tests-workflow) - [CodeQL scan](#codeql-scan) - [Publishing documentation](#publishing-documentation) @@ -86,16 +83,16 @@ run in the context of the "apache/airflow" repository and has WRITE access to the GitHub Container Registry. When the PR changes important files (for example `generated/provider_depdencies.json` or -`pyproject.toml`), the PR is run in "upgrade to newer dependencies" mode - where instead -of using constraints to build images, attempt is made to upgrade all dependencies to latest -versions and build images with them. This way we check how Airflow behaves when the +`pyproject.toml` or `hatch_build.py`), the PR is run in "upgrade to newer dependencies" mode - +where instead of using constraints to build images, attempt is made to upgrade +all dependencies to latest versions and build images with them. This way we check how Airflow behaves when the dependencies are upgraded. This can also be forced by setting the `upgrade to newer dependencies` label in the PR if you are a committer and want to force dependency upgrade. ## Canary run -This workflow is triggered when a pull request is merged into the "main" -branch or pushed to any of the "v2-\*-test" branches. The "Canary" run +This workflow is triggered when a pull request is merged into the `main` +branch or pushed to any of the `v*-*-test` branches. The `canary` run aims to upgrade dependencies to their latest versions and promptly pushes a preview of the CI/PROD image cache to the GitHub Registry. This allows pull requests to quickly utilize the new cache, which is @@ -106,84 +103,36 @@ updates the constraint files in the "constraints-main" branch with the latest constraints and pushes both the cache and the latest CI/PROD images to the GitHub Registry. -If the "Canary" build fails, it often indicates that a new version of +If the `canary` build fails, it often indicates that a new version of our dependencies is incompatible with the current tests or Airflow code. Alternatively, it could mean that a breaking change has been merged into -"main". Both scenarios require prompt attention from the maintainers. +`main`. Both scenarios require prompt attention from the maintainers. While a "broken main" due to our code should be fixed quickly, "broken dependencies" may take longer to resolve. Until the tests pass, the constraints will not be updated, meaning that regular PRs will continue using the older version of dependencies that passed one of the previous -"Canary" runs. +`canary` runs. -## Scheduled run - -The "scheduled" workflow, which is designed to run regularly (typically -overnight), is triggered when a scheduled run occurs. This workflow is -largely identical to the "Canary" run, with one key difference: the -image is always built from scratch, not from a cache. This approach -ensures that we can verify whether any "system" dependencies in the -Debian base image have changed, and confirm that the build process -remains reproducible. Since the process for a scheduled run mirrors that -of a "Canary" run, no separate diagram is necessary to illustrate it. +The `canary` runs are executed 6 times a day on schedule, you can also +trigger the `canary` run manually via `workflow-dispatch` mechanism. # Workflows -A general note about cancelling duplicated workflows: for the -`Build Images`, `Tests` and `CodeQL` workflows, we use the `concurrency` -feature of GitHub actions to automatically cancel "old" workflow runs of +A general note about cancelling duplicated workflows: for `Tests` and `CodeQL` workflows, +we use the `concurrency` feature of GitHub actions to automatically cancel "old" workflow runs of each type. This means that if you push a new commit to a branch or to a pull request while a workflow is already running, GitHub Actions will automatically cancel the old workflow run. -## Build Images Workflow - -This workflow builds images for the CI Workflow for pull requests coming -from forks. - -The GitHub Actions event that trigger this workflow is `pull_request_target`, which means that -it is triggered when a pull request is opened. This also means that the -workflow has Write permission to push to the GitHub registry the images, which are -used by CI jobs. As a result, the images can be built only once and -reused by all CI jobs (including matrix jobs). We've implemented -it so that the `Tests` workflow waits for the images to be built by the -`Build Images` workflow before running. - -Those "Build Image" steps are skipped for pull requests that do not come -from "forks" (i.e. internal PRs for the Apache Airflow repository). -This is because, in case of PRs originating from Apache Airflow (which only -committers can create those) the "pull_request" workflows have enough -permission to push images to GitHub Registry. - -This workflow is not triggered by normal pushes to our "main" branches, -i.e., after a pull request is merged or when a `scheduled` run is -triggered. In these cases, the "CI" workflow has enough permissions -to push the images, so this workflow is simply not run. - -The workflow has the following jobs: - -| Job | Description | -|-------------------|---------------------------------------------| -| Build Info | Prints detailed information about the build | -| Build CI images | Builds all configured CI images | -| Build PROD images | Builds all configured PROD images | - -The images are stored in the [GitHub Container -Registry](https://github.com/orgs/apache/packages?repo_name=airflow), and their names follow the patterns -described in [Images](02_images.md#naming-conventions) - -Image building is configured in "fail-fast" mode. If any image -fails to build, it cancels the other builds and the `Tests` workflow -run that triggered it. - -## Differences for main and release branches +## Differences for `main` and `v*-*-test` branches The type of tests executed varies depending on the version or branch being tested. For the "main" development branch, we run all tests to maintain the quality of Airflow. However, when releasing patch-level updates on older branches, we only run a subset of tests. This is because older branches are used exclusively for releasing Airflow and -its corresponding image, not for releasing providers or Helm charts. +its corresponding image, not for releasing providers or Helm charts, +so all those tests are skipped there by default. This behaviour is controlled by `default-branch` output of the build-info job. Whenever we create a branch for an older version, we update @@ -192,90 +141,75 @@ the new branch. In several places, the selection of tests is based on whether this output is `main`. They are marked in the "Release branches" column of the table below. -## Committer vs. Non-committer PRs - -Please refer to the appropriate section in [selective CI checks](04_selective_checks.md#committer-vs-non-committer-prs) docs. - ## Tests Workflow -This workflow is a regular workflow that performs all checks of Airflow -code. - -| Job | Description | PR | Canary | Scheduled | Release branches | -|---------------------------------|----------------------------------------------------------|----------|----------|------------|------------------| -| Build info | Prints detailed information about the build | Yes | Yes | Yes | Yes | -| Push early cache & images | Pushes early cache/images to GitHub Registry | | Yes | | | -| Check that image builds quickly | Checks that image builds quickly | | Yes | | Yes | -| Build CI images | Builds images in-workflow (not in the build images) | | Yes | Yes (1) | Yes (4) | -| Generate constraints/CI verify | Generate constraints for the build and verify CI image | Yes (2) | Yes (2) | Yes (2) | Yes (2) | -| Build PROD images | Builds images in-workflow (not in the build images) | | Yes | Yes (1) | Yes (4) | -| Run breeze tests | Run unit tests for Breeze | Yes | Yes | Yes | Yes | -| Test OpenAPI client gen | Tests if OpenAPIClient continues to generate | Yes | Yes | Yes | Yes | -| React WWW tests | React UI tests for new Airflow UI | Yes | Yes | Yes | Yes | -| Test examples image building | Tests if PROD image build examples work | Yes | Yes | Yes | Yes | -| Test git clone on Windows | Tests if Git clone for for Windows | Yes (5) | Yes (5) | Yes (5) | Yes (5) | -| Waits for CI Images | Waits for and verify CI Images | Yes (2) | Yes (2) | Yes (2) | Yes (2) | -| Upgrade checks | Performs checks if there are some pending upgrades | | Yes | Yes | Yes | -| Static checks | Performs full static checks | Yes (6) | Yes | Yes | Yes (7) | -| Basic static checks | Performs basic static checks (no image) | Yes (6) | | | | -| Build docs | Builds and tests publishing of the documentation | Yes | Yes (11) | Yes | Yes | -| Spellcheck docs | Spellcheck docs | Yes | Yes | Yes | Yes | -| Tests wheel provider packages | Tests if provider packages can be built and released | Yes | Yes | Yes | | -| Tests Airflow compatibility | Compatibility of provider packages with older Airflow | Yes | Yes | Yes | | -| Tests dist provider packages | Tests if dist provider packages can be built | | Yes | Yes | | -| Tests airflow release commands | Tests if airflow release command works | | Yes | Yes | | -| Tests (Backend/Python matrix) | Run the Pytest unit DB tests (Backend/Python matrix) | Yes | Yes | Yes | Yes (8) | -| No DB tests | Run the Pytest unit Non-DB tests (with pytest-xdist) | Yes | Yes | Yes | Yes (8) | -| Integration tests | Runs integration tests (Postgres/Mysql) | Yes | Yes | Yes | Yes (9) | -| Quarantined tests | Runs quarantined tests (with flakiness and side-effects) | Yes | Yes | Yes | Yes (8) | -| Test airflow packages | Tests that Airflow package can be built and released | Yes | Yes | Yes | Yes | -| Helm tests | Run the Helm integration tests | Yes | Yes | Yes | | -| Helm release tests | Run the tests for Helm releasing | Yes | Yes | Yes | | -| Summarize warnings | Summarizes warnings from all other tests | Yes | Yes | Yes | Yes | -| Wait for PROD Images | Waits for and verify PROD Images | Yes (2) | Yes (2) | Yes (2) | Yes (2) | -| Docker Compose test/PROD verify | Tests quick-start Docker Compose and verify PROD image | Yes | Yes | Yes | Yes | -| Tests Kubernetes | Run Kubernetes test | Yes | Yes | Yes | | -| Update constraints | Upgrade constraints to latest ones | Yes (3) | Yes (3) | Yes (3) | Yes (3) | -| Push cache & images | Pushes cache/images to GitHub Registry (3) | | Yes (3) | | Yes | -| Build CI ARM images | Builds CI images for ARM | Yes (10) | | Yes | | +This workflow is a regular workflow that performs all checks of Airflow code. The `main` and `v*-*-test` +pushes are `canary` runs. + +| Job | Description | PR | main | v*-*-test | +|---------------------------------|----------------------------------------------------------|---------|---------|-----------| +| Build info | Prints detailed information about the build | Yes | Yes | Yes | +| Push early cache & images | Pushes early cache/images to GitHub Registry | | Yes (2) | Yes (2) | +| Check that image builds quickly | Checks that image builds quickly | | Yes | Yes | +| Build CI images | Builds images | Yes | Yes | Yes | +| Generate constraints/CI verify | Generate constraints for the build and verify CI image | Yes | Yes | Yes | +| Build PROD images | Builds images | Yes | Yes | Yes (3) | +| Run breeze tests | Run unit tests for Breeze | Yes | Yes | Yes | +| Test OpenAPI client gen | Tests if OpenAPIClient continues to generate | Yes | Yes | Yes | +| React WWW tests | React UI tests for new Airflow UI | Yes | Yes | Yes | +| Test examples image building | Tests if PROD image build examples work | Yes | Yes | Yes | +| Test git clone on Windows | Tests if Git clone for for Windows | Yes (4) | Yes (4) | Yes (4) | +| Upgrade checks | Performs checks if there are some pending upgrades | | Yes | Yes | +| Static checks | Performs full static checks | Yes (5) | Yes | Yes (6) | +| Basic static checks | Performs basic static checks (no image) | Yes (5) | | | +| Build and publish docs | Builds and tests publishing of the documentation | Yes (8) | Yes (8) | Yes (8) | +| Spellcheck docs | Spellcheck docs | Yes | Yes | Yes (7) | +| Tests wheel provider packages | Tests if provider packages can be built and released | Yes | Yes | | +| Tests Airflow compatibility | Compatibility of provider packages with older Airflow | Yes | Yes | | +| Tests dist provider packages | Tests if dist provider packages can be built | | Yes | | +| Tests airflow release commands | Tests if airflow release command works | | Yes | Yes | +| DB tests matrix | Run the Pytest unit DB tests | Yes | Yes | Yes (7) | +| No DB tests | Run the Pytest unit Non-DB tests (with pytest-xdist) | Yes | Yes | Yes (7) | +| Integration tests | Runs integration tests (Postgres/Mysql) | Yes | Yes | Yes (7) | +| Quarantined tests | Runs quarantined tests (with flakiness and side-effects) | Yes | Yes | Yes (7) | +| Test airflow packages | Tests that Airflow package can be built and released | Yes | Yes | Yes | +| Helm tests | Run the Helm integration tests | Yes | Yes | | +| Helm release tests | Run the tests for Helm releasing | Yes | Yes | | +| Summarize warnings | Summarizes warnings from all other tests | Yes | Yes | Yes | +| Docker Compose test/PROD verify | Tests quick-start Docker Compose and verify PROD image | Yes | Yes | Yes | +| Tests Kubernetes | Run Kubernetes test | Yes | Yes | | +| Update constraints | Upgrade constraints to latest ones | Yes | Yes (2) | Yes (2) | +| Push cache & images | Pushes cache/images to GitHub Registry (3) | | Yes (3) | | +| Build CI ARM images | Builds CI images for ARM | Yes (9) | | | `(1)` Scheduled jobs builds images from scratch - to test if everything works properly for clean builds -`(2)` The jobs wait for CI images to be available. It only actually runs when build image is needed (in -case of simpler PRs that do not change dependencies or source code, -images are not build) - -`(3)` PROD and CI cache & images are pushed as "cache" (both AMD and -ARM) and "latest" (only AMD) to GitHub Container registry and +`(2)` PROD and CI cache & images are pushed as "cache" (both AMD and +ARM) and "latest" (only AMD) to GitHub Container Registry and constraints are upgraded only if all tests are successful. The images are rebuilt in this step using constraints pushed in the previous step. -Constraints are only actually pushed in the `canary/scheduled` runs. +Constraints are only actually pushed in the `canary` runs. -`(4)` In main, PROD image uses locally build providers using "latest" +`(3)` In main, PROD image uses locally build providers using "latest" version of the provider code. In the non-main version of the build, the latest released providers from PyPI are used. -`(5)` Always run with public runners to test if Git clone works on +`(4)` Always run with public runners to test if Git clone works on Windows. -`(6)` Run full set of static checks when selective-checks determine that +`(5)` Run full set of static checks when selective-checks determine that they are needed (basically, when Python code has been modified). -`(7)` On non-main builds some of the static checks that are related to +`(6)` On non-main builds some of the static checks that are related to Providers are skipped via selective checks (`skip-pre-commits` check). -`(8)` On non-main builds the unit tests for providers are skipped via -selective checks removing the "Providers" test type. - -`(9)` On non-main builds the integration tests for providers are skipped -via `skip-providers-tests` selective check output. +`(7)` On non-main builds the unit tests, docs and integration tests +for providers are skipped via selective checks. -`(10)` Only run the builds in case PR is run by a committer from -"apache" repository and in scheduled build. +`(8)` Docs publishing is only done in Canary run. -`(11)` Docs publishing is only done in Canary run, to handle the case where -cloning whole airflow site on Public Runner cannot complete due to the size of the repository. +`(9)` ARM images are not currently built - until we have ARM runners available. ## CodeQL scan @@ -285,8 +219,7 @@ violations. It is run for JavaScript and Python code. ## Publishing documentation -Documentation from the `main` branch is automatically published on -Amazon S3. +Documentation from the `main` branch is automatically published on Amazon S3. To make this possible, GitHub Action has secrets set up with credentials for an Amazon Web Service account - `DOCS_AWS_ACCESS_KEY_ID` and @@ -303,4 +236,4 @@ Website endpoint: ----- -Read next about [Diagrams](06_diagrams.md) +Read next about [Debugging CI builds](06_debugging.md) diff --git a/dev/breeze/doc/ci/07_debugging.md b/dev/breeze/doc/ci/06_debugging.md similarity index 93% rename from dev/breeze/doc/ci/07_debugging.md rename to dev/breeze/doc/ci/06_debugging.md index 9e7173ae84721..8d030034728c7 100644 --- a/dev/breeze/doc/ci/07_debugging.md +++ b/dev/breeze/doc/ci/06_debugging.md @@ -34,10 +34,7 @@ either run in our Self-Hosted runners (with 64 GB RAM 8 CPUs) or in the GitHub Public runners (6 GB of RAM, 2 CPUs) and the results will vastly differ depending on which environment is used. We are utilizing parallelism to make use of all the available CPU/Memory but sometimes -you need to enable debugging and force certain environments. Additional -difficulty is that `Build Images` workflow is `pull-request-target` -type, which means that it will always run using the `main` version - no -matter what is in your Pull Request. +you need to enable debugging and force certain environments. There are several ways how you can debug the CI jobs and modify their behaviour when you are maintainer. @@ -64,4 +61,4 @@ the PR to apply the label to the PR. ----- -Read next about [Running CI locally](08_running_ci_locally.md) +Read next about [Running CI locally](07_running_ci_locally.md) diff --git a/dev/breeze/doc/ci/06_diagrams.md b/dev/breeze/doc/ci/06_diagrams.md deleted file mode 100644 index afe51a309e8eb..0000000000000 --- a/dev/breeze/doc/ci/06_diagrams.md +++ /dev/null @@ -1,466 +0,0 @@ - - - - -**Table of Contents** *generated with [DocToc](https://github.com/thlorenz/doctoc)* - -- [CI Sequence diagrams](#ci-sequence-diagrams) - - [Pull request flow from fork](#pull-request-flow-from-fork) - - [Pull request flow from "apache/airflow" repo](#pull-request-flow-from-apacheairflow-repo) - - [Merge "Canary" run](#merge-canary-run) - - [Scheduled run](#scheduled-run) - - - -# CI Sequence diagrams - -You can see here the sequence diagrams of the flow happening during the CI Jobs. - -## Pull request flow from fork - -This is the flow that happens when a pull request is created from a fork - which is the most frequent -pull request flow that happens in Airflow. The "pull_request" workflow does not have write access -to the GitHub Registry, so it cannot push the CI/PROD images there. Instead, we push the images -from the "pull_request_target" workflow, which has write access to the GitHub Registry. Note that -this workflow always uses scripts and workflows from the "target" branch of the "apache/airflow" -repository, so the user submitting such pull request cannot override our build scripts and inject malicious -code into the workflow that has potentially write access to the GitHub Registry (and can override cache). - -Security is the main reason why we have two workflows for pull requests and such complex workflows. - -```mermaid -sequenceDiagram - Note over Airflow Repo: pull request - Note over Tests: pull_request
[Read Token] - Note over Build Images: pull_request_target
[Write Token] - activate Airflow Repo - Airflow Repo -->> Tests: Trigger 'pull_request' - activate Tests - Tests -->> Build Images: Trigger 'pull_request_target' - activate Build Images - Note over Tests: Build info - Note over Tests: Selective checks
Decide what to do - Note over Build Images: Build info - Note over Build Images: Selective checks
Decide what to do - Note over Tests: Skip Build
(Runs in 'Build Images')
CI Images - Note over Tests: Skip Build
(Runs in 'Build Images')
PROD Images - par - GitHub Registry ->> Build Images: Use cache from registry - Airflow Repo ->> Build Images: Use constraints from `constraints-BRANCH` - Note over Build Images: Build CI Images
[COMMIT_SHA]
Upgrade to newer dependencies if deps changed - Build Images ->> GitHub Registry: Push CI Images
[COMMIT_SHA] - Build Images ->> Artifacts: Upload source constraints - and - Note over Tests: OpenAPI client gen - and - Note over Tests: React WWW tests - and - Note over Tests: Test git clone on Windows - and - Note over Tests: Helm release tests - and - opt - Note over Tests: Run basic
static checks - end - end - loop Wait for CI images - GitHub Registry ->> Tests: Pull CI Images
[COMMIT_SHA] - end - par - GitHub Registry ->> Tests: Pull CI Images
[COMMIT_SHA] - Note over Tests: Verify CI Images
[COMMIT_SHA] - Note over Tests: Generate constraints
source,pypi,no-providers - Tests ->> Artifacts: Upload source,pypi,no-providers constraints - and - Artifacts ->> Build Images: Download source constraints - GitHub Registry ->> Build Images: Use cache from registry - Note over Build Images: Build PROD Images
[COMMIT_SHA] - Build Images ->> GitHub Registry: Push PROD Images
[COMMIT_SHA] - and - opt - GitHub Registry ->> Tests: Pull CI Images
[COMMIT_SHA] - Note over Tests: Run static checks - end - and - opt - GitHub Registry ->> Tests: Pull CI Images
[COMMIT_SHA] - Note over Tests: Build docs - end - and - opt - GitHub Registry ->> Tests: Pull CI Images
[COMMIT_SHA] - Note over Tests: Spellcheck docs - end - and - opt - GitHub Registry ->> Tests: Pull CI Images
[COMMIT_SHA] - Note over Tests: Unit Tests
Python/DB matrix - end - and - opt - GitHub Registry ->> Tests: Pull CI Images
[COMMIT_SHA] - Note over Tests: Unit Tests
Python/Non-DB matrix - end - and - opt - GitHub Registry ->> Tests: Pull CI Images
[COMMIT_SHA] - Note over Tests: Integration Tests - end - and - opt - GitHub Registry ->> Tests: Pull CI Images
[COMMIT_SHA] - Note over Tests: Quarantined Tests - end - and - opt - GitHub Registry ->> Tests: Pull CI Images
[COMMIT_SHA] - Note over Tests: Build/test provider packages
wheel, sdist, old airflow - end - and - opt - GitHub Registry ->> Tests: Pull CI Images
[COMMIT_SHA] - Note over Tests: Test airflow
release commands - end - and - opt - GitHub Registry ->> Tests: Pull CI Images
[COMMIT_SHA] - Note over Tests: Helm tests - end - end - par - Note over Tests: Summarize Warnings - and - opt - Artifacts ->> Tests: Download source,pypi,no-providers constraints - Note over Tests: Display constraints diff - end - and - opt - loop Wait for PROD images - GitHub Registry ->> Tests: Pull PROD Images
[COMMIT_SHA] - end - end - and - opt - Note over Tests: Build ARM CI images - end - end - par - opt - GitHub Registry ->> Tests: Pull PROD Images
[COMMIT_SHA] - Note over Tests: Test examples
PROD image building - end - and - opt - GitHub Registry ->> Tests: Pull PROD Images
[COMMIT_SHA] - Note over Tests: Run Kubernetes
tests - end - and - opt - GitHub Registry ->> Tests: Pull PROD Images
[COMMIT_SHA] - Note over Tests: Verify PROD Images
[COMMIT_SHA] - Note over Tests: Run docker-compose
tests - end - end - Tests -->> Airflow Repo: Status update - deactivate Airflow Repo - deactivate Tests -``` - -## Pull request flow from "apache/airflow" repo - -The difference between this flow and the previous one is that the CI/PROD images are built in the -CI workflow and pushed to the GitHub Registry from there. This cannot be done in case of fork -pull request, because Pull Request from forks cannot have "write" access to GitHub Registry. All the steps -except "Build Info" from the "Build Images" workflows are skipped in this case. - -THis workflow can be used by maintainers in case they have a Pull Request that changes the scripts and -CI workflows used to build images, because in this case the "Build Images" workflow will use them -from the Pull Request. This is safe, because the Pull Request is from the "apache/airflow" repository -and only maintainers can push to that repository and create Pull Requests from it. - -```mermaid -sequenceDiagram - Note over Airflow Repo: pull request - Note over Tests: pull_request
[Write Token] - Note over Build Images: pull_request_target
[Unused Token] - activate Airflow Repo - Airflow Repo -->> Tests: Trigger 'pull_request' - activate Tests - Tests -->> Build Images: Trigger 'pull_request_target' - activate Build Images - Note over Tests: Build info - Note over Tests: Selective checks
Decide what to do - Note over Build Images: Build info - Note over Build Images: Selective checks
Decide what to do - Note over Build Images: Skip Build
(Runs in 'Tests')
CI Images - Note over Build Images: Skip Build
(Runs in 'Tests')
PROD Images - deactivate Build Images - Note over Tests: Build info - Note over Tests: Selective checks
Decide what to do - par - GitHub Registry ->> Tests: Use cache from registry - Airflow Repo ->> Tests: Use constraints from `constraints-BRANCH` - Note over Tests: Build CI Images
[COMMIT_SHA]
Upgrade to newer dependencies if deps changed - Tests ->> GitHub Registry: Push CI Images
[COMMIT_SHA] - Tests ->> Artifacts: Upload source constraints - and - Note over Tests: OpenAPI client gen - and - Note over Tests: React WWW tests - and - Note over Tests: Test examples
PROD image building - and - Note over Tests: Test git clone on Windows - and - Note over Tests: Helm release tests - and - opt - Note over Tests: Run basic
static checks - end - end - Note over Tests: Skip waiting for CI images - par - GitHub Registry ->> Tests: Pull CI Images
[COMMIT_SHA] - Note over Tests: Verify CI Images
[COMMIT_SHA] - Note over Tests: Generate constraints
source,pypi,no-providers - Tests ->> Artifacts: Upload source,pypi,no-providers constraints - and - Artifacts ->> Tests: Download source constraints - GitHub Registry ->> Tests: Use cache from registry - Note over Tests: Build PROD Images
[COMMIT_SHA] - Tests ->> GitHub Registry: Push PROD Images
[COMMIT_SHA] - and - opt - GitHub Registry ->> Tests: Pull CI Images
[COMMIT_SHA] - Note over Tests: Run static checks - end - and - opt - GitHub Registry ->> Tests: Pull CI Images
[COMMIT_SHA] - Note over Tests: Build docs - end - and - opt - GitHub Registry ->> Tests: Pull CI Images
[COMMIT_SHA] - Note over Tests: Spellcheck docs - end - and - opt - GitHub Registry ->> Tests: Pull CI Images
[COMMIT_SHA] - Note over Tests: Unit Tests
Python/DB matrix - end - and - opt - GitHub Registry ->> Tests: Pull CI Images
[COMMIT_SHA] - Note over Tests: Unit Tests
Python/Non-DB matrix - end - and - opt - GitHub Registry ->> Tests: Pull CI Images
[COMMIT_SHA] - Note over Tests: Integration Tests - end - and - opt - GitHub Registry ->> Tests: Pull CI Images
[COMMIT_SHA] - Note over Tests: Quarantined Tests - end - and - opt - GitHub Registry ->> Tests: Pull CI Images
[COMMIT_SHA] - Note over Tests: Build/test provider packages
wheel, sdist, old airflow - end - and - opt - GitHub Registry ->> Tests: Pull CI Images
[COMMIT_SHA] - Note over Tests: Test airflow
release commands - end - and - opt - GitHub Registry ->> Tests: Pull CI Images
[COMMIT_SHA] - Note over Tests: Helm tests - end - end - Note over Tests: Skip waiting for PROD images - par - Note over Tests: Summarize Warnings - and - opt - Artifacts ->> Tests: Download source,pypi,no-providers constraints - Note over Tests: Display constraints diff - end - and - Note over Tests: Build ARM CI images - and - opt - GitHub Registry ->> Tests: Pull PROD Images
[COMMIT_SHA] - Note over Tests: Run Kubernetes
tests - end - and - opt - GitHub Registry ->> Tests: Pull PROD Images
[COMMIT_SHA] - Note over Tests: Verify PROD Images
[COMMIT_SHA] - Note over Tests: Run docker-compose
tests - end - end - Tests -->> Airflow Repo: Status update - deactivate Airflow Repo - deactivate Tests -``` - -## Merge "Canary" run - -This is the flow that happens when a pull request is merged to the "main" branch or pushed to any of -the "v2-*-test" branches. The "Canary" run attempts to upgrade dependencies to the latest versions -and quickly pushes an early cache the CI/PROD images to the GitHub Registry - so that pull requests -can quickly use the new cache - this is useful when Dockerfile or installation scripts change because such -cache will already have the latest Dockerfile and scripts pushed even if some tests will fail. -When successful, the run updates the constraints files in the "constraints-BRANCH" branch with the latest -constraints and pushes both cache and latest CI/PROD images to the GitHub Registry. - -```mermaid -sequenceDiagram - Note over Airflow Repo: push/merge - Note over Tests: push
[Write Token] - activate Airflow Repo - Airflow Repo -->> Tests: Trigger 'push' - activate Tests - Note over Tests: Build info - Note over Tests: Selective checks
Decide what to do - par - GitHub Registry ->> Tests: Use cache from registry
(Not for scheduled run) - Airflow Repo ->> Tests: Use constraints from `constraints-BRANCH` - Note over Tests: Build CI Images
[COMMIT_SHA]
Always upgrade to newer deps - Tests ->> GitHub Registry: Push CI Images
[COMMIT_SHA] - Tests ->> Artifacts: Upload source constraints - and - GitHub Registry ->> Tests: Use cache from registry
(Not for scheduled run) - Note over Tests: Check that image builds quickly - and - GitHub Registry ->> Tests: Use cache from registry
(Not for scheduled run) - Note over Tests: Push early CI Image cache - Tests ->> GitHub Registry: Push CI cache Images - and - Note over Tests: OpenAPI client gen - and - Note over Tests: React WWW tests - and - Note over Tests: Test git clone on Windows - and - Note over Tests: Run upgrade checks - end - Note over Tests: Skip waiting for CI images - par - GitHub Registry ->> Tests: Pull CI Images
[COMMIT_SHA] - Note over Tests: Verify CI Images
[COMMIT_SHA] - Note over Tests: Generate constraints
source,pypi,no-providers - Tests ->> Artifacts: Upload source,pypi,no-providers constraints - and - Artifacts ->> Tests: Download source constraints - GitHub Registry ->> Tests: Use cache from registry - Note over Tests: Build PROD Images
[COMMIT_SHA] - Tests ->> GitHub Registry: Push PROD Images
[COMMIT_SHA] - and - Artifacts ->> Tests: Download source constraints - and - GitHub Registry ->> Tests: Pull CI Images
[COMMIT_SHA] - Note over Tests: Run static checks - and - GitHub Registry ->> Tests: Pull CI Images
[COMMIT_SHA] - Note over Tests: Build docs - and - GitHub Registry ->> Tests: Pull CI Images
[COMMIT_SHA] - Note over Tests: Spellcheck docs - and - GitHub Registry ->> Tests: Pull CI Images
[COMMIT_SHA] - Note over Tests: Unit Tests
Python/DB matrix - and - GitHub Registry ->> Tests: Pull CI Images
[COMMIT_SHA] - Note over Tests: Unit Tests
Python/Non-DB matrix - and - GitHub Registry ->> Tests: Pull CI Images
[COMMIT_SHA] - Note over Tests: Integration Tests - and - GitHub Registry ->> Tests: Pull CI Images
[COMMIT_SHA] - Note over Tests: Quarantined Tests - and - GitHub Registry ->> Tests: Pull CI Images
[COMMIT_SHA] - Note over Tests: Build/test provider packages
wheel, sdist, old airflow - and - GitHub Registry ->> Tests: Pull CI Images
[COMMIT_SHA] - Note over Tests: Test airflow
release commands - and - GitHub Registry ->> Tests: Pull CI Images
[COMMIT_SHA] - Note over Tests: Helm tests - end - Note over Tests: Skip waiting for PROD images - par - Note over Tests: Summarize Warnings - and - Artifacts ->> Tests: Download source,pypi,no-providers constraints - Note over Tests: Display constraints diff - Tests ->> Airflow Repo: Push constraints if changed to 'constraints-BRANCH' - and - GitHub Registry ->> Tests: Pull PROD Images
[COMMIT_SHA] - Note over Tests: Test examples
PROD image building - and - GitHub Registry ->> Tests: Pull PROD Image
[COMMIT_SHA] - Note over Tests: Run Kubernetes
tests - and - GitHub Registry ->> Tests: Pull PROD Image
[COMMIT_SHA] - Note over Tests: Verify PROD Images
[COMMIT_SHA] - Note over Tests: Run docker-compose
tests - end - par - GitHub Registry ->> Tests: Use cache from registry - Airflow Repo ->> Tests: Get latest constraints from 'constraints-BRANCH' - Note over Tests: Build CI latest images/cache - Tests ->> GitHub Registry: Push CI latest images/cache - GitHub Registry ->> Tests: Use cache from registry - Airflow Repo ->> Tests: Get latest constraints from 'constraints-BRANCH' - Note over Tests: Build PROD latest images/cache - Tests ->> GitHub Registry: Push PROD latest images/cache - and - GitHub Registry ->> Tests: Use cache from registry - Airflow Repo ->> Tests: Get latest constraints from 'constraints-BRANCH' - Note over Tests: Build ARM CI cache - Tests ->> GitHub Registry: Push ARM CI cache - GitHub Registry ->> Tests: Use cache from registry - Airflow Repo ->> Tests: Get latest constraints from 'constraints-BRANCH' - Note over Tests: Build ARM PROD cache - Tests ->> GitHub Registry: Push ARM PROD cache - end - Tests -->> Airflow Repo: Status update - deactivate Airflow Repo - deactivate Tests -``` - -## Scheduled run - -This is the flow that happens when a scheduled run is triggered. The "scheduled" workflow is aimed to -run regularly (overnight) even if no new PRs are merged to "main". Scheduled run is generally the -same as "Canary" run, with the difference that the image used to run the tests is built without using -cache - it's always built from the scratch. This way we can check that no "system" dependencies in debian -base image have changed and that the build is still reproducible. No separate diagram is needed for -scheduled run as it is identical to that of "Canary" run. - ------ - -Read next about [Debugging](07_debugging.md) diff --git a/dev/breeze/doc/ci/07_running_ci_locally.md b/dev/breeze/doc/ci/07_running_ci_locally.md new file mode 100644 index 0000000000000..21df374b0e16c --- /dev/null +++ b/dev/breeze/doc/ci/07_running_ci_locally.md @@ -0,0 +1,129 @@ + + + + +**Table of Contents** *generated with [DocToc](https://github.com/thlorenz/doctoc)* + +- [Running the CI Jobs locally](#running-the-ci-jobs-locally) + - [Basic variables](#basic-variables) + - [Host & GIT variables](#host--git-variables) + - [In-container environment initialization](#in-container-environment-initialization) +- [Image build variables](#image-build-variables) +- [Upgrade to newer dependencies](#upgrade-to-newer-dependencies) + + + +# Running the CI Jobs locally + +The main goal of the CI philosophy we have that no matter how complex +the test and integration infrastructure, as a developer you should be +able to reproduce and re-run any of the failed checks locally. One part +of it are pre-commit checks, that allow you to run the same static +checks in CI and locally, but another part is the CI environment which +is replicated locally with Breeze. + +You can read more about Breeze in +[README.rst](../README.rst) but in essence it is a python wrapper around +docker commands that allows you (among others) to re-create CI environment +in your local development instance and interact with it. +In its basic form, when you do development you can run all the same +tests that will be run in CI - but +locally, before you submit them as PR. Another use case where Breeze is +useful is when tests fail on CI. + +All our CI jobs are executed via `breeze` commands. You can replicate +exactly what our CI is doing by running the sequence of corresponding +`breeze` command. Make sure however that you look at both: + +- flags passed to `breeze` commands +- environment variables used when `breeze` command is run - this is + useful when we want to set a common flag for all `breeze` commands in + the same job or even the whole workflow. For example `VERBOSE` + variable is set to `true` for all our workflows so that more detailed + information about internal commands executed in CI is printed. + +In the output of the CI jobs, you will find both - the flags passed and +environment variables set. + +Every contributor can also pull and run images being result of a specific +CI run in GitHub Actions. This is a powerful tool that allows to +reproduce CI failures locally, enter the images and fix them much +faster. It is enough to download and uncompress the artifact that stores the +image and run ``breeze ci-image load -i --python python`` +to load the image and mark the image as refreshed in the local cache. + +You can read more about it in [Breeze](../README.rst) and +[Testing](../../../../contributing-docs/09_testing.rst) + +Depending whether the scripts are run locally via +[Breeze](../README.rst) or whether they are run in +`Build Images` or `Tests` workflows they can take different values. + +You can use those variables when you try to reproduce the build locally +(alternatively you can pass those via corresponding command line flags +passed to `breeze shell` command. + +## Basic variables + +| Variable | Local dev | CI | Comment | +|-----------------------------|-----------|------|------------------------------------------------------------------------------| +| PYTHON_MAJOR_MINOR_VERSION | | | Major/Minor version of Python used. | +| DB_RESET | false | true | Determines whether database should be reset at the container entry. | +| ANSWER | | yes | This variable determines if answer to questions should be automatically set. | + +## Host & GIT variables + +| Variable | Local dev | CI | Comment | +|-------------------|-----------|------------|-----------------------------------------| +| HOST_USER_ID | Host UID | | User id of the host user. | +| HOST_GROUP_ID | Host GID | | Group id of the host user. | +| HOST_OS | | linux | OS of the Host (darwin/linux/windows). | +| COMMIT_SHA | | GITHUB_SHA | SHA of the commit of the build is run | + +## In-container environment initialization + +| Variable | Local dev | CI | Comment | +|---------------------------------|-----------|-----------|-----------------------------------------------------------------------------| +| SKIP_ENVIRONMENT_INITIALIZATION | false (*) | false (*) | Skip initialization of test environment (*) set to true in pre-commits. | +| SKIP_IMAGE_UPGRADE_CHECK | false (*) | false (*) | Skip checking if image should be upgraded (*) set to true in pre-commits. | +| SKIP_PROVIDERS_TESTS | false | false | Skip running provider integration tests. | +| SKIP_SSH_SETUP | false | false (*) | Skip setting up SSH server for tests. (*) set to true in GitHub CodeSpaces. | +| VERBOSE_COMMANDS | false | false | Whether every command executed in docker should be printed. | + +# Image build variables + +| Variable | Local dev | CI | Comment | +|---------------------------------|-----------|-----------|--------------------------------------------------------------------| +| UPGRADE_TO_NEWER_DEPENDENCIES | false | false (*) | Whether dependencies should be upgraded. (*) set in CI when needed | + +# Upgrade to newer dependencies + +By default, we are using a tested set of dependency constraints stored in separated "orphan" branches of the airflow repository +("constraints-main, "constraints-2-0") but when this flag is set to anything but false (for example random value), +they are not used and "eager" upgrade strategy is used when installing dependencies. We set it to true in case of direct +pushes (merges) to main and scheduled builds so that the constraints are tested. In those builds, in case we determine +that the tests pass we automatically push latest set of "tested" constraints to the repository. Setting the value to random +value is best way to assure that constraints are upgraded even if there is no change to pyproject.toml +This way our constraints are automatically tested and updated whenever new versions of libraries are released. +(*) true in case of direct pushes and scheduled builds + +---- + +**Thank you** for reading this far. We hope that you have learned a lot about Airflow's CI. diff --git a/dev/breeze/doc/ci/08_running_ci_locally.md b/dev/breeze/doc/ci/08_running_ci_locally.md deleted file mode 100644 index 4fd0a7c993799..0000000000000 --- a/dev/breeze/doc/ci/08_running_ci_locally.md +++ /dev/null @@ -1,141 +0,0 @@ - - - - -**Table of Contents** *generated with [DocToc](https://github.com/thlorenz/doctoc)* - -- [Running the CI Jobs locally](#running-the-ci-jobs-locally) -- [Upgrade to newer dependencies](#upgrade-to-newer-dependencies) - - - -# Running the CI Jobs locally - -The main goal of the CI philosophy we have that no matter how complex -the test and integration infrastructure, as a developer you should be -able to reproduce and re-run any of the failed checks locally. One part -of it are pre-commit checks, that allow you to run the same static -checks in CI and locally, but another part is the CI environment which -is replicated locally with Breeze. - -You can read more about Breeze in -[README.rst](../README.rst) but in essence it is a script -that allows you to re-create CI environment in your local development -instance and interact with it. In its basic form, when you do -development you can run all the same tests that will be run in CI - but -locally, before you submit them as PR. Another use case where Breeze is -useful is when tests fail on CI. You can take the full `COMMIT_SHA` of -the failed build pass it as `--image-tag` parameter of Breeze and it -will download the very same version of image that was used in CI and run -it locally. This way, you can very easily reproduce any failed test that -happens in CI - even if you do not check out the sources connected with -the run. - -All our CI jobs are executed via `breeze` commands. You can replicate -exactly what our CI is doing by running the sequence of corresponding -`breeze` command. Make sure however that you look at both: - -- flags passed to `breeze` commands -- environment variables used when `breeze` command is run - this is - useful when we want to set a common flag for all `breeze` commands in - the same job or even the whole workflow. For example `VERBOSE` - variable is set to `true` for all our workflows so that more detailed - information about internal commands executed in CI is printed. - -In the output of the CI jobs, you will find both - the flags passed and -environment variables set. - -You can read more about it in [Breeze](../README.rst) and -[Testing](../../../../contributing-docs/09_testing.rst) - -Since we store images from every CI run, you should be able easily -reproduce any of the CI tests problems locally. You can do it by pulling -and using the right image and running it with the right docker command, -For example knowing that the CI job was for commit -`cd27124534b46c9688a1d89e75fcd137ab5137e3`: - -``` bash -docker pull ghcr.io/apache/airflow/main/ci/python3.9:cd27124534b46c9688a1d89e75fcd137ab5137e3 - -docker run -it ghcr.io/apache/airflow/main/ci/python3.9:cd27124534b46c9688a1d89e75fcd137ab5137e3 -``` - -But you usually need to pass more variables and complex setup if you -want to connect to a database or enable some integrations. Therefore it -is easiest to use [Breeze](../README.rst) for that. For -example if you need to reproduce a MySQL environment in python 3.9 -environment you can run: - -``` bash -breeze --image-tag cd27124534b46c9688a1d89e75fcd137ab5137e3 --python 3.9 --backend mysql -``` - -You will be dropped into a shell with the exact version that was used -during the CI run and you will be able to run pytest tests manually, -easily reproducing the environment that was used in CI. Note that in -this case, you do not need to checkout the sources that were used for -that run - they are already part of the image - but remember that any -changes you make in those sources are lost when you leave the image as -the sources are not mapped from your host machine. - -Depending whether the scripts are run locally via -[Breeze](../README.rst) or whether they are run in -`Build Images` or `Tests` workflows they can take different values. - -You can use those variables when you try to reproduce the build locally -(alternatively you can pass those via corresponding command line flags -passed to `breeze shell` command. - -| Variable | Local development | Build Images workflow | CI Workflow | Comment | -|-----------------------------------------|--------------------|------------------------|--------------|--------------------------------------------------------------------------------| -| Basic variables | | | | | -| PYTHON_MAJOR_MINOR_VERSION | | | | Major/Minor version of Python used. | -| DB_RESET | false | true | true | Determines whether database should be reset at the container entry. | -| Forcing answer | | | | | -| ANSWER | | yes | yes | This variable determines if answer to questions should be automatically given. | -| Host variables | | | | | -| HOST_USER_ID | | | | User id of the host user. | -| HOST_GROUP_ID | | | | Group id of the host user. | -| HOST_OS | | linux | linux | OS of the Host (darwin/linux/windows). | -| Git variables | | | | | -| COMMIT_SHA | | GITHUB_SHA | GITHUB_SHA | SHA of the commit of the build is run | -| In container environment initialization | | | | | -| SKIP_ENVIRONMENT_INITIALIZATION | false* | false* | false* | Skip initialization of test environment * set to true in pre-commits | -| SKIP_IMAGE_UPGRADE_CHECK | false* | false* | false* | Skip checking if image should be upgraded * set to true in pre-commits | -| SKIP_PROVIDERS_TESTS | false* | false* | false* | Skip running provider integration tests | -| SKIP_SSH_SETUP | false* | false* | false* | Skip setting up SSH server for tests. * set to true in GitHub CodeSpaces | -| VERBOSE_COMMANDS | false | false | false | Determines whether every command executed in docker should be printed. | -| Image build variables | | | | | -| UPGRADE_TO_NEWER_DEPENDENCIES | false | false | false* | Determines whether the build should attempt to upgrade dependencies. | - -# Upgrade to newer dependencies - -By default we are using a tested set of dependency constraints stored in separated "orphan" branches of the airflow repository -("constraints-main, "constraints-2-0") but when this flag is set to anything but false (for example random value), -they are not used used and "eager" upgrade strategy is used when installing dependencies. We set it to true in case of direct -pushes (merges) to main and scheduled builds so that the constraints are tested. In those builds, in case we determine -that the tests pass we automatically push latest set of "tested" constraints to the repository. Setting the value to random -value is best way to assure that constraints are upgraded even if there is no change to pyproject.toml -This way our constraints are automatically tested and updated whenever new versions of libraries are released. -(*) true in case of direct pushes and scheduled builds - ----- - -**Thank you** for reading this far. We hope that you have learned a lot about Airflow's CI. diff --git a/dev/breeze/doc/ci/README.md b/dev/breeze/doc/ci/README.md index f52376e18b125..bf20a3a700923 100644 --- a/dev/breeze/doc/ci/README.md +++ b/dev/breeze/doc/ci/README.md @@ -24,6 +24,5 @@ This directory contains detailed design of the Airflow CI setup. * [GitHub Variables](03_github_variables.md) - contains description of the GitHub variables used in CI * [Selective checks](04_selective_checks.md) - contains description of the selective checks performed in CI * [Workflows](05_workflows.md) - contains description of the workflows used in CI -* [Diagrams](06_diagrams.md) - contains diagrams of the CI workflows -* [Debugging](07_debugging.md) - contains description of debugging CI issues -* [Running CI Locally](08_running_ci_locally.md) - contains description of running CI locally +* [Debugging](06_debugging.md) - contains description of debugging CI issues +* [Running CI Locally](07_running_ci_locally.md) - contains description of running CI locally diff --git a/dev/breeze/doc/images/image_artifacts.png b/dev/breeze/doc/images/image_artifacts.png new file mode 100644 index 0000000000000000000000000000000000000000..485a6a2c9cf10ebdc0241b317c089d5be40f9079 GIT binary patch literal 47666 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 literal 0 HcmV?d00001 diff --git a/dev/breeze/doc/images/output_ci-image.svg b/dev/breeze/doc/images/output_ci-image.svg index 8d7c7893f55c8..6fc5b425c5c7a 100644 --- a/dev/breeze/doc/images/output_ci-image.svg +++ b/dev/breeze/doc/images/output_ci-image.svg @@ -1,4 +1,4 @@ - +