diff --git a/frontend/src/__mocks__/mockImageStreamK8sResource.ts b/frontend/src/__mocks__/mockImageStreamK8sResource.ts index d8734f397b..08238010c7 100644 --- a/frontend/src/__mocks__/mockImageStreamK8sResource.ts +++ b/frontend/src/__mocks__/mockImageStreamK8sResource.ts @@ -6,6 +6,8 @@ type MockResourceConfigType = { name?: string; namespace?: string; displayName?: string; + imageTag?: string; + tagName?: string; opts?: RecursivePartial; }; @@ -13,6 +15,8 @@ export const mockImageStreamK8sResource = ({ name = 'test-imagestream', namespace = 'test-project', displayName = 'Test Image', + imageTag = 'quay.io/opendatahub/notebooks@sha256:a138838e1c9acd7708462e420bf939e03296b97e9cf6c0aa0fd9a5d20361ab75', + tagName = '1.2', opts = {}, }: MockResourceConfigType): ImageStreamKind => _.mergeWith( @@ -48,7 +52,7 @@ export const mockImageStreamK8sResource = ({ }, tags: [ { - name: '1.2', + name: tagName, annotations: { 'opendatahub.io/notebook-python-dependencies': '[{"name":"JupyterLab","version": "3.2"}, {"name": "Notebook","version": "6.4"}]', @@ -56,7 +60,7 @@ export const mockImageStreamK8sResource = ({ }, from: { kind: 'DockerImage', - name: 'quay.io/opendatahub/notebooks@sha256:a138838e1c9acd7708462e420bf939e03296b97e9cf6c0aa0fd9a5d20361ab75', + name: imageTag, }, }, ], @@ -72,7 +76,7 @@ export const mockImageStreamK8sResource = ({ created: '2023-06-30T15:07:36Z', dockerImageReference: 'quay.io/opendatahub/notebooks@sha256:a138838e1c9acd7708462e420bf939e03296b97e9cf6c0aa0fd9a5d20361ab75', - image: 'sha256:a138838e1c9acd7708462e420bf939e03296b97e9cf6c0aa0fd9a5d20361ab75', + image: imageTag, generation: 2, }, ], diff --git a/frontend/src/__mocks__/mockNotebookK8sResource.ts b/frontend/src/__mocks__/mockNotebookK8sResource.ts index 14418e9b54..7c959f6092 100644 --- a/frontend/src/__mocks__/mockNotebookK8sResource.ts +++ b/frontend/src/__mocks__/mockNotebookK8sResource.ts @@ -12,6 +12,8 @@ type MockResourceConfigType = { user?: string; description?: string; resources?: ContainerResources; + image?: string; + lastImageSelection?: string; opts?: RecursivePartial; }; @@ -22,6 +24,8 @@ export const mockNotebookK8sResource = ({ user = 'test-user', description = '', resources = DEFAULT_NOTEBOOK_SIZES[0].resources, + image = 'test-imagestream:1.2', + lastImageSelection = 's2i-minimal-notebook:py3.8-v1', opts = {}, }: MockResourceConfigType): NotebookKind => _.merge( @@ -32,7 +36,7 @@ export const mockNotebookK8sResource = ({ annotations: { 'notebooks.kubeflow.org/last-activity': '2023-02-14T21:45:14Z', 'notebooks.opendatahub.io/inject-oauth': 'true', - 'notebooks.opendatahub.io/last-image-selection': 's2i-minimal-notebook:py3.8-v1', + 'notebooks.opendatahub.io/last-image-selection': lastImageSelection, 'notebooks.opendatahub.io/last-size-selection': 'Small', 'notebooks.opendatahub.io/oauth-logout-url': 'http://localhost:4010/projects/project?notebookLogout=workbench', @@ -96,8 +100,7 @@ export const mockNotebookK8sResource = ({ }, }, ], - image: - 'image-registry.openshift-image-registry.svc:5000/redhat-ods-applications/s2i-minimal-notebook:py3.8-v1', + image, imagePullPolicy: 'Always', livenessProbe: { failureThreshold: 3, diff --git a/frontend/src/__tests__/cypress/cypress/e2e/projects/projectDetails.cy.ts b/frontend/src/__tests__/cypress/cypress/e2e/projects/projectDetails.cy.ts index 0ea14efb73..b226866015 100644 --- a/frontend/src/__tests__/cypress/cypress/e2e/projects/projectDetails.cy.ts +++ b/frontend/src/__tests__/cypress/cypress/e2e/projects/projectDetails.cy.ts @@ -18,6 +18,10 @@ import { projectDetails } from '~/__tests__/cypress/cypress/pages/projects'; type HandlersProps = { isEmpty?: boolean; imageStreamName?: string; + imageStreamTag?: string; + disableKServeConfig?: boolean; + disableKServeMetrics?: boolean; + disableModelConfig?: boolean; isEnabled?: string; isUnknown?: boolean; }; @@ -25,6 +29,7 @@ type HandlersProps = { const initIntercepts = ({ isEmpty = false, imageStreamName = 'test-image', + imageStreamTag = 'latest', isEnabled = 'true', isUnknown = false, }: HandlersProps) => { @@ -99,14 +104,14 @@ const initIntercepts = ({ spec: { tags: [ { - name: 'latest', + name: imageStreamTag, }, ], }, status: { tags: [ { - tag: 'latest', + tag: imageStreamTag, }, ], }, @@ -185,7 +190,7 @@ describe('Project Details', () => { }); it('Notebook with deleted image', () => { - initIntercepts({ imageStreamName: 'test' }); + initIntercepts({ imageStreamName: 'test', imageStreamTag: 'failing-tag' }); projectDetails.visit('test-project'); const notebookRow = projectDetails.getNotebookRow('test-notebook'); notebookRow.shouldHaveNotebookImageName('Test image'); diff --git a/frontend/src/__tests__/cypress/cypress/pages/modelServing.ts b/frontend/src/__tests__/cypress/cypress/pages/modelServing.ts index e0c2b4920f..9f8e5d1e21 100644 --- a/frontend/src/__tests__/cypress/cypress/pages/modelServing.ts +++ b/frontend/src/__tests__/cypress/cypress/pages/modelServing.ts @@ -173,7 +173,10 @@ class ServingRuntimeModal extends Modal { } } +// Expect KServeModal to inherit both modal classes. +// eslint-disable-next-line @typescript-eslint/no-unsafe-declaration-merging interface KServeModal extends ServingRuntimeModal, InferenceServiceModal {} +// eslint-disable-next-line @typescript-eslint/no-unsafe-declaration-merging class KServeModal extends InferenceServiceModal {} mixin(KServeModal, [ServingRuntimeModal, InferenceServiceModal]); diff --git a/frontend/src/pages/projects/screens/detail/notebooks/__tests__/useNotebookImageData.spec.ts b/frontend/src/pages/projects/screens/detail/notebooks/__tests__/useNotebookImageData.spec.ts new file mode 100644 index 0000000000..cf23daa0bb --- /dev/null +++ b/frontend/src/pages/projects/screens/detail/notebooks/__tests__/useNotebookImageData.spec.ts @@ -0,0 +1,109 @@ +import { mockNotebookK8sResource } from '~/__mocks__/mockNotebookK8sResource'; +import { mockImageStreamK8sResource } from '~/__mocks__/mockImageStreamK8sResource'; +import { getNotebookImageData } from '~/pages/projects/screens/detail/notebooks/useNotebookImageData'; +import { NotebookImageAvailability } from '~/pages/projects/screens/detail/notebooks/const'; + +describe('getNotebookImageData', () => { + it('should return image data when image stream exists and image version exists with internal registry', () => { + const notebook = mockNotebookK8sResource({ + image: + 'quay.io/opendatahub/notebooks@sha256:a138838e1c9acd7708462e420bf939e03296b97e9cf6c0aa0fd9a5d20361ab75', + lastImageSelection: 'jupyter-datascience-notebook', + }); + const images = [ + mockImageStreamK8sResource({ + imageTag: + 'quay.io/opendatahub/notebooks@sha256:a138838e1c9acd7708462e420bf939e03296b97e9cf6c0aa0fd9a5d20361ab75', + name: 'jupyter-datascience-notebook', + }), + ]; + const result = getNotebookImageData(notebook, images); + expect(result?.imageAvailability).toBe(NotebookImageAvailability.ENABLED); + }); + + it('should return image data when image stream exists and image version exists without internal registry', () => { + const imageName = 'jupyter-datascience-notebook'; + const tagName = '2024.1'; + const notebook = mockNotebookK8sResource({ + image: `image-registry.openshift-image-registry.svc:5000/opendatahub/${imageName}:${tagName}`, + lastImageSelection: 'jupyter-datascience-notebook', + }); + const images = [ + mockImageStreamK8sResource({ + imageTag: + 'quay.io/opendatahub/notebooks@sha256:a138838e1c9acd7708462e420bf939e03296b97e9cf6c0aa0fd9a5d20361ab75', + tagName, + name: imageName, + }), + ]; + const result = getNotebookImageData(notebook, images); + expect(result?.imageAvailability).toBe(NotebookImageAvailability.ENABLED); + }); + + it('should return the right image if multiple ImageStreams have the same image with internal registry', () => { + const displayName = 'Jupyter Data Science Notebook'; + const notebook = mockNotebookK8sResource({ + image: + 'quay.io/opendatahub/notebooks@sha256:a138838e1c9acd7708462e420bf939e03296b97e9cf6c0aa0fd9a5d20361ab75', + lastImageSelection: 'jupyter-datascience-notebook', + }); + const images = [ + mockImageStreamK8sResource({ + imageTag: + 'quay.io/opendatahub/notebooks@sha256:a138838e1c9acd7708462e420bf939e03296b97e9cf6c0aa0fd9a5d20361ab75', + name: 'jupyter-datascience-notebook', + displayName, + }), + mockImageStreamK8sResource({ + imageTag: + 'quay.io/opendatahub/notebooks@sha256:a138838e1c9acd7708462e420bf939e03296b97e9cf6c0aa0fd9a5d20361ab75', + name: 'custom-notebook', + displayName: 'Custom Notebook', + }), + ]; + const result = getNotebookImageData(notebook, images); + expect(result?.imageDisplayName).toBe(displayName); + }); + + it('should return the right image if multiple ImageStreams have the same tag without internal registry', () => { + const imageName = 'jupyter-datascience-notebook'; + const tagName = '2024.1'; + const displayName = 'Jupyter Data Science Notebook'; + const notebook = mockNotebookK8sResource({ + image: `image-registry.openshift-image-registry.svc:5000/opendatahub/${imageName}:${tagName}`, + lastImageSelection: 'jupyter-datascience-notebook', + }); + const images = [ + mockImageStreamK8sResource({ + imageTag: + 'quay.io/opendatahub/notebooks@sha256:a138838e1c9acd7708462e420bf939e03296b97e9cf6c0aa0fd9a5d20361ab75', + tagName, + name: 'code-server', + displayName: 'Code Server', + }), + mockImageStreamK8sResource({ + imageTag: + 'quay.io/opendatahub/notebooks@sha256:a138838e1c9acd7708462e420bf939e03296b97e9cf6c0aa0fd9a5d20361ab75', + tagName, + name: imageName, + displayName, + }), + ]; + const result = getNotebookImageData(notebook, images); + expect(result?.imageDisplayName).toBe(displayName); + }); + + it('should return image data when image stream exists and image version does not exist', () => { + const notebook = mockNotebookK8sResource({ + image: + 'quay.io/opendatahub/notebooks@sha256:a138838e1c9acd7708462e420bf939e03296b97e9cf6c0aa0fd9a5d20361ab75', + }); + const images = [ + mockImageStreamK8sResource({ + imageTag: 'quay.io/opendatahub/notebooks@sha256:invalid', + }), + ]; + const result = getNotebookImageData(notebook, images); + expect(result?.imageAvailability).toBe(NotebookImageAvailability.DELETED); + }); +}); diff --git a/frontend/src/pages/projects/screens/detail/notebooks/useNotebookImageData.ts b/frontend/src/pages/projects/screens/detail/notebooks/useNotebookImageData.ts index 0795287e5e..17f4078783 100644 --- a/frontend/src/pages/projects/screens/detail/notebooks/useNotebookImageData.ts +++ b/frontend/src/pages/projects/screens/detail/notebooks/useNotebookImageData.ts @@ -1,5 +1,5 @@ import * as React from 'react'; -import { NotebookKind } from '~/k8sTypes'; +import { ImageStreamKind, NotebookKind } from '~/k8sTypes'; import useNamespaces from '~/pages/notebookController/useNamespaces'; import useImageStreams from '~/pages/projects/screens/spawner/useImageStreams'; import { PodContainer } from '~/types'; @@ -7,81 +7,94 @@ import { getImageStreamDisplayName } from '~/pages/projects/screens/spawner/spaw import { NotebookImageAvailability } from './const'; import { NotebookImageData } from './types'; -const useNotebookImageData = (notebook?: NotebookKind): NotebookImageData => { - const { dashboardNamespace } = useNamespaces(); - const [images, loaded, loadError] = useImageStreams(dashboardNamespace, true); +export const getNotebookImageData = ( + notebook: NotebookKind, + images: ImageStreamKind[], +): NotebookImageData[0] => { + const container: PodContainer | undefined = notebook.spec.template.spec.containers.find( + (currentContainer) => currentContainer.name === notebook.metadata.name, + ); + const imageTag = container?.image.split('/').at(-1)?.split(':'); - return React.useMemo(() => { - if (!loaded || !notebook) { - return [null, false, loadError]; - } + // if image could not be parsed from the container, consider it deleted because the image tag is invalid + if (!imageTag || imageTag.length < 2 || !container) { + return { + imageAvailability: NotebookImageAvailability.DELETED, + }; + } + + const [imageName, versionName] = imageTag; + const [lastImageSelectionName] = + notebook.metadata.annotations?.['notebooks.opendatahub.io/last-image-selection']?.split(':') ?? + []; - const container: PodContainer | undefined = notebook.spec.template.spec.containers.find( - (currentContainer) => currentContainer.name === notebook.metadata.name, + // Fallback for non internal registry clusters + const imageStream = + images.find((image) => image.metadata.name === imageName) || + images.find((image) => + image.spec.tags + ? image.spec.tags.find( + (version) => + version.from?.name === container.image && + image.metadata.name === lastImageSelectionName, + ) + : false, ); - const imageTag = container?.image.split('/').at(-1)?.split(':'); - // if image could not be parsed from the container, consider it deleted because the image tag is invalid - if (!imageTag || imageTag.length < 2 || !container) { - return [ - { - imageAvailability: NotebookImageAvailability.DELETED, - }, - true, - undefined, - ]; - } + // if the image stream is not found, consider it deleted + if (!imageStream) { + // Get the image display name from the notebook metadata if we can't find the image stream. (this is a fallback and could still be undefined) + const imageDisplayName = notebook.metadata.annotations?.['opendatahub.io/image-display-name']; - const [imageName, versionName] = imageTag; - const imageStream = images.find((image) => image.metadata.name === imageName); + return { + imageAvailability: NotebookImageAvailability.DELETED, + imageDisplayName, + }; + } - // if the image stream is not found, consider it deleted - if (!imageStream) { - // Get the image display name from the notebook metadata if we can't find the image stream. (this is a fallback and could still be undefined) - const imageDisplayName = notebook.metadata.annotations?.['opendatahub.io/image-display-name']; + const versions = imageStream.spec.tags || []; + const imageVersion = versions.find( + (version) => version.name === versionName || version.from?.name === container.image, + ); - return [ - { - imageAvailability: NotebookImageAvailability.DELETED, - imageDisplayName, - }, - true, - undefined, - ]; - } + // because the image stream was found, get its display name + const imageDisplayName = getImageStreamDisplayName(imageStream); - const versions = imageStream.spec.tags || []; - const imageVersion = versions.find((version) => version.name === versionName); + // if the image version is not found, consider the image stream deleted + if (!imageVersion) { + return { + imageAvailability: NotebookImageAvailability.DELETED, + imageDisplayName, + }; + } - // because the image stream was found, get its display name - const imageDisplayName = getImageStreamDisplayName(imageStream); + // if the image stream exists and the image version exists, return the image data + return { + imageStream, + imageVersion, + imageAvailability: + imageStream.metadata.labels?.['opendatahub.io/notebook-image'] === 'true' + ? NotebookImageAvailability.ENABLED + : NotebookImageAvailability.DISABLED, + imageDisplayName, + }; +}; - // if the image version is not found, consider the image stream deleted - if (!imageVersion) { - return [ - { - imageAvailability: NotebookImageAvailability.DELETED, - imageDisplayName, - }, - true, - undefined, - ]; +const useNotebookImageData = (notebook?: NotebookKind): NotebookImageData => { + const { dashboardNamespace } = useNamespaces(); + const [images, loaded, loadError] = useImageStreams(dashboardNamespace, true); + + return React.useMemo(() => { + if (!loaded || !notebook) { + return [null, false, loadError]; + } + const data = getNotebookImageData(notebook, images); + + if (data === null) { + return [null, false, loadError]; } - // if the image stream exists and the image version exists, return the image data - return [ - { - imageStream, - imageVersion, - imageAvailability: - imageStream.metadata.labels?.['opendatahub.io/notebook-image'] === 'true' - ? NotebookImageAvailability.ENABLED - : NotebookImageAvailability.DISABLED, - imageDisplayName, - }, - true, - undefined, - ]; + return [data, true, undefined]; }, [images, notebook, loaded, loadError]); };