diff --git a/.github/workflows/build-image-pr.yml b/.github/workflows/build-image-pr.yml index 14ff73f5b..7c8806b2b 100644 --- a/.github/workflows/build-image-pr.yml +++ b/.github/workflows/build-image-pr.yml @@ -1,5 +1,6 @@ name: Test container image build and deployment on: + workflow_dispatch: pull_request: paths-ignore: - "LICENSE*" @@ -9,7 +10,7 @@ on: - ".github/ISSUE_TEMPLATE/**" - ".github/dependabot.yml" - "docs/**" - - "clients/python/docs/**" + - "clients/python/**" env: IMG_ORG: kubeflow IMG_REPO: model-registry @@ -31,7 +32,7 @@ jobs: shell: bash env: VERSION: ${{ steps.tags.outputs.tag }} - run: ./scripts/build_deploy.sh + run: make image/build - name: Start Kind Cluster uses: helm/kind-action@v1.10.0 with: @@ -41,25 +42,20 @@ jobs: IMG: "${{ env.IMG_ORG }}/${{ env.IMG_REPO }}:${{ steps.tags.outputs.tag }}" run: | kind load docker-image -n chart-testing ${IMG} - - name: Create Test Registry - env: - IMG: "${{ env.IMG_ORG }}/${{ env.IMG_REPO }}:${{ steps.tags.outputs.tag }}" + - name: Setup kustomize run: | echo "Download kustomize 5.2.1" mkdir $GITHUB_WORKSPACE/kustomize curl -s "https://raw.githubusercontent.com/kubernetes-sigs/kustomize/master/hack/install_kustomize.sh" | bash -s "5.2.1" "$GITHUB_WORKSPACE/kustomize" PATH=$GITHUB_WORKSPACE/kustomize:$PATH echo "Display Kustomize version" - kustomize version - echo "Deploying Model Registry using Manifests; branch ${BRANCH}" - kubectl create namespace kubeflow - cd manifests/kustomize/overlays/db - kustomize edit set image kubeflow/model-registry:latest $IMG - kustomize build | kubectl apply -f - - - name: Wait for Test Registry Deployment + kustomize version + - name: Deploy Model Registry using manifests + env: + IMG: "${{ env.IMG_ORG }}/${{ env.IMG_REPO }}:${{ steps.tags.outputs.tag }}" + run: ./scripts/deploy_on_kind.sh + - name: Deployment logs run: | - kubectl wait --for=condition=available -n kubeflow deployment/model-registry-db --timeout=5m - kubectl wait --for=condition=available -n kubeflow deployment/model-registry-deployment --timeout=5m kubectl logs -n kubeflow deployment/model-registry-deployment - name: Set up Python uses: actions/setup-python@v5 diff --git a/.github/workflows/python-tests.yml b/.github/workflows/python-tests.yml index 750a3b5f4..75c338bad 100644 --- a/.github/workflows/python-tests.yml +++ b/.github/workflows/python-tests.yml @@ -3,6 +3,7 @@ on: push: branches: - "main" + workflow_dispatch: pull_request: paths-ignore: - "LICENSE*" @@ -13,25 +14,17 @@ on: - ".github/dependabot.yml" - "docs/**" jobs: - tests: - name: ${{ matrix.session }} ${{ matrix.python }} + lint: + name: ${{ matrix.session }} runs-on: ubuntu-latest strategy: fail-fast: false matrix: python: ["3.12"] - session: [lint, tests, mypy, docs-build] - include: - - python: "3.9" - session: tests - - python: "3.10" - session: tests - - python: "3.11" - session: tests + session: [lint, mypy] env: NOXSESSION: ${{ matrix.session }} FORCE_COLOR: "1" - PRE_COMMIT_COLOR: "always" steps: - name: Check out the repository uses: actions/checkout@v4 @@ -61,27 +54,154 @@ jobs: pipx install --pip-args=--constraint=${{ github.workspace }}/.github/workflows/constraints.txt nox pipx inject --pip-args=--constraint=${{ github.workspace }}/.github/workflows/constraints.txt nox nox-poetry nox --version - - name: Run Nox + - name: Nox lint working-directory: clients/python run: | - if [[ ${{ matrix.session }} == "tests" ]]; then - make build-mr - nox --python=${{ matrix.python }} -- --cov-report=xml - poetry build - elif [[ ${{ matrix.session }} == "mypy" ]]; then + if [[ ${{ matrix.session }} == "mypy" ]]; then nox --python=${{ matrix.python }} ||\ echo "::error title='mypy failure'::Check the logs for more details" else nox --python=${{ matrix.python }} fi + + test: + name: Test against Py ${{ matrix.python }} + runs-on: ubuntu-latest + strategy: + fail-fast: false + matrix: + python: ["3.12", "3.11", "3.10", "3.9"] + env: + FORCE_COLOR: "1" + IMG_ORG: kubeflow + IMG_REPO: model-registry + steps: + - name: Check out the repository + uses: actions/checkout@v4 + - name: Set up Python ${{ matrix.python }} + uses: actions/setup-python@v5 + with: + python-version: ${{ matrix.python }} + - name: Upgrade pip + run: | + pip install --constraint=.github/workflows/constraints.txt pip + pip --version + - name: Upgrade pip in virtual environments + shell: python + run: | + import os + import pip + + with open(os.environ["GITHUB_ENV"], mode="a") as io: + print(f"VIRTUALENV_PIP={pip.__version__}", file=io) + - name: Install Poetry + # use absolute path as recommended with: https://github.com/pypa/pipx/issues/1331 + run: | + pipx install --pip-args=--constraint=${{ github.workspace }}/.github/workflows/constraints.txt poetry + poetry --version + - name: Install Nox + run: | + pipx install --pip-args=--constraint=${{ github.workspace }}/.github/workflows/constraints.txt nox + pipx inject --pip-args=--constraint=${{ github.workspace }}/.github/workflows/constraints.txt nox nox-poetry + nox --version + - name: Nox test + working-directory: clients/python + run: | + kubectl port-forward -n kubeflow service/model-registry-service 8080:8080 & + sleep 2 + nox --python=${{ matrix.python }} --session=tests -- --cov-report=xml + - name: Generate Tag + shell: bash + id: tags + run: | + commit_sha=${{ github.event.after }} + tag=main-${commit_sha:0:7} + echo "tag=${tag}" >> $GITHUB_OUTPUT + - name: Build Image + shell: bash + env: + IMG_VERSION: ${{ steps.tags.outputs.tag }} + run: make image/build + - name: Start Kind Cluster + uses: helm/kind-action@v1.10.0 + with: + node_image: "kindest/node:v1.27.11" + - name: Load Local Registry Test Image + if: matrix.session == 'tests' + env: + IMG: "docker.io/${{ env.IMG_ORG }}/${{ env.IMG_REPO }}:${{ steps.tags.outputs.tag }}" + run: | + kind load docker-image -n chart-testing ${IMG} + - name: Setup kustomize + run: | + echo "Download kustomize 5.2.1" + mkdir $GITHUB_WORKSPACE/kustomize + curl -s "https://raw.githubusercontent.com/kubernetes-sigs/kustomize/master/hack/install_kustomize.sh" | bash -s "5.2.1" "$GITHUB_WORKSPACE/kustomize" + PATH=$GITHUB_WORKSPACE/kustomize:$PATH + echo "Display Kustomize version" + kustomize version + - name: Deploy Model Registry using manifests + env: + IMG: "docker.io/${{ env.IMG_ORG }}/${{ env.IMG_REPO }}:${{ steps.tags.outputs.tag }}" + run: ./scripts/deploy_on_kind.sh + - name: Nox test end-to-end + working-directory: clients/python + run: | + kubectl port-forward -n kubeflow service/model-registry-service 8080:8080 & + sleep 2 + nox --python=${{ matrix.python }} --session=e2e-tests -- --cov-report=xml + + docs-build: + name: ${{ matrix.session }} + runs-on: ubuntu-latest + strategy: + fail-fast: false + matrix: + python: ["3.12"] + session: [docs-build] + env: + NOXSESSION: ${{ matrix.session }} + FORCE_COLOR: "1" + steps: + - name: Check out the repository + uses: actions/checkout@v4 + - name: Set up Python ${{ matrix.python }} + uses: actions/setup-python@v5 + with: + python-version: ${{ matrix.python }} + - name: Upgrade pip + run: | + pip install --constraint=.github/workflows/constraints.txt pip + pip --version + - name: Upgrade pip in virtual environments + shell: python + run: | + import os + import pip + + with open(os.environ["GITHUB_ENV"], mode="a") as io: + print(f"VIRTUALENV_PIP={pip.__version__}", file=io) + - name: Install Poetry + # use absolute path as recommended with: https://github.com/pypa/pipx/issues/1331 + run: | + pipx install --pip-args=--constraint=${{ github.workspace }}/.github/workflows/constraints.txt poetry + poetry --version + - name: Install Nox + run: | + pipx install --pip-args=--constraint=${{ github.workspace }}/.github/workflows/constraints.txt nox + pipx inject --pip-args=--constraint=${{ github.workspace }}/.github/workflows/constraints.txt nox nox-poetry + nox --version + - name: Run Nox + working-directory: clients/python + run: | + nox --python=${{ matrix.python }} + poetry build - name: Upload dist - if: matrix.session == 'tests' && matrix.python == '3.12' uses: actions/upload-artifact@v4 with: name: py-dist path: clients/python/dist - name: Upload documentation - if: matrix.session == 'docs-build' uses: actions/upload-artifact@v4 with: name: py-docs diff --git a/clients/python/Makefile b/clients/python/Makefile index e5412a9bb..dde6612ab 100644 --- a/clients/python/Makefile +++ b/clients/python/Makefile @@ -1,6 +1,5 @@ all: install tidy -IMG_REGISTRY ?= docker.io IMG_VERSION ?= latest .PHONY: install @@ -14,12 +13,13 @@ install: clean: rm -rf src/mr_openapi -.PHONY: build-mr -build-mr: - cd ../../ && IMG_REGISTRY=${IMG_REGISTRY} IMG_VERSION=${IMG_VERSION} make image/build +.PHONY: deploy-latest-mr +deploy-latest-mr: + cd ../../ && IMG_VERSION=${IMG_VERSION} make image/build && LOCAL=1 ./scripts/deploy_on_kind.sh + kubectl port-forward -n kubeflow services/model-registry-service 8080:8080 & .PHONY: test-e2e -test-e2e: build-mr +test-e2e: deploy-latest-mr poetry run pytest --e2e -s .PHONY: test diff --git a/clients/python/noxfile.py b/clients/python/noxfile.py index 195631e5a..807acb9f4 100644 --- a/clients/python/noxfile.py +++ b/clients/python/noxfile.py @@ -52,22 +52,30 @@ def mypy(session: Session) -> None: @session(python=python_versions) def tests(session: Session) -> None: + """Run the test suite.""" + session.install(".") + session.install( + "pytest", + "pytest-asyncio", + ) + session.run( + "pytest", + *session.posargs, + ) + + +@session(python=python_versions) +def e2e_tests(session: Session) -> None: """Run the test suite.""" session.install(".") session.install( "coverage[toml]", "pytest", "pytest-asyncio", - "nest-asyncio", "pytest-cov", - "pygments", "huggingface-hub", ) try: - session.run( - "pytest", - *session.posargs, - ) session.run( "pytest", "--e2e", diff --git a/clients/python/tests/conftest.py b/clients/python/tests/conftest.py index b480379ce..d92628efb 100644 --- a/clients/python/tests/conftest.py +++ b/clients/python/tests/conftest.py @@ -6,7 +6,7 @@ import time from contextlib import asynccontextmanager from pathlib import Path -from time import sleep +from urllib.parse import urlparse import pytest import requests @@ -29,13 +29,14 @@ def pytest_collection_modifyitems(config, items): continue -REGISTRY_HOST = "http://localhost" -REGISTRY_PORT = 8080 -REGISTRY_URL = f"{REGISTRY_HOST}:{REGISTRY_PORT}" -COMPOSE_FILE = "docker-compose.yaml" -MAX_POLL_TIME = 1200 # the first build is extremely slow if using docker-compose-*local*.yaml for bootstrap of builder image +REGISTRY_URL = os.environ.get("MR_URL", "http://localhost:8080") +parsed = urlparse(REGISTRY_URL) +host, port = parsed.netloc.split(":") +REGISTRY_HOST = f"{parsed.scheme}://{host}" +REGISTRY_PORT = int(port) + +MAX_POLL_TIME = 10 POLL_INTERVAL = 1 -DOCKER = os.getenv("DOCKER", "docker") start_time = time.time() @@ -64,38 +65,8 @@ def poll_for_ready(): time.sleep(POLL_INTERVAL) -@pytest.fixture(scope="session") -def _compose_mr(root): - print("Assuming this is the Model Registry root directory:", root) - shared_volume = root / "test/config/ml-metadata" - sqlite_db_file = shared_volume / "metadata.sqlite.db" - if sqlite_db_file.exists(): - msg = f"The file {sqlite_db_file} already exists; make sure to cancel it before running these tests." - raise FileExistsError(msg) - print(f" Starting Docker Compose in folder {root}") - p = subprocess.Popen( # noqa: S602 - f"{DOCKER} compose -f {COMPOSE_FILE} up", - shell=True, - cwd=root, - ) - yield - - p.kill() - print(f" Closing Docker Compose in folder {root}") - subprocess.call( # noqa: S602 - f"{DOCKER} compose -f {COMPOSE_FILE} down", - shell=True, - cwd=root, - ) - try: - os.remove(sqlite_db_file) - print(f"Removed {sqlite_db_file} successfully.") - except Exception as e: - print(f"An error occurred while removing {sqlite_db_file}: {e}") - - def cleanup(client): - async def yield_and_restart(_compose_mr, root): + async def yield_and_restart(root): poll_for_ready() if inspect.iscoroutinefunction(client) or inspect.isasyncgenfunction(client): async with asynccontextmanager(client)() as async_client: @@ -103,18 +74,9 @@ async def yield_and_restart(_compose_mr, root): else: yield client() - sqlite_db_file = root / "test/config/ml-metadata/metadata.sqlite.db" - try: - os.remove(sqlite_db_file) - print(f"Removed {sqlite_db_file} successfully.") - except Exception as e: - print(f"An error occurred while removing {sqlite_db_file}: {e}") - # we have to wait to make sure the server restarts after the file is gone - sleep(1) - - print("Restarting model-registry...") + print("Cleaning DB...") subprocess.call( # noqa: S602 - f"{DOCKER} compose -f {COMPOSE_FILE} restart model-registry", + "./scripts/cleanup.sh", shell=True, cwd=root, ) @@ -135,6 +97,7 @@ def event_loop(): def client() -> ModelRegistry: return ModelRegistry(REGISTRY_HOST, REGISTRY_PORT, author="author", is_secure=False) + @pytest.fixture(scope="module") def setup_env_user_token(): with tempfile.NamedTemporaryFile(delete=False) as token_file: diff --git a/clients/python/tests/regression_test.py b/clients/python/tests/regression_test.py index 035a22577..0310f3879 100644 --- a/clients/python/tests/regression_test.py +++ b/clients/python/tests/regression_test.py @@ -26,7 +26,7 @@ def test_create_tagged_version(client: ModelRegistry): @pytest.mark.e2e -def test_get_model_without_user_token(setup_env_user_token, client): +def test_get_model_without_user_token(setup_env_user_token, client: ModelRegistry): """Test regression for using client methods without an user_token in the init arguments. Reported on: https://github.com/kubeflow/model-registry/issues/340 diff --git a/clients/python/tests/test_client.py b/clients/python/tests/test_client.py index bb958cf50..4cd93764a 100644 --- a/clients/python/tests/test_client.py +++ b/clients/python/tests/test_client.py @@ -393,6 +393,9 @@ def test_get_model_versions(client: ModelRegistry): @pytest.mark.e2e +@pytest.mark.xfail( + reason="MLMD issue tracked on: https://github.com/kubeflow/model-registry/issues/358" +) def test_get_model_versions_order_by(client: ModelRegistry): name = "test_model" models = 5 diff --git a/scripts/cleanup.sh b/scripts/cleanup.sh new file mode 100755 index 000000000..a018c7a62 --- /dev/null +++ b/scripts/cleanup.sh @@ -0,0 +1,34 @@ +#!/usr/bin/env bash + +set -e + +MR_NAMESPACE="${MR_NAMESPACE:-kubeflow}" +TEST_DB_NAME="${TEST_DB_NAME:-metadb}" + +SQL_CMD=$( + cat <