Skip to content

Commit

Permalink
Handle deleted container size in workbenches
Browse files Browse the repository at this point in the history
  • Loading branch information
manaswinidas committed Jul 3, 2024
1 parent e687eb8 commit da7d2d9
Show file tree
Hide file tree
Showing 6 changed files with 153 additions and 97 deletions.
137 changes: 70 additions & 67 deletions frontend/src/__mocks__/mockDashboardConfig.ts
Original file line number Diff line number Diff line change
@@ -1,4 +1,5 @@
import { DashboardConfigKind, KnownLabels } from '~/k8sTypes';
import { NotebookSize } from '~/types';

type MockDashboardConfigType = {
disableInfo?: boolean;
Expand Down Expand Up @@ -26,6 +27,7 @@ type MockDashboardConfigType = {
disableDistributedWorkloads?: boolean;
disableModelRegistry?: boolean;
disableNotebookController?: boolean;
notebookSizes?: NotebookSize[];
};

export const mockDashboardConfig = ({
Expand Down Expand Up @@ -54,6 +56,73 @@ export const mockDashboardConfig = ({
disableDistributedWorkloads = false,
disableModelRegistry = true,
disableNotebookController = false,
notebookSizes = [
{
name: 'XSmall',
resources: {
limits: {
cpu: '0.5',
memory: '500Mi',
},
requests: {
cpu: '0.1',
memory: '100Mi',
},
},
},
{
name: 'Small',
resources: {
limits: {
cpu: '2',
memory: '8Gi',
},
requests: {
cpu: '1',
memory: '8Gi',
},
},
},
{
name: 'Medium',
resources: {
limits: {
cpu: '6',
memory: '24Gi',
},
requests: {
cpu: '3',
memory: '24Gi',
},
},
},
{
name: 'Large',
resources: {
limits: {
cpu: '14',
memory: '56Gi',
},
requests: {
cpu: '7',
memory: '56Gi',
},
},
},
{
name: 'X Large',
resources: {
limits: {
cpu: '30',
memory: '120Gi',
},
requests: {
cpu: '15',
memory: '120Gi',
},
},
},
],
}: MockDashboardConfigType): DashboardConfigKind => ({
apiVersion: 'opendatahub.io/v1alpha',
kind: 'OdhDashboardConfig',
Expand Down Expand Up @@ -147,73 +216,7 @@ export const mockDashboardConfig = ({
},
},
],
notebookSizes: [
{
name: 'XSmall',
resources: {
limits: {
cpu: '0.5',
memory: '500Mi',
},
requests: {
cpu: '0.1',
memory: '100Mi',
},
},
},
{
name: 'Small',
resources: {
limits: {
cpu: '2',
memory: '8Gi',
},
requests: {
cpu: '1',
memory: '8Gi',
},
},
},
{
name: 'Medium',
resources: {
limits: {
cpu: '6',
memory: '24Gi',
},
requests: {
cpu: '3',
memory: '24Gi',
},
},
},
{
name: 'Large',
resources: {
limits: {
cpu: '14',
memory: '56Gi',
},
requests: {
cpu: '7',
memory: '56Gi',
},
},
},
{
name: 'X Large',
resources: {
limits: {
cpu: '30',
memory: '120Gi',
},
requests: {
cpu: '15',
memory: '120Gi',
},
},
},
],
notebookSizes,
templateOrder: ['test-model'],
templateDisablement: ['test-model'],
},
Expand Down
Original file line number Diff line number Diff line change
@@ -1,4 +1,5 @@
import {
mockDashboardConfig,
mockDscStatus,
mockK8sResourceList,
mockNotebookK8sResource,
Expand Down Expand Up @@ -399,10 +400,66 @@ describe('Workbench page', () => {
});

cy.get('@editWorkbench.all').then((interceptions) => {
expect(interceptions).to.have.length(2); // 1 dry run request and 1 actaul request
expect(interceptions).to.have.length(2); // 1 dry run request and 1 actual request
});
});

it('Handle deleted notebook sizes in workbenches table', () => {
initIntercepts({});
cy.interceptOdh(
'GET /api/config',
mockDashboardConfig({
notebookSizes: [
{
name: 'Medium',
resources: {
limits: {
cpu: '6',
memory: '24Gi',
},
requests: {
cpu: '3',
memory: '24Gi',
},
},
},
{
name: 'Large',
resources: {
limits: {
cpu: '14',
memory: '56Gi',
},
requests: {
cpu: '7',
memory: '56Gi',
},
},
},
{
name: 'X Large',
resources: {
limits: {
cpu: '30',
memory: '120Gi',
},
requests: {
cpu: '15',
memory: '120Gi',
},
},
},
],
}),
);
workbenchPage.visit('test-project');
const notebookRow = workbenchPage.getNotebookRow('Test Notebook');
notebookRow.shouldHaveNotebookImageName('Test Image');
notebookRow.shouldHaveContainerSize('Custom');
notebookRow.findKebabAction('Edit workbench').click();
editSpawnerPage.shouldHaveContainerSizeInput('Custom');
});

it('Validate that updating invalid workbench will navigate to the new page with an error message', () => {
initIntercepts({});
notFoundSpawnerPage.visit('updated-notebook');
Expand Down
Original file line number Diff line number Diff line change
@@ -1,17 +1,8 @@
import * as React from 'react';
import { ActionsColumn, ExpandableRowContent, Tbody, Td, Tr } from '@patternfly/react-table';
import {
Button,
Flex,
FlexItem,
Icon,
Popover,
Split,
SplitItem,
Tooltip,
} from '@patternfly/react-core';
import { Button, Flex, FlexItem, Icon, Popover, Split, SplitItem } from '@patternfly/react-core';
import { useNavigate } from 'react-router-dom';
import { ExclamationCircleIcon, InfoCircleIcon } from '@patternfly/react-icons';
import { InfoCircleIcon } from '@patternfly/react-icons';
import { NotebookState } from '~/pages/projects/notebook/types';
import NotebookRouteLink from '~/pages/projects/notebook/NotebookRouteLink';
import NotebookStatusToggle from '~/pages/projects/notebook/NotebookStatusToggle';
Expand Down Expand Up @@ -51,7 +42,7 @@ const NotebookTableRow: React.FC<NotebookTableRowProps> = ({
const { currentProject } = React.useContext(ProjectDetailsContext);
const navigate = useNavigate();
const [isExpanded, setExpanded] = React.useState(false);
const { size: notebookSize, error: sizeError } = useNotebookDeploymentSize(obj.notebook);
const { size: notebookSize } = useNotebookDeploymentSize(obj.notebook);
const [notebookImage, loaded, loadError] = useNotebookImage(obj.notebook);

return (
Expand Down Expand Up @@ -124,7 +115,7 @@ const NotebookTableRow: React.FC<NotebookTableRowProps> = ({
}
>
<DashboardPopupIconButton
aria-label="Notebook image has out of date Elrya version"
aria-label="Notebook image has out of date Elyra version"
data-testid="outdated-elyra-info"
icon={
<Icon status="info">
Expand All @@ -143,14 +134,7 @@ const NotebookTableRow: React.FC<NotebookTableRowProps> = ({
spaceItems={{ default: 'spaceItemsXs' }}
alignItems={{ default: 'alignItemsCenter' }}
>
<FlexItem>{notebookSize?.name ?? 'Unknown'}</FlexItem>
{sizeError && (
<Tooltip content={sizeError}>
<Icon aria-label="error icon" role="button" status="danger" tabIndex={0}>
<ExclamationCircleIcon />
</Icon>
</Tooltip>
)}
<FlexItem>{notebookSize?.name ?? 'Custom'}</FlexItem>
</Flex>
</Td>
) : null}
Expand Down
12 changes: 9 additions & 3 deletions frontend/src/pages/projects/screens/spawner/SpawnerPage.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -11,7 +11,7 @@ import {
StackItem,
} from '@patternfly/react-core';
import ApplicationsPage from '~/pages/ApplicationsPage';
import { ImageStreamAndVersion } from '~/types';
import { ImageStreamAndVersion, NotebookSize } from '~/types';
import GenericSidebar from '~/components/GenericSidebar';
import NameDescriptionField from '~/concepts/k8s/NameDescriptionField';
import { ProjectDetailsContext } from '~/pages/projects/ProjectDetailsContext';
Expand Down Expand Up @@ -61,7 +61,7 @@ const SpawnerPage: React.FC<SpawnerPageProps> = ({ existingNotebook }) => {
imageStream: undefined,
imageVersion: undefined,
});
const { selectedSize, setSelectedSize, sizes } = useNotebookSize();
const { selectedSize, setSelectedSize, sizes } = useNotebookSize(!!existingNotebook);
const [supportedAcceleratorProfiles, setSupportedAcceleratorProfiles] = React.useState<
string[] | undefined
>();
Expand Down Expand Up @@ -97,10 +97,16 @@ const SpawnerPage: React.FC<SpawnerPageProps> = ({ existingNotebook }) => {

const { size: notebookSize } = useNotebookDeploymentSize(existingNotebook);
React.useEffect(() => {
const customSize: NotebookSize = { name: 'Custom', resources: {} };
if (notebookSize) {
setSelectedSize(notebookSize.name);
} else {
if (existingNotebook && !sizes.some((size) => size.name === 'Custom')) {
sizes.unshift(customSize);
}
setSelectedSize(customSize.name);
}
}, [notebookSize, setSelectedSize]);
}, [notebookSize, setSelectedSize, sizes, existingNotebook]);

const [notebookAcceleratorProfileState, setNotebookAcceleratorProfileState] =
useNotebookAcceleratorProfile(existingNotebook);
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -35,7 +35,11 @@ const ContainerSizeSelector: React.FC<ContainerSizeSelectorProps> = ({
{sizes.map((size) => {
const { name } = size;
const desc = getSizeDescription(size);
return <SelectOption key={name} value={name} description={desc} />;
return name === 'Custom' ? (
<SelectOption key={name} value={name} />
) : (
<SelectOption key={name} value={name} description={desc} />
);
})}
</Select>

Expand Down
10 changes: 6 additions & 4 deletions frontend/src/pages/projects/screens/spawner/useNotebookSize.ts
Original file line number Diff line number Diff line change
Expand Up @@ -14,7 +14,9 @@ export const getNotebookSizes = (config: DashboardConfigKind): NotebookSize[] =>
return sizes;
};

export const useNotebookSize = (): {
export const useNotebookSize = (
isExistingNotebook: boolean,
): {
selectedSize: NotebookSize;
setSelectedSize: (size: string) => void;
sizes: NotebookSize[];
Expand All @@ -28,15 +30,15 @@ export const useNotebookSize = (): {
const setSelectedSizeSafe = React.useCallback(
(name: string) => {
let foundSize = sizes.find((size) => size.name === name);
if (!foundSize) {
if (!foundSize && isExistingNotebook) {
[foundSize] = sizes;
notification.warning(
'The size you select is no longer available, we have set the size to the default one.',
);
}
setSelectedSize(foundSize);
setSelectedSize(foundSize ?? sizes[0]);
},
[notification, sizes],
[notification, sizes, isExistingNotebook],
);

return { selectedSize, setSelectedSize: setSelectedSizeSafe, sizes };
Expand Down

0 comments on commit da7d2d9

Please sign in to comment.