| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432 |
- # Workflow that builds, tests and then pushes the docker images to the ghcr.io repository
- name: Build Publish and Test Runtime Image
- # Only run one workflow of the same group at a time.
- # There can be at most one running and one pending job in a concurrency group at any time.
- concurrency:
- group: ${{ github.workflow }}-${{ github.ref }}
- cancel-in-progress: ${{ github.ref != 'refs/heads/main' }}
- on:
- push:
- branches:
- - main
- tags:
- - '*'
- pull_request:
- workflow_dispatch:
- inputs:
- reason:
- description: 'Reason for manual trigger'
- required: true
- default: ''
- jobs:
- # Builds the OpenDevin Docker images
- ghcr_build:
- runs-on: ubuntu-latest
- outputs:
- tags: ${{ steps.capture-tags.outputs.tags }}
- permissions:
- contents: read
- packages: write
- strategy:
- matrix:
- image: ['opendevin']
- platform: ['amd64', 'arm64']
- 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: Set up Docker Buildx
- id: buildx
- uses: docker/setup-buildx-action@v3
- - name: Build and export image
- id: build
- run: ./containers/build.sh ${{ matrix.image }} ${{ github.repository_owner }} ${{ matrix.platform }}
- - name: Capture tags
- id: capture-tags
- run: |
- tags=$(cat tags.txt)
- echo "tags=$tags"
- echo "tags=$tags" >> $GITHUB_OUTPUT
- - name: Upload Docker image as artifact
- uses: actions/upload-artifact@v4
- with:
- name: ${{ matrix.image }}-docker-image-${{ matrix.platform }}
- path: /tmp/${{ matrix.image }}_image_${{ matrix.platform }}.tar
- retention-days: 14
- # Builds the runtime Docker images
- ghcr_build_runtime:
- runs-on: ubuntu-latest
- outputs:
- tags: ${{ steps.capture-tags.outputs.tags }}
- permissions:
- contents: read
- packages: write
- strategy:
- matrix:
- image: ['od_runtime']
- base_image: ['nikolaik/python-nodejs:python3.11-nodejs22']
- platform: ['amd64', 'arm64']
- 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: Set up Docker Buildx
- id: buildx
- uses: docker/setup-buildx-action@v3
- - name: Install poetry via pipx
- run: pipx install poetry
- - name: Set up Python
- uses: actions/setup-python@v5
- with:
- python-version: '3.11'
- cache: 'poetry'
- - name: Install Python dependencies using Poetry
- run: make install-python-dependencies
- - name: Create source distribution and Dockerfile
- run: poetry run python3 opendevin/runtime/utils/runtime_build.py --base_image ${{ matrix.base_image }} --build_folder containers/runtime --force_rebuild
- - name: Build and export image
- id: build
- run: |
- if [ -f 'containers/runtime/Dockerfile' ]; then
- echo 'Dockerfile detected, building runtime image...'
- ./containers/build.sh ${{ matrix.image }} ${{ github.repository_owner }} ${{ matrix.platform }}
- else
- echo 'No Dockerfile detected which means an exact image is already built. Pulling the image and saving it to a tar file...'
- source containers/runtime/config.sh
- echo "$DOCKER_IMAGE_TAG $DOCKER_IMAGE_HASH_TAG" >> tags.txt
- echo "Pulling image $DOCKER_IMAGE/$DOCKER_IMAGE_HASH_TAG to /tmp/${{ matrix.image }}_image_${{ matrix.platform }}.tar"
- docker pull $DOCKER_IMAGE:$DOCKER_IMAGE_HASH_TAG
- docker save $DOCKER_IMAGE:$DOCKER_IMAGE_HASH_TAG -o /tmp/${{ matrix.image }}_image_${{ matrix.platform }}.tar
- fi
- - name: Capture tags
- id: capture-tags
- run: |
- tags=$(cat tags.txt)
- echo "tags=$tags"
- echo "tags=$tags" >> $GITHUB_OUTPUT
- - name: Upload Docker image as artifact
- uses: actions/upload-artifact@v4
- with:
- name: ${{ matrix.image }}-docker-image-${{ matrix.platform }}
- path: /tmp/${{ matrix.image }}_image_${{ matrix.platform }}.tar
- retention-days: 14
- # Run unit tests with the EventStream and Server runtime Docker images
- test_runtime:
- name: Test Runtime
- runs-on: ubuntu-latest
- needs: [ghcr_build_runtime, ghcr_build]
- strategy:
- matrix:
- runtime_type: ['eventstream']
- steps:
- - uses: actions/checkout@v4
- - name: Free Disk Space (Ubuntu)
- uses: jlumbroso/free-disk-space@main
- with:
- # this might remove tools that are actually needed,
- # when 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
- swap-storage: true
- - name: Install poetry via pipx
- run: pipx install poetry
- - name: Set up Python
- uses: actions/setup-python@v5
- with:
- python-version: '3.11'
- cache: 'poetry'
- - name: Install Python dependencies using Poetry
- run: make install-python-dependencies
- - name: Download Runtime Docker image
- if: matrix.runtime_type == 'eventstream'
- uses: actions/download-artifact@v4
- with:
- name: od_runtime-docker-image-amd64
- path: /tmp/
- - name: Download Sandbox Docker image
- if: matrix.runtime_type == 'server'
- uses: actions/download-artifact@v4
- with:
- name: sandbox-docker-image-amd64
- path: /tmp/
- - name: Load Runtime image and run runtime tests
- run: |
- # Load the Docker image and capture the output
- if [ "${{ matrix.runtime_type }}" == "eventstream" ]; then
- output=$(docker load -i /tmp/od_runtime_image_amd64.tar)
- else
- output=$(docker load -i /tmp/sandbox_image_amd64.tar)
- fi
- # Extract the first image name from the output
- image_name=$(echo "$output" | grep -oP 'Loaded image: \K.*' | head -n 1)
- # Print the full name of the image
- echo "Loaded Docker image: $image_name"
- TEST_RUNTIME=${{ matrix.runtime_type }} SANDBOX_USER_ID=$(id -u) SANDBOX_CONTAINER_IMAGE=$image_name TEST_IN_CI=true poetry run pytest --cov=agenthub --cov=opendevin --cov-report=xml -s ./tests/unit/test_runtime.py
- - name: Upload coverage to Codecov
- uses: codecov/codecov-action@v4
- env:
- CODECOV_TOKEN: ${{ secrets.CODECOV_TOKEN }}
- # Run integration tests with the eventstream runtime Docker image
- runtime_integration_tests_on_linux:
- name: Runtime Integration Tests on Linux
- runs-on: ubuntu-latest
- needs: [ghcr_build_runtime]
- strategy:
- fail-fast: false
- matrix:
- python-version: ['3.11']
- # server is tested in a separate workflow
- runtime_type: ['eventstream']
- steps:
- - uses: actions/checkout@v4
- - name: Install poetry via pipx
- run: pipx install poetry
- - name: Set up Python
- uses: actions/setup-python@v5
- with:
- python-version: ${{ matrix.python-version }}
- cache: 'poetry'
- - name: Install Python dependencies using Poetry
- run: make install-python-dependencies
- - name: Download Runtime Docker image
- uses: actions/download-artifact@v4
- with:
- name: od_runtime-docker-image-amd64
- path: /tmp/
- - name: Load runtime image and run integration tests
- run: |
- # Load the Docker image and capture the output
- if [ "${{ matrix.runtime_type }}" == "eventstream" ]; then
- output=$(docker load -i /tmp/od_runtime_image_amd64.tar)
- else
- echo "No Runtime Docker image to load"
- exit 1
- fi
- # Extract the first image name from the output
- image_name=$(echo "$output" | grep -oP 'Loaded image: \K.*' | head -n 1)
- # Print the full name of the image
- echo "Loaded Docker image: $image_name"
- TEST_RUNTIME=${{ matrix.runtime_type }} SANDBOX_USER_ID=$(id -u) SANDBOX_CONTAINER_IMAGE=$image_name TEST_IN_CI=true TEST_ONLY=true ./tests/integration/regenerate.sh
- - name: Upload coverage to Codecov
- uses: codecov/codecov-action@v4
- env:
- CODECOV_TOKEN: ${{ secrets.CODECOV_TOKEN }}
- # Push the OpenDevin and sandbox Docker images to the ghcr.io repository
- ghcr_push:
- runs-on: ubuntu-latest
- needs: [ghcr_build]
- if: github.ref == 'refs/heads/main' || startsWith(github.ref, 'refs/tags/')
- env:
- tags: ${{ needs.ghcr_build.outputs.tags }}
- permissions:
- contents: read
- packages: write
- strategy:
- matrix:
- image: ['opendevin']
- platform: ['amd64', 'arm64']
- steps:
- - name: Checkout code
- uses: actions/checkout@v4
- - name: Login to GHCR
- uses: docker/login-action@v2
- with:
- registry: ghcr.io
- username: ${{ github.repository_owner }}
- password: ${{ secrets.GITHUB_TOKEN }}
- - name: Download Docker images
- uses: actions/download-artifact@v4
- with:
- name: ${{ matrix.image }}-docker-image-${{ matrix.platform }}
- path: /tmp/${{ matrix.platform }}
- - name: Load images and push to registry
- run: |
- mv /tmp/${{ matrix.platform }}/${{ matrix.image }}_image_${{ matrix.platform }}.tar .
- loaded_image=$(docker load -i ${{ matrix.image }}_image_${{ matrix.platform }}.tar | grep "Loaded image:" | head -n 1 | awk '{print $3}')
- echo "loaded image = $loaded_image"
- tags=$(echo ${tags} | tr ' ' '\n')
- image_name=$(echo "ghcr.io/${{ github.repository_owner }}/${{ matrix.image }}" | tr '[:upper:]' '[:lower:]')
- echo "image name = $image_name"
- for tag in $tags; do
- echo "tag = $tag"
- docker tag $loaded_image $image_name:${tag}_${{ matrix.platform }}
- docker push $image_name:${tag}_${{ matrix.platform }}
- done
- # Push the runtime Docker images to the ghcr.io repository
- ghcr_push_runtime:
- runs-on: ubuntu-latest
- needs: [ghcr_build_runtime, test_runtime, runtime_integration_tests_on_linux]
- if: github.ref == 'refs/heads/main' || startsWith(github.ref, 'refs/tags/')
- env:
- RUNTIME_TAGS: ${{ needs.ghcr_build_runtime.outputs.tags }}
- permissions:
- contents: read
- packages: write
- strategy:
- matrix:
- image: ['od_runtime']
- platform: ['amd64', 'arm64']
- steps:
- - name: Checkout code
- uses: actions/checkout@v4
- - name: Free Disk Space (Ubuntu)
- uses: jlumbroso/free-disk-space@main
- with:
- tool-cache: true
- android: true
- dotnet: true
- haskell: true
- large-packages: true
- docker-images: false
- swap-storage: true
- - name: Login to GHCR
- uses: docker/login-action@v2
- with:
- registry: ghcr.io
- username: ${{ github.repository_owner }}
- password: ${{ secrets.GITHUB_TOKEN }}
- - name: Download Docker images
- uses: actions/download-artifact@v4
- with:
- name: ${{ matrix.image }}-docker-image-${{ matrix.platform }}
- path: /tmp/${{ matrix.platform }}
- - name: List downloaded files
- run: |
- ls -la /tmp/${{ matrix.platform }}
- file /tmp/${{ matrix.platform }}/*
- - name: Load images and push to registry
- run: |
- mv /tmp/${{ matrix.platform }}/${{ matrix.image }}_image_${{ matrix.platform }}.tar ./${{ matrix.image }}_image_${{ matrix.platform }}.tar
- if ! loaded_image=$(docker load -i ${{ matrix.image }}_image_${{ matrix.platform }}.tar | grep "Loaded image:" | head -n 1 | awk '{print $3}'); then
- echo "Failed to load Docker image"
- exit 1
- fi
- echo "loaded image = $loaded_image"
- image_name=$(echo "ghcr.io/${{ github.repository_owner }}/${{ matrix.image }}" | tr '[:upper:]' '[:lower:]')
- echo "image name = $image_name"
- echo "$RUNTIME_TAGS" | tr ' ' '\n' | while read -r tag; do
- echo "tag = $tag"
- if [ -n "$image_name" ] && [ -n "$tag" ]; then
- docker tag $loaded_image $image_name:${tag}_${{ matrix.platform }}
- docker push $image_name:${tag}_${{ matrix.platform }}
- else
- echo "Skipping tag and push due to empty image_name or tag"
- fi
- done
- # Creates and pushes the OpenDevin and sandbox Docker image manifests
- create_manifest:
- runs-on: ubuntu-latest
- needs: [ghcr_build, ghcr_push]
- if: github.ref == 'refs/heads/main' || startsWith(github.ref, 'refs/tags/')
- env:
- tags: ${{ needs.ghcr_build.outputs.tags }}
- strategy:
- matrix:
- image: ['opendevin']
- permissions:
- contents: read
- packages: write
- steps:
- - name: Checkout code
- uses: actions/checkout@v4
- - name: Login to GHCR
- uses: docker/login-action@v2
- with:
- registry: ghcr.io
- username: ${{ github.repository_owner }}
- password: ${{ secrets.GITHUB_TOKEN }}
- - name: Create and push multi-platform manifest
- run: |
- image_name=$(echo "ghcr.io/${{ github.repository_owner }}/${{ matrix.image }}" | tr '[:upper:]' '[:lower:]')
- echo "image name = $image_name"
- tags=$(echo ${tags} | tr ' ' '\n')
- for tag in $tags; do
- echo 'tag = $tag'
- docker buildx imagetools create --tag $image_name:$tag \
- $image_name:${tag}_amd64 \
- $image_name:${tag}_arm64
- done
- # Creates and pushes the runtime Docker image manifest
- create_manifest_runtime:
- runs-on: ubuntu-latest
- needs: [ghcr_build_runtime, ghcr_push_runtime]
- if: github.ref == 'refs/heads/main' || startsWith(github.ref, 'refs/tags/')
- env:
- tags: ${{ needs.ghcr_build_runtime.outputs.tags }}
- strategy:
- matrix:
- image: ['od_runtime']
- permissions:
- contents: read
- packages: write
- steps:
- - name: Checkout code
- uses: actions/checkout@v4
- - name: Login to GHCR
- uses: docker/login-action@v2
- with:
- registry: ghcr.io
- username: ${{ github.repository_owner }}
- password: ${{ secrets.GITHUB_TOKEN }}
- - name: Create and push multi-platform manifest
- run: |
- image_name=$(echo "ghcr.io/${{ github.repository_owner }}/${{ matrix.image }}" | tr '[:upper:]' '[:lower:]')
- echo "image name = $image_name"
- tags=$(echo ${tags} | tr ' ' '\n')
- for tag in $tags; do
- echo 'tag = $tag'
- docker buildx imagetools create --tag $image_name:$tag \
- $image_name:${tag}_amd64 \
- $image_name:${tag}_arm64
- done
|