Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Add first Model Registry tests #54

Merged
merged 22 commits into from
Jan 15, 2025
Merged
Show file tree
Hide file tree
Changes from 3 commits
Commits
File filter

Filter by extension

Filter by extension


Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
2 changes: 1 addition & 1 deletion pyproject.toml
Original file line number Diff line number Diff line change
Expand Up @@ -42,7 +42,7 @@ classifiers = [
dependencies = [
"ipython>=8.18.1",
"openshift-python-utilities>=5.0.71",
"openshift-python-wrapper>=10.0.100",
"openshift-python-wrapper>=11.0.6",
"pytest-dependency>=0.6.0",
"pytest-progress",
"python-simple-logger",
Expand Down
11 changes: 10 additions & 1 deletion tests/conftest.py
Original file line number Diff line number Diff line change
@@ -1,10 +1,11 @@
from typing import Tuple, Any, Generator

import shlex
import pytest
from pytest import FixtureRequest, Config
from kubernetes.dynamic import DynamicClient
from ocp_resources.namespace import Namespace
from ocp_resources.resource import get_client
from pyhelper_utils.shell import run_command

from utilities.infra import create_ns

Expand All @@ -14,6 +15,14 @@ def admin_client() -> DynamicClient:
return get_client()


@pytest.fixture(scope="session")
lugi0 marked this conversation as resolved.
Show resolved Hide resolved
def admin_client_token(admin_client: DynamicClient) -> str:
cmd = "oc whoami -t"
lugi0 marked this conversation as resolved.
Show resolved Hide resolved
_, out, _ = run_command(command=shlex.split(cmd), verify_stderr=False, check=False)
# `\n` appended to token in out, return without that
return out[:-1]


@pytest.fixture(scope="class")
def model_namespace(request: FixtureRequest, admin_client: DynamicClient) -> Generator[Namespace, Any, Any]:
with create_ns(admin_client=admin_client, name=request.param["name"]) as ns:
Expand Down
Empty file.
307 changes: 307 additions & 0 deletions tests/model_registry/conftest.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,307 @@
import pytest
from typing import Generator, Any
from ocp_resources.secret import Secret
from ocp_resources.namespace import Namespace
from ocp_resources.service import Service
from ocp_resources.persistent_volume_claim import PersistentVolumeClaim
from ocp_resources.deployment import Deployment
from ocp_resources.model_registry import ModelRegistry
from simple_logger.logger import get_logger
from kubernetes.dynamic import DynamicClient

from tests.model_registry.utils import get_endpoint_from_mr_service, get_mr_service_by_label
from utilities.infra import create_ns
from utilities.constants import Protocols


LOGGER = get_logger(name=__name__)

DB_RESOURCES_NAME = "model-registry-db"


@pytest.fixture(scope="class")
def model_registry_namespace(admin_client: DynamicClient) -> Generator[Namespace, Any, Any]:
# This namespace should exist after Model Registry is enabled, but it can also be deleted
# from the cluster and does not get reconciled. Fetch if it exists, create otherwise.
lugi0 marked this conversation as resolved.
Show resolved Hide resolved
ns = Namespace(name="rhoai-model-registries", client=admin_client)
if ns.exists:
yield ns
else:
with create_ns(
lugi0 marked this conversation as resolved.
Show resolved Hide resolved
name="rhoai-model-registries",
admin_client=admin_client,
teardown=False,
rnetser marked this conversation as resolved.
Show resolved Hide resolved
ensure_exists=True,
) as ns:
yield ns


@pytest.fixture(scope="class")
def model_registry_db_service(
admin_client: DynamicClient, model_registry_namespace: Namespace
) -> Generator[Service, Any, Any]:
with Service(
client=admin_client,
name=DB_RESOURCES_NAME,
namespace=model_registry_namespace.name,
ports=[
{
"name": "mysql",
"nodePort": 0,
"port": 3306,
"protocol": "TCP",
"appProtocol": "tcp",
"targetPort": 3306,
}
],
selector={
"name": DB_RESOURCES_NAME,
},
label={
"app.kubernetes.io/name": DB_RESOURCES_NAME,
"app.kubernetes.io/instance": DB_RESOURCES_NAME,
"app.kubernetes.io/part-of": DB_RESOURCES_NAME,
"app.kubernetes.io/managed-by": "kustomize",
},
annotations={
"template.openshift.io/expose-uri": "mysql://{.spec.clusterIP}:{.spec.ports[?(.name==\mysql\)].port}",
},
) as mr_db_service:
yield mr_db_service


@pytest.fixture(scope="class")
def model_registry_db_pvc(
admin_client: DynamicClient,
model_registry_namespace: Namespace,
) -> Generator[PersistentVolumeClaim, Any, Any]:
pvc_kwargs = {
lugi0 marked this conversation as resolved.
Show resolved Hide resolved
"accessmodes": "ReadWriteOnce",
"name": DB_RESOURCES_NAME,
"namespace": model_registry_namespace.name,
"client": admin_client,
"size": "5Gi",
"label": {
"app.kubernetes.io/name": DB_RESOURCES_NAME,
"app.kubernetes.io/instance": DB_RESOURCES_NAME,
"app.kubernetes.io/part-of": DB_RESOURCES_NAME,
"app.kubernetes.io/managed-by": "kustomize",
},
}

with PersistentVolumeClaim(**pvc_kwargs) as pvc:
yield pvc
adolfo-ab marked this conversation as resolved.
Show resolved Hide resolved


@pytest.fixture(scope="class")
def model_registry_db_secret(
admin_client: DynamicClient,
model_registry_namespace: Namespace,
) -> Generator[Secret, Any, Any]:
with Secret(
client=admin_client,
name=DB_RESOURCES_NAME,
namespace=model_registry_namespace.name,
string_data={
"database-name": "model_registry",
"database-password": "TheBlurstOfTimes", # pragma: allowlist secret
"database-user": "mlmduser", # pragma: allowlist secret
},
label={
"app.kubernetes.io/name": DB_RESOURCES_NAME,
"app.kubernetes.io/instance": DB_RESOURCES_NAME,
"app.kubernetes.io/part-of": DB_RESOURCES_NAME,
"app.kubernetes.io/managed-by": "kustomize",
},
annotations={
"template.openshift.io/expose-database_name": "'{.data[''database-name'']}'",
"template.openshift.io/expose-password": "'{.data[''database-password'']}'",
"template.openshift.io/expose-username": "'{.data[''database-user'']}'",
},
) as mr_db_secret:
yield mr_db_secret


@pytest.fixture(scope="class")
def model_registry_db_deployment(
admin_client: DynamicClient,
model_registry_namespace: Namespace,
model_registry_db_secret: Secret,
model_registry_db_pvc: PersistentVolumeClaim,
model_registry_db_service: Service,
) -> Generator[Deployment, Any, Any]:
with Deployment(
lugi0 marked this conversation as resolved.
Show resolved Hide resolved
name=DB_RESOURCES_NAME,
namespace=model_registry_namespace.name,
annotations={
"template.alpha.openshift.io/wait-for-ready": "true",
},
label={
"app.kubernetes.io/name": DB_RESOURCES_NAME,
"app.kubernetes.io/instance": DB_RESOURCES_NAME,
"app.kubernetes.io/part-of": DB_RESOURCES_NAME,
"app.kubernetes.io/managed-by": "kustomize",
},
replicas=1,
revision_history_limit=0,
selector={"matchLabels": {"name": DB_RESOURCES_NAME}},
strategy={"type": "Recreate"},
template={
"metadata": {
"labels": {
"name": DB_RESOURCES_NAME,
"sidecar.istio.io/inject": "false",
}
},
"spec": {
"containers": [
{
"env": [
{
"name": "MYSQL_USER",
"valueFrom": {
"secretKeyRef": {
"key": "database-user",
"name": f"{model_registry_db_secret.name}",
}
},
},
{
"name": "MYSQL_PASSWORD",
"valueFrom": {
"secretKeyRef": {
"key": "database-password",
"name": f"{model_registry_db_secret.name}",
}
},
},
{
"name": "MYSQL_ROOT_PASSWORD",
"valueFrom": {
"secretKeyRef": {
"key": "database-password",
"name": f"{model_registry_db_secret.name}",
}
},
},
{
"name": "MYSQL_DATABASE",
"valueFrom": {
"secretKeyRef": {
"key": "database-name",
"name": f"{model_registry_db_secret.name}",
}
},
},
],
"args": [
"--datadir",
"/var/lib/mysql/datadir",
"--default-authentication-plugin=mysql_native_password",
],
"image": "mysql:8.3.0",
"imagePullPolicy": "IfNotPresent",
"livenessProbe": {
"exec": {
"command": [
"/bin/bash",
"-c",
"mysqladmin -u${MYSQL_USER} -p${MYSQL_ROOT_PASSWORD} ping",
]
},
"initialDelaySeconds": 15,
"periodSeconds": 10,
"timeoutSeconds": 5,
},
"name": "mysql",
"ports": [{"containerPort": 3306, "protocol": "TCP"}],
"readinessProbe": {
"exec": {
"command": [
"/bin/bash",
"-c",
'mysql -D ${MYSQL_DATABASE} -u${MYSQL_USER} -p${MYSQL_ROOT_PASSWORD} -e "SELECT 1"',
]
},
"initialDelaySeconds": 10,
"timeoutSeconds": 5,
},
"securityContext": {"capabilities": {}, "privileged": False},
"terminationMessagePath": "/dev/termination-log",
"volumeMounts": [
{
"mountPath": "/var/lib/mysql",
"name": f"{DB_RESOURCES_NAME}-data",
}
],
}
],
"dnsPolicy": "ClusterFirst",
"restartPolicy": "Always",
"volumes": [
{
"name": f"{DB_RESOURCES_NAME}-data",
"persistentVolumeClaim": {"claimName": DB_RESOURCES_NAME},
}
],
},
},
wait_for_resource=True,
) as mr_db_deployment:
mr_db_deployment.wait_for_replicas(deployed=True)
yield mr_db_deployment


@pytest.fixture(scope="class")
def model_registry_instance(
admin_client: DynamicClient,
model_registry_namespace: Namespace,
model_registry_db_deployment: Deployment,
model_registry_db_secret: Secret,
model_registry_db_service: Service,
) -> Generator[ModelRegistry, Any, Any]:
with ModelRegistry(
name="model-registry",
namespace=model_registry_namespace.name,
label={
"app.kubernetes.io/name": "model-registry",
lugi0 marked this conversation as resolved.
Show resolved Hide resolved
"app.kubernetes.io/instance": "model-registry",
"app.kubernetes.io/part-of": "model-registry-operator",
"app.kubernetes.io/managed-by": "kustomize",
lugi0 marked this conversation as resolved.
Show resolved Hide resolved
"app.kubernetes.io/created-by": "model-registry-operator",
},
grpc={},
rest={},
istio={
"authProvider": "redhat-ods-applications-auth-provider",
"gateway": {"grpc": {"tls": {}}, "rest": {"tls": {}}},
},
mysql={
"host": f"{model_registry_db_deployment.name}.{model_registry_db_deployment.namespace}.svc.cluster.local",
"database": model_registry_db_secret.string_data["database-name"],
"passwordSecret": {"key": "database-password", "name": DB_RESOURCES_NAME},
"port": 3306,
"skipDBCreation": False,
"username": model_registry_db_secret.string_data["database-user"],
},
wait_for_resource=True,
) as mr:
mr.wait_for_condition(condition="Available", status="True")
yield mr


@pytest.fixture(scope="class")
def model_registry_instance_service(
admin_client: DynamicClient,
model_registry_namespace: Namespace,
model_registry_instance: ModelRegistry,
) -> Service:
return get_mr_service_by_label(admin_client, model_registry_namespace, model_registry_instance)
lugi0 marked this conversation as resolved.
Show resolved Hide resolved


@pytest.fixture(scope="class")
def model_registry_instance_rest_endpoint(
admin_client: DynamicClient,
model_registry_instance_service: Service,
) -> str:
return get_endpoint_from_mr_service(admin_client, model_registry_instance_service, Protocols.REST)
2 changes: 2 additions & 0 deletions tests/model_registry/constants.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,2 @@
class ModelRegistryEndpoints:
REGISTERED_MODELS: str = "/api/model_registry/v1alpha3/registered_models"
31 changes: 31 additions & 0 deletions tests/model_registry/test_model_registry_creation.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,31 @@
import shlex
from typing import Self
from simple_logger.logger import get_logger
from ocp_resources.model_registry import ModelRegistry
from pyhelper_utils.shell import run_command

from tests.model_registry.utils import generate_register_model_command

LOGGER = get_logger(name=__name__)


class TestModelRegistryCreation:
"""Tests the creation of a model registry"""

# TODO: Enable Model Registry in DSC if needed

def test_model_registry_instance_creation(self: Self, model_registry_instance: ModelRegistry):
lugi0 marked this conversation as resolved.
Show resolved Hide resolved
assert model_registry_instance.name == "model-registry"
lugi0 marked this conversation as resolved.
Show resolved Hide resolved

def test_registering_model(self: Self, model_registry_instance_rest_endpoint: str, admin_client_token: str):
cmd = generate_register_model_command(model_registry_instance_rest_endpoint, admin_client_token)
res, out, _ = run_command(command=shlex.split(cmd), verify_stderr=False, check=False)
assert res
lugi0 marked this conversation as resolved.
Show resolved Hide resolved
out_dict = eval(out)
lugi0 marked this conversation as resolved.
Show resolved Hide resolved
assert out_dict["name"] == "model-name"
assert out_dict["description"] == "test-model"
assert out_dict["externalId"] == "1"
assert out_dict["owner"] == "opendatahub-tests-client"
assert out_dict["state"] == "LIVE"
lugi0 marked this conversation as resolved.
Show resolved Hide resolved

# TODO: Query for a registered Model
Loading