|
@@ -25,7 +25,69 @@ on:
|
|
|
required: true
|
|
required: true
|
|
|
default: ''
|
|
default: ''
|
|
|
|
|
|
|
|
|
|
+env:
|
|
|
|
|
+ BASE_IMAGE_FOR_HASH_EQUIVALENCE_TEST: nikolaik/python-nodejs:python3.11-nodejs22
|
|
|
|
|
+
|
|
|
jobs:
|
|
jobs:
|
|
|
|
|
+ # Builds the OpenHands Docker images
|
|
|
|
|
+ ghcr_build_app:
|
|
|
|
|
+ name: Build App Image
|
|
|
|
|
+ runs-on: ubuntu-latest
|
|
|
|
|
+ permissions:
|
|
|
|
|
+ contents: read
|
|
|
|
|
+ packages: write
|
|
|
|
|
+ outputs:
|
|
|
|
|
+ hash_from_app_image: ${{ steps.get_hash_in_app_image.outputs.hash_from_app_image }}
|
|
|
|
|
+ steps:
|
|
|
|
|
+ - name: Checkout
|
|
|
|
|
+ uses: actions/checkout@v4
|
|
|
|
|
+ - name: Free Disk Space (Ubuntu)
|
|
|
|
|
+ uses: jlumbroso/free-disk-space@main
|
|
|
|
|
+ with:
|
|
|
|
|
+ # this might remove tools that are actually needed,
|
|
|
|
|
+ # if set to "true" but frees about 6 GB
|
|
|
|
|
+ tool-cache: true
|
|
|
|
|
+ # all of these default to true, but feel free to set to
|
|
|
|
|
+ # "false" if necessary for your workflow
|
|
|
|
|
+ android: true
|
|
|
|
|
+ dotnet: true
|
|
|
|
|
+ haskell: true
|
|
|
|
|
+ large-packages: true
|
|
|
|
|
+ docker-images: false
|
|
|
|
|
+ swap-storage: true
|
|
|
|
|
+ - name: Set up QEMU
|
|
|
|
|
+ uses: docker/setup-qemu-action@v3
|
|
|
|
|
+ - name: Login to GHCR
|
|
|
|
|
+ uses: docker/login-action@v3
|
|
|
|
|
+ with:
|
|
|
|
|
+ registry: ghcr.io
|
|
|
|
|
+ username: ${{ github.repository_owner }}
|
|
|
|
|
+ password: ${{ secrets.GITHUB_TOKEN }}
|
|
|
|
|
+ - name: Set up Docker Buildx
|
|
|
|
|
+ id: buildx
|
|
|
|
|
+ uses: docker/setup-buildx-action@v3
|
|
|
|
|
+ - name: Build and push app image
|
|
|
|
|
+ if: "!github.event.pull_request.head.repo.fork"
|
|
|
|
|
+ run: |
|
|
|
|
|
+ ./containers/build.sh openhands ${{ github.repository_owner }} --push
|
|
|
|
|
+ - name: Build app image
|
|
|
|
|
+ if: "github.event.pull_request.head.repo.fork"
|
|
|
|
|
+ run: |
|
|
|
|
|
+ ./containers/build.sh openhands image ${{ github.repository_owner }}
|
|
|
|
|
+ - name: Get hash in App Image
|
|
|
|
|
+ id: get_hash_in_app_image
|
|
|
|
|
+ run: |
|
|
|
|
|
+ # Lowercase the repository owner
|
|
|
|
|
+ export REPO_OWNER=${{ github.repository_owner }}
|
|
|
|
|
+ REPO_OWNER=$(echo $REPO_OWNER | tr '[:upper:]' '[:lower:]')
|
|
|
|
|
+ # Run the build script in the app image
|
|
|
|
|
+ docker run -e SANDBOX_USER_ID=0 -v /var/run/docker.sock:/var/run/docker.sock ghcr.io/${REPO_OWNER}/openhands:${{ github.sha }} /bin/bash -c "mkdir -p containers/runtime; python3 openhands/runtime/utils/runtime_build.py --base_image ${{ env.BASE_IMAGE_FOR_HASH_EQUIVALENCE_TEST }} --build_folder containers/runtime --force_rebuild" 2>&1 | tee docker-outputs.txt
|
|
|
|
|
+ # Get the hash from the build script
|
|
|
|
|
+ hash_from_app_image=$(cat docker-outputs.txt | grep "Hash for docker build directory" | awk -F "): " '{print $2}' | uniq | head -n1)
|
|
|
|
|
+ echo "hash_from_app_image=$hash_from_app_image" >> $GITHUB_OUTPUT
|
|
|
|
|
+ echo "Hash from app image: $hash_from_app_image"
|
|
|
|
|
+
|
|
|
|
|
+
|
|
|
# Builds the runtime Docker images
|
|
# Builds the runtime Docker images
|
|
|
ghcr_build_runtime:
|
|
ghcr_build_runtime:
|
|
|
name: Build Image
|
|
name: Build Image
|
|
@@ -104,6 +166,56 @@ jobs:
|
|
|
name: runtime-${{ matrix.base_image.tag }}
|
|
name: runtime-${{ matrix.base_image.tag }}
|
|
|
path: /tmp/runtime-${{ matrix.base_image.tag }}.tar
|
|
path: /tmp/runtime-${{ matrix.base_image.tag }}.tar
|
|
|
|
|
|
|
|
|
|
+ verify_hash_equivalence_in_runtime_and_app:
|
|
|
|
|
+ name: Verify Hash Equivalence in Runtime and Docker images
|
|
|
|
|
+ runs-on: ubuntu-latest
|
|
|
|
|
+ needs: [ghcr_build_runtime, ghcr_build_app]
|
|
|
|
|
+ strategy:
|
|
|
|
|
+ fail-fast: false
|
|
|
|
|
+ matrix:
|
|
|
|
|
+ base_image: ['nikolaik']
|
|
|
|
|
+ steps:
|
|
|
|
|
+ - uses: actions/checkout@v4
|
|
|
|
|
+ - name: Cache Poetry dependencies
|
|
|
|
|
+ uses: actions/cache@v4
|
|
|
|
|
+ with:
|
|
|
|
|
+ path: |
|
|
|
|
|
+ ~/.cache/pypoetry
|
|
|
|
|
+ ~/.virtualenvs
|
|
|
|
|
+ key: ${{ runner.os }}-poetry-${{ hashFiles('**/poetry.lock') }}
|
|
|
|
|
+ restore-keys: |
|
|
|
|
|
+ ${{ runner.os }}-poetry-
|
|
|
|
|
+ - name: Set up Python
|
|
|
|
|
+ uses: actions/setup-python@v5
|
|
|
|
|
+ with:
|
|
|
|
|
+ python-version: '3.11'
|
|
|
|
|
+ - name: Install poetry via pipx
|
|
|
|
|
+ run: pipx install poetry
|
|
|
|
|
+ - name: Install Python dependencies using Poetry
|
|
|
|
|
+ run: make install-python-dependencies
|
|
|
|
|
+ - name: Get hash in App Image
|
|
|
|
|
+ run: |
|
|
|
|
|
+ echo "Hash from app image: ${{ needs.ghcr_build_app.outputs.hash_from_app_image }}"
|
|
|
|
|
+ echo "hash_from_app_image=${{ needs.ghcr_build_app.outputs.hash_from_app_image }}" >> $GITHUB_ENV
|
|
|
|
|
+
|
|
|
|
|
+ - name: Get hash using code (development mode)
|
|
|
|
|
+ run: |
|
|
|
|
|
+ mkdir -p containers/runtime
|
|
|
|
|
+ poetry run python3 openhands/runtime/utils/runtime_build.py --base_image ${{ env.BASE_IMAGE_FOR_HASH_EQUIVALENCE_TEST }} --build_folder containers/runtime --force_rebuild > output.txt 2>&1
|
|
|
|
|
+ hash_from_code=$(cat output.txt | grep "Hash for docker build directory" | awk -F "): " '{print $2}' | uniq | head -n1)
|
|
|
|
|
+ echo "hash_from_code=$hash_from_code" >> $GITHUB_ENV
|
|
|
|
|
+
|
|
|
|
|
+ - name: Compare hashes
|
|
|
|
|
+ run: |
|
|
|
|
|
+ echo "Hash from App Image: ${{ env.hash_from_app_image }}"
|
|
|
|
|
+ echo "Hash from Code: ${{ env.hash_from_code }}"
|
|
|
|
|
+ if [ "${{ env.hash_from_app_image }}" = "${{ env.hash_from_code }}" ]; then
|
|
|
|
|
+ echo "Hashes match!"
|
|
|
|
|
+ else
|
|
|
|
|
+ echo "Hashes do not match!"
|
|
|
|
|
+ exit 1
|
|
|
|
|
+ fi
|
|
|
|
|
+
|
|
|
# Run unit tests with the EventStream runtime Docker images as root
|
|
# Run unit tests with the EventStream runtime Docker images as root
|
|
|
test_runtime_root:
|
|
test_runtime_root:
|
|
|
name: RT Unit Tests (Root)
|
|
name: RT Unit Tests (Root)
|
|
@@ -341,7 +453,7 @@ jobs:
|
|
|
name: All Runtime Tests Passed
|
|
name: All Runtime Tests Passed
|
|
|
if: ${{ !cancelled() && !contains(needs.*.result, 'failure') && !contains(needs.*.result, 'cancelled') }}
|
|
if: ${{ !cancelled() && !contains(needs.*.result, 'failure') && !contains(needs.*.result, 'cancelled') }}
|
|
|
runs-on: ubuntu-latest
|
|
runs-on: ubuntu-latest
|
|
|
- needs: [test_runtime_root, test_runtime_oh, runtime_integration_tests_on_linux]
|
|
|
|
|
|
|
+ needs: [test_runtime_root, test_runtime_oh, runtime_integration_tests_on_linux, verify_hash_equivalence_in_runtime_and_app]
|
|
|
steps:
|
|
steps:
|
|
|
- name: All tests passed
|
|
- name: All tests passed
|
|
|
run: echo "All runtime tests have passed successfully!"
|
|
run: echo "All runtime tests have passed successfully!"
|
|
@@ -350,7 +462,7 @@ jobs:
|
|
|
name: All Runtime Tests Passed
|
|
name: All Runtime Tests Passed
|
|
|
if: ${{ cancelled() || contains(needs.*.result, 'failure') || contains(needs.*.result, 'cancelled') }}
|
|
if: ${{ cancelled() || contains(needs.*.result, 'failure') || contains(needs.*.result, 'cancelled') }}
|
|
|
runs-on: ubuntu-latest
|
|
runs-on: ubuntu-latest
|
|
|
- needs: [test_runtime_root, test_runtime_oh, runtime_integration_tests_on_linux]
|
|
|
|
|
|
|
+ needs: [test_runtime_root, test_runtime_oh, runtime_integration_tests_on_linux, verify_hash_equivalence_in_runtime_and_app]
|
|
|
steps:
|
|
steps:
|
|
|
- name: Some tests failed
|
|
- name: Some tests failed
|
|
|
run: |
|
|
run: |
|