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

Open
wants to merge 16 commits into
base: main
Choose a base branch
from
Open
Show file tree
Hide file tree
Changes from all 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
3 changes: 3 additions & 0 deletions .gitignore
Original file line number Diff line number Diff line change
Expand Up @@ -162,3 +162,6 @@ cython_debug/
# OS generated files #
.DS_Store
.DS_Store?

# VSCode config
.vscode/
4 changes: 0 additions & 4 deletions CONTRIBUTING.md
Original file line number Diff line number Diff line change
Expand Up @@ -32,7 +32,3 @@ pre-commit install

- Add typing to new code; typing is enforced using [mypy](https://mypy-lang.org/)
- Rules are defined in [our pyproject.toml file](//pyproject.toml#L10)

If you use Visual Studio Code as your IDE, we recommend using the [Mypy Type Checker](https://marketplace.visualstudio.com/items?itemName=ms-python.mypy-type-checker) extension.
After installing it, make sure to update the `Mypy-type-checkers: Args` setting
to `"mypy-type-checker.args" = ["--config-file=pyproject.toml"]`.
35 changes: 35 additions & 0 deletions VSCODE_CONFIG.md
Original file line number Diff line number Diff line change
@@ -0,0 +1,35 @@
# Helpful VS Code Settings and Extensions

The following are some helpful tips on how to set up your VS Code environment for working with this repository.

## Mypy Type Checker in Visual Studio Code

If you use Visual Studio Code as your IDE, we recommend using the [Mypy Type Checker](https://marketplace.visualstudio.com/items?itemName=ms-python.mypy-type-checker) extension.
After installing it, make sure to update the `Mypy-type-checkers: Args` setting
to `"mypy-type-checker.args" = ["--config-file=pyproject.toml"]`.

## Debugging in Visual Studio Code

If you use Visual Studio Code and want to debug your test execution with its "Run and Debug" feature, you'll want to use
a `launch.json` file similar to this one:

```
{
// Use IntelliSense to learn about possible attributes.
// Hover to view descriptions of existing attributes.
// For more information, visit: https://go.microsoft.com/fwlink/?linkid=830387
"version": "0.2.0",
"configurations": [
{
"justMyCode": false, #set to false if you want to debug dependent libraries too
"name": "uv_pytest_debugger",
"type": "debugpy",
"request": "launch",
"program": ".venv/bin/pytest", #or your path to pytest's bin in the venv
"python": "${command:python.interpreterPath}", #make sure uv's python interpreter is selected in vscode
"console": "integratedTerminal",
"args": "path/to/test.py" #the args for pytest, can be a list, in this example runs a single file
}
]
}
```
3 changes: 2 additions & 1 deletion pyproject.toml
Original file line number Diff line number Diff line change
Expand Up @@ -43,13 +43,14 @@ classifiers = [
dependencies = [
"ipython>=8.18.1",
"openshift-python-utilities>=5.0.71",
"openshift-python-wrapper>=11.0.6",
"openshift-python-wrapper>=11.0.7",
"pytest-dependency>=0.6.0",
"pytest-progress",
"python-simple-logger",
"pyyaml",
"tenacity",
"types-requests>=2.32.0.20241016",
"schemathesis>=3.38.10",
"requests",
"pytest-asyncio",
"syrupy",
Expand Down
10 changes: 9 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
from utilities.constants import AcceleratorType
Expand All @@ -15,6 +16,13 @@ def admin_client() -> DynamicClient:
return get_client()


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


@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.
305 changes: 305 additions & 0 deletions tests/model_registry/conftest.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,305 @@
import pytest
import schemathesis
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, KubernetesAnnotations


LOGGER = get_logger(name=__name__)

DB_RESOURCES_NAME: str = "model-registry-db"
MR_INSTANCE_NAME: str = "model-registry"
MR_OPERATOR_NAME: str = "model-registry-operator"
MR_NAMESPACE: str = "rhoai-model-registries"
DEFAULT_LABEL_DICT_DB: dict[str, str] = {
KubernetesAnnotations.NAME: DB_RESOURCES_NAME,
KubernetesAnnotations.INSTANCE: DB_RESOURCES_NAME,
KubernetesAnnotations.PART_OF: DB_RESOURCES_NAME,
}


@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=MR_NAMESPACE, client=admin_client)
if ns.exists:
yield ns
else:
LOGGER.warning(f"{MR_NAMESPACE} namespace was not present, creating it")
with create_ns(
lugi0 marked this conversation as resolved.
Show resolved Hide resolved
name=MR_NAMESPACE,
admin_client=admin_client,
teardown=False,
rnetser marked this conversation as resolved.
Show resolved Hide resolved
) 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=DEFAULT_LABEL_DICT_DB,
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]:
with PersistentVolumeClaim(
accessmodes="ReadWriteOnce",
name=DB_RESOURCES_NAME,
namespace=model_registry_namespace.name,
client=admin_client,
size="5Gi",
label=DEFAULT_LABEL_DICT_DB,
) as pvc:
yield pvc


@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=DEFAULT_LABEL_DICT_DB,
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=DEFAULT_LABEL_DICT_DB,
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=MR_INSTANCE_NAME,
namespace=model_registry_namespace.name,
label={
KubernetesAnnotations.NAME: MR_INSTANCE_NAME,
KubernetesAnnotations.INSTANCE: MR_INSTANCE_NAME,
KubernetesAnnotations.PART_OF: MR_OPERATOR_NAME,
KubernetesAnnotations.CREATED_BY: MR_OPERATOR_NAME,
},
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(
client=admin_client, ns=model_registry_namespace, mr_instance=model_registry_instance
)


@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(
client=admin_client, svc=model_registry_instance_service, protocol=Protocols.REST
)


@pytest.fixture(scope="class")
def generated_schema(model_registry_instance_rest_endpoint):
return schemathesis.from_uri(
uri="https://raw.githubusercontent.com/kubeflow/model-registry/main/api/openapi/model-registry.yaml",
base_url=f"https://{model_registry_instance_rest_endpoint}/",
)
Loading