From a4395bf2a00b4036a2b109cacb2d28de29a0beb2 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. Fixes: #42999 Fixes: #43268 --- .../actions/checkout_target_commit/action.yml | 81 ------ .../actions/prepare_all_ci_images/action.yml | 94 +++++++ .../prepare_breeze_and_image/action.yml | 37 ++- .../actions/prepare_single_image/action.yml | 51 ++++ .../workflows/additional-ci-image-checks.yml | 9 +- .../workflows/additional-prod-image-tests.yml | 28 +- .github/workflows/basic-tests.yml | 1 - .github/workflows/build-images.yml | 264 ------------------ .github/workflows/ci-image-build.yml | 65 ++--- .github/workflows/ci-image-checks.yml | 35 +-- .github/workflows/ci.yml | 163 ++--------- .github/workflows/finalize-tests.yml | 9 - .github/workflows/generate-constraints.yml | 25 +- .github/workflows/helm-tests.yml | 11 +- .../workflows/integration-system-tests.yml | 25 +- .github/workflows/k8s-tests.yml | 64 ++--- .github/workflows/prod-image-build.yml | 103 +++---- .github/workflows/prod-image-extra-checks.yml | 5 - .github/workflows/release_dockerhub_image.yml | 2 +- .github/workflows/run-unit-tests.yml | 11 +- .github/workflows/special-tests.yml | 10 - .github/workflows/task-sdk-tests.yml | 11 +- .github/workflows/test-provider-packages.yml | 19 +- Dockerfile | 129 ++------- Dockerfile.ci | 195 +++---------- dev/breeze/doc/06_managing_docker_images.rst | 64 ++++- dev/breeze/doc/ci/02_images.md | 21 +- dev/breeze/doc/ci/05_workflows.md | 4 +- dev/breeze/doc/ci/08_running_ci_locally.md | 57 +--- 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 | 136 +++++++++ .../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 | 124 ++++++++ .../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 | 146 ++++++---- .../commands/ci_image_commands_config.py | 29 +- .../commands/common_image_options.py | 30 -- .../airflow_breeze/commands/common_options.py | 18 +- .../commands/developer_commands.py | 24 +- .../commands/developer_commands_config.py | 3 - .../commands/kubernetes_commands.py | 44 +-- .../commands/kubernetes_commands_config.py | 2 - .../commands/production_image_commands.py | 120 ++++++-- .../production_image_commands_config.py | 28 +- .../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/install_airflow.sh | 16 +- ...ll_airflow_dependencies_from_branch_tip.sh | 103 ------- scripts/in_container/_in_container_utils.sh | 10 +- 126 files changed, 2175 insertions(+), 2765 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_image/action.yml delete mode 100644 .github/workflows/build-images.yml 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..893c1094295ee --- /dev/null +++ b/.github/actions/prepare_all_ci_images/action.yml @@ -0,0 +1,94 @@ +# 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 images' +description: 'Recreates current python image from artifacts' +inputs: + pull-image-type: + description: 'Which image type to prepare' + default: "CI" + 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 + - name: "Restore CI docker images ${{ inputs.platform }}-3.8" + uses: ./.github/actions/prepare_single_image + with: + pull-image-type: ${{ inputs.pull-image-type }} + platform: ${{ inputs.platform }} + python: "3.8" + python-versions-list-as-string: ${{ inputs.python-versions-list-as-string }} + - name: "Restore CI docker images ${{ inputs.platform }}-3.9" + uses: ./.github/actions/prepare_single_image + with: + pull-image-type: ${{ inputs.pull-image-type }} + platform: ${{ inputs.platform }} + python: "3.9" + python-versions-list-as-string: ${{ inputs.python-versions-list-as-string }} + - name: "Restore CI docker images ${{ inputs.platform }}-3.10" + uses: ./.github/actions/prepare_single_image + with: + pull-image-type: ${{ inputs.pull-image-type }} + platform: ${{ inputs.platform }} + python: "3.10" + python-versions-list-as-string: ${{ inputs.python-versions-list-as-string }} + - name: "Restore CI docker images ${{ inputs.platform }}-3.11" + uses: ./.github/actions/prepare_single_image + with: + pull-image-type: ${{ inputs.pull-image-type }} + platform: ${{ inputs.platform }} + python: "3.11" + python-versions-list-as-string: ${{ inputs.python-versions-list-as-string }} + - name: "Restore CI docker images ${{ inputs.platform }}-3.12" + uses: ./.github/actions/prepare_single_image + with: + pull-image-type: ${{ inputs.pull-image-type }} + platform: ${{ inputs.platform }} + python: "3.12" + python-versions-list-as-string: ${{ inputs.python-versions-list-as-string }} + - name: "Load CI image ${{ inputs.platform }}:${{ inputs.python-versions-list-as-string }}" + run: | + for PYTHON in ${{ inputs.python-versions-list-as-string }}; do + breeze ci-image load --platform ${{ inputs.platform }} --python ${PYTHON} + rm -rf /tmp/-*${PYTHON}.tar + done + shell: bash + if: inputs.pull-image-type == 'CI' + - name: "Load PROD image ${{ inputs.platform }}${{ env.PYTHON_MAJOR_MINOR_VERSION }}" + run: | + for PYTHON in ${{ inputs.python-versions-list-as-string }}; do + breeze ci-image load --platform ${{ inputs.platform }} --python ${PYTHON} + rm -rf /tmp/-*${PYTHON}.tar + done + shell: bash + if: inputs.pull-image-type == 'PROD' diff --git a/.github/actions/prepare_breeze_and_image/action.yml b/.github/actions/prepare_breeze_and_image/action.yml index 41aa17092d589..ca06266c6ef90 100644 --- a/.github/actions/prepare_breeze_and_image/action.yml +++ b/.github/actions/prepare_breeze_and_image/action.yml @@ -16,12 +16,15 @@ # 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 + description: 'Which image type to prepare' + 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 +32,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 }}-${{ 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/" + if: inputs.pull-image-type == 'CI' + - name: "Load CI image ${{ inputs.platform }}${{ env.PYTHON_MAJOR_MINOR_VERSION }}" + run: breeze ci-image load --platform ${{ inputs.platform }} 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 }} + - 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/" + if: inputs.pull-image-type == 'PROD' + - name: "Load PROD image ${{ inputs.platform }}${{ env.PYTHON_MAJOR_MINOR_VERSION }}" + run: breeze prod-image load --platform ${{ inputs.platform }} shell: bash - run: breeze prod-image pull --tag-as-latest if: inputs.pull-image-type == 'PROD' diff --git a/.github/actions/prepare_single_image/action.yml b/.github/actions/prepare_single_image/action.yml new file mode 100644 index 0000000000000..011564b8e4ebc --- /dev/null +++ b/.github/actions/prepare_single_image/action.yml @@ -0,0 +1,51 @@ +# 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 }}${{ env.PYTHON_MAJOR_MINOR_VERSION }}" + run: breeze ci-image load --platform "${{ inputs.platform }}" --python "${{ inputs.python }}" + shell: bash + if: contains(inputs.python-versions-list-as-string, inputs.python) + - name: "Remove saved image ${{ inputs.platform }}-${{ inputs.python }}" + run: rm -f /tmp/ci-image-save-*-${{ 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..b6d9e25a2b802 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 }} @@ -122,11 +116,10 @@ jobs: - 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 + - name: Pull PROD image ${{ inputs.default-python-version}} + run: breeze prod-image pull env: PYTHON_MAJOR_MINOR_VERSION: "${{ inputs.default-python-version }}" - IMAGE_TAG: "${{ inputs.image-tag }}" - name: "Setup python" uses: actions/setup-python@v5 with: @@ -138,7 +131,7 @@ jobs: 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: @@ -146,7 +139,6 @@ jobs: 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 +153,10 @@ 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 & CI image: ${{ inputs.default-python-version}}" + uses: ./.github/actions/prepare_breeze_and_image + with: + platform: "linux/amd64" + 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..f99cbd65cb2e6 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 @@ -105,13 +99,9 @@ jobs: 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 || '' }}" + name: "Build CI ${{ inputs.platform }} image ${{ matrix.python-version }}" # 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 @@ -122,7 +112,7 @@ ${{ inputs.do-build == 'true' && inputs.image-tag || '' }}" # 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 DEFAULT_BRANCH: ${{ inputs.branch }} @@ -137,41 +127,21 @@ ${{ 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" - 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' - name: > Build ${{ inputs.push-image == 'true' && ' & push ' || '' }} - ${{ inputs.platform }}:${{ matrix.python-version }}:${{ inputs.image-tag }} + ${{ inputs.platform }}:${{ matrix.python-version }} run: > - breeze ci-image build --builder airflow_cache --tag-as-latest --image-tag "${{ inputs.image-tag }}" + breeze ci-image build --builder airflow_cache --python "${{ matrix.python-version }}" --platform "${{ inputs.platform }}" env: DOCKER_CACHE: ${{ inputs.docker-cache }} @@ -189,7 +159,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 ${{ matrix.python-version }}" + run: breeze ci-image save --python "${{ matrix.python-version }}" --platform "${{ inputs.platform }}" + if: inputs.upload-image-artifact == 'true' + - name: "Stash CI docker image ${{ matrix.python-version }}" + uses: apache/infrastructure-actions/stash/save@c94b890bbedc2fc61466d28e6bd9966bc6c6643c + with: + key: "ci-image-save-${{ inputs.platform }}-${{ matrix.python-version }}" + path: "/tmp/ci-image-save-*-${{ matrix.python-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..b5eb98e136f91 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,10 @@ 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" id: breeze - name: "Install pre-commit" uses: ./.github/actions/install-pre-commit @@ -165,7 +160,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 +169,10 @@ 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" id: breeze - name: "Install pre-commit" uses: ./.github/actions/install-pre-commit @@ -208,7 +202,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 +214,10 @@ 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" - uses: actions/cache@v4 id: cache-doc-inventories with: @@ -254,7 +247,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 +275,10 @@ 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" - name: "Publish docs" run: > breeze release-management publish-docs --override-versioned --run-in-parallel @@ -331,7 +325,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 +346,10 @@ 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" - 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..4420152c8c8b8 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,18 @@ 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: 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 +578,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 +602,6 @@ jobs: needs: - build-info - generate-constraints - - wait-for-ci-images - - wait-for-prod-images - ci-image-checks - tests-sqlite - tests-mysql @@ -723,13 +612,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..d73a1d37dc760 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 CI images: ${{ inputs.python-versions-list-as-string}}" + uses: ./.github/actions/prepare_all_images + with: + platform: "linux/amd64" + python-versions-list-as-string: ${{ inputs.python-versions-list-as-string }} + - name: "Verify 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..a54de11a4a933 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,10 @@ 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" - 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..67bc0320a5ee0 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,10 @@ 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" - 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 +116,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 +135,10 @@ 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" - name: "Integration: providers ${{ matrix.integration }}" run: ./scripts/ci/testing/run_integration_tests_with_retry.sh providers "${{ matrix.integration }}" - name: "Post Tests success" @@ -162,7 +156,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 +175,10 @@ 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" - 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..26fc00b87d8da 100644 --- a/.github/workflows/k8s-tests.yml +++ b/.github/workflows/k8s-tests.yml @@ -24,20 +24,12 @@ 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 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 +47,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 +66,25 @@ 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: Restore PROD image ${{ 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/" + - name: Import PROD image ${{ env.PYTHON_MAJOR_MINOR_VERSION }} + run: breeze prod-image load --platform "${{ inputs.platform }}" - name: "Cache bin folder with tools for kubernetes testing" uses: actions/cache@v4 with: @@ -103,26 +95,34 @@ jobs: - 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..9f4a422e2704d 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,20 +178,15 @@ 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 @@ -231,54 +211,34 @@ ${{ 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: "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 ${{ matrix.python-version }}" shell: bash run: > - breeze prod-image build --tag-as-latest --image-tag "${{ inputs.image-tag }}" + breeze prod-image build --commit-sha "${{ github.sha }}" --install-packages-from-context --airflow-constraints-mode constraints-source-providers --use-constraints-for-context-packages --python "${{ matrix.python-version }}" @@ -290,12 +250,12 @@ ${{ 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 ${{ matrix.python-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 }}" env: @@ -306,9 +266,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 ${{ matrix.python-version }} + run: breeze prod-image verify --python "${{ matrix.python-version }}" + - name: "Export PROD docker image ${{ matrix.python-version }}" run: > - breeze prod-image verify --image-tag "${{ inputs.image-tag }}" - --python "${{ matrix.python-version }}" - if: inputs.do-build == 'true' + breeze prod-image save --python "${{ matrix.python-version }}" --platform "${{ inputs.platform }}" + if: inputs.upload-image-artifact == 'true' + - name: "Stash PROD docker image ${{ matrix.python-version }}" + uses: apache/infrastructure-actions/stash/save@c94b890bbedc2fc61466d28e6bd9966bc6c6643c + with: + key: "prod-image-save-${{ inputs.platform }}-${{ matrix.python-version }}" + path: "/tmp/prod-image-save-*-${{ matrix.python-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/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..3a449a375a8c7 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,10 @@ 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" - 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..9baabed3b545c 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,10 @@ 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" - 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..8e8099cc907dd 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,10 @@ 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" - name: "Cleanup dist files" run: rm -fv ./dist/* - name: "Prepare provider documentation" @@ -161,7 +155,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 }}" VERSION_SUFFIX_FOR_PYPI: "dev0" @@ -176,10 +169,10 @@ 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.default-python-version}}" uses: ./.github/actions/prepare_breeze_and_image + with: + platform: "linux/amd64" - 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..74d8eb59435da 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 @@ -1202,7 +1115,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 +1198,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 +1226,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 +1237,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 +1246,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 +1265,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 +1310,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/02_images.md b/dev/breeze/doc/ci/02_images.md index eb3af6ae6ce87..263b37d62b82b 100644 --- a/dev/breeze/doc/ci/02_images.md +++ b/dev/breeze/doc/ci/02_images.md @@ -329,22 +329,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. +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. -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 -``` - -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 +419,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 +429,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 | diff --git a/dev/breeze/doc/ci/05_workflows.md b/dev/breeze/doc/ci/05_workflows.md index 130774a730cb6..63c94871c5d9d 100644 --- a/dev/breeze/doc/ci/05_workflows.md +++ b/dev/breeze/doc/ci/05_workflows.md @@ -206,9 +206,9 @@ code. | 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) | +| Build CI images | Builds 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) | +| Build PROD images | Builds 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 | diff --git a/dev/breeze/doc/ci/08_running_ci_locally.md b/dev/breeze/doc/ci/08_running_ci_locally.md index 4fd0a7c993799..7300c8fde0b84 100644 --- a/dev/breeze/doc/ci/08_running_ci_locally.md +++ b/dev/breeze/doc/ci/08_running_ci_locally.md @@ -36,17 +36,13 @@ 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 +[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. 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. +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 @@ -62,39 +58,16 @@ exactly what our CI is doing by running the sequence of corresponding 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) -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. @@ -127,9 +100,9 @@ passed to `breeze shell` command. # Upgrade to newer dependencies -By default we are using a tested set of dependency constraints stored in separated "orphan" branches of the airflow repository +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 +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 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 @@ - +