From d339863050e72be23156d0c44a9851a581463b28 Mon Sep 17 00:00:00 2001 From: Pushpa Padti <99261071+ppadti@users.noreply.github.com> Date: Fri, 10 Jan 2025 20:25:55 +0530 Subject: [PATCH 1/6] Add cypress tests to submit logic (#3631) --- frontend/src/__mocks__/mockModelRegistry.ts | 74 +- .../cypress/pages/modelRegistrySettings.ts | 36 + .../cypress/cypress/support/commands/odh.ts | 6 +- .../modelRegistrySettings/mockCertificate.pem | 1 + .../modelRegistrySettings.cy.ts | 824 +++++++++++++++++- .../modelRegistrySettings/unSupportedFile.txt | 1 + .../CreateMRSecureDBSection.tsx | 18 +- .../modelRegistrySettings/CreateModal.tsx | 45 +- .../modelRegistrySettings/PemFileUpload.tsx | 3 + .../src/pages/modelRegistrySettings/utils.ts | 19 +- 10 files changed, 966 insertions(+), 61 deletions(-) create mode 100644 frontend/src/__tests__/cypress/cypress/tests/mocked/modelRegistrySettings/mockCertificate.pem create mode 100644 frontend/src/__tests__/cypress/cypress/tests/mocked/modelRegistrySettings/unSupportedFile.txt diff --git a/frontend/src/__mocks__/mockModelRegistry.ts b/frontend/src/__mocks__/mockModelRegistry.ts index d734162a3a..6b3f53d939 100644 --- a/frontend/src/__mocks__/mockModelRegistry.ts +++ b/frontend/src/__mocks__/mockModelRegistry.ts @@ -4,6 +4,8 @@ type MockModelRegistryType = { name?: string; namespace?: string; conditions?: K8sCondition[]; + sslRootCertificateConfigMap?: { name: string; key: string } | null; + sslRootCertificateSecret?: { name: string; key: string } | null; }; export const mockModelRegistry = ({ @@ -25,37 +27,47 @@ export const mockModelRegistry = ({ type: 'Available', }, ], -}: MockModelRegistryType): ModelRegistryKind => ({ - apiVersion: 'modelregistry.opendatahub.io/v1alpha1', - kind: 'ModelRegistry', - metadata: { - name, - creationTimestamp: '2024-03-14T08:01:42Z', - namespace, - }, - spec: { - grpc: {}, - rest: {}, - istio: { - gateway: { - grpc: { tls: {} }, - rest: { tls: {} }, - }, + sslRootCertificateConfigMap = null, + sslRootCertificateSecret = null, +}: MockModelRegistryType): ModelRegistryKind => { + const data: ModelRegistryKind = { + apiVersion: 'modelregistry.opendatahub.io/v1alpha1', + kind: 'ModelRegistry', + metadata: { + name, + creationTimestamp: '2024-03-14T08:01:42Z', + namespace, }, - postgres: { - database: 'model-registry', - host: 'model-registry-db', - passwordSecret: { - key: 'database-password', - name: 'model-registry-db', + spec: { + grpc: {}, + rest: {}, + istio: { + gateway: { + grpc: { tls: {} }, + rest: { tls: {} }, + }, + }, + mysql: { + database: 'model-registry', + host: 'model-registry-db', + passwordSecret: { + key: 'database-password', + name: 'model-registry-db', + }, + port: 5432, + skipDBCreation: false, + username: 'mlmduser', }, - port: 5432, - skipDBCreation: false, - sslMode: 'disable', - username: 'mlmduser', }, - }, - status: { - conditions, - }, -}); + status: { + conditions, + }, + }; + + if (sslRootCertificateConfigMap && data.spec.mysql) { + data.spec.mysql.sslRootCertificateConfigMap = sslRootCertificateConfigMap; + } else if (sslRootCertificateSecret && data.spec.mysql) { + data.spec.mysql.sslRootCertificateSecret = sslRootCertificateSecret; + } + return data; +}; diff --git a/frontend/src/__tests__/cypress/cypress/pages/modelRegistrySettings.ts b/frontend/src/__tests__/cypress/cypress/pages/modelRegistrySettings.ts index a4024eb399..efe7ea5f51 100644 --- a/frontend/src/__tests__/cypress/cypress/pages/modelRegistrySettings.ts +++ b/frontend/src/__tests__/cypress/cypress/pages/modelRegistrySettings.ts @@ -1,5 +1,7 @@ import { appChrome } from './appChrome'; +import { Contextual } from './components/Contextual'; import { K8sNameDescriptionField } from './components/subComponents/K8sNameDescriptionField'; +import { SearchSelector } from './components/subComponents/SearchSelector'; export enum FormFieldSelector { NAME = '#mr-name', @@ -29,6 +31,10 @@ export enum DatabaseDetailsTestId { class ModelRegistrySettings { k8sNameDescription = new K8sNameDescriptionField('mr'); + resourceNameSelect = new SearchSelector('existing-ca-resource-selector'); + + keySelect = new SearchSelector('existing-ca-key-selector'); + visit(wait = true) { cy.visitWithLogin('/modelRegistrySettings'); if (wait) { @@ -130,6 +136,10 @@ class ModelRegistrySettings { return cy.findByTestId('existing-ca-radio'); } + findUploadNewCertificateRadio() { + return cy.findByTestId('new-certificate-ca-radio'); + } + findAddSecureDbMRCheckbox() { return cy.findByTestId('add-secure-db-mr-checkbox'); } @@ -141,6 +151,32 @@ class ModelRegistrySettings { findExistingCAKeyInputToggle() { return cy.findByTestId('existing-ca-key-selector-toggle'); } + + getNewCertificateUpload() { + return new CertificateUpload(() => cy.findByTestId('certificate-upload')); + } + + findErrorFetchingResourceAlert() { + return cy.findByTestId('error-fetching-resource-alert'); + } + + findCertificateNote() { + return cy.findByTestId('certificate-note'); + } +} + +class CertificateUpload extends Contextual { + findUploadCertificateInput() { + return this.find().find('[data-testid="new-certificate-upload"] input[type="file"]'); + } + + uploadPemFile(filePath: string) { + this.findUploadCertificateInput().selectFile([filePath], { force: true }); + } + + findRestrictedFileUploadHelptext() { + return cy.findByTestId('restricted-file-example-helpText'); + } } export const modelRegistrySettings = new ModelRegistrySettings(); diff --git a/frontend/src/__tests__/cypress/cypress/support/commands/odh.ts b/frontend/src/__tests__/cypress/cypress/support/commands/odh.ts index 1da4c6a9f2..296dc1aab2 100644 --- a/frontend/src/__tests__/cypress/cypress/support/commands/odh.ts +++ b/frontend/src/__tests__/cypress/cypress/support/commands/odh.ts @@ -368,6 +368,10 @@ declare global { type: 'GET /api/modelRegistries', response: OdhResponse>, ) => Cypress.Chainable) & + (( + type: 'POST /api/modelRegistries', + response: OdhResponse, + ) => Cypress.Chainable) & (( type: 'PATCH /api/modelRegistries/:modelRegistryName', options: { @@ -656,7 +660,7 @@ declare global { ) => Cypress.Chainable) & (( type: 'GET /api/modelRegistryCertificates', - response: ListConfigSecretsResponse, + response: OdhResponse, ) => Cypress.Chainable) & (( type: 'POST /api/connection-types', diff --git a/frontend/src/__tests__/cypress/cypress/tests/mocked/modelRegistrySettings/mockCertificate.pem b/frontend/src/__tests__/cypress/cypress/tests/mocked/modelRegistrySettings/mockCertificate.pem new file mode 100644 index 0000000000..01f9fc9acf --- /dev/null +++ b/frontend/src/__tests__/cypress/cypress/tests/mocked/modelRegistrySettings/mockCertificate.pem @@ -0,0 +1 @@ +sample certificate \ No newline at end of file diff --git a/frontend/src/__tests__/cypress/cypress/tests/mocked/modelRegistrySettings/modelRegistrySettings.cy.ts b/frontend/src/__tests__/cypress/cypress/tests/mocked/modelRegistrySettings/modelRegistrySettings.cy.ts index 6bd6911495..049112cd94 100644 --- a/frontend/src/__tests__/cypress/cypress/tests/mocked/modelRegistrySettings/modelRegistrySettings.cy.ts +++ b/frontend/src/__tests__/cypress/cypress/tests/mocked/modelRegistrySettings/modelRegistrySettings.cy.ts @@ -28,18 +28,23 @@ const groupSubjects: RoleBindingSubject[] = [ }, ]; +const sampleCertificatePath = './cypress/tests/mocked/modelRegistrySettings/mockCertificate.pem'; +const unSupportedFilePath = './cypress/tests/mocked/modelRegistrySettings/unSupportedFile.txt'; + const setupMocksForMRSettingAccess = ({ hasModelRegistries = true, hasDatabasePassword = true, disableModelRegistrySecureDB = true, - secrets = [{ name: 'foo', keys: ['foo.crt', 'bar.crt'] }], + secrets = [{ name: 'sampleSecret', keys: ['foo.crt', 'bar.crt'] }], configMaps = [{ name: 'foo-bar', keys: ['bar.crt'] }], + failedToLoadCertificates = false, }: { hasModelRegistries?: boolean; hasDatabasePassword?: boolean; disableModelRegistrySecureDB?: boolean; secrets?: ConfigSecretItem[]; configMaps?: ConfigSecretItem[]; + failedToLoadCertificates?: boolean; }) => { asProductAdminUser(); cy.interceptOdh( @@ -64,6 +69,7 @@ const setupMocksForMRSettingAccess = ({ requiredCapabilities: [StackCapability.SERVICE_MESH, StackCapability.SERVICE_MESH_AUTHZ], }), ); + cy.interceptOdh('POST /api/modelRegistries', mockModelRegistry({})).as('createModelRegistry'); cy.interceptOdh( 'GET /api/modelRegistries', mockK8sResourceList( @@ -84,6 +90,25 @@ const setupMocksForMRSettingAccess = ({ }, ], }), + mockModelRegistry({ + name: 'test-registry-4', + sslRootCertificateConfigMap: { name: 'odh-trusted-ca-bundle', key: 'ca-bundle.crt' }, + }), + mockModelRegistry({ + name: 'test-registry-5', + sslRootCertificateConfigMap: { + name: 'odh-trusted-ca-bundle', + key: 'odh-ca-bundle.crt', + }, + }), + mockModelRegistry({ + name: 'test-registry-7', + sslRootCertificateSecret: { name: 'sampleSecret', key: 'foo.crt' }, + }), + mockModelRegistry({ + name: 'test-registry-8', + sslRootCertificateConfigMap: { name: 'new-certificate-db-credential', key: 'ca.crt' }, + }), ] : [], ), @@ -107,6 +132,59 @@ const setupMocksForMRSettingAccess = ({ req.reply(500); // Something went wrong on the backend when decoding the secret }, ); + cy.interceptOdh( + 'GET /api/modelRegistries/:modelRegistryName', + { + path: { modelRegistryName: 'test-registry-4' }, + }, + { + modelRegistry: mockModelRegistry({ + name: 'test-registry-4', + sslRootCertificateConfigMap: { name: 'odh-trusted-ca-bundle', key: 'ca-bundle.crt' }, + }), + databasePassword: hasDatabasePassword ? 'test-password' : undefined, + }, + ); + cy.interceptOdh( + 'GET /api/modelRegistries/:modelRegistryName', + { + path: { modelRegistryName: 'test-registry-5' }, + }, + { + modelRegistry: mockModelRegistry({ + name: 'test-registry-5', + sslRootCertificateConfigMap: { name: 'odh-trusted-ca-bundle', key: 'odh-ca-bundle.crt' }, + }), + databasePassword: hasDatabasePassword ? 'test-password' : undefined, + }, + ); + + cy.interceptOdh( + 'GET /api/modelRegistries/:modelRegistryName', + { + path: { modelRegistryName: 'test-registry-7' }, + }, + { + modelRegistry: mockModelRegistry({ + name: 'test-registry-7', + sslRootCertificateSecret: { name: 'sampleSecret', key: 'foo.crt' }, + }), + databasePassword: hasDatabasePassword ? 'test-password' : undefined, + }, + ); + cy.interceptOdh( + 'GET /api/modelRegistries/:modelRegistryName', + { + path: { modelRegistryName: 'test-registry-8' }, + }, + { + modelRegistry: mockModelRegistry({ + name: 'test-registry-8', + sslRootCertificateConfigMap: { name: 'new-certificate-db-credential', key: 'ca.crt' }, + }), + databasePassword: hasDatabasePassword ? 'test-password' : undefined, + }, + ); cy.interceptOdh( 'PATCH /api/modelRegistries/:modelRegistryName', @@ -117,7 +195,63 @@ const setupMocksForMRSettingAccess = ({ modelRegistry: mockModelRegistry({ name: 'test-registry-1' }), databasePassword: 'test-password', }, - ); + ).as('updateModelRegistry'); + + cy.interceptOdh( + 'PATCH /api/modelRegistries/:modelRegistryName', + { + path: { modelRegistryName: 'test-registry-4' }, + }, + { + modelRegistry: mockModelRegistry({ + name: 'test-registry-4', + sslRootCertificateConfigMap: { name: 'odh-trusted-ca-bundle', key: 'odh-ca-bundle.crt' }, + }), + databasePassword: 'test-password', + }, + ).as('updateTestRegistry-4'); + + cy.interceptOdh( + 'PATCH /api/modelRegistries/:modelRegistryName', + { + path: { modelRegistryName: 'test-registry-5' }, + }, + { + modelRegistry: mockModelRegistry({ + name: 'test-registry-5', + sslRootCertificateConfigMap: { name: 'foo-bar', key: 'bar.crt' }, + }), + databasePassword: 'test-password', + }, + ).as('updateTestRegistry-5'); + + cy.interceptOdh( + 'PATCH /api/modelRegistries/:modelRegistryName', + { + path: { modelRegistryName: 'test-registry-7' }, + }, + { + modelRegistry: mockModelRegistry({ + name: 'test-registry-7', + sslRootCertificateConfigMap: { name: 'foo-bar', key: 'bar.crt' }, + }), + databasePassword: 'test-password', + }, + ).as('updateTestRegistry-7'); + + cy.interceptOdh( + 'PATCH /api/modelRegistries/:modelRegistryName', + { + path: { modelRegistryName: 'test-registry-8' }, + }, + { + modelRegistry: mockModelRegistry({ + name: 'test-registry-8', + sslRootCertificateConfigMap: { name: 'foo-bar', key: 'bar.crt' }, + }), + databasePassword: 'test-password', + }, + ).as('updateTestRegistry-8'); cy.interceptOdh( 'GET /api/modelRegistryRoleBindings', @@ -136,12 +270,37 @@ const setupMocksForMRSettingAccess = ({ roleRefName: 'registry-user-test-registry-2', modelRegistryName: 'test-registry-2', }), + mockRoleBindingK8sResource({ + namespace: 'odh-model-registries', + name: 'test-registry-4-user', + subjects: groupSubjects, + roleRefName: 'registry-user-test-registry-4', + modelRegistryName: 'test-registry-4', + }), + mockRoleBindingK8sResource({ + namespace: 'odh-model-registries', + name: 'test-registry-8-user', + subjects: groupSubjects, + roleRefName: 'registry-user-test-registry-8', + modelRegistryName: 'test-registry-8', + }), + mockRoleBindingK8sResource({ + namespace: 'odh-model-registries', + name: 'test-registry-5-user', + subjects: groupSubjects, + roleRefName: 'registry-user-test-registry-5', + modelRegistryName: 'test-registry-5', + }), ]), ); cy.interceptOdh( 'GET /api/modelRegistryCertificates', - mockConfigMapsSecrets({ secrets, configMaps }), + failedToLoadCertificates + ? (req) => { + req.reply(500); // Something went wrong + } + : mockConfigMapsSecrets({ secrets, configMaps }), ); }; @@ -207,16 +366,91 @@ describe('CreateModal', () => { modelRegistrySettings.findSubmitButton().should('be.enabled'); modelRegistrySettings.shouldHaveNoErrors(); + modelRegistrySettings.findSubmitButton().click(); + + cy.wait('@createModelRegistry').then((interception) => { + expect(interception.request.body).to.containSubset({ + modelRegistry: { + metadata: { + name: 'image', + namespace: 'odh-model-registries', + annotations: { + 'openshift.io/description': '', + 'openshift.io/display-name': 'valid-mr-name', + }, + }, + spec: { + mysql: { + host: 'host', + port: 1234, + database: 'myDatabase', + username: 'validUser', + skipDBCreation: false, + }, + }, + }, + databasePassword: 'strongPassword', + }); + }); }); - it('checks whether the secure DB section exists and both first and second radio options are disabled', () => { - setupMocksForMRSettingAccess({ disableModelRegistrySecureDB: false }); + it('show error when certificates fails to load', () => { + setupMocksForMRSettingAccess({ + disableModelRegistrySecureDB: false, + failedToLoadCertificates: true, + }); modelRegistrySettings.visit(true); cy.findByText('Create model registry').click(); + modelRegistrySettings.findFormField(FormFieldSelector.NAME).type('valid-mr-name'); + modelRegistrySettings.findFormField(FormFieldSelector.HOST).type('host'); + modelRegistrySettings.findFormField(FormFieldSelector.PORT).type('1234'); + modelRegistrySettings.findFormField(FormFieldSelector.USERNAME).type('validUser'); + modelRegistrySettings.findFormField(FormFieldSelector.PASSWORD).type('strongPassword'); + modelRegistrySettings.findFormField(FormFieldSelector.DATABASE).type('myDatabase'); + modelRegistrySettings.findAddSecureDbMRCheckbox().should('exist'); + modelRegistrySettings.findAddSecureDbMRCheckbox().check(); + modelRegistrySettings.findErrorFetchingResourceAlert().should('exist'); + modelRegistrySettings + .findErrorFetchingResourceAlert() + .should('have.text', 'Danger alert:Error fetching config maps and secrets'); + modelRegistrySettings.findSubmitButton().should('be.disabled'); + }); + + it('checks whether the secure DB section exists, both first and second radio options are disabled and third option is checked', () => { + setupMocksForMRSettingAccess({ + disableModelRegistrySecureDB: false, + }); + modelRegistrySettings.visit(true); + cy.findByText('Create model registry').click(); + modelRegistrySettings.findAddSecureDbMRCheckbox().should('exist'); modelRegistrySettings.findAddSecureDbMRCheckbox().check(); modelRegistrySettings.findClusterWideCARadio().should('be.disabled'); modelRegistrySettings.findOpenshiftCARadio().should('be.disabled'); + modelRegistrySettings.findSubmitButton().should('be.disabled'); + modelRegistrySettings.findExistingCARadio().should('be.checked'); + }); + + it('when Use cluster-wide CA bundle is disabled, it should check Use Open Data Hub CA bundle', () => { + setupMocksForMRSettingAccess({ + disableModelRegistrySecureDB: false, + configMaps: [{ name: 'odh-trusted-ca-bundle', keys: ['odh-ca-bundle.crt'] }], + }); + modelRegistrySettings.visit(true); + cy.findByText('Create model registry').click(); + modelRegistrySettings.findFormField(FormFieldSelector.NAME).type('valid-mr-name'); + modelRegistrySettings.findFormField(FormFieldSelector.HOST).type('host'); + modelRegistrySettings.findFormField(FormFieldSelector.PORT).type('1234'); + modelRegistrySettings.findFormField(FormFieldSelector.USERNAME).type('validUser'); + modelRegistrySettings.findFormField(FormFieldSelector.PASSWORD).type('strongPassword'); + modelRegistrySettings.findFormField(FormFieldSelector.DATABASE).type('myDatabase'); + + modelRegistrySettings.findAddSecureDbMRCheckbox().should('exist'); + modelRegistrySettings.findAddSecureDbMRCheckbox().check(); + modelRegistrySettings.findClusterWideCARadio().should('be.disabled'); + modelRegistrySettings.findOpenshiftCARadio().should('be.enabled'); + modelRegistrySettings.findOpenshiftCARadio().should('be.checked'); + modelRegistrySettings.findSubmitButton().should('be.enabled'); }); it('both first and second radio options are enabled', () => { @@ -226,10 +460,300 @@ describe('CreateModal', () => { }); modelRegistrySettings.visit(true); cy.findByText('Create model registry').click(); + modelRegistrySettings.findFormField(FormFieldSelector.NAME).type('valid-mr-name'); + modelRegistrySettings.findFormField(FormFieldSelector.HOST).type('host'); + modelRegistrySettings.findFormField(FormFieldSelector.PORT).type('1234'); + modelRegistrySettings.findFormField(FormFieldSelector.USERNAME).type('validUser'); + modelRegistrySettings.findFormField(FormFieldSelector.PASSWORD).type('strongPassword'); + modelRegistrySettings.findFormField(FormFieldSelector.DATABASE).type('myDatabase'); + modelRegistrySettings.findAddSecureDbMRCheckbox().should('exist'); modelRegistrySettings.findAddSecureDbMRCheckbox().check(); modelRegistrySettings.findClusterWideCARadio().should('be.enabled'); + modelRegistrySettings.findClusterWideCARadio().should('be.checked'); + modelRegistrySettings.findOpenshiftCARadio().should('be.enabled'); + modelRegistrySettings.findSubmitButton().should('be.enabled'); + }); + + it('create a model registry with Use cluster-wide CA bundle option', () => { + setupMocksForMRSettingAccess({ + disableModelRegistrySecureDB: false, + configMaps: [{ name: 'odh-trusted-ca-bundle', keys: ['ca-bundle.crt'] }], + }); + modelRegistrySettings.visit(true); + cy.findByText('Create model registry').click(); + modelRegistrySettings.findFormField(FormFieldSelector.NAME).type('valid-mr-name'); + modelRegistrySettings.findFormField(FormFieldSelector.HOST).type('host'); + modelRegistrySettings.findFormField(FormFieldSelector.PORT).type('1234'); + modelRegistrySettings.findFormField(FormFieldSelector.USERNAME).type('validUser'); + modelRegistrySettings.findFormField(FormFieldSelector.PASSWORD).type('strongPassword'); + modelRegistrySettings.findFormField(FormFieldSelector.DATABASE).type('myDatabase'); + modelRegistrySettings.findAddSecureDbMRCheckbox().should('exist'); + modelRegistrySettings.findAddSecureDbMRCheckbox().check(); + modelRegistrySettings.findClusterWideCARadio().should('be.checked'); + modelRegistrySettings.findSubmitButton().should('be.enabled'); + modelRegistrySettings.findSubmitButton().click(); + + cy.wait('@createModelRegistry').then((interception) => { + expect(interception.request.body).to.containSubset({ + modelRegistry: { + metadata: { + name: 'valid-mr-name', + namespace: 'odh-model-registries', + annotations: { + 'openshift.io/description': '', + 'openshift.io/display-name': 'valid-mr-name', + }, + }, + spec: { + mysql: { + host: 'host', + port: 1234, + database: 'myDatabase', + username: 'validUser', + skipDBCreation: false, + sslRootCertificateConfigMap: { name: 'odh-trusted-ca-bundle', key: 'ca-bundle.crt' }, + }, + }, + }, + databasePassword: 'strongPassword', + }); + }); + }); + + it('create a model registry with Use Open Data Hub CA bundle option', () => { + setupMocksForMRSettingAccess({ + disableModelRegistrySecureDB: false, + configMaps: [{ name: 'odh-trusted-ca-bundle', keys: ['odh-ca-bundle.crt'] }], + }); + modelRegistrySettings.visit(true); + cy.findByText('Create model registry').click(); + modelRegistrySettings.findFormField(FormFieldSelector.NAME).type('valid-mr-name'); + modelRegistrySettings.findFormField(FormFieldSelector.HOST).type('host'); + modelRegistrySettings.findFormField(FormFieldSelector.PORT).type('1234'); + modelRegistrySettings.findFormField(FormFieldSelector.USERNAME).type('validUser'); + modelRegistrySettings.findFormField(FormFieldSelector.PASSWORD).type('strongPassword'); + modelRegistrySettings.findFormField(FormFieldSelector.DATABASE).type('myDatabase'); + modelRegistrySettings.findAddSecureDbMRCheckbox().should('exist'); + modelRegistrySettings.findAddSecureDbMRCheckbox().check(); + modelRegistrySettings.findClusterWideCARadio().should('be.disabled'); modelRegistrySettings.findOpenshiftCARadio().should('be.enabled'); + modelRegistrySettings.findOpenshiftCARadio().should('be.checked'); + modelRegistrySettings.findSubmitButton().should('be.enabled'); + modelRegistrySettings.findSubmitButton().click(); + + cy.wait('@createModelRegistry').then((interception) => { + expect(interception.request.body).to.containSubset({ + modelRegistry: { + metadata: { + name: 'valid-mr-name', + namespace: 'odh-model-registries', + annotations: { + 'openshift.io/description': '', + 'openshift.io/display-name': 'valid-mr-name', + }, + }, + spec: { + mysql: { + host: 'host', + port: 1234, + database: 'myDatabase', + username: 'validUser', + skipDBCreation: false, + sslRootCertificateConfigMap: { + name: 'odh-trusted-ca-bundle', + key: 'odh-ca-bundle.crt', + }, + }, + }, + }, + databasePassword: 'strongPassword', + }); + }); + }); + + it('create a model registry with Choose from existing certificates option - secret', () => { + setupMocksForMRSettingAccess({ + disableModelRegistrySecureDB: false, + }); + modelRegistrySettings.visit(true); + cy.findByText('Create model registry').click(); + modelRegistrySettings.findFormField(FormFieldSelector.NAME).type('valid-mr-name'); + modelRegistrySettings.findFormField(FormFieldSelector.HOST).type('host'); + modelRegistrySettings.findFormField(FormFieldSelector.PORT).type('1234'); + modelRegistrySettings.findFormField(FormFieldSelector.USERNAME).type('validUser'); + modelRegistrySettings.findFormField(FormFieldSelector.PASSWORD).type('strongPassword'); + modelRegistrySettings.findFormField(FormFieldSelector.DATABASE).type('myDatabase'); + modelRegistrySettings.findAddSecureDbMRCheckbox().should('exist'); + modelRegistrySettings.findAddSecureDbMRCheckbox().check(); + modelRegistrySettings.findClusterWideCARadio().should('be.disabled'); + modelRegistrySettings.findOpenshiftCARadio().should('be.disabled'); + modelRegistrySettings.findExistingCARadio().should('be.enabled'); + modelRegistrySettings.findExistingCARadio().should('be.checked'); + + modelRegistrySettings.findExistingCAKeyInputToggle().should('be.disabled'); + modelRegistrySettings.findExistingCAResourceInputToggle().should('be.enabled'); + modelRegistrySettings.resourceNameSelect.openAndSelectItem('sampleSecret', true); + modelRegistrySettings.findExistingCAKeyInputToggle().should('be.enabled'); + modelRegistrySettings.keySelect.openAndSelectItem('bar.crt', true); + + modelRegistrySettings.findSubmitButton().should('be.enabled'); + modelRegistrySettings.findSubmitButton().click(); + + cy.wait('@createModelRegistry').then((interception) => { + expect(interception.request.body).to.containSubset({ + modelRegistry: { + metadata: { + name: 'valid-mr-name', + namespace: 'odh-model-registries', + annotations: { + 'openshift.io/description': '', + 'openshift.io/display-name': 'valid-mr-name', + }, + }, + spec: { + mysql: { + host: 'host', + port: 1234, + database: 'myDatabase', + username: 'validUser', + skipDBCreation: false, + sslRootCertificateSecret: { name: 'sampleSecret', key: 'bar.crt' }, + }, + }, + }, + databasePassword: 'strongPassword', + }); + }); + }); + + it('create a model registry with Choose from existing certificates - config map', () => { + setupMocksForMRSettingAccess({ + disableModelRegistrySecureDB: false, + }); + modelRegistrySettings.visit(true); + cy.findByText('Create model registry').click(); + modelRegistrySettings.findFormField(FormFieldSelector.NAME).type('valid-mr-name'); + modelRegistrySettings.findFormField(FormFieldSelector.HOST).type('host'); + modelRegistrySettings.findFormField(FormFieldSelector.PORT).type('1234'); + modelRegistrySettings.findFormField(FormFieldSelector.USERNAME).type('validUser'); + modelRegistrySettings.findFormField(FormFieldSelector.PASSWORD).type('strongPassword'); + modelRegistrySettings.findFormField(FormFieldSelector.DATABASE).type('myDatabase'); + modelRegistrySettings.findAddSecureDbMRCheckbox().should('exist'); + modelRegistrySettings.findAddSecureDbMRCheckbox().check(); + modelRegistrySettings.findExistingCARadio().should('be.enabled'); + + modelRegistrySettings.findClusterWideCARadio().should('be.disabled'); + modelRegistrySettings.findOpenshiftCARadio().should('be.disabled'); + modelRegistrySettings.findExistingCARadio().should('be.checked'); + + modelRegistrySettings.findExistingCAKeyInputToggle().should('be.disabled'); + modelRegistrySettings.findExistingCAResourceInputToggle().should('be.enabled'); + modelRegistrySettings.resourceNameSelect.openAndSelectItem('foo-bar', true); + modelRegistrySettings.findExistingCAKeyInputToggle().should('be.enabled'); + modelRegistrySettings.keySelect.openAndSelectItem('bar.crt', true); + + modelRegistrySettings.findSubmitButton().should('be.enabled'); + modelRegistrySettings.findSubmitButton().click(); + + cy.wait('@createModelRegistry').then((interception) => { + expect(interception.request.body).to.containSubset({ + modelRegistry: { + metadata: { + name: 'valid-mr-name', + namespace: 'odh-model-registries', + annotations: { + 'openshift.io/description': '', + 'openshift.io/display-name': 'valid-mr-name', + }, + }, + spec: { + mysql: { + host: 'host', + port: 1234, + database: 'myDatabase', + username: 'validUser', + skipDBCreation: false, + sslRootCertificateConfigMap: { name: 'foo-bar', key: 'bar.crt' }, + }, + }, + }, + databasePassword: 'strongPassword', + }); + }); + }); + + it('create a model registry with Upload new certificate option', () => { + setupMocksForMRSettingAccess({ + disableModelRegistrySecureDB: false, + }); + modelRegistrySettings.visit(true); + cy.findByText('Create model registry').click(); + modelRegistrySettings.findFormField(FormFieldSelector.NAME).type('valid-mr-name'); + modelRegistrySettings.findFormField(FormFieldSelector.HOST).type('host'); + modelRegistrySettings.findFormField(FormFieldSelector.PORT).type('1234'); + modelRegistrySettings.findFormField(FormFieldSelector.USERNAME).type('validUser'); + modelRegistrySettings.findFormField(FormFieldSelector.PASSWORD).type('strongPassword'); + modelRegistrySettings.findFormField(FormFieldSelector.DATABASE).type('myDatabase'); + modelRegistrySettings.findAddSecureDbMRCheckbox().should('exist'); + modelRegistrySettings.findAddSecureDbMRCheckbox().check(); + modelRegistrySettings.findUploadNewCertificateRadio().check(); + const certificateUploadSection = modelRegistrySettings.getNewCertificateUpload(); + certificateUploadSection.uploadPemFile(sampleCertificatePath); + modelRegistrySettings.findSubmitButton().should('be.enabled'); + modelRegistrySettings.findSubmitButton().click(); + + cy.wait('@createModelRegistry').then((interception) => { + expect(interception.request.body).to.containSubset({ + modelRegistry: { + metadata: { + name: 'valid-mr-name', + namespace: 'odh-model-registries', + annotations: { + 'openshift.io/description': '', + 'openshift.io/display-name': 'valid-mr-name', + }, + }, + spec: { + mysql: { + host: 'host', + port: 1234, + database: 'myDatabase', + username: 'validUser', + skipDBCreation: false, + sslRootCertificateConfigMap: { name: 'valid-mr-name-db-credential', key: '' }, + }, + }, + }, + databasePassword: 'strongPassword', + newDatabaseCACertificate: 'sample certificate', + }); + }); + }); + + it('Show error when creating a model registry with Upload new certificate option with unsupported file', () => { + setupMocksForMRSettingAccess({ + disableModelRegistrySecureDB: false, + }); + modelRegistrySettings.visit(true); + cy.findByText('Create model registry').click(); + modelRegistrySettings.findFormField(FormFieldSelector.NAME).type('valid-mr-name'); + modelRegistrySettings.findFormField(FormFieldSelector.HOST).type('host'); + modelRegistrySettings.findFormField(FormFieldSelector.PORT).type('1234'); + modelRegistrySettings.findFormField(FormFieldSelector.USERNAME).type('validUser'); + modelRegistrySettings.findFormField(FormFieldSelector.PASSWORD).type('strongPassword'); + modelRegistrySettings.findFormField(FormFieldSelector.DATABASE).type('myDatabase'); + modelRegistrySettings.findAddSecureDbMRCheckbox().should('exist'); + modelRegistrySettings.findAddSecureDbMRCheckbox().check(); + modelRegistrySettings.findUploadNewCertificateRadio().check(); + modelRegistrySettings.findCertificateNote().should('exist'); + const certificateUploadSection = modelRegistrySettings.getNewCertificateUpload(); + certificateUploadSection.uploadPemFile(unSupportedFilePath); + certificateUploadSection.findRestrictedFileUploadHelptext().should('exist'); + certificateUploadSection + .findRestrictedFileUploadHelptext() + .should('have.text', 'Must be a PEM file: error status;'); + modelRegistrySettings.findSubmitButton().should('be.disabled'); }); }); @@ -270,6 +794,10 @@ describe('EditModelRegistry', () => { .should('have.value', 'model-registry'); modelRegistrySettings.findSubmitButton().should('be.enabled'); modelRegistrySettings.findSubmitButton().click(); + + cy.wait('@updateModelRegistry').then((interception) => { + expect(interception.request.body).to.containSubset({}); + }); }); it('Shows skeleton, when password is loading', () => { @@ -318,6 +846,292 @@ describe('EditModelRegistry', () => { .should('have.value', 'model-registry'); modelRegistrySettings.findSubmitButton().should('be.disabled'); }); + + it('Edit model registry(without CA certificate) to add CA certificate with Use cluster-wide CA bundle option', () => { + setupMocksForMRSettingAccess({ + disableModelRegistrySecureDB: false, + configMaps: [{ name: 'odh-trusted-ca-bundle', keys: ['ca-bundle.crt'] }], + }); + modelRegistrySettings.visit(true); + modelRegistrySettings + .findModelRegistryRow('test-registry-1') + .findKebabAction('Edit model registry') + .click(); + modelRegistrySettings.findAddSecureDbMRCheckbox().should('exist'); + modelRegistrySettings.findAddSecureDbMRCheckbox().should('not.be.checked'); + modelRegistrySettings.findAddSecureDbMRCheckbox().check(); + + modelRegistrySettings.findSubmitButton().should('be.enabled'); + modelRegistrySettings.findSubmitButton().click(); + + cy.wait('@updateModelRegistry').then((interception) => { + expect(interception.request.body).to.containSubset({ + modelRegistry: { + metadata: { + annotations: { + 'openshift.io/description': '', + 'openshift.io/display-name': 'test-registry-1', + }, + }, + spec: { + mysql: { + host: 'model-registry-db', + port: 5432, + database: 'model-registry', + username: 'mlmduser', + sslRootCertificateConfigMap: { name: 'odh-trusted-ca-bundle', key: 'ca-bundle.crt' }, + sslRootCertificateSecret: null, + }, + }, + }, + databasePassword: 'test-password', + }); + }); + }); + + it('Edit model registry with Use cluster-wide CA bundle option selected to add CA certificate with Open Data Hub CA bundle option', () => { + setupMocksForMRSettingAccess({ + disableModelRegistrySecureDB: false, + configMaps: [{ name: 'odh-trusted-ca-bundle', keys: ['ca-bundle.crt', 'odh-ca-bundle.crt'] }], + }); + modelRegistrySettings.visit(true); + modelRegistrySettings + .findModelRegistryRow('test-registry-4') + .findKebabAction('Edit model registry') + .click(); + modelRegistrySettings.findAddSecureDbMRCheckbox().should('be.checked'); + modelRegistrySettings.findClusterWideCARadio().should('be.checked'); + modelRegistrySettings.findSubmitButton().should('be.enabled'); + + modelRegistrySettings.findOpenshiftCARadio().should('be.enabled'); + modelRegistrySettings.findOpenshiftCARadio().check(); + + modelRegistrySettings.findSubmitButton().should('be.enabled'); + modelRegistrySettings.findSubmitButton().click(); + + cy.wait('@updateTestRegistry-4').then((interception) => { + expect(interception.request.body).to.containSubset({ + modelRegistry: { + metadata: { + annotations: { + 'openshift.io/description': '', + 'openshift.io/display-name': 'test-registry-4', + }, + }, + spec: { + mysql: { + host: 'model-registry-db', + port: 5432, + database: 'model-registry', + username: 'mlmduser', + sslRootCertificateConfigMap: { + name: 'odh-trusted-ca-bundle', + key: 'odh-ca-bundle.crt', + }, + sslRootCertificateSecret: null, + }, + }, + }, + databasePassword: 'test-password', + }); + }); + }); + + it('Edit model registry with Open Data Hub CA bundle option selected to add CA certificate with Choose from existing certificates option - secret', () => { + setupMocksForMRSettingAccess({ + disableModelRegistrySecureDB: false, + configMaps: [{ name: 'odh-trusted-ca-bundle', keys: ['odh-ca-bundle.crt'] }], + secrets: [{ name: 'sampleSecret', keys: ['bar.crt'] }], + }); + modelRegistrySettings.visit(true); + modelRegistrySettings + .findModelRegistryRow('test-registry-5') + .findKebabAction('Edit model registry') + .click(); + modelRegistrySettings.findAddSecureDbMRCheckbox().should('be.checked'); + modelRegistrySettings.findOpenshiftCARadio().should('be.checked'); + modelRegistrySettings.findSubmitButton().should('be.enabled'); + + modelRegistrySettings.findExistingCARadio().should('be.enabled'); + modelRegistrySettings.findExistingCARadio().click(); + + modelRegistrySettings.findExistingCAKeyInputToggle().should('be.disabled'); + modelRegistrySettings.findExistingCAResourceInputToggle().should('be.enabled'); + modelRegistrySettings.resourceNameSelect.openAndSelectItem('sampleSecret', true); + modelRegistrySettings.findExistingCAKeyInputToggle().should('be.enabled'); + modelRegistrySettings.keySelect.openAndSelectItem('bar.crt', true); + + modelRegistrySettings.findSubmitButton().should('be.enabled'); + modelRegistrySettings.findSubmitButton().click(); + + cy.wait('@updateTestRegistry-5').then((interception) => { + expect(interception.request.body).to.containSubset({ + modelRegistry: { + metadata: { + annotations: { + 'openshift.io/description': '', + 'openshift.io/display-name': 'test-registry-5', + }, + }, + spec: { + mysql: { + host: 'model-registry-db', + port: 5432, + database: 'model-registry', + username: 'mlmduser', + sslRootCertificateSecret: { name: 'sampleSecret', key: 'bar.crt' }, + sslRootCertificateConfigMap: null, + }, + }, + }, + databasePassword: 'test-password', + }); + }); + }); + + it('Edit model registry with Secrets selected to add CA certificate with Choose from existing certificates option - secret', () => { + setupMocksForMRSettingAccess({ + disableModelRegistrySecureDB: false, + secrets: [ + { name: 'sampleSecret', keys: ['foo.crt'] }, + { name: 'new-secret', keys: ['ca.crt'] }, + ], + }); + modelRegistrySettings.visit(true); + modelRegistrySettings + .findModelRegistryRow('test-registry-7') + .findKebabAction('Edit model registry') + .click(); + modelRegistrySettings.findAddSecureDbMRCheckbox().should('be.checked'); + + modelRegistrySettings.findSubmitButton().should('be.enabled'); + + modelRegistrySettings.resourceNameSelect.openAndSelectItem('new-secret', true); + modelRegistrySettings.keySelect.openAndSelectItem('ca.crt', true); + + modelRegistrySettings.findSubmitButton().should('be.enabled'); + modelRegistrySettings.findSubmitButton().click(); + + cy.wait('@updateTestRegistry-7').then((interception) => { + expect(interception.request.body).to.containSubset({ + modelRegistry: { + metadata: { + annotations: { + 'openshift.io/description': '', + 'openshift.io/display-name': 'test-registry-7', + }, + }, + spec: { + mysql: { + host: 'model-registry-db', + port: 5432, + database: 'model-registry', + username: 'mlmduser', + sslRootCertificateConfigMap: null, + sslRootCertificateSecret: { name: 'new-secret', key: 'ca.crt' }, + }, + }, + }, + databasePassword: 'test-password', + }); + }); + }); + + it('Edit model registry with Upload new certificate option selected to add CA certificate with Choose from existing certificates option - config map', () => { + setupMocksForMRSettingAccess({ + disableModelRegistrySecureDB: false, + configMaps: [ + { name: 'foo-bar', keys: ['bar.crt'] }, + { name: 'new-certificate-db-credentilas', keys: ['ca.crt'] }, + ], + }); + modelRegistrySettings.visit(true); + modelRegistrySettings + .findModelRegistryRow('test-registry-8') + .findKebabAction('Edit model registry') + .click(); + modelRegistrySettings.findAddSecureDbMRCheckbox().should('be.checked'); + + // checking the MR form should naturally change to show that they are using "Choose from existing certificates" with the right configmap/key selected + modelRegistrySettings.findExistingCARadio().should('be.checked'); + modelRegistrySettings.findSubmitButton().should('be.enabled'); + + modelRegistrySettings.resourceNameSelect.openAndSelectItem('foo-bar', true); + modelRegistrySettings.keySelect.openAndSelectItem('bar.crt', true); + + modelRegistrySettings.findSubmitButton().should('be.enabled'); + modelRegistrySettings.findSubmitButton().click(); + + cy.wait('@updateTestRegistry-8').then((interception) => { + expect(interception.request.body).to.containSubset({ + modelRegistry: { + metadata: { + annotations: { + 'openshift.io/description': '', + 'openshift.io/display-name': 'test-registry-8', + }, + }, + spec: { + mysql: { + host: 'model-registry-db', + port: 5432, + database: 'model-registry', + username: 'mlmduser', + sslRootCertificateConfigMap: { name: 'foo-bar', key: 'bar.crt' }, + sslRootCertificateSecret: null, + }, + }, + }, + databasePassword: 'test-password', + }); + }); + }); + + it('Edit model registry with Upload new certificate option selected to add CA certificate with Choose from existing certificates option - secret', () => { + setupMocksForMRSettingAccess({ + disableModelRegistrySecureDB: false, + }); + modelRegistrySettings.visit(true); + modelRegistrySettings + .findModelRegistryRow('test-registry-8') + .findKebabAction('Edit model registry') + .click(); + modelRegistrySettings.findAddSecureDbMRCheckbox().should('be.checked'); + + // checking the MR form should naturally change to show that they are using "Choose from existing certificates" with the right configmap/key selected + modelRegistrySettings.findExistingCARadio().should('be.checked'); + modelRegistrySettings.findSubmitButton().should('be.enabled'); + + modelRegistrySettings.resourceNameSelect.openAndSelectItem('sampleSecret', true); + modelRegistrySettings.keySelect.openAndSelectItem('foo.crt', true); + + modelRegistrySettings.findSubmitButton().should('be.enabled'); + modelRegistrySettings.findSubmitButton().click(); + + cy.wait('@updateTestRegistry-8').then((interception) => { + expect(interception.request.body).to.containSubset({ + modelRegistry: { + metadata: { + annotations: { + 'openshift.io/description': '', + 'openshift.io/display-name': 'test-registry-8', + }, + }, + spec: { + mysql: { + host: 'model-registry-db', + port: 5432, + database: 'model-registry', + username: 'mlmduser', + sslRootCertificateSecret: { name: 'sampleSecret', key: 'foo.crt' }, + sslRootCertificateConfigMap: null, + }, + }, + }, + databasePassword: 'test-password', + }); + }); + }); }); describe('ManagePermissions', () => { diff --git a/frontend/src/__tests__/cypress/cypress/tests/mocked/modelRegistrySettings/unSupportedFile.txt b/frontend/src/__tests__/cypress/cypress/tests/mocked/modelRegistrySettings/unSupportedFile.txt new file mode 100644 index 0000000000..8773f39855 --- /dev/null +++ b/frontend/src/__tests__/cypress/cypress/tests/mocked/modelRegistrySettings/unSupportedFile.txt @@ -0,0 +1 @@ +file content \ No newline at end of file diff --git a/frontend/src/pages/modelRegistrySettings/CreateMRSecureDBSection.tsx b/frontend/src/pages/modelRegistrySettings/CreateMRSecureDBSection.tsx index da3068115c..6869ce2824 100644 --- a/frontend/src/pages/modelRegistrySettings/CreateMRSecureDBSection.tsx +++ b/frontend/src/pages/modelRegistrySettings/CreateMRSecureDBSection.tsx @@ -12,6 +12,7 @@ import { ResourceType, SecureDBRType, } from './const'; +import { isClusterWideCABundleEnabled, isOpenshiftCAbundleEnabled } from './utils'; export interface SecureDBInfo { type: SecureDBRType; @@ -71,18 +72,8 @@ export const CreateMRSecureDBSection: React.FC = ( return false; }; - const clusterWideCABundle = existingCertConfigMaps.find( - (configMap) => configMap.name === ODH_TRUSTED_BUNDLE && configMap.keys.includes(CA_BUNDLE_CRT), - ); - - const isClusterWideCABundleAvailable = !!clusterWideCABundle; - - const openshiftCAbundle = existingCertConfigMaps.find( - (configMap) => - configMap.name === ODH_TRUSTED_BUNDLE && configMap.keys.includes(ODH_CA_BUNDLE_CRT), - ); - - const isProductCABundleAvailable = !!openshiftCAbundle; + const isClusterWideCABundleAvailable = isClusterWideCABundleEnabled(existingCertConfigMaps); + const isProductCABundleAvailable = isOpenshiftCAbundleEnabled(existingCertConfigMaps); const getKeysByName = (configMapsSecrets: ConfigSecretItem[], targetName: string): string[] => { const configMapSecret = configMapsSecrets.find( @@ -306,6 +297,7 @@ export const CreateMRSecureDBSection: React.FC = ( handleSecureDBTypeChange(SecureDBRType.NEW)} label="Upload new certificate" id="new-ca" @@ -316,6 +308,7 @@ export const CreateMRSecureDBSection: React.FC = ( isInline title="Note" variant="info" + data-testid="certificate-note" style={{ marginLeft: 'var(--pf-t--global--spacer--lg)' }} > Uploading a certificate below creates the {newConfigMapName} ConfigMap @@ -324,6 +317,7 @@ export const CreateMRSecureDBSection: React.FC = ( diff --git a/frontend/src/pages/modelRegistrySettings/CreateModal.tsx b/frontend/src/pages/modelRegistrySettings/CreateModal.tsx index cf4c1aab50..2e1769accc 100644 --- a/frontend/src/pages/modelRegistrySettings/CreateModal.tsx +++ b/frontend/src/pages/modelRegistrySettings/CreateModal.tsx @@ -31,6 +31,8 @@ import { findConfigMap, findSecureDBType, constructRequestBody, + isClusterWideCABundleEnabled, + isOpenshiftCAbundleEnabled, } from '~/pages/modelRegistrySettings/utils'; import { RecursivePartial } from '~/typeHelpers'; import { CreateMRSecureDBSection, SecureDBInfo } from './CreateMRSecureDBSection'; @@ -55,14 +57,6 @@ const CreateModal: React.FC = ({ onClose, refresh, modelRegist const [password, setPassword] = React.useState(''); const [database, setDatabase] = React.useState(''); const [addSecureDB, setAddSecureDB] = React.useState(false); - const [secureDBInfo, setSecureDBInfo] = React.useState({ - type: SecureDBRType.CLUSTER_WIDE, - nameSpace: '', - resourceName: '', - certificate: '', - key: '', - isValid: true, - }); const [isHostTouched, setIsHostTouched] = React.useState(false); const [isPortTouched, setIsPortTouched] = React.useState(false); const [isUsernameTouched, setIsUsernameTouched] = React.useState(false); @@ -74,9 +68,33 @@ const CreateModal: React.FC = ({ onClose, refresh, modelRegist const [configSecrets, configSecretsLoaded, configSecretsError] = useModelRegistryCertificateNames( !addSecureDB, ); - + const [secureDBInfo, setSecureDBInfo] = React.useState({ + type: SecureDBRType.CLUSTER_WIDE, + nameSpace: '', + resourceName: '', + certificate: '', + key: '', + isValid: true, + }); const modelRegistryNamespace = dscStatus?.components?.modelregistry?.registriesNamespace || ''; + React.useEffect(() => { + if (configSecretsLoaded && !configSecretsError && !mr) { + setSecureDBInfo((prev) => ({ + ...prev, + type: isClusterWideCABundleEnabled(configSecrets.configMaps) + ? SecureDBRType.CLUSTER_WIDE + : isOpenshiftCAbundleEnabled(configSecrets.configMaps) + ? SecureDBRType.OPENSHIFT + : SecureDBRType.EXISTING, + isValid: !!( + isClusterWideCABundleEnabled(configSecrets.configMaps) || + isOpenshiftCAbundleEnabled(configSecrets.configMaps) + ), + })); + } + }, [configSecretsLoaded, configSecrets.configMaps, mr, configSecretsError]); + React.useEffect(() => { if (mr) { const dbSpec = mr.spec.mysql || mr.spec.postgres; @@ -227,7 +245,7 @@ const CreateModal: React.FC = ({ onClose, refresh, modelRegist hasContent(port) && hasContent(username) && hasContent(database) && - (!addSecureDB || secureDBInfo.isValid); + (!addSecureDB || (secureDBInfo.isValid && !configSecretsError)); return ( = ({ onClose, refresh, modelRegist setSecureDBInfo={setSecureDBInfo} /> ) : ( - + {configSecretsError?.message} ))} diff --git a/frontend/src/pages/modelRegistrySettings/PemFileUpload.tsx b/frontend/src/pages/modelRegistrySettings/PemFileUpload.tsx index 85a1c2df5f..5684a6daee 100644 --- a/frontend/src/pages/modelRegistrySettings/PemFileUpload.tsx +++ b/frontend/src/pages/modelRegistrySettings/PemFileUpload.tsx @@ -49,7 +49,9 @@ export const PemFileUpload: React.FC<{ onChange: (value: string) => void }> = ({ <> void }> = ({ {isRejected ? 'Must be a PEM file' : 'Upload a PEM file'} diff --git a/frontend/src/pages/modelRegistrySettings/utils.ts b/frontend/src/pages/modelRegistrySettings/utils.ts index e268a1e686..a06b2517a9 100644 --- a/frontend/src/pages/modelRegistrySettings/utils.ts +++ b/frontend/src/pages/modelRegistrySettings/utils.ts @@ -1,5 +1,5 @@ import { RecursivePartial } from '~/typeHelpers'; -import { ModelRegistryKind } from '~/k8sTypes'; +import { ConfigSecretItem, ModelRegistryKind } from '~/k8sTypes'; import { CA_BUNDLE_CRT, ODH_CA_BUNDLE_CRT, @@ -51,3 +51,20 @@ export const constructRequestBody = ( return mr; }; + +export const isClusterWideCABundleEnabled = ( + existingCertConfigMaps: ConfigSecretItem[], +): boolean => { + const clusterWideCABundle = existingCertConfigMaps.find( + (configMap) => configMap.name === ODH_TRUSTED_BUNDLE && configMap.keys.includes(CA_BUNDLE_CRT), + ); + return !!clusterWideCABundle; +}; + +export const isOpenshiftCAbundleEnabled = (existingCertConfigMaps: ConfigSecretItem[]): boolean => { + const openshiftCAbundle = existingCertConfigMaps.find( + (configMap) => + configMap.name === ODH_TRUSTED_BUNDLE && configMap.keys.includes(ODH_CA_BUNDLE_CRT), + ); + return !!openshiftCAbundle; +}; From f03527b178e5a86fdf4917e460a4e9f7965ea51b Mon Sep 17 00:00:00 2001 From: Sean Pryor Date: Fri, 10 Jan 2025 10:59:10 -0500 Subject: [PATCH 2/6] Add in proper handling of spaces and quotes in ISVC parsing (#3592) This adds a rudimentary parser that should handle most cases for inferenceservice argument parsing. Note, I'm not a JS expert, so it's possible there's some degerenate edge case with this, but in testing it worked fine --- frontend/src/api/k8s/inferenceServices.ts | 8 +++++--- frontend/src/api/k8s/utils.ts | 21 +++++++++++++++++++++ 2 files changed, 26 insertions(+), 3 deletions(-) diff --git a/frontend/src/api/k8s/inferenceServices.ts b/frontend/src/api/k8s/inferenceServices.ts index ff91464399..a1f1d5340c 100644 --- a/frontend/src/api/k8s/inferenceServices.ts +++ b/frontend/src/api/k8s/inferenceServices.ts @@ -15,7 +15,7 @@ import { ContainerResources } from '~/types'; import { AcceleratorProfileFormData } from '~/utilities/useAcceleratorProfileFormState'; import { AcceleratorProfileState } from '~/utilities/useReadAcceleratorState'; import { getModelServingProjects } from './projects'; -import { assemblePodSpecOptions } from './utils'; +import { assemblePodSpecOptions, parseCommandLine } from './utils'; const applyAuthToInferenceService = ( inferenceService: InferenceServiceKind, @@ -100,6 +100,8 @@ export const assembleInferenceService = ( const dataConnectionKey = secretKey || dataConnection; const nonEmptyArgs = servingRuntimeArgs?.filter(Boolean) || []; + // Ensure that we properly handle separating args + const splitArgs: string[] = nonEmptyArgs.flatMap(parseCommandLine); const nonEmptyEnvVars = servingRuntimeEnvVars?.filter((ev) => ev.name) || []; let updateInferenceService: InferenceServiceKind = inferenceService @@ -143,7 +145,7 @@ export const assembleInferenceService = ( path, }, }), - args: nonEmptyArgs, + args: splitArgs, env: nonEmptyEnvVars, }, }, @@ -192,7 +194,7 @@ export const assembleInferenceService = ( path, }, }), - args: nonEmptyArgs, + args: splitArgs, env: nonEmptyEnvVars, }, }, diff --git a/frontend/src/api/k8s/utils.ts b/frontend/src/api/k8s/utils.ts index 2399f3eca8..caec06914d 100644 --- a/frontend/src/api/k8s/utils.ts +++ b/frontend/src/api/k8s/utils.ts @@ -74,3 +74,24 @@ export const getshmVolume = (sizeLimit?: string): Volume => ({ name: 'shm', emptyDir: { medium: 'Memory', ...(sizeLimit && { sizeLimit }) }, }); + +export const parseCommandLine = (input: string): string[] => { + const args: string[] = []; + const regex = /(?:[^\s"']+|"[^"]*"|'[^']*')+/g; + let match: RegExpExecArray | null; + + while ((match = regex.exec(input)) !== null) { + let arg: string = match[0]; + + // Remove surrounding quotes if any + if (arg.startsWith('"') && arg.endsWith('"')) { + arg = arg.slice(1, -1).replace(/\\"/g, '"'); // Unescape double quotes + } else if (arg.startsWith("'") && arg.endsWith("'")) { + arg = arg.slice(1, -1); // Remove single quotes + } + + args.push(arg); + } + + return args; +}; From 4c87176b2b0c3c987dd87f097a54684a729a04e7 Mon Sep 17 00:00:00 2001 From: Manaswini Das Date: Fri, 10 Jan 2025 23:19:08 +0530 Subject: [PATCH 3/6] Unit tests for parseCommandLine util (#3633) --- frontend/src/api/k8s/__tests__/utils.spec.ts | 60 +++++++++++++++++++- 1 file changed, 59 insertions(+), 1 deletion(-) diff --git a/frontend/src/api/k8s/__tests__/utils.spec.ts b/frontend/src/api/k8s/__tests__/utils.spec.ts index fbc4d46a62..ac4631ea45 100644 --- a/frontend/src/api/k8s/__tests__/utils.spec.ts +++ b/frontend/src/api/k8s/__tests__/utils.spec.ts @@ -1,6 +1,11 @@ import { mockAcceleratorProfile } from '~/__mocks__/mockAcceleratorProfile'; import { AcceleratorProfileState } from '~/utilities/useReadAcceleratorState'; -import { assemblePodSpecOptions, getshmVolume, getshmVolumeMount } from '~/api/k8s/utils'; +import { + assemblePodSpecOptions, + getshmVolume, + getshmVolumeMount, + parseCommandLine, +} from '~/api/k8s/utils'; import { ContainerResources, TolerationEffect, TolerationOperator } from '~/types'; import { AcceleratorProfileFormData } from '~/utilities/useAcceleratorProfileFormState'; @@ -328,3 +333,56 @@ describe('getshmVolume', () => { }); }); }); + +describe('parseCommandLine', () => { + test('parses simple space-separated arguments', () => { + expect(parseCommandLine('arg1 arg2 arg3')).toEqual(['arg1', 'arg2', 'arg3']); + }); + + test('handles double-quoted arguments', () => { + expect(parseCommandLine('"arg1 with spaces" arg2')).toEqual(['arg1 with spaces', 'arg2']); + }); + + test('handles single-quoted arguments', () => { + expect(parseCommandLine("'arg1 with spaces' arg2")).toEqual(['arg1 with spaces', 'arg2']); + }); + + test('handles mixed quotes', () => { + expect(parseCommandLine('\'single quoted\' "double quoted"')).toEqual([ + 'single quoted', + 'double quoted', + ]); + }); + + test('handles nested quotes', () => { + expect(parseCommandLine("'nested\"'")).toEqual(['nested"']); + }); + + test('handles empty input', () => { + expect(parseCommandLine('')).toEqual([]); + }); + + test('handles arguments with special characters', () => { + expect(parseCommandLine('arg1 arg2!@#$%^&*()')).toEqual(['arg1', 'arg2!@#$%^&*()']); + }); + + test('handles arguments with escaped spaces', () => { + expect(parseCommandLine('arg1 "arg with spaces" arg3')).toEqual([ + 'arg1', + 'arg with spaces', + 'arg3', + ]); + }); + + test('removes surrounding quotes from single-quoted arguments', () => { + expect(parseCommandLine("'arg with spaces'")).toEqual(['arg with spaces']); + }); + + test('removes surrounding quotes from double-quoted arguments', () => { + expect(parseCommandLine('"arg with spaces"')).toEqual(['arg with spaces']); + }); + + test('handles multiple consecutive spaces', () => { + expect(parseCommandLine('arg1 arg2 arg3')).toEqual(['arg1', 'arg2', 'arg3']); + }); +}); From bbb0bc9ce3d7e84efd7c55ddfba2a3608d6b045d Mon Sep 17 00:00:00 2001 From: Mike Turley Date: Fri, 10 Jan 2025 14:33:38 -0500 Subject: [PATCH 4/6] parseCommandLine: Add unit test for multi-line input, remove incorrect help text (#3634) Signed-off-by: Mike Turley --- frontend/src/api/k8s/__tests__/utils.spec.ts | 4 ++++ .../projects/kServeModal/ServingRuntimeArgsSection.tsx | 4 ++-- 2 files changed, 6 insertions(+), 2 deletions(-) diff --git a/frontend/src/api/k8s/__tests__/utils.spec.ts b/frontend/src/api/k8s/__tests__/utils.spec.ts index ac4631ea45..a59b0f5542 100644 --- a/frontend/src/api/k8s/__tests__/utils.spec.ts +++ b/frontend/src/api/k8s/__tests__/utils.spec.ts @@ -385,4 +385,8 @@ describe('parseCommandLine', () => { test('handles multiple consecutive spaces', () => { expect(parseCommandLine('arg1 arg2 arg3')).toEqual(['arg1', 'arg2', 'arg3']); }); + + test('handles multi-line input', () => { + expect(parseCommandLine('arg1 arg2\narg3 arg4')).toEqual(['arg1', 'arg2', 'arg3', 'arg4']); + }); }); diff --git a/frontend/src/pages/modelServing/screens/projects/kServeModal/ServingRuntimeArgsSection.tsx b/frontend/src/pages/modelServing/screens/projects/kServeModal/ServingRuntimeArgsSection.tsx index 76e070e9da..edd7a36ce2 100644 --- a/frontend/src/pages/modelServing/screens/projects/kServeModal/ServingRuntimeArgsSection.tsx +++ b/frontend/src/pages/modelServing/screens/projects/kServeModal/ServingRuntimeArgsSection.tsx @@ -107,8 +107,8 @@ const ServingRuntimeArgsSection: React.FC = ({ - {`Enter one argument and its values per line. Overwriting the runtime's predefined - listening port or model location will likely result in a failed deployment.`} + {`Overwriting the runtime's predefined listening port or + model location will likely result in a failed deployment.`} From ada286912e6554766ad21becc97cf678e9429d9d Mon Sep 17 00:00:00 2001 From: Juntao Wang <37624318+DaoDaoNoCode@users.noreply.github.com> Date: Fri, 10 Jan 2025 15:14:51 -0500 Subject: [PATCH 5/6] Add create/edit/duplicate hardware profile page (#3604) * Add create/edit hardware profile page * address UX feedbacks * Add tests for node resources table * add tests for size fields and update the validation * add new type in crd * change node resources table based per requested * change CRD --- frontend/src/__mocks__/mockHardwareProfile.ts | 14 +- .../cypress/cypress/pages/hardwareProfile.ts | 307 ++++++++++ .../manageHardwareProfiles.cy.ts | 573 ++++++++++++++++++ .../k8s/__tests__/hardwareProfiles.spec.ts | 20 +- frontend/src/api/k8s/hardwareProfiles.ts | 12 +- .../hardwareProfiles/HardwareProfiles.tsx | 6 +- .../HardwareProfilesRoutes.tsx | 8 + .../HardwareProfilesTableRow.tsx | 24 +- .../HardwareProfilesToolbar.tsx | 89 +-- .../ManageNodeResourceSection.tsx | 69 --- frontend/src/pages/hardwareProfiles/const.ts | 117 ++-- .../manage/ManageHardwareProfile.tsx | 142 +++++ .../manage/ManageHardwareProfileFooter.tsx | 103 ++++ .../manage/ManageHardwareProfileWrapper.tsx | 75 +++ .../manage/ManageNodeResourceSection.tsx | 93 +++ .../manage/ManageNodeSelectorSection.tsx | 73 +++ .../manage/ManageTolerationSection.tsx | 73 +++ .../pages/hardwareProfiles/manage/types.ts | 14 + .../nodeResource/CountFormField.tsx | 15 +- .../nodeResource/ManageNodeResourceModal.tsx | 14 +- .../nodeResource/NodeResourceForm.tsx | 64 +- .../nodeResource/NodeResourceTable.tsx | 4 +- .../nodeResource/NodeResourceTableRow.tsx | 4 +- .../hardwareProfiles/nodeResource/const.ts | 17 + .../hardwareProfiles/nodeResource/utils.ts | 15 +- .../nodeSelector/ManageNodeSelectorModal.tsx | 66 ++ .../nodeSelector/NodeSelectorTable.tsx | 67 ++ .../nodeSelector/NodeSelectorTableRow.tsx | 48 ++ .../nodeSelector/NodeSelectorsTable.tsx | 27 - .../hardwareProfiles/nodeSelector/const.ts | 25 + .../toleration/ManageTolerationModal.tsx | 193 ++++++ .../toleration/TolerationTable.tsx | 67 ++ .../toleration/TolerationTableRow.tsx | 55 ++ .../toleration/TolerationsTable.tsx | 32 - .../hardwareProfiles/toleration/const.ts | 79 +++ frontend/src/types.ts | 6 + .../hardwareprofiles.opendatahub.io.crd.yaml | 10 +- 37 files changed, 2311 insertions(+), 309 deletions(-) create mode 100644 frontend/src/__tests__/cypress/cypress/tests/mocked/hardwareProfiles/manageHardwareProfiles.cy.ts delete mode 100644 frontend/src/pages/hardwareProfiles/ManageNodeResourceSection.tsx create mode 100644 frontend/src/pages/hardwareProfiles/manage/ManageHardwareProfile.tsx create mode 100644 frontend/src/pages/hardwareProfiles/manage/ManageHardwareProfileFooter.tsx create mode 100644 frontend/src/pages/hardwareProfiles/manage/ManageHardwareProfileWrapper.tsx create mode 100644 frontend/src/pages/hardwareProfiles/manage/ManageNodeResourceSection.tsx create mode 100644 frontend/src/pages/hardwareProfiles/manage/ManageNodeSelectorSection.tsx create mode 100644 frontend/src/pages/hardwareProfiles/manage/ManageTolerationSection.tsx create mode 100644 frontend/src/pages/hardwareProfiles/manage/types.ts create mode 100644 frontend/src/pages/hardwareProfiles/nodeSelector/ManageNodeSelectorModal.tsx create mode 100644 frontend/src/pages/hardwareProfiles/nodeSelector/NodeSelectorTable.tsx create mode 100644 frontend/src/pages/hardwareProfiles/nodeSelector/NodeSelectorTableRow.tsx delete mode 100644 frontend/src/pages/hardwareProfiles/nodeSelector/NodeSelectorsTable.tsx create mode 100644 frontend/src/pages/hardwareProfiles/nodeSelector/const.ts create mode 100644 frontend/src/pages/hardwareProfiles/toleration/ManageTolerationModal.tsx create mode 100644 frontend/src/pages/hardwareProfiles/toleration/TolerationTable.tsx create mode 100644 frontend/src/pages/hardwareProfiles/toleration/TolerationTableRow.tsx delete mode 100644 frontend/src/pages/hardwareProfiles/toleration/TolerationsTable.tsx create mode 100644 frontend/src/pages/hardwareProfiles/toleration/const.ts diff --git a/frontend/src/__mocks__/mockHardwareProfile.ts b/frontend/src/__mocks__/mockHardwareProfile.ts index 01039a1ed7..1fff33fa4b 100644 --- a/frontend/src/__mocks__/mockHardwareProfile.ts +++ b/frontend/src/__mocks__/mockHardwareProfile.ts @@ -18,6 +18,7 @@ type MockResourceConfigType = { enabled?: boolean; nodeSelectors?: NodeSelector[]; tolerations?: Toleration[]; + annotations?: Record; }; export const mockHardwareProfile = ({ @@ -29,10 +30,17 @@ export const mockHardwareProfile = ({ { displayName: 'Memory', identifier: 'memory', - minCount: '5Gi', - maxCount: '2Gi', + minCount: '2Gi', + maxCount: '5Gi', defaultCount: '2Gi', }, + { + displayName: 'CPU', + identifier: 'cpu', + minCount: '1', + maxCount: '2', + defaultCount: '1', + }, ], description = '', enabled = true, @@ -49,6 +57,7 @@ export const mockHardwareProfile = ({ value: 'va;ue', }, ], + annotations, }: MockResourceConfigType): HardwareProfileKind => ({ apiVersion: 'dashboard.opendatahub.io/v1alpha1', kind: 'HardwareProfile', @@ -59,6 +68,7 @@ export const mockHardwareProfile = ({ namespace, resourceVersion: '1309350', uid, + annotations, }, spec: { identifiers, diff --git a/frontend/src/__tests__/cypress/cypress/pages/hardwareProfile.ts b/frontend/src/__tests__/cypress/cypress/pages/hardwareProfile.ts index 03d07b8aa5..dea94bfa4b 100644 --- a/frontend/src/__tests__/cypress/cypress/pages/hardwareProfile.ts +++ b/frontend/src/__tests__/cypress/cypress/pages/hardwareProfile.ts @@ -1,4 +1,6 @@ import { Contextual } from '~/__tests__/cypress/cypress/pages/components/Contextual'; +import { K8sNameDescriptionField } from '~/__tests__/cypress/cypress/pages/components/subComponents/K8sNameDescriptionField'; +import { Modal } from '~/__tests__/cypress/cypress/pages/components/Modal'; import { TableRow } from './components/table'; class HardwareProfileTableToolbar extends Contextual { @@ -104,4 +106,309 @@ class HardwareProfile { } } +class NodeResourceRow extends TableRow { + shouldHaveResourceLabel(name: string) { + this.find().find(`[data-label="Resource label"]`).should('have.text', name); + return this; + } + + shouldHaveResourceIdentifier(name: string) { + this.find().find(`[data-label="Resource identifier"]`).should('have.text', name); + return this; + } + + findEditAction() { + return this.find().findByRole('button', { name: 'Edit node resource' }); + } + + findDeleteAction() { + return this.find().findByRole('button', { name: 'Remove node resource' }); + } +} + +class NodeSelectorRow extends TableRow { + shouldHaveKey(name: string) { + this.find().find(`[data-label=Key]`).should('have.text', name); + return this; + } + + shouldHaveValue(name: string) { + this.find().find(`[data-label=Value]`).should('have.text', name); + return this; + } + + findEditAction() { + return this.find().findByRole('button', { name: 'Edit node selector' }); + } + + findDeleteAction() { + return this.find().findByRole('button', { name: 'Remove node selector' }); + } +} + +class TolerationRow extends TableRow { + shouldHaveOperator(name: string) { + this.find().find(`[data-label=Operator]`).should('have.text', name); + return this; + } + + shouldHaveEffect(name: string) { + this.find().find(`[data-label=Effect]`).should('have.text', name); + return this; + } + + shouldHaveTolerationSeconds(toleration: string) { + this.find().find(`[data-label="Toleration seconds"]`).should('have.text', toleration); + return this; + } + + findEditAction() { + return this.find().findByRole('button', { name: 'Edit toleration' }); + } + + findDeleteAction() { + return this.find().findByRole('button', { name: 'Remove toleration' }); + } +} + +class ManageHardwareProfile { + k8sNameDescription = new K8sNameDescriptionField('hardware-profile-name-desc'); + + findAddTolerationButton() { + return cy.findByTestId('add-toleration-button'); + } + + findAddNodeSelectorButton() { + return cy.findByTestId('add-node-selector-button'); + } + + findAddNodeResourceButton() { + return cy.findByTestId('add-node-resource-button'); + } + + findSubmitButton() { + return cy.findByTestId('hardware-profile-create-button'); + } + + findTolerationTable() { + return cy.findByTestId('hardware-profile-tolerations-table'); + } + + findNodeSelectorTable() { + return cy.findByTestId('hardware-profile-node-selectors-table'); + } + + findNodeResourceTable() { + return cy.findByTestId('hardware-profile-node-resources-table'); + } + + findNodeResourceTableAlert() { + return cy.findByTestId('node-resource-table-alert'); + } + + getTolerationTableRow(name: string) { + return new TolerationRow(() => + this.findTolerationTable().find(`[data-label=Key]`).contains(name).parents('tr'), + ); + } + + getNodeSelectorTableRow(name: string) { + return new NodeSelectorRow(() => + this.findNodeSelectorTable().find(`[data-label=Key]`).contains(name).parents('tr'), + ); + } + + getNodeResourceTableRow(name: string) { + return new NodeResourceRow(() => + this.findNodeResourceTable() + .find(`[data-label="Resource identifier"]`) + .contains(name) + .parents('tr'), + ); + } +} + +class CreateHardwareProfile extends ManageHardwareProfile { + visit() { + cy.visitWithLogin('/hardwareProfiles/Create'); + this.wait(); + } + + private wait() { + this.findSubmitButton().contains('Create hardware profile'); + cy.testA11y(); + } +} + +class EditHardwareProfile extends ManageHardwareProfile { + visit(name: string) { + cy.visitWithLogin(`/hardwareProfiles/edit/${name}`); + cy.testA11y(); + } + + findErrorText() { + return cy.findByTestId('problem-loading-hardware-profile'); + } + + findViewAllHardwareProfilesButton() { + return cy.findByTestId('view-all-hardware-profiles'); + } +} + +class DuplicateHardwareProfile extends ManageHardwareProfile { + visit(name: string) { + cy.visitWithLogin(`/hardwareProfiles/duplicate/${name}`); + cy.testA11y(); + } + + findErrorText() { + return cy.findByTestId('problem-loading-hardware-profile'); + } + + findViewAllHardwareProfilesButton() { + return cy.findByTestId('view-all-hardware-profiles'); + } +} + +class TolerationModal extends Modal { + constructor(edit = false) { + super(edit ? 'Edit toleration' : 'Add toleration'); + } + + findTolerationKeyInput() { + return this.find().findByTestId('toleration-key-input'); + } + + findTolerationValueAlert() { + return this.find().findByTestId('toleration-value-alert'); + } + + findTolerationSecondRadioCustom() { + return this.find().findByTestId('toleration-seconds-radio-custom'); + } + + findTolerationSecondAlert() { + return this.find().findByTestId('toleration-seconds-alert'); + } + + findTolerationValueInput() { + return this.find().findByTestId('toleration-value-input'); + } + + private findTolerationOperatorSelect() { + return this.find().findByTestId('toleration-operator-select'); + } + + private findTolerationEffectSelect() { + return this.find().findByTestId('toleration-effect-select'); + } + + findOperatorOptionExist() { + return this.findTolerationOperatorSelect().findSelectOption( + 'Exists A toleration "matches" a taint if the keys are the same and the effects are the same. No value should be specified.', + ); + } + + findOperatorOptionEqual() { + return this.findTolerationOperatorSelect().findSelectOption( + 'Equal A toleration "matches" a taint if the keys are the same, the effects are the same, and the values are equal.', + ); + } + + findEffectOptionNoExecute() { + return this.findTolerationEffectSelect().findSelectOption( + 'NoExecute Pods will be evicted from the node if they do not tolerate the taint.', + ); + } + + findPlusButton() { + return this.find().findByRole('button', { name: 'Plus' }); + } + + findTolerationSubmitButton() { + return this.find().findByTestId('modal-submit-button'); + } +} + +class NodeSelectorModal extends Modal { + constructor(edit = false) { + super(edit ? 'Edit node selector' : 'Add node selector'); + } + + findNodeSelectorKeyInput() { + return this.find().findByTestId('node-selector-key-input'); + } + + findNodeSelectorValueInput() { + return this.find().findByTestId('node-selector-value-input'); + } + + findNodeSelectorSubmitButton() { + return this.find().findByTestId('modal-submit-button'); + } +} + +class NodeResourceModal extends Modal { + constructor(edit = false) { + super(edit ? 'Edit node resource' : 'Add node resource'); + } + + findNodeResourceLabelInput() { + return this.find().findByTestId('node-resource-label-input'); + } + + findNodeResourceIdentifierInput() { + return this.find().findByTestId('node-resource-identifier-input'); + } + + findNodeResourceTypeSelect() { + return this.find().findByTestId('node-resource-type-select'); + } + + findNodeResourceExistingErrorMessage() { + return this.find().findByTestId('resource-identifier-error'); + } + + findNodeResourceDefaultInput() { + return this.find().findByTestId('node-resource-size-default').findByLabelText('Input'); + } + + selectNodeResourceDefaultUnit(name: string) { + this.find() + .findByTestId('node-resource-size-default') + .findByTestId('value-unit-select') + .findDropdownItem(name) + .click(); + } + + findNodeResourceDefaultErrorMessage() { + return this.find().findByTestId('node-resource-size-default-error'); + } + + findNodeResourceMinInput() { + return this.find().findByTestId('node-resource-size-minimum-allowed').findByLabelText('Input'); + } + + findNodeResourceMinErrorMessage() { + return this.find().findByTestId('node-resource-size-minimum-allowed-error'); + } + + findNodeResourceMaxInput() { + return this.find().findByTestId('node-resource-size-maximum-allowed').findByLabelText('Input'); + } + + findNodeResourceSubmitButton() { + return this.find().findByTestId('modal-submit-button'); + } +} + export const hardwareProfile = new HardwareProfile(); +export const createHardwareProfile = new CreateHardwareProfile(); +export const createTolerationModal = new TolerationModal(false); +export const editTolerationModal = new TolerationModal(true); +export const createNodeSelectorModal = new NodeSelectorModal(false); +export const editNodeSelectorModal = new NodeSelectorModal(true); +export const createNodeResourceModal = new NodeResourceModal(false); +export const editNodeResourceModal = new NodeResourceModal(true); +export const editHardwareProfile = new EditHardwareProfile(); +export const duplicateHardwareProfile = new DuplicateHardwareProfile(); diff --git a/frontend/src/__tests__/cypress/cypress/tests/mocked/hardwareProfiles/manageHardwareProfiles.cy.ts b/frontend/src/__tests__/cypress/cypress/tests/mocked/hardwareProfiles/manageHardwareProfiles.cy.ts new file mode 100644 index 0000000000..eeee29b89e --- /dev/null +++ b/frontend/src/__tests__/cypress/cypress/tests/mocked/hardwareProfiles/manageHardwareProfiles.cy.ts @@ -0,0 +1,573 @@ +import { TolerationEffect, TolerationOperator } from '~/types'; +import { mockK8sResourceList } from '~/__mocks__/mockK8sResourceList'; +import { HardwareProfileModel } from '~/__tests__/cypress/cypress/utils/models'; +import { asProductAdminUser } from '~/__tests__/cypress/cypress/utils/mockUsers'; +import { mockHardwareProfile } from '~/__mocks__/mockHardwareProfile'; +import { + createHardwareProfile, + createNodeResourceModal, + createNodeSelectorModal, + createTolerationModal, + duplicateHardwareProfile, + editHardwareProfile, + editNodeResourceModal, + editNodeSelectorModal, + editTolerationModal, +} from '~/__tests__/cypress/cypress/pages/hardwareProfile'; + +type HandlersProps = { + isPresent?: boolean; +}; + +const initIntercepts = ({ isPresent = true }: HandlersProps) => { + cy.interceptK8s( + { model: HardwareProfileModel, ns: 'opendatahub', name: 'test-hardware-profile' }, + isPresent + ? mockHardwareProfile({ + namespace: 'opendatahub', + name: 'test-hardware-profile', + displayName: 'Test Hardware Profile', + description: 'Test description', + identifiers: [ + { + displayName: 'Memory', + identifier: 'memory', + minCount: '2Gi', + maxCount: '5Gi', + defaultCount: '2Gi', + }, + { + displayName: 'CPU', + identifier: 'cpu', + minCount: '1', + maxCount: '2', + defaultCount: '1', + }, + { + identifier: 'nvidia.com/gpu', + displayName: 'GPU', + maxCount: 2, + minCount: 1, + defaultCount: 1, + }, + ], + tolerations: [ + { + key: 'nvidia.com/gpu', + operator: TolerationOperator.EXISTS, + effect: TolerationEffect.NO_SCHEDULE, + }, + ], + nodeSelectors: [{ key: 'test-key', value: 'test-value' }], + }) + : { + statusCode: 404, + }, + ); +}; + +describe('Manage Hardware Profile', () => { + beforeEach(() => { + asProductAdminUser(); + }); + + it('create hardware profile', () => { + initIntercepts({}); + createHardwareProfile.visit(); + createHardwareProfile.findSubmitButton().should('be.disabled'); + + // test required fields + createHardwareProfile.k8sNameDescription.findDisplayNameInput().fill('Test hardware profile'); + createHardwareProfile.findSubmitButton().should('be.enabled'); + + // test resource name validation + createHardwareProfile.k8sNameDescription.findResourceEditLink().click(); + createHardwareProfile.k8sNameDescription + .findResourceNameInput() + .should('have.attr', 'aria-invalid', 'false'); + createHardwareProfile.k8sNameDescription + .findResourceNameInput() + .should('have.value', 'test-hardware-profile'); + // Invalid character k8s names fail + createHardwareProfile.k8sNameDescription.findResourceNameInput().clear().type('InVaLiD vAlUe!'); + createHardwareProfile.k8sNameDescription + .findResourceNameInput() + .should('have.attr', 'aria-invalid', 'true'); + createHardwareProfile.findSubmitButton().should('be.disabled'); + createHardwareProfile.k8sNameDescription + .findResourceNameInput() + .clear() + .type('test-hardware-profile-name'); + createHardwareProfile.findSubmitButton().should('be.enabled'); + createHardwareProfile.k8sNameDescription.findDescriptionInput().fill('Test description'); + + cy.interceptK8s( + 'POST', + { + model: HardwareProfileModel, + ns: 'opendatahub', + name: 'test-hardware-profile', + }, + mockHardwareProfile({ name: 'test-hardware-profile', namespace: 'opendatahub' }), + ).as('createHardwareProfile'); + createHardwareProfile.findSubmitButton().click(); + + cy.wait('@createHardwareProfile').then((interception) => { + expect(interception.request.body.spec.displayName).to.be.eql('Test hardware profile'); + expect(interception.request.body.spec.description).to.be.eql('Test description'); + }); + }); + + it('test node resources section', () => { + initIntercepts({}); + createHardwareProfile.visit(); + createHardwareProfile.k8sNameDescription.findDisplayNameInput().fill('test-hardware-profile'); + + // test node resource table + createHardwareProfile.findNodeResourceTable().should('exist'); + // open node resource modal + createHardwareProfile.findAddNodeResourceButton().click(); + // fill in form required fields + createNodeResourceModal.findNodeResourceSubmitButton().should('be.disabled'); + createNodeResourceModal.findNodeResourceLabelInput().fill('Test GPU'); + // test duplicated identifier + createNodeResourceModal.findNodeResourceIdentifierInput().fill('cpu'); + createNodeResourceModal.findNodeResourceExistingErrorMessage().should('exist'); + createNodeResourceModal.findNodeResourceSubmitButton().should('be.disabled'); + createNodeResourceModal.findNodeResourceIdentifierInput().fill('test-gpu'); + createNodeResourceModal.findNodeResourceTypeSelect().should('contain.text', 'Other'); + createNodeResourceModal.findNodeResourceSubmitButton().should('be.enabled'); + createNodeResourceModal.findNodeResourceSubmitButton().click(); + // test that values were added correctly + createHardwareProfile.getNodeResourceTableRow('test-gpu').shouldHaveResourceLabel('Test GPU'); + + // test edit node resource + createHardwareProfile.getNodeResourceTableRow('cpu').findEditAction().click(); + editNodeResourceModal.findNodeResourceTypeSelect().should('contain.text', 'CPU'); + // test default value should be within min and max value + editNodeResourceModal.selectNodeResourceDefaultUnit('Milicores'); + editNodeResourceModal.findNodeResourceDefaultErrorMessage().should('exist'); + editNodeResourceModal.selectNodeResourceDefaultUnit('Cores'); + editNodeResourceModal.findNodeResourceDefaultErrorMessage().should('not.exist'); + // test min value should not exceed max value + editNodeResourceModal.findNodeResourceMinInput().type('3'); + editNodeResourceModal.findNodeResourceMinErrorMessage().should('exist'); + editNodeResourceModal.findNodeResourceMinInput().clear(); + editNodeResourceModal.findNodeResourceMinErrorMessage().should('not.exist'); + editNodeResourceModal.findCancelButton().click(); + + createHardwareProfile.getNodeResourceTableRow('memory').findEditAction().click(); + editNodeResourceModal.findNodeResourceTypeSelect().should('contain.text', 'Memory'); + // test default value should be within min and max value + editNodeResourceModal.selectNodeResourceDefaultUnit('MiB'); + editNodeResourceModal.findNodeResourceDefaultErrorMessage().should('exist'); + editNodeResourceModal.selectNodeResourceDefaultUnit('GiB'); + editNodeResourceModal.findNodeResourceDefaultErrorMessage().should('not.exist'); + // test min value should not exceed max value + editNodeResourceModal.findNodeResourceMinInput().type('3'); + editNodeResourceModal.findNodeResourceMinErrorMessage().should('exist'); + editNodeResourceModal.findNodeResourceMinInput().clear(); + editNodeResourceModal.findNodeResourceMinErrorMessage().should('not.exist'); + editNodeResourceModal.findCancelButton().click(); + + createHardwareProfile.getNodeResourceTableRow('test-gpu').findEditAction().click(); + editNodeResourceModal.findNodeResourceLabelInput().fill('Test GPU Edited'); + editNodeResourceModal.findNodeResourceIdentifierInput().fill('test-gpu-edited'); + // test default value should be within min and max value + editNodeResourceModal.findNodeResourceDefaultInput().type('3'); + editNodeResourceModal.findNodeResourceDefaultErrorMessage().should('exist'); + editNodeResourceModal.findNodeResourceSubmitButton().should('be.disabled'); + editNodeResourceModal.findNodeResourceDefaultInput().type('{backspace}'); + editNodeResourceModal.findNodeResourceDefaultErrorMessage().should('not.exist'); + editNodeResourceModal.findNodeResourceSubmitButton().should('be.enabled'); + // test min value should not exceed max value + editNodeResourceModal.findNodeResourceMinInput().type('3'); + editNodeResourceModal.findNodeResourceMinErrorMessage().should('exist'); + editNodeResourceModal.findNodeResourceSubmitButton().should('be.disabled'); + editNodeResourceModal.findNodeResourceMinInput().type('{backspace}'); + editNodeResourceModal.findNodeResourceMinErrorMessage().should('not.exist'); + editNodeResourceModal.findNodeResourceSubmitButton().should('be.enabled'); + editNodeResourceModal.findNodeResourceMaxInput().type('3'); + editNodeResourceModal.findNodeResourceSubmitButton().click(); + createHardwareProfile + .getNodeResourceTableRow('test-gpu-edited') + .shouldHaveResourceLabel('Test GPU Edited') + .shouldHaveResourceIdentifier('test-gpu-edited'); + + // test deleting the last CPU trigger the alert shown + createHardwareProfile.getNodeResourceTableRow('cpu').findDeleteAction().click(); + createHardwareProfile.findNodeResourceTableAlert().should('exist'); + createHardwareProfile.findAddNodeResourceButton().click(); + createNodeResourceModal.findNodeResourceLabelInput().fill('CPU'); + createNodeResourceModal.findNodeResourceIdentifierInput().fill('cpu'); + createNodeResourceModal.findNodeResourceTypeSelect().findSelectOption('CPU').click(); + createNodeResourceModal.findNodeResourceSubmitButton().should('be.enabled'); + createNodeResourceModal.findNodeResourceSubmitButton().click(); + + createHardwareProfile.findNodeResourceTableAlert().should('not.exist'); + + // test deleting the last Memory trigger the alert shown + createHardwareProfile.getNodeResourceTableRow('memory').findDeleteAction().click(); + createHardwareProfile.findNodeResourceTableAlert().should('exist'); + + cy.interceptK8s( + 'POST', + { + model: HardwareProfileModel, + ns: 'opendatahub', + name: 'test-hardware-profile', + }, + mockHardwareProfile({ name: 'test-hardware-profile', namespace: 'opendatahub' }), + ).as('createHardwareProfile'); + createHardwareProfile.findSubmitButton().click(); + + cy.wait('@createHardwareProfile').then((interception) => { + expect(interception.request.body.spec.identifiers).to.be.eql([ + { + displayName: 'Test GPU Edited', + identifier: 'test-gpu-edited', + minCount: 1, + maxCount: 13, + defaultCount: 1, + }, + { + identifier: 'cpu', + displayName: 'CPU', + defaultCount: 2, + maxCount: 4, + minCount: 1, + resourceType: 'CPU', + }, + ]); + }); + }); + + it('test node selectors section', () => { + initIntercepts({}); + createHardwareProfile.visit(); + createHardwareProfile.findSubmitButton().should('be.disabled'); + createHardwareProfile.k8sNameDescription.findDisplayNameInput().fill('test-hardware-profile'); + + // test node selectors empty state + createHardwareProfile.findNodeSelectorTable().should('not.exist'); + // open node selector modal + createHardwareProfile.findAddNodeSelectorButton().click(); + // fill in form required fields + createNodeSelectorModal.findNodeSelectorSubmitButton().should('be.disabled'); + createNodeSelectorModal.findNodeSelectorKeyInput().fill('test-key'); + createNodeSelectorModal.findNodeSelectorSubmitButton().should('be.disabled'); + createNodeSelectorModal.findNodeSelectorValueInput().fill('test-value'); + createNodeSelectorModal.findNodeSelectorSubmitButton().should('be.enabled'); + createNodeSelectorModal.findNodeSelectorSubmitButton().click(); + // test that values were added correctly + createHardwareProfile + .getNodeSelectorTableRow('test-key') + .shouldHaveKey('test-key') + .shouldHaveValue('test-value'); + // test edit node selector + let nodeSelectorTableRow = createHardwareProfile.getNodeSelectorTableRow('test-key'); + nodeSelectorTableRow.findEditAction().click(); + editNodeSelectorModal.findNodeSelectorKeyInput().fill('test-update'); + editNodeSelectorModal.findCancelButton().click(); + nodeSelectorTableRow.findEditAction().click(); + editNodeSelectorModal.findNodeSelectorKeyInput().fill('test-update'); + editNodeSelectorModal.findNodeSelectorSubmitButton().click(); + nodeSelectorTableRow = createHardwareProfile.getNodeSelectorTableRow('test-update'); + nodeSelectorTableRow.shouldHaveValue('test-value'); + // test cancel clears fields + nodeSelectorTableRow.findEditAction().click(); + editNodeSelectorModal.findCancelButton().click(); + createHardwareProfile.findAddNodeSelectorButton().click(); + createNodeSelectorModal.findNodeSelectorSubmitButton().should('be.disabled'); + createNodeSelectorModal.findCancelButton().click(); + // add another field + createHardwareProfile.findAddNodeSelectorButton().click(); + createNodeSelectorModal.findNodeSelectorKeyInput().fill('new-test-node-selector'); + createNodeSelectorModal.findNodeSelectorValueInput().fill('new-test-value'); + createNodeSelectorModal.findNodeSelectorSubmitButton().click(); + // delete the previous one + nodeSelectorTableRow.findDeleteAction().click(); + + cy.interceptK8s( + 'POST', + { + model: HardwareProfileModel, + ns: 'opendatahub', + name: 'test-hardware-profile', + }, + mockHardwareProfile({ name: 'test-hardware-profile', namespace: 'opendatahub' }), + ).as('createHardwareProfile'); + createHardwareProfile.findSubmitButton().click(); + + cy.wait('@createHardwareProfile').then((interception) => { + expect(interception.request.body.spec.nodeSelectors).to.be.eql([ + { key: 'new-test-node-selector', value: 'new-test-value' }, + ]); + }); + }); + + it('test tolerations section', () => { + initIntercepts({}); + createHardwareProfile.visit(); + createHardwareProfile.findSubmitButton().should('be.disabled'); + createHardwareProfile.k8sNameDescription.findDisplayNameInput().fill('test-hardware-profile'); + + // test tolerations empty state + createHardwareProfile.findTolerationTable().should('not.exist'); + // open toleration modal + createHardwareProfile.findAddTolerationButton().click(); + // fill in form required fields + createTolerationModal.findTolerationSubmitButton().should('be.disabled'); + createTolerationModal.findTolerationKeyInput().fill('test-key'); + createTolerationModal.findTolerationSubmitButton().should('be.enabled'); + // test value info warning when operator is Exists + createTolerationModal.findOperatorOptionExist().click(); + createTolerationModal.findTolerationValueInput().fill('test-value'); + createTolerationModal.findTolerationValueAlert().should('exist'); + createTolerationModal.findOperatorOptionEqual().click(); + createTolerationModal.findTolerationValueAlert().should('not.exist'); + // test toleration seconds warning when effect is not NoExecute + createTolerationModal.findTolerationSecondRadioCustom().click(); + createTolerationModal.findTolerationSecondAlert().should('exist'); + createTolerationModal.findEffectOptionNoExecute().click(); + createTolerationModal.findTolerationSecondAlert().should('not.exist'); + createTolerationModal.findPlusButton().click(); + createTolerationModal.findTolerationSubmitButton().click(); + // test that values were added correctly + createHardwareProfile + .getTolerationTableRow('test-key') + .shouldHaveOperator('Equal') + .shouldHaveEffect('NoExecute') + .shouldHaveTolerationSeconds('1 second(s)'); + // test bare minimum fields + createHardwareProfile.findAddTolerationButton().click(); + createTolerationModal.findTolerationKeyInput().fill('toleration-key'); + createTolerationModal.findTolerationSubmitButton().click(); + createHardwareProfile + .getTolerationTableRow('toleration-key') + .shouldHaveOperator('Equal') + .shouldHaveEffect('-') + .shouldHaveTolerationSeconds('-'); + // test edit toleration + let tolerationTableRow = createHardwareProfile.getTolerationTableRow('test-key'); + tolerationTableRow.findEditAction().click(); + editTolerationModal.findTolerationKeyInput().fill('test-update'); + editTolerationModal.findCancelButton().click(); + tolerationTableRow = createHardwareProfile.getTolerationTableRow('test-key'); + tolerationTableRow.findEditAction().click(); + editTolerationModal.findTolerationKeyInput().fill('updated-test'); + editTolerationModal.findTolerationSubmitButton().click(); + tolerationTableRow = createHardwareProfile.getTolerationTableRow('updated-test'); + tolerationTableRow.shouldHaveOperator('Equal'); + // test cancel clears fields + tolerationTableRow.findEditAction().click(); + editTolerationModal.findCancelButton().click(); + createHardwareProfile.findAddTolerationButton().click(); + createTolerationModal.findTolerationSubmitButton().should('be.disabled'); + createTolerationModal.findCancelButton().click(); + // test delete + tolerationTableRow.findDeleteAction().click(); + createHardwareProfile.getTolerationTableRow('toleration-key').findDeleteAction().click(); + createHardwareProfile.findTolerationTable().should('not.exist'); + + cy.interceptK8s( + 'POST', + { + model: HardwareProfileModel, + ns: 'opendatahub', + name: 'test-hardware-profile', + }, + mockHardwareProfile({ name: 'test-hardware-profile', namespace: 'opendatahub' }), + ).as('createHardwareProfile'); + createHardwareProfile.findSubmitButton().click(); + + cy.wait('@createHardwareProfile').then((interception) => { + expect(interception.request.body.spec.tolerations).to.be.eql([]); + }); + }); + + it('edit page has expected values', () => { + initIntercepts({}); + //update the description intercept + cy.interceptK8s( + 'PUT', + { + model: HardwareProfileModel, + ns: 'opendatahub', + name: 'test-hardware-profile', + }, + mockHardwareProfile({ + name: 'test-hardware-profile', + namespace: 'opendatahub', + description: 'Updated description', + }), + ).as('updatedHardwareProfile'); + editHardwareProfile.visit('test-hardware-profile'); + editHardwareProfile.k8sNameDescription + .findDisplayNameInput() + .should('have.value', 'Test Hardware Profile'); + editHardwareProfile.k8sNameDescription + .findDescriptionInput() + .should('have.value', 'Test description'); + + editHardwareProfile.findNodeResourceTable().should('exist'); + editHardwareProfile + .getNodeResourceTableRow('nvidia.com/gpu') + .shouldHaveResourceLabel('GPU') + .findDeleteAction() + .click(); + + editHardwareProfile.findTolerationTable().should('exist'); + editHardwareProfile + .getTolerationTableRow('nvidia.com/gpu') + .shouldHaveEffect('NoSchedule') + .shouldHaveOperator('Exists') + .shouldHaveTolerationSeconds('-'); + + editHardwareProfile.findNodeSelectorTable().should('exist'); + editHardwareProfile + .getNodeSelectorTableRow('test-key') + .shouldHaveValue('test-value') + .findDeleteAction() + .click(); + + editHardwareProfile.k8sNameDescription.findDescriptionInput().fill('Updated description'); + editHardwareProfile.findSubmitButton().click(); + cy.wait('@updatedHardwareProfile').then((interception) => { + expect(interception.request.body.spec).to.eql({ + identifiers: [ + { + displayName: 'Memory', + identifier: 'memory', + minCount: '2Gi', + maxCount: '5Gi', + defaultCount: '2Gi', + }, + { + displayName: 'CPU', + identifier: 'cpu', + minCount: '1', + maxCount: '2', + defaultCount: '1', + }, + ], + displayName: 'Test Hardware Profile', + enabled: true, + tolerations: [ + { + key: 'nvidia.com/gpu', + operator: 'Exists', + effect: 'NoSchedule', + }, + ], + nodeSelectors: [], + description: 'Updated description', + }); + }); + }); + + it('duplicate page has expected values', () => { + initIntercepts({}); + cy.interceptK8s( + 'POST', + { + model: HardwareProfileModel, + ns: 'opendatahub', + name: 'duplicate-hardware-profile', + }, + mockHardwareProfile({ + name: 'duplicate-hardware-profile', + namespace: 'opendatahub', + description: 'Updated description', + }), + ).as('createHardwareProfile'); + duplicateHardwareProfile.visit('test-hardware-profile'); + duplicateHardwareProfile.findSubmitButton().should('be.disabled'); + duplicateHardwareProfile.k8sNameDescription.findDisplayNameInput().should('have.value', ''); + duplicateHardwareProfile.k8sNameDescription.findDescriptionInput().should('have.value', ''); + + editHardwareProfile.findNodeResourceTable().should('exist'); + editHardwareProfile.getNodeResourceTableRow('nvidia.com/gpu').shouldHaveResourceLabel('GPU'); + + duplicateHardwareProfile.findTolerationTable().should('exist'); + duplicateHardwareProfile + .getTolerationTableRow('nvidia.com/gpu') + .shouldHaveEffect('NoSchedule') + .shouldHaveOperator('Exists') + .shouldHaveTolerationSeconds('-'); + + duplicateHardwareProfile.findNodeSelectorTable().should('exist'); + duplicateHardwareProfile + .getNodeSelectorTableRow('test-key') + .shouldHaveValue('test-value') + .findDeleteAction() + .click(); + + duplicateHardwareProfile.k8sNameDescription + .findDisplayNameInput() + .fill('duplicate hardware profile'); + duplicateHardwareProfile.findSubmitButton().should('be.enabled').click(); + cy.wait('@createHardwareProfile').then((interception) => { + expect(interception.request.body.spec).to.eql({ + identifiers: [ + { + displayName: 'Memory', + identifier: 'memory', + minCount: '2Gi', + maxCount: '5Gi', + defaultCount: '2Gi', + }, + { + displayName: 'CPU', + identifier: 'cpu', + minCount: '1', + maxCount: '2', + defaultCount: '1', + }, + { + identifier: 'nvidia.com/gpu', + displayName: 'GPU', + maxCount: 2, + minCount: 1, + defaultCount: 1, + }, + ], + displayName: 'duplicate hardware profile', + enabled: true, + tolerations: [ + { + key: 'nvidia.com/gpu', + operator: 'Exists', + effect: 'NoSchedule', + }, + ], + nodeSelectors: [], + description: '', + }); + }); + }); + + it('invalid id in edit page', () => { + initIntercepts({ isPresent: false }); + editHardwareProfile.visit('test-hardware-profile'); + editHardwareProfile.findErrorText().should('exist'); + cy.interceptK8sList( + HardwareProfileModel, + mockK8sResourceList([mockHardwareProfile({ namespace: 'opendatahub', uid: 'test-12' })]), + ).as('listHardwareProfiles'); + editHardwareProfile.findViewAllHardwareProfilesButton().click(); + cy.wait('@listHardwareProfiles'); + }); + + it('invalid id in duplicate page', () => { + initIntercepts({ isPresent: false }); + duplicateHardwareProfile.visit('test-hardware-profile'); + duplicateHardwareProfile.findErrorText().should('exist'); + cy.interceptK8sList( + HardwareProfileModel, + mockK8sResourceList([mockHardwareProfile({ namespace: 'opendatahub', uid: 'test-12' })]), + ).as('listHardwareProfiles'); + duplicateHardwareProfile.findViewAllHardwareProfilesButton().click(); + cy.wait('@listHardwareProfiles'); + }); +}); diff --git a/frontend/src/api/k8s/__tests__/hardwareProfiles.spec.ts b/frontend/src/api/k8s/__tests__/hardwareProfiles.spec.ts index 39b0fa22d7..d93eb9f6da 100644 --- a/frontend/src/api/k8s/__tests__/hardwareProfiles.spec.ts +++ b/frontend/src/api/k8s/__tests__/hardwareProfiles.spec.ts @@ -34,16 +34,25 @@ const k8sCreateResourceMock = jest.mocked(k8sCreateResource const k8sUpdateResourceMock = jest.mocked(k8sUpdateResource); const k8sDeleteResourceMock = jest.mocked(k8sDeleteResource); +global.structuredClone = (val: unknown) => JSON.parse(JSON.stringify(val)); + const data: HardwareProfileKind['spec'] = { displayName: 'test', identifiers: [ { displayName: 'Memory', identifier: 'memory', - minCount: '5Gi', - maxCount: '2Gi', + minCount: '2Gi', + maxCount: '5Gi', defaultCount: '2Gi', }, + { + displayName: 'CPU', + identifier: 'cpu', + minCount: '1', + maxCount: '2', + defaultCount: '1', + }, ], description: 'test description', enabled: true, @@ -62,6 +71,7 @@ const assembleHardwareProfileResult: HardwareProfileKind = { metadata: { name: 'test-1', namespace: 'namespace', + annotations: expect.anything(), }, spec: data, }; @@ -135,7 +145,7 @@ describe('getHardwareProfile', () => { }); describe('createHardwareProfiles', () => { - it('should create hadware profile', async () => { + it('should create hardware profile', async () => { k8sCreateResourceMock.mockResolvedValue(mockHardwareProfile({ uid: 'test' })); const result = await createHardwareProfile('test-1', data, 'namespace'); expect(k8sCreateResourceMock).toHaveBeenCalledWith({ @@ -184,6 +194,8 @@ describe('updateHardwareProfile', () => { namespace: 'namespace', description: 'test description', displayName: 'test', + nodeSelectors: [], + annotations: expect.anything(), }), }); expect(k8sUpdateResourceMock).toHaveBeenCalledTimes(1); @@ -211,6 +223,8 @@ describe('updateHardwareProfile', () => { namespace: 'namespace', description: 'test description', displayName: 'test', + nodeSelectors: [], + annotations: expect.anything(), }), }); }); diff --git a/frontend/src/api/k8s/hardwareProfiles.ts b/frontend/src/api/k8s/hardwareProfiles.ts index b091adb6b9..11f2ac6524 100644 --- a/frontend/src/api/k8s/hardwareProfiles.ts +++ b/frontend/src/api/k8s/hardwareProfiles.ts @@ -37,6 +37,9 @@ export const assembleHardwareProfile = ( metadata: { name: hardwareProfileName || translateDisplayNameForK8s(data.displayName), namespace, + annotations: { + 'opendatahub.io/modified-date': new Date().toISOString(), + }, }, spec: data, }); @@ -66,7 +69,14 @@ export const updateHardwareProfile = ( opts?: K8sAPIOptions, ): Promise => { const resource = assembleHardwareProfile(existingHardwareProfile.metadata.name, data, namespace); - const hardwareProfileResource = _.merge({}, existingHardwareProfile, resource); + + const oldHardwareProfile = structuredClone(existingHardwareProfile); + // clean up the resources from the old hardware profile + oldHardwareProfile.spec.identifiers = []; + oldHardwareProfile.spec.nodeSelectors = []; + oldHardwareProfile.spec.tolerations = []; + + const hardwareProfileResource = _.merge({}, oldHardwareProfile, resource); return k8sUpdateResource( applyK8sAPIOptions({ model: HardwareProfileModel, resource: hardwareProfileResource }, opts), diff --git a/frontend/src/pages/hardwareProfiles/HardwareProfiles.tsx b/frontend/src/pages/hardwareProfiles/HardwareProfiles.tsx index e834bb9f05..237ab22616 100644 --- a/frontend/src/pages/hardwareProfiles/HardwareProfiles.tsx +++ b/frontend/src/pages/hardwareProfiles/HardwareProfiles.tsx @@ -11,6 +11,7 @@ import { EmptyStateFooter, } from '@patternfly/react-core'; import { PlusCircleIcon } from '@patternfly/react-icons'; +import { useNavigate } from 'react-router-dom'; import ApplicationsPage from '~/pages/ApplicationsPage'; import { useDashboardNamespace } from '~/redux/selectors'; import { ODH_PRODUCT_NAME } from '~/utilities/const'; @@ -22,6 +23,7 @@ const description = `Manage hardware profile settings for users in your organiza const HardwareProfiles: React.FC = () => { const { dashboardNamespace } = useDashboardNamespace(); const [hardwareProfiles, loaded, loadError, refresh] = useHardwareProfiles(dashboardNamespace); + const navigate = useNavigate(); const isEmpty = hardwareProfiles.length === 0; @@ -45,9 +47,7 @@ const HardwareProfiles: React.FC = () => { diff --git a/frontend/src/pages/hardwareProfiles/HardwareProfilesRoutes.tsx b/frontend/src/pages/hardwareProfiles/HardwareProfilesRoutes.tsx index 8e4ea99a3d..c36c2471dd 100644 --- a/frontend/src/pages/hardwareProfiles/HardwareProfilesRoutes.tsx +++ b/frontend/src/pages/hardwareProfiles/HardwareProfilesRoutes.tsx @@ -1,10 +1,18 @@ import * as React from 'react'; import { Navigate, Route, Routes } from 'react-router-dom'; +import ManageHardwareProfile from '~/pages/hardwareProfiles/manage/ManageHardwareProfile'; +import { + DuplicateHardwareProfile, + EditHardwareProfile, +} from '~/pages/hardwareProfiles/manage/ManageHardwareProfileWrapper'; import HardwareProfiles from './HardwareProfiles'; const HardwareProfilesRoutes: React.FC = () => ( } /> + } /> + } /> + } /> } /> ); diff --git a/frontend/src/pages/hardwareProfiles/HardwareProfilesTableRow.tsx b/frontend/src/pages/hardwareProfiles/HardwareProfilesTableRow.tsx index cdd0a2c55a..9db3aecd0f 100644 --- a/frontend/src/pages/hardwareProfiles/HardwareProfilesTableRow.tsx +++ b/frontend/src/pages/hardwareProfiles/HardwareProfilesTableRow.tsx @@ -7,13 +7,14 @@ import { TimestampTooltipVariant, } from '@patternfly/react-core'; import { ActionsColumn, ExpandableRowContent, Tbody, Td, Tr } from '@patternfly/react-table'; +import { useNavigate } from 'react-router-dom'; import { relativeTime } from '~/utilities/time'; import { TableRowTitleDescription } from '~/components/table'; import HardwareProfileEnableToggle from '~/pages/hardwareProfiles/HardwareProfileEnableToggle'; import { HardwareProfileKind } from '~/k8sTypes'; import NodeResourceTable from '~/pages/hardwareProfiles/nodeResource/NodeResourceTable'; -import NodeSelectorTable from '~/pages/hardwareProfiles/nodeSelector/NodeSelectorsTable'; -import TolerationTable from '~/pages/hardwareProfiles/toleration/TolerationsTable'; +import NodeSelectorTable from '~/pages/hardwareProfiles/nodeSelector/NodeSelectorTable'; +import TolerationTable from '~/pages/hardwareProfiles/toleration/TolerationTable'; import { isHardwareProfileOOTB } from '~/pages/hardwareProfiles/utils'; type HardwareProfilesTableRowProps = { @@ -31,6 +32,7 @@ const HardwareProfilesTableRow: React.FC = ({ }) => { const modifiedDate = hardwareProfile.metadata.annotations?.['opendatahub.io/modified-date']; const [isExpanded, setExpanded] = React.useState(false); + const navigate = useNavigate(); return ( @@ -73,15 +75,19 @@ const HardwareProfilesTableRow: React.FC = ({ undefined, - }, + ...(isHardwareProfileOOTB(hardwareProfile) + ? [] + : [ + { + title: 'Edit', + onClick: () => + navigate(`/hardwareProfiles/edit/${hardwareProfile.metadata.name}`), + }, + ]), { title: 'Duplicate', - // TODO: add duplicate - onClick: () => undefined, + onClick: () => + navigate(`/hardwareProfiles/duplicate/${hardwareProfile.metadata.name}`), }, ...(isHardwareProfileOOTB(hardwareProfile) ? [] diff --git a/frontend/src/pages/hardwareProfiles/HardwareProfilesToolbar.tsx b/frontend/src/pages/hardwareProfiles/HardwareProfilesToolbar.tsx index ffb1c17323..231932920c 100644 --- a/frontend/src/pages/hardwareProfiles/HardwareProfilesToolbar.tsx +++ b/frontend/src/pages/hardwareProfiles/HardwareProfilesToolbar.tsx @@ -1,5 +1,6 @@ import React from 'react'; import { Button, SearchInput, ToolbarGroup, ToolbarItem } from '@patternfly/react-core'; +import { useNavigate } from 'react-router-dom'; import FilterToolbar from '~/components/FilterToolbar'; import { HardwareProfileEnableType, @@ -17,46 +18,52 @@ type HardwareProfilesToolbarProps = { const HardwareProfilesToolbar: React.FC = ({ filterData, onFilterUpdate, -}) => ( - - data-testid="hardware-profiles-table-toolbar" - filterOptions={hardwareProfileFilterOptions} - filterOptionRenders={{ - [HardwareProfileFilterOptions.name]: ({ onChange, ...props }) => ( - onChange(value)} - /> - ), - [HardwareProfileFilterOptions.enabled]: ({ value, onChange, ...props }) => ( - ({ - key: v, - label: v, - }))} - onChange={(v) => onChange(v)} - popperProps={{ maxWidth: undefined }} - /> - ), - }} - filterData={filterData} - onFilterUpdate={onFilterUpdate} - > - - - {/* TODO: navigate to creation page */} - - - - -); +}) => { + const navigate = useNavigate(); + + return ( + + data-testid="hardware-profiles-table-toolbar" + filterOptions={hardwareProfileFilterOptions} + filterOptionRenders={{ + [HardwareProfileFilterOptions.name]: ({ onChange, ...props }) => ( + onChange(value)} + /> + ), + [HardwareProfileFilterOptions.enabled]: ({ value, onChange, ...props }) => ( + ({ + key: v, + label: v, + }))} + onChange={(v) => onChange(v)} + popperProps={{ maxWidth: undefined }} + /> + ), + }} + filterData={filterData} + onFilterUpdate={onFilterUpdate} + > + + + + + + + ); +}; export default HardwareProfilesToolbar; diff --git a/frontend/src/pages/hardwareProfiles/ManageNodeResourceSection.tsx b/frontend/src/pages/hardwareProfiles/ManageNodeResourceSection.tsx deleted file mode 100644 index 0d0933fd0e..0000000000 --- a/frontend/src/pages/hardwareProfiles/ManageNodeResourceSection.tsx +++ /dev/null @@ -1,69 +0,0 @@ -import React from 'react'; -import { - FormSection, - Flex, - FlexItem, - Button, - Content, - Stack, - StackItem, -} from '@patternfly/react-core'; -import { Identifier } from '~/types'; -import NodeResourceTable from './nodeResource/NodeResourceTable'; -import ManageNodeResourceModal from './nodeResource/ManageNodeResourceModal'; - -type ManageNodeResourceSectionProps = { - nodeResources: Identifier[]; - setNodeResources: (identifiers: Identifier[]) => void; -}; - -const ManageNodeResourceSection: React.FC = ({ - nodeResources, - setNodeResources, -}) => { - const [isNodeResourceModalOpen, setIsNodeResourceModalOpen] = React.useState(false); - return ( - <> - - Node resources - - - - - - Every hardware profile must include CPU and memory resources. Additional resources, - such as GPUs, can be added here. - - - - } - > - - - setNodeResources(newIdentifiers)} - /> - - - - {isNodeResourceModalOpen ? ( - setIsNodeResourceModalOpen(false)} - onSave={(identifier) => setNodeResources([...nodeResources, identifier])} - nodeResources={nodeResources} - /> - ) : null} - - ); -}; - -export default ManageNodeResourceSection; diff --git a/frontend/src/pages/hardwareProfiles/const.ts b/frontend/src/pages/hardwareProfiles/const.ts index 8aef7ea046..ed003dfb92 100644 --- a/frontend/src/pages/hardwareProfiles/const.ts +++ b/frontend/src/pages/hardwareProfiles/const.ts @@ -1,6 +1,10 @@ import { SortableData } from '~/components/table'; import { HardwareProfileKind } from '~/k8sTypes'; -import { Identifier, NodeSelector, Toleration } from '~/types'; +import { + ManageHardwareProfileSectionID, + ManageHardwareProfileSectionTitlesType, +} from '~/pages/hardwareProfiles/manage/types'; +import { IdentifierResourceType } from '~/types'; export const hardwareProfileColumns: SortableData[] = [ { @@ -43,87 +47,6 @@ export const hardwareProfileColumns: SortableData[] = [ }, ]; -export const nodeResourceColumns: SortableData[] = [ - { - field: 'name', - label: 'Resource label', - sortable: false, - width: 20, - }, - { - field: 'identifier', - label: 'Resource identifier', - sortable: false, - width: 20, - }, - { - field: 'default', - label: 'Default', - sortable: false, - width: 20, - }, - { - field: 'min_allowed', - label: 'Minimum allowed', - sortable: false, - width: 20, - }, - { - field: 'max_allowed', - label: 'Maximum allowed', - sortable: false, - width: 20, - }, -]; - -export const nodeSelectorColumns: SortableData[] = [ - { - field: 'key', - label: 'Key', - sortable: false, - width: 50, - }, - { - field: 'value', - label: 'Value', - sortable: false, - width: 50, - }, -]; - -export const tolerationColumns: SortableData[] = [ - { - field: 'operator', - label: 'Operator', - sortable: false, - width: 20, - }, - { - field: 'key', - label: 'Key', - sortable: false, - width: 20, - }, - { - field: 'value', - label: 'Value', - sortable: false, - width: 20, - }, - { - field: 'effect', - label: 'Effect', - sortable: false, - width: 20, - }, - { - field: 'toleration_seconds', - label: 'Toleration seconds', - sortable: false, - width: 20, - }, -]; - export enum HardwareProfileEnableType { enabled = 'Enabled', disabled = 'Disabled', @@ -148,3 +71,33 @@ export const initialHardwareProfileFilterData: HardwareProfileFilterDataType = { [HardwareProfileFilterOptions.name]: '', [HardwareProfileFilterOptions.enabled]: undefined, }; + +export const ManageHardwareProfileSectionTitles: ManageHardwareProfileSectionTitlesType = { + [ManageHardwareProfileSectionID.DETAILS]: 'Details', + [ManageHardwareProfileSectionID.IDENTIFIERS]: 'Node resources', + [ManageHardwareProfileSectionID.NODE_SELECTORS]: 'Node selectors', + [ManageHardwareProfileSectionID.TOLERATIONS]: 'Tolerations', +}; + +export const DEFAULT_HARDWARE_PROFILE_SPEC: HardwareProfileKind['spec'] = { + displayName: '', + enabled: true, + identifiers: [ + { + identifier: 'cpu', + displayName: 'CPU', + defaultCount: 2, + maxCount: 4, + minCount: 1, + resourceType: IdentifierResourceType.CPU, + }, + { + identifier: 'memory', + displayName: 'Memory', + defaultCount: '4Gi', + minCount: '2Gi', + maxCount: '8Gi', + resourceType: IdentifierResourceType.MEMORY, + }, + ], +}; diff --git a/frontend/src/pages/hardwareProfiles/manage/ManageHardwareProfile.tsx b/frontend/src/pages/hardwareProfiles/manage/ManageHardwareProfile.tsx new file mode 100644 index 0000000000..70f3153d88 --- /dev/null +++ b/frontend/src/pages/hardwareProfiles/manage/ManageHardwareProfile.tsx @@ -0,0 +1,142 @@ +import React from 'react'; +import { Link } from 'react-router-dom'; +import { Breadcrumb, BreadcrumbItem, Form, FormSection, PageSection } from '@patternfly/react-core'; +import ApplicationsPage from '~/pages/ApplicationsPage'; +import useGenericObjectState from '~/utilities/useGenericObjectState'; +import { HardwareProfileKind } from '~/k8sTypes'; +import K8sNameDescriptionField, { + useK8sNameDescriptionFieldData, +} from '~/concepts/k8s/K8sNameDescriptionField/K8sNameDescriptionField'; +import { isK8sNameDescriptionDataValid } from '~/concepts/k8s/K8sNameDescriptionField/utils'; +import { + DEFAULT_HARDWARE_PROFILE_SPEC, + ManageHardwareProfileSectionTitles, +} from '~/pages/hardwareProfiles/const'; +import ManageNodeSelectorSection from '~/pages/hardwareProfiles/manage/ManageNodeSelectorSection'; +import ManageTolerationSection from '~/pages/hardwareProfiles/manage/ManageTolerationSection'; +import ManageHardwareProfileFooter from '~/pages/hardwareProfiles/manage/ManageHardwareProfileFooter'; +import ManageNodeResourceSection from '~/pages/hardwareProfiles/manage/ManageNodeResourceSection'; +import { HardwareProfileFormData, ManageHardwareProfileSectionID } from './types'; + +type ManageHardwareProfileProps = { + existingHardwareProfile?: HardwareProfileKind; + duplicatedHardwareProfile?: HardwareProfileKind; +}; + +const ManageHardwareProfile: React.FC = ({ + existingHardwareProfile, + duplicatedHardwareProfile, +}) => { + const [state, setState] = useGenericObjectState( + DEFAULT_HARDWARE_PROFILE_SPEC, + ); + const { data: profileNameDesc, onDataChange: setProfileNameDesc } = + useK8sNameDescriptionFieldData({ + initialData: existingHardwareProfile + ? { + name: existingHardwareProfile.spec.displayName, + k8sName: existingHardwareProfile.metadata.name, + description: existingHardwareProfile.spec.description, + } + : undefined, + }); + + React.useEffect(() => { + if (existingHardwareProfile) { + setState('identifiers', existingHardwareProfile.spec.identifiers); + setState('enabled', existingHardwareProfile.spec.enabled); + setState('nodeSelectors', existingHardwareProfile.spec.nodeSelectors); + setState('tolerations', existingHardwareProfile.spec.tolerations); + } + }, [existingHardwareProfile, setState]); + + React.useEffect(() => { + if (duplicatedHardwareProfile) { + setState('identifiers', duplicatedHardwareProfile.spec.identifiers); + setState('enabled', duplicatedHardwareProfile.spec.enabled); + setState('nodeSelectors', duplicatedHardwareProfile.spec.nodeSelectors); + setState('tolerations', duplicatedHardwareProfile.spec.tolerations); + } + }, [duplicatedHardwareProfile, setState]); + + const formState: HardwareProfileFormData = React.useMemo( + () => ({ + ...state, + name: profileNameDesc.k8sName.value, + displayName: profileNameDesc.name, + description: profileNameDesc.description, + }), + [state, profileNameDesc], + ); + + const validFormData = isK8sNameDescriptionDataValid(profileNameDesc); + + return ( + + Hardware profiles} /> + + {existingHardwareProfile ? 'Edit' : duplicatedHardwareProfile ? 'Duplicate' : 'Create'}{' '} + hardware profile + + + } + loaded + empty={false} + > + +
+ + + + setState('identifiers', identifiers)} + /> + setState('nodeSelectors', nodeSelectors)} + /> + setState('tolerations', tolerations)} + /> + +
+ + + +
+ ); +}; + +export default ManageHardwareProfile; diff --git a/frontend/src/pages/hardwareProfiles/manage/ManageHardwareProfileFooter.tsx b/frontend/src/pages/hardwareProfiles/manage/ManageHardwareProfileFooter.tsx new file mode 100644 index 0000000000..871f625201 --- /dev/null +++ b/frontend/src/pages/hardwareProfiles/manage/ManageHardwareProfileFooter.tsx @@ -0,0 +1,103 @@ +import { + Stack, + StackItem, + Alert, + ActionList, + ActionListItem, + Button, +} from '@patternfly/react-core'; +import React from 'react'; +import { useNavigate } from 'react-router'; +import { HardwareProfileKind } from '~/k8sTypes'; +import { HardwareProfileFormData } from '~/pages/hardwareProfiles/manage/types'; +import { createHardwareProfile, updateHardwareProfile } from '~/api'; +import { useDashboardNamespace } from '~/redux/selectors'; + +type ManageHardwareProfileFooterProps = { + state: HardwareProfileFormData; + existingHardwareProfile?: HardwareProfileKind; + validFormData: boolean; +}; + +const ManageHardwareProfileFooter: React.FC = ({ + state, + existingHardwareProfile, + validFormData, +}) => { + const [errorMessage, setErrorMessage] = React.useState(''); + const [isLoading, setIsLoading] = React.useState(false); + const { dashboardNamespace } = useDashboardNamespace(); + const navigate = useNavigate(); + + const { name, ...spec } = state; + + const onCreateHardwareProfile = async () => { + setIsLoading(true); + createHardwareProfile(name, spec, dashboardNamespace) + .then(() => navigate('/hardwareProfiles')) + .catch((err) => { + setErrorMessage(err.message); + }) + .finally(() => { + setIsLoading(false); + }); + }; + + const onUpdateHardwareProfile = async () => { + if (existingHardwareProfile) { + setIsLoading(true); + updateHardwareProfile(spec, existingHardwareProfile, dashboardNamespace) + .then(() => navigate(`/hardwareProfiles`)) + .catch((err) => { + setErrorMessage(err.message); + }) + .finally(() => { + setIsLoading(false); + }); + } + }; + + return ( + + {errorMessage && ( + + + {errorMessage} + + + )} + + + + + + + + + + + + ); +}; + +export default ManageHardwareProfileFooter; diff --git a/frontend/src/pages/hardwareProfiles/manage/ManageHardwareProfileWrapper.tsx b/frontend/src/pages/hardwareProfiles/manage/ManageHardwareProfileWrapper.tsx new file mode 100644 index 0000000000..2976ee3c5f --- /dev/null +++ b/frontend/src/pages/hardwareProfiles/manage/ManageHardwareProfileWrapper.tsx @@ -0,0 +1,75 @@ +import * as React from 'react'; +import { useParams } from 'react-router'; +import { + Bullseye, + Button, + EmptyState, + EmptyStateBody, + Spinner, + Title, +} from '@patternfly/react-core'; +import { ExclamationCircleIcon } from '@patternfly/react-icons'; +import { useNavigate } from 'react-router-dom'; +import { useDashboardNamespace } from '~/redux/selectors'; +import ManageHardwareProfile from '~/pages/hardwareProfiles/manage/ManageHardwareProfile'; +import useHardwareProfile from '~/pages/hardwareProfiles/useHardwareProfile'; +import { HardwareProfileKind } from '~/k8sTypes'; + +type ManageHardwareProfileWrapperProps = { + children: (data: HardwareProfileKind) => React.ReactNode; +}; + +const ManageHardwareProfileWrapper: React.FC = ({ + children, +}) => { + const navigate = useNavigate(); + const { hardwareProfileName } = useParams(); + const { dashboardNamespace } = useDashboardNamespace(); + const [data, , error] = useHardwareProfile(dashboardNamespace, hardwareProfileName); + + if (error) { + return ( + + + Problem loading hardware profile + + } + icon={ExclamationCircleIcon} + > + {error.message} + + + + ); + } + + if (!data) { + return ( + + + + ); + } + + return children(data); +}; + +export const EditHardwareProfile: React.FC = () => ( + + {(data) => } + +); + +export const DuplicateHardwareProfile: React.FC = () => ( + + {(data) => } + +); diff --git a/frontend/src/pages/hardwareProfiles/manage/ManageNodeResourceSection.tsx b/frontend/src/pages/hardwareProfiles/manage/ManageNodeResourceSection.tsx new file mode 100644 index 0000000000..1b6cec6068 --- /dev/null +++ b/frontend/src/pages/hardwareProfiles/manage/ManageNodeResourceSection.tsx @@ -0,0 +1,93 @@ +import React from 'react'; +import { FormSection, Flex, FlexItem, Button, Alert, AlertVariant } from '@patternfly/react-core'; +import { AddCircleOIcon } from '@patternfly/react-icons'; +import { Identifier, IdentifierResourceType } from '~/types'; +import NodeResourceTable from '~/pages/hardwareProfiles/nodeResource/NodeResourceTable'; +import ManageNodeResourceModal from '~/pages/hardwareProfiles/nodeResource/ManageNodeResourceModal'; +import { ManageHardwareProfileSectionTitles } from '~/pages/hardwareProfiles/const'; +import { ManageHardwareProfileSectionID } from '~/pages/hardwareProfiles/manage/types'; + +type ManageNodeResourceSectionProps = { + nodeResources: Identifier[]; + setNodeResources: (identifiers: Identifier[]) => void; +}; + +const ManageNodeResourceSection: React.FC = ({ + nodeResources, + setNodeResources, +}) => { + const [isNodeResourceModalOpen, setIsNodeResourceModalOpen] = React.useState(false); + const isEmpty = nodeResources.length === 0; + return ( + <> + + + {ManageHardwareProfileSectionTitles[ManageHardwareProfileSectionID.IDENTIFIERS]} + + {!isEmpty && ( + + + + )} + + } + > + Every hardware profile is highly recommended to include CPU and memory resources. Additional + resources, such as GPUs, can be added here, too. + {!( + nodeResources.some( + (identifier) => identifier.resourceType === IdentifierResourceType.CPU, + ) && + nodeResources.some( + (identifier) => identifier.resourceType === IdentifierResourceType.MEMORY, + ) + ) && ( + + It is not recommended to remove the CPU or Memory. The resources that use this hardware + profile will schedule, but will be very unstable due to not having any lower or upper + resource bounds. + + )} + {!isEmpty && ( + setNodeResources(newResources)} + /> + )} + + {isNodeResourceModalOpen && ( + setIsNodeResourceModalOpen(false)} + onSave={(identifier) => setNodeResources([...nodeResources, identifier])} + nodeResources={nodeResources} + /> + )} + {isEmpty && ( + + )} + + ); +}; + +export default ManageNodeResourceSection; diff --git a/frontend/src/pages/hardwareProfiles/manage/ManageNodeSelectorSection.tsx b/frontend/src/pages/hardwareProfiles/manage/ManageNodeSelectorSection.tsx new file mode 100644 index 0000000000..284f0fbaf3 --- /dev/null +++ b/frontend/src/pages/hardwareProfiles/manage/ManageNodeSelectorSection.tsx @@ -0,0 +1,73 @@ +import React from 'react'; +import { FormSection, Button, Flex, FlexItem } from '@patternfly/react-core'; +import { AddCircleOIcon } from '@patternfly/react-icons'; +import { NodeSelector } from '~/types'; +import ManageNodeSelectorModal from '~/pages/hardwareProfiles/nodeSelector/ManageNodeSelectorModal'; +import NodeSelectorTable from '~/pages/hardwareProfiles/nodeSelector/NodeSelectorTable'; +import { ManageHardwareProfileSectionTitles } from '~/pages/hardwareProfiles/const'; +import { ManageHardwareProfileSectionID } from '~/pages/hardwareProfiles/manage/types'; + +type ManageNodeSelectorSectionProps = { + nodeSelectors: NodeSelector[]; + setNodeSelectors: (nodeSelectors: NodeSelector[]) => void; +}; + +const ManageNodeSelectorSection: React.FC = ({ + nodeSelectors, + setNodeSelectors, +}) => { + const [isNodeSelectorModalOpen, setIsNodeSelectorModalOpen] = React.useState(false); + const isEmpty = nodeSelectors.length === 0; + return ( + <> + + + {ManageHardwareProfileSectionTitles[ManageHardwareProfileSectionID.NODE_SELECTORS]} + + {!isEmpty && ( + + + + )} + + } + > + Node selectors are added to a pod spec to allow the pod to be scheduled on nodes with + matching labels. + {!isEmpty && ( + setNodeSelectors(newNodeSelectors)} + /> + )} + + {isNodeSelectorModalOpen && ( + setIsNodeSelectorModalOpen(false)} + onSave={(nodeSelector) => setNodeSelectors([...nodeSelectors, nodeSelector])} + /> + )} + {isEmpty && ( + + )} + + ); +}; + +export default ManageNodeSelectorSection; diff --git a/frontend/src/pages/hardwareProfiles/manage/ManageTolerationSection.tsx b/frontend/src/pages/hardwareProfiles/manage/ManageTolerationSection.tsx new file mode 100644 index 0000000000..d574c51396 --- /dev/null +++ b/frontend/src/pages/hardwareProfiles/manage/ManageTolerationSection.tsx @@ -0,0 +1,73 @@ +import React from 'react'; +import { FormSection, Button, Flex, FlexItem } from '@patternfly/react-core'; +import { AddCircleOIcon } from '@patternfly/react-icons'; +import { Toleration } from '~/types'; +import ManageTolerationModal from '~/pages/hardwareProfiles/toleration/ManageTolerationModal'; +import TolerationTable from '~/pages/hardwareProfiles/toleration/TolerationTable'; +import { ManageHardwareProfileSectionTitles } from '~/pages/hardwareProfiles/const'; +import { ManageHardwareProfileSectionID } from '~/pages/hardwareProfiles/manage/types'; + +type ManageTolerationSectionProps = { + tolerations: Toleration[]; + setTolerations: (tolerations: Toleration[]) => void; +}; + +const ManageTolerationSection: React.FC = ({ + tolerations, + setTolerations, +}) => { + const [isTolerationModalOpen, setIsTolerationModalOpen] = React.useState(false); + const isEmpty = tolerations.length === 0; + return ( + <> + + + {ManageHardwareProfileSectionTitles[ManageHardwareProfileSectionID.TOLERATIONS]} + + {!isEmpty && ( + + + + )} + + } + > + Tolerations are applied to pods and allow the scheduler to schedule pods on nodes with + matching taints. + {tolerations.length !== 0 && ( + setTolerations(newTolerations)} + /> + )} + + {isTolerationModalOpen && ( + setIsTolerationModalOpen(false)} + onSave={(toleration) => setTolerations([...tolerations, toleration])} + /> + )} + {isEmpty && ( + + )} + + ); +}; + +export default ManageTolerationSection; diff --git a/frontend/src/pages/hardwareProfiles/manage/types.ts b/frontend/src/pages/hardwareProfiles/manage/types.ts new file mode 100644 index 0000000000..478da51707 --- /dev/null +++ b/frontend/src/pages/hardwareProfiles/manage/types.ts @@ -0,0 +1,14 @@ +import { HardwareProfileKind } from '~/k8sTypes'; + +export enum ManageHardwareProfileSectionID { + DETAILS = 'details', + IDENTIFIERS = 'identifiers', + NODE_SELECTORS = 'node-selectors', + TOLERATIONS = 'tolerations', +} + +export type ManageHardwareProfileSectionTitlesType = { + [key in ManageHardwareProfileSectionID]: string; +}; + +export type HardwareProfileFormData = { name: string } & HardwareProfileKind['spec']; diff --git a/frontend/src/pages/hardwareProfiles/nodeResource/CountFormField.tsx b/frontend/src/pages/hardwareProfiles/nodeResource/CountFormField.tsx index 5c03c07c27..3fdca5f935 100644 --- a/frontend/src/pages/hardwareProfiles/nodeResource/CountFormField.tsx +++ b/frontend/src/pages/hardwareProfiles/nodeResource/CountFormField.tsx @@ -3,13 +3,14 @@ import { FormGroup, FormHelperText, HelperText, HelperTextItem } from '@patternf import MemoryField from '~/components/MemoryField'; import CPUField from '~/components/CPUField'; import NumberInputWrapper from '~/components/NumberInputWrapper'; +import { IdentifierResourceType } from '~/types'; type CountFormFieldProps = { label: string; fieldId: string; size: number | string; setSize: (value: number | string) => void; - identifier: string; + type?: IdentifierResourceType; errorMessage?: string; isValid?: boolean; }; @@ -19,15 +20,15 @@ const CountFormField: React.FC = ({ fieldId, size, setSize, - identifier, + type, errorMessage, isValid = true, }) => { const renderInputField = () => { - switch (identifier) { - case 'cpu': + switch (type) { + case IdentifierResourceType.CPU: return setSize(value)} value={size} />; - case 'memory': + case IdentifierResourceType.MEMORY: return setSize(value)} value={String(size)} />; default: return ( @@ -45,12 +46,12 @@ const CountFormField: React.FC = ({ }; return ( - + {renderInputField()} {!isValid && errorMessage && ( - + {errorMessage} diff --git a/frontend/src/pages/hardwareProfiles/nodeResource/ManageNodeResourceModal.tsx b/frontend/src/pages/hardwareProfiles/nodeResource/ManageNodeResourceModal.tsx index 9e4971ded7..2d296fb59c 100644 --- a/frontend/src/pages/hardwareProfiles/nodeResource/ManageNodeResourceModal.tsx +++ b/frontend/src/pages/hardwareProfiles/nodeResource/ManageNodeResourceModal.tsx @@ -1,7 +1,7 @@ import React from 'react'; import { Modal } from '@patternfly/react-core/deprecated'; import DashboardModalFooter from '~/concepts/dashboard/DashboardModalFooter'; -import { Identifier } from '~/types'; +import { Identifier, IdentifierResourceType } from '~/types'; import useGenericObjectState from '~/utilities/useGenericObjectState'; import { CPU_UNITS, MEMORY_UNITS_FOR_SELECTION, UnitOption } from '~/utilities/valueUnits'; import { EMPTY_IDENTIFIER } from './const'; @@ -32,11 +32,11 @@ const ManageNodeResourceModal: React.FC = ({ !nodeResources.some((i) => i.identifier === identifier.identifier); React.useEffect(() => { - switch (identifier.identifier) { - case 'cpu': + switch (identifier.resourceType) { + case IdentifierResourceType.CPU: setUnitOptions(CPU_UNITS); break; - case 'memory': + case IdentifierResourceType.MEMORY: setUnitOptions(MEMORY_UNITS_FOR_SELECTION); break; default: @@ -44,9 +44,8 @@ const ManageNodeResourceModal: React.FC = ({ } }, [identifier]); - const isValidCounts = unitOptions - ? validateDefaultCount(identifier, unitOptions) && validateMinCount(identifier, unitOptions) - : true; + const isValidCounts = + validateDefaultCount(identifier, unitOptions) && validateMinCount(identifier, unitOptions); const isButtonDisabled = !identifier.displayName || !identifier.identifier || !isUniqueIdentifier || !isValidCounts; @@ -75,7 +74,6 @@ const ManageNodeResourceModal: React.FC = ({ identifier={identifier} setIdentifier={setIdentifier} unitOptions={unitOptions} - isExistingIdentifier={!!existingIdentifier} isUniqueIdentifier={isUniqueIdentifier} />
diff --git a/frontend/src/pages/hardwareProfiles/nodeResource/NodeResourceForm.tsx b/frontend/src/pages/hardwareProfiles/nodeResource/NodeResourceForm.tsx index 90b7858259..29b600efee 100644 --- a/frontend/src/pages/hardwareProfiles/nodeResource/NodeResourceForm.tsx +++ b/frontend/src/pages/hardwareProfiles/nodeResource/NodeResourceForm.tsx @@ -7,9 +7,16 @@ import { HelperText, HelperTextItem, } from '@patternfly/react-core'; -import { Identifier } from '~/types'; +import { Identifier, IdentifierResourceType } from '~/types'; import { UpdateObjectAtPropAndValue } from '~/pages/projects/types'; import { UnitOption } from '~/utilities/valueUnits'; +import SimpleSelect from '~/components/SimpleSelect'; +import { asEnumMember } from '~/utilities/utils'; +import { + DEFAULT_CPU_SIZE, + DEFAULT_MEMORY_SIZE, + EMPTY_IDENTIFIER, +} from '~/pages/hardwareProfiles/nodeResource/const'; import { validateDefaultCount, validateMinCount } from './utils'; import CountFormField from './CountFormField'; @@ -17,7 +24,6 @@ type NodeResourceFormProps = { identifier: Identifier; setIdentifier: UpdateObjectAtPropAndValue; unitOptions?: UnitOption[]; - isExistingIdentifier?: boolean; isUniqueIdentifier?: boolean; }; @@ -25,7 +31,6 @@ const NodeResourceForm: React.FC = ({ identifier, setIdentifier, unitOptions, - isExistingIdentifier, isUniqueIdentifier, }) => { const validated = isUniqueIdentifier ? 'default' : 'error'; @@ -36,6 +41,7 @@ const NodeResourceForm: React.FC = ({ setIdentifier('displayName', value)} + data-testid="node-resource-label-input" />
@@ -43,11 +49,8 @@ const NodeResourceForm: React.FC = ({ setIdentifier('identifier', value)} - isDisabled={ - isExistingIdentifier && - (identifier.identifier === 'cpu' || identifier.identifier === 'memory') - } validated={validated} + data-testid="node-resource-identifier-input" /> {!isUniqueIdentifier && ( @@ -61,30 +64,67 @@ const NodeResourceForm: React.FC = ({ )} + + ({ + key: v, + label: v, + })), + { key: 'Other', label: 'Other' }, + ]} + value={identifier.resourceType || 'Other'} + onChange={(value) => { + const resourceType = asEnumMember(value, IdentifierResourceType); + switch (resourceType) { + case IdentifierResourceType.CPU: + setIdentifier('resourceType', resourceType); + setIdentifier('minCount', DEFAULT_CPU_SIZE.minCount); + setIdentifier('maxCount', DEFAULT_CPU_SIZE.maxCount); + setIdentifier('defaultCount', DEFAULT_CPU_SIZE.defaultCount); + break; + case IdentifierResourceType.MEMORY: + setIdentifier('resourceType', resourceType); + setIdentifier('minCount', DEFAULT_MEMORY_SIZE.minCount); + setIdentifier('maxCount', DEFAULT_MEMORY_SIZE.maxCount); + setIdentifier('defaultCount', DEFAULT_MEMORY_SIZE.defaultCount); + break; + default: + setIdentifier('resourceType', undefined); + setIdentifier('minCount', EMPTY_IDENTIFIER.minCount); + setIdentifier('maxCount', EMPTY_IDENTIFIER.maxCount); + setIdentifier('defaultCount', EMPTY_IDENTIFIER.defaultCount); + } + }} + /> + + setIdentifier('defaultCount', value)} - isValid={unitOptions ? validateDefaultCount(identifier, unitOptions) : true} + isValid={validateDefaultCount(identifier, unitOptions)} errorMessage="Default must be equal to or between the minimum and maximum allowed limits." /> setIdentifier('minCount', value)} - isValid={unitOptions ? validateMinCount(identifier, unitOptions) : true} + isValid={validateMinCount(identifier, unitOptions)} errorMessage="Minimum allowed value cannot exceed the maximum allowed value." /> setIdentifier('maxCount', value)} /> diff --git a/frontend/src/pages/hardwareProfiles/nodeResource/NodeResourceTable.tsx b/frontend/src/pages/hardwareProfiles/nodeResource/NodeResourceTable.tsx index 640e648ba2..94143eb72d 100644 --- a/frontend/src/pages/hardwareProfiles/nodeResource/NodeResourceTable.tsx +++ b/frontend/src/pages/hardwareProfiles/nodeResource/NodeResourceTable.tsx @@ -7,7 +7,7 @@ import ManageNodeResourceModal from './ManageNodeResourceModal'; type NodeResourceTableProps = { nodeResources: Identifier[]; - onUpdate?: (identifiers: Identifier[]) => void; + onUpdate?: (nodeResources: Identifier[]) => void; }; const NodeResourceTable: React.FC = ({ nodeResources, onUpdate }) => { @@ -29,7 +29,7 @@ const NodeResourceTable: React.FC = ({ nodeResources, on } rowRenderer={(identifier, rowIndex) => ( { setEditIdentifier(newIdentifier); diff --git a/frontend/src/pages/hardwareProfiles/nodeResource/NodeResourceTableRow.tsx b/frontend/src/pages/hardwareProfiles/nodeResource/NodeResourceTableRow.tsx index dff5b3d099..85c296658b 100644 --- a/frontend/src/pages/hardwareProfiles/nodeResource/NodeResourceTableRow.tsx +++ b/frontend/src/pages/hardwareProfiles/nodeResource/NodeResourceTableRow.tsx @@ -20,6 +20,7 @@ const NodeResourceTableRow: React.FC = ({ {identifier.displayName} {identifier.identifier} + {identifier.resourceType ?? 'Other'} {identifier.defaultCount} {identifier.minCount} {identifier.maxCount} @@ -29,7 +30,6 @@ const NodeResourceTableRow: React.FC = ({