diff --git a/cypress/mocks/compression.ts b/cypress/mocks/compression.ts new file mode 100644 index 000000000..09a382533 --- /dev/null +++ b/cypress/mocks/compression.ts @@ -0,0 +1,141 @@ +export const compressionPool = { + apiVersion: 'ceph.rook.io/v1', + kind: 'CephBlockPool', + metadata: { name: 'plasma-reactor-pool', namespace: 'openshift-storage' }, + spec: { + compressionMode: 'aggressive', + deviceClass: 'ssd', + failureDomain: 'zone', + parameters: { compression_mode: 'aggressive' }, + replicated: { size: 2 }, + }, +}; + +export const compressionStorageClass = { + kind: 'StorageClass', + apiVersion: 'storage.k8s.io/v1', + metadata: { + name: 'compression-class', + namespace: 'openshift-storage', + }, + provisioner: 'openshift-storage.rbd.csi.ceph.com', + parameters: { + 'csi.storage.k8s.io/fstype': 'ext4', + 'csi.storage.k8s.io/provisioner-secret-namespace': 'openshift-storage', + 'csi.storage.k8s.io/provisioner-secret-name': 'rook-csi-rbd-provisioner', + 'csi.storage.k8s.io/node-stage-secret-name': 'rook-csi-rbd-node', + 'csi.storage.k8s.io/controller-expand-secret-name': + 'rook-csi-rbd-provisioner', + imageFormat: '2', + clusterID: 'openshift-storage', + imageFeatures: 'layering,deep-flatten,exclusive-lock,object-map,fast-diff', + 'csi.storage.k8s.io/controller-expand-secret-namespace': + 'openshift-storage', + pool: 'plasma-reactor-pool', + 'csi.storage.k8s.io/node-stage-secret-namespace': 'openshift-storage', + }, + reclaimPolicy: 'Delete', + allowVolumeExpansion: true, + volumeBindingMode: 'Immediate', +}; + +export const fioPVC = { + kind: 'PersistentVolumeClaim', + apiVersion: 'v1', + metadata: { + name: 'fio-claim', + namespace: 'openshift-storage', + }, + spec: { + storageClassName: 'compression-class', + accessModes: ['ReadWriteOnce'], + volumeMode: 'Filesystem', + resources: { + requests: { + storage: '5Gi', + }, + }, + }, +}; + +export const fioConfig = `kind: ConfigMap +apiVersion: v1 +metadata: + name: fio-job-config + namespace: openshift-storage +data: + fio.job: |- + [global] + ioengine=psync + direct=1 + buffered=0 + size=1G + iodepth=1000 + numjobs=2 + group_reporting + refill_buffers + rwmixread=80 + norandommap + randrepeat=0 + percentage_random=0 + bs=512K + buffer_compress_percentage=50 + rw=read + [testjob] +`; + +export const fioJob = { + apiVersion: 'batch/v1', + kind: 'Job', + metadata: { + name: 'fio', + namespace: 'openshift-storage', + labels: { + app: 'fio', + }, + }, + spec: { + template: { + spec: { + containers: [ + { + name: 'fio', + image: 'quay.io/badhikar/fio:latest', + command: ['sh'], + args: [ + '-c', + // eslint-disable-next-line no-template-curly-in-string + 'echo ${HOSTNAME} && mkdir -p /scratch/${HOSTNAME} && fio /configs/fio.job --eta=never --directory=/scratch/${HOSTNAME}', + ], + volumeMounts: [ + { + name: 'fio-config-vol', + mountPath: '/configs', + }, + { + name: 'fio-data', + mountPath: '/scratch', + }, + ], + imagePullPolicy: 'Always', + }, + ], + restartPolicy: 'OnFailure', + volumes: [ + { + name: 'fio-config-vol', + configMap: { + name: 'fio-job-config', + }, + }, + { + name: 'fio-data', + persistentVolumeClaim: { + claimName: 'fio-claim', + }, + }, + ], + }, + }, + }, +}; diff --git a/cypress/support.ts b/cypress/support.ts index fe44d73f0..5e32e7ab0 100644 --- a/cypress/support.ts +++ b/cypress/support.ts @@ -2,11 +2,13 @@ import { CLUSTER_NAMESPACE, STORAGE_SYSTEM_NAME, OCS_SC_STATE } from './consts'; import './support/selectors'; import './support/login'; +import { commandPoll } from './views/common'; declare global { namespace Cypress { interface Chainable { install(encrypted?: boolean): Chainable; + enableToolboxPod(): void; } } } @@ -87,3 +89,19 @@ Cypress.Commands.add('install', () => { } }); }); + +Cypress.Commands.add('enableToolboxPod', () => { + cy.exec( + `oc patch ocsinitialization ocsinit -n openshift-storage --type json --patch '[{ "op": "replace", "path": "/spec/enableCephTools", "value": true }]'` + ); + commandPoll( + 'oc get pod -l "app=rook-ceph-tools" -n openshift-storage', + null, + false, + 30, + 0 + ); + cy.exec( + `oc wait --for=condition=ready pod -l "app=rook-ceph-tools" -n openshift-storage` + ); +}); diff --git a/cypress/tests/compression.spec.ts b/cypress/tests/compression.spec.ts new file mode 100644 index 000000000..7d52c6076 --- /dev/null +++ b/cypress/tests/compression.spec.ts @@ -0,0 +1,59 @@ +import { + compressionPool, + compressionStorageClass, + fioConfig, + fioJob, + fioPVC, +} from '../mocks/compression'; +import { navigateToBlockPool } from '../views/block-pool'; +import { commandPoll } from '../views/common'; + +describe('Test Pool Compression', () => { + before(() => { + cy.login(); + cy.visit('/'); + cy.install(); + cy.enableToolboxPod(); + cy.exec(`echo '${JSON.stringify(compressionPool)}' | oc apply -f -`); + cy.exec( + `echo '${JSON.stringify(compressionStorageClass)}' | oc apply -f -` + ); + cy.exec(`echo '${fioConfig}' | oc apply -f -`); + cy.exec(`echo '${JSON.stringify(fioPVC)}' | oc apply -f -`); + cy.exec(`echo '${JSON.stringify(fioJob)}' | oc apply -f -`); + commandPoll('oc get job fio', null, false, 30, 0); + cy.exec(`oc wait --for=condition=complete job fio`, { timeout: 120000 }); + }); + + after(() => { + cy.exec(`echo ${JSON.stringify(fioJob)} | oc delete -f -`); + cy.exec(`echo ${JSON.stringify(fioPVC)} | oc delete -f -`); + cy.exec(`echo ${JSON.stringify(fioConfig)} | oc delete -f -`); + cy.exec(`echo ${JSON.stringify(compressionPool)} | oc delete -f -`); + cy.exec(`echo ${JSON.stringify(compressionStorageClass)} | oc delete -f -`); + cy.logout(); + }); + + it('Tests compression statistics are correct on the BlockPool list Page', () => { + navigateToBlockPool(); + cy.byTestID('name-filter-input').type(compressionPool.metadata.name); + // TableData does not support data-test ids + // eslint-disable-next-line cypress/require-data-selectors + cy.get('#compressionStatus').should('have.text', 'Enabled'); + cy.exec( + `oc exec -it $(oc get pods -n openshift-storage -l app=rook-ceph-tools -o name) -n openshift-storage -- /usr/bin/rados df -f json` + ).then(({ stdout }) => { + const poolData = JSON.parse(stdout).pools; + const plasmaPool = poolData.find( + (pool) => pool.name === compressionPool.metadata.name + ); + const compressionSavings = plasmaPool.compress_bytes_used; + // TableData does not support data-test ids + // eslint-disable-next-line cypress/require-data-selectors + cy.get('#compressionSavings').should( + 'have.text', + (compressionSavings / 1024 ** 2).toFixed(2) + ); + }); + }); +}); diff --git a/cypress/views/block-pool.ts b/cypress/views/block-pool.ts index b614e3765..8d45156ee 100644 --- a/cypress/views/block-pool.ts +++ b/cypress/views/block-pool.ts @@ -23,7 +23,7 @@ export const navigateToBlockPool = () => { cy.byLegacyTestID('horizontal-link-Storage Systems').click(); cy.byLegacyTestID('item-filter').type('ocs-storagecluster-storagesystem'); cy.byTestRows('resource-row').get('td a').first().click(); - cy.byLegacyTestID('horizontal-link-BlockPools').click(); + cy.byTestID('horizontal-link-BlockPools').click(); }; export const populateBlockPoolForm = () => { diff --git a/cypress/views/common.ts b/cypress/views/common.ts index 5cbabafe8..e245993c4 100644 --- a/cypress/views/common.ts +++ b/cypress/views/common.ts @@ -2,16 +2,22 @@ export const commonFlows = { navigateToODF: () => { cy.clickNavLink(['Storage', 'Data Foundation']); }, + // eslint-disable-next-line cypress/require-data-selectors + checkAll: () => cy.get('input[name=check-all]'), }; export const commandPoll = ( cmd: string, expected: string, failOnNonZeroExit: boolean = true, - retry: number = 300 + retry: number = 300, + expectedCode?: number ) => { cy.exec(cmd, { failOnNonZeroExit }).then((res) => { - if (res.stdout === expected) { + if ( + (res.stdout === expected || expectedCode === res.code) && + res.stderr === '' + ) { assert(true); return; } @@ -20,6 +26,6 @@ export const commandPoll = ( return; } - commandPoll(cmd, expected, failOnNonZeroExit, retry - 1); + commandPoll(cmd, expected, failOnNonZeroExit, retry - 1, expectedCode); }); };