From f78f8dda364d065d4165a93c6ac2af0956d5d94f Mon Sep 17 00:00:00 2001 From: Jovan Cvetkovic Date: Mon, 9 Jan 2023 10:28:00 +0100 Subject: [PATCH 1/9] [FEATURE] Detector must have at least one alert set #288 Signed-off-by: Jovan Cvetkovic --- .../ConfigureAlerts/containers/ConfigureAlerts.tsx | 10 +++++++--- 1 file changed, 7 insertions(+), 3 deletions(-) diff --git a/public/pages/CreateDetector/components/ConfigureAlerts/containers/ConfigureAlerts.tsx b/public/pages/CreateDetector/components/ConfigureAlerts/containers/ConfigureAlerts.tsx index 9cf0ddbc1..5f4ce02b8 100644 --- a/public/pages/CreateDetector/components/ConfigureAlerts/containers/ConfigureAlerts.tsx +++ b/public/pages/CreateDetector/components/ConfigureAlerts/containers/ConfigureAlerts.tsx @@ -130,9 +130,13 @@ export default class ConfigureAlerts extends Component this.onDelete(index)}> - Remove alert trigger - + triggers.length > 1 ? ( + this.onDelete(index)}> + Remove alert trigger + + ) : ( + <> + ) } > From 16d9d4614a887ddee761e1d31ecf02b282259f02 Mon Sep 17 00:00:00 2001 From: Jovan Cvetkovic Date: Fri, 20 Jan 2023 14:38:11 +0100 Subject: [PATCH 2/9] [BUG] Create detector | Interval field can be empty #378 Signed-off-by: Jovan Cvetkovic --- .../components/DetectorSchedule/Interval.tsx | 22 +++++++++++++++++-- .../containers/DefineDetector.tsx | 3 ++- 2 files changed, 22 insertions(+), 3 deletions(-) diff --git a/public/pages/CreateDetector/components/DefineDetector/components/DetectorSchedule/Interval.tsx b/public/pages/CreateDetector/components/DefineDetector/components/DetectorSchedule/Interval.tsx index 1c9b2530d..171b32711 100644 --- a/public/pages/CreateDetector/components/DefineDetector/components/DetectorSchedule/Interval.tsx +++ b/public/pages/CreateDetector/components/DefineDetector/components/DetectorSchedule/Interval.tsx @@ -20,14 +20,25 @@ export interface IntervalProps { onDetectorScheduleChange(schedule: PeriodSchedule): void; } +export interface IntervalState { + isIntervalValid: boolean; +} + const unitOptions: EuiSelectOption[] = [ { value: 'MINUTES', text: 'Minutes' }, { value: 'HOURS', text: 'Hours' }, { value: 'DAYS', text: 'Days' }, ]; -export class Interval extends React.Component { +export class Interval extends React.Component { + state = { + isIntervalValid: true, + }; + onTimeIntervalChange = (event: React.ChangeEvent) => { + this.setState({ + isIntervalValid: !!event.target.value, + }); this.props.onDetectorScheduleChange({ period: { ...this.props.detector.schedule.period, @@ -46,9 +57,14 @@ export class Interval extends React.Component { }; render() { + const { isIntervalValid } = this.state; const { period } = this.props.detector.schedule; return ( - }> + } + isInvalid={!isIntervalValid} + error={'Enter schedule interval.'} + > { value={period.interval} onChange={this.onTimeIntervalChange} data-test-subj={'detector-schedule-number-select'} + required={true} + isInvalid={!isIntervalValid} /> diff --git a/public/pages/CreateDetector/components/DefineDetector/containers/DefineDetector.tsx b/public/pages/CreateDetector/components/DefineDetector/containers/DefineDetector.tsx index 36ad37b3e..858e68250 100644 --- a/public/pages/CreateDetector/components/DefineDetector/containers/DefineDetector.tsx +++ b/public/pages/CreateDetector/components/DefineDetector/containers/DefineDetector.tsx @@ -43,7 +43,8 @@ export default class DefineDetector extends Component= MIN_NUM_DATA_SOURCES; + detector.inputs[0].detector_input.indices.length >= MIN_NUM_DATA_SOURCES && + !!detector.schedule.period.interval; this.props.changeDetector(detector); this.props.updateDataValidState(DetectorCreationStep.DEFINE_DETECTOR, isDataValid); } From e06484f967afc7085b5424337905145a64833ad3 Mon Sep 17 00:00:00 2001 From: Jovan Cvetkovic Date: Tue, 24 Jan 2023 10:31:10 +0100 Subject: [PATCH 3/9] fix PR failed tests Signed-off-by: Jovan Cvetkovic --- cypress.json | 1 + cypress/integration/1_detectors.spec.js | 27 +++++++++++++++---------- cypress/integration/2_rules.spec.js | 6 ++++-- cypress/integration/3_alerts.spec.js | 3 +++ cypress/support/detectors.js | 3 +++ cypress/support/indexes.js | 9 ++++++++- cypress/support/rules.js | 6 +++++- cypress/support/typings.js | 2 +- 8 files changed, 41 insertions(+), 16 deletions(-) diff --git a/cypress.json b/cypress.json index 46e089817..ff5f1e803 100644 --- a/cypress.json +++ b/cypress.json @@ -2,6 +2,7 @@ "viewportHeight": 900, "viewportWidth": 1440, "defaultCommandTimeout": 20000, + "requestTimeout": 20000, "baseUrl": "http://localhost:5601", "env": { "opensearch_url": "localhost:9200", diff --git a/cypress/integration/1_detectors.spec.js b/cypress/integration/1_detectors.spec.js index 9cbb4cac7..4f5a80090 100644 --- a/cypress/integration/1_detectors.spec.js +++ b/cypress/integration/1_detectors.spec.js @@ -38,24 +38,29 @@ describe('Detectors', () => { before(() => { cy.cleanUpTests(); // Create test index - cy.createIndex(indexName, sample_index_settings); - cy.request('POST', '_plugins/_security_analytics/rules/_search?prePackaged=true', { - from: 0, - size: 5000, - query: { - nested: { - path: 'rule', - query: { bool: { must: [{ match: { 'rule.category': 'windows' } }] } }, - }, - }, - }); + cy.createIndex(indexName, sample_index_settings).then(() => + cy + .request('POST', '_plugins/_security_analytics/rules/_search?prePackaged=true', { + from: 0, + size: 5000, + query: { + nested: { + path: 'rule', + query: { bool: { must: [{ match: { 'rule.category': 'windows' } }] } }, + }, + }, + }) + .should('have.property', 'status', 200) + ); cy.contains(detectorName).should('not.exist'); }); beforeEach(() => { + cy.intercept('/detectors/_search').as('detectorsSearch'); // Visit Detectors page cy.visit(`${OPENSEARCH_DASHBOARDS_URL}/detectors`); + cy.wait('@detectorsSearch').should('have.property', 'state', 'Complete'); // Check that correct page is showing cy.waitForPageLoad('detectors', { diff --git a/cypress/integration/2_rules.spec.js b/cypress/integration/2_rules.spec.js index a29b0d6ee..e0d34deda 100644 --- a/cypress/integration/2_rules.spec.js +++ b/cypress/integration/2_rules.spec.js @@ -131,8 +131,10 @@ const checkRulesFlyout = () => { describe('Rules', () => { before(() => cy.cleanUpTests()); beforeEach(() => { + cy.intercept('/rules/_search').as('rulesSearch'); // Visit Rules page cy.visit(`${OPENSEARCH_DASHBOARDS_URL}/rules`); + cy.wait('@rulesSearch').should('have.property', 'state', 'Complete'); // Check that correct page is showing cy.waitForPageLoad('rules', { @@ -194,12 +196,12 @@ describe('Rules', () => { force: true, }); + cy.wait('@getRules'); + cy.waitForPageLoad('rules', { contains: 'Rules', }); - cy.wait('@getRules'); - checkRulesFlyout(); }); diff --git a/cypress/integration/3_alerts.spec.js b/cypress/integration/3_alerts.spec.js index 332328c82..17ca56530 100644 --- a/cypress/integration/3_alerts.spec.js +++ b/cypress/integration/3_alerts.spec.js @@ -84,7 +84,10 @@ describe('Alerts', () => { beforeEach(() => { // Visit Alerts table page + cy.intercept('/detectors/_search').as('detectorsSearch'); + // Visit Detectors page cy.visit(`${OPENSEARCH_DASHBOARDS_URL}/alerts`); + cy.wait('@detectorsSearch').should('have.property', 'state', 'Complete'); // Wait for page to load cy.waitForPageLoad('alerts', { diff --git a/cypress/support/detectors.js b/cypress/support/detectors.js index b0ec6c60f..22f53c45b 100644 --- a/cypress/support/detectors.js +++ b/cypress/support/detectors.js @@ -68,5 +68,8 @@ Cypress.Commands.add('deleteAllDetectors', () => { method: 'DELETE', url: `${Cypress.env('opensearch')}/.opensearch-sap-detectors-config`, failOnStatusCode: false, + }).as('deleteAllDetectors'); + cy.get('@deleteAllDetectors').should((response) => { + expect(response.status).to.be.oneOf([200, 404]); }); }); diff --git a/cypress/support/indexes.js b/cypress/support/indexes.js index 1ffc22116..ed041fe71 100644 --- a/cypress/support/indexes.js +++ b/cypress/support/indexes.js @@ -6,7 +6,11 @@ const { NODE_API } = require('./constants'); Cypress.Commands.add('createIndex', (index, settings = {}) => { - cy.request('PUT', `${Cypress.env('opensearch')}/${index}`, settings); + cy.request('PUT', `${Cypress.env('opensearch')}/${index}`, settings).should( + 'have.property', + 'status', + 200 + ); }); Cypress.Commands.add('createIndexTemplate', (name, template) => { @@ -43,5 +47,8 @@ Cypress.Commands.add('deleteAllIndices', () => { method: 'DELETE', url: `${Cypress.env('opensearch')}/index*,sample*,opensearch_dashboards*,test*,cypress*`, failOnStatusCode: false, + }).as('deleteAllIndices'); + cy.get('@deleteAllIndices').should((response) => { + expect(response.status).to.be.oneOf([200, 404]); }); }); diff --git a/cypress/support/rules.js b/cypress/support/rules.js index 46f01b3b7..8dada4fd1 100644 --- a/cypress/support/rules.js +++ b/cypress/support/rules.js @@ -51,10 +51,14 @@ Cypress.Commands.add('deleteRule', (ruleName) => { }); Cypress.Commands.add('deleteAllCustomRules', () => { + const url = `${Cypress.env('opensearch')}/.opensearch-sap-custom-rules-config`; cy.request({ method: 'DELETE', - url: `${Cypress.env('opensearch')}/.opensearch-sap-custom-rules-config`, + url: url, failOnStatusCode: false, body: { query: { match_all: {} } }, + }).as('deleteAllCustomRules'); + cy.get('@deleteAllCustomRules').should((response) => { + expect(response.status).to.be.oneOf([200, 404]); }); }); diff --git a/cypress/support/typings.js b/cypress/support/typings.js index 4345d7e0b..347c47c78 100644 --- a/cypress/support/typings.js +++ b/cypress/support/typings.js @@ -9,7 +9,7 @@ Cypress.Commands.add( prevSubject: true, }, (subject, text) => { - return cy.get(subject).ospType(text).realPress('Enter'); + return cy.get(subject).clear().ospType(text).realPress('Enter'); } ); From bc088d22e4ad62f32bb46ed0d31f841fef4a01ee Mon Sep 17 00:00:00 2001 From: Jovan Cvetkovic Date: Tue, 24 Jan 2023 11:32:26 +0100 Subject: [PATCH 4/9] fix PR failed tests Signed-off-by: Jovan Cvetkovic --- cypress/support/indexes.js | 1 + 1 file changed, 1 insertion(+) diff --git a/cypress/support/indexes.js b/cypress/support/indexes.js index ed041fe71..1e9f5b498 100644 --- a/cypress/support/indexes.js +++ b/cypress/support/indexes.js @@ -49,6 +49,7 @@ Cypress.Commands.add('deleteAllIndices', () => { failOnStatusCode: false, }).as('deleteAllIndices'); cy.get('@deleteAllIndices').should((response) => { + // Both statuses are a pass, 200 means deleted successfully and 404 there was no index to delete expect(response.status).to.be.oneOf([200, 404]); }); }); From 7dd83f7c97284be8bc93a6df3d25c97ec4e9f1e3 Mon Sep 17 00:00:00 2001 From: Jovan Cvetkovic Date: Tue, 24 Jan 2023 12:37:30 +0100 Subject: [PATCH 5/9] unit tests Signed-off-by: Jovan Cvetkovic --- cypress.json | 5 +++-- 1 file changed, 3 insertions(+), 2 deletions(-) diff --git a/cypress.json b/cypress.json index ff5f1e803..457d0faee 100644 --- a/cypress.json +++ b/cypress.json @@ -1,8 +1,9 @@ { "viewportHeight": 900, "viewportWidth": 1440, - "defaultCommandTimeout": 20000, - "requestTimeout": 20000, + "defaultCommandTimeout": 30000, + "requestTimeout": 60000, + "responseTimeout": 120000, "baseUrl": "http://localhost:5601", "env": { "opensearch_url": "localhost:9200", From 20c505576985128aa952affc149bc08dcad741e0 Mon Sep 17 00:00:00 2001 From: Jovan Cvetkovic Date: Tue, 24 Jan 2023 14:06:15 +0100 Subject: [PATCH 6/9] unit tests Signed-off-by: Jovan Cvetkovic --- cypress.json | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/cypress.json b/cypress.json index 457d0faee..48f248ec8 100644 --- a/cypress.json +++ b/cypress.json @@ -2,8 +2,8 @@ "viewportHeight": 900, "viewportWidth": 1440, "defaultCommandTimeout": 30000, - "requestTimeout": 60000, - "responseTimeout": 120000, + "requestTimeout": 300000, + "responseTimeout": 300000, "baseUrl": "http://localhost:5601", "env": { "opensearch_url": "localhost:9200", From f58ec30c80ab0e6d6df75eebba1d23400464fa8a Mon Sep 17 00:00:00 2001 From: Jovan Cvetkovic Date: Tue, 24 Jan 2023 21:54:54 +0100 Subject: [PATCH 7/9] testing github-action v5 Signed-off-by: Jovan Cvetkovic --- .github/workflows/cypress-workflow.yml | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/.github/workflows/cypress-workflow.yml b/.github/workflows/cypress-workflow.yml index 10ea8e774..c7a09169b 100644 --- a/.github/workflows/cypress-workflow.yml +++ b/.github/workflows/cypress-workflow.yml @@ -127,7 +127,7 @@ jobs: # for now just chrome, use matrix to do all browsers later - name: Cypress tests - uses: cypress-io/github-action@v2 + uses: cypress-io/github-action@v5 with: working-directory: OpenSearch-Dashboards/plugins/security-analytics-dashboards-plugin command: yarn run cypress run From 9a304fa9cd2ce573026d3f07f85528087e2f0e0e Mon Sep 17 00:00:00 2001 From: Jovan Cvetkovic Date: Tue, 24 Jan 2023 22:54:32 +0100 Subject: [PATCH 8/9] testing github-action v5 Signed-off-by: Jovan Cvetkovic --- .github/workflows/cypress-workflow.yml | 1 + 1 file changed, 1 insertion(+) diff --git a/.github/workflows/cypress-workflow.yml b/.github/workflows/cypress-workflow.yml index c7a09169b..860d8d208 100644 --- a/.github/workflows/cypress-workflow.yml +++ b/.github/workflows/cypress-workflow.yml @@ -133,6 +133,7 @@ jobs: command: yarn run cypress run wait-on: 'http://localhost:5601' browser: chrome + video: false env: CYPRESS_CACHE_FOLDER: ${{ matrix.cypress_cache_folder }} From 66ed328aa555dea2e1d8e78648f566b88cf8585a Mon Sep 17 00:00:00 2001 From: Jovan Cvetkovic Date: Wed, 25 Jan 2023 08:19:46 +0100 Subject: [PATCH 9/9] testing github-action v5 Signed-off-by: Jovan Cvetkovic --- .github/workflows/cypress-workflow.yml | 3 ++- 1 file changed, 2 insertions(+), 1 deletion(-) diff --git a/.github/workflows/cypress-workflow.yml b/.github/workflows/cypress-workflow.yml index 860d8d208..edf76c580 100644 --- a/.github/workflows/cypress-workflow.yml +++ b/.github/workflows/cypress-workflow.yml @@ -124,6 +124,8 @@ jobs: with: path: ${{ matrix.cypress_cache_folder }} key: cypress-cache-v2-${{ runner.os }}-${{ hashFiles('**/package.json') }} + restore-keys: | + cypress-cache-v2-${{ runner.os }}-${{ hashFiles('**/package.json') }} # for now just chrome, use matrix to do all browsers later - name: Cypress tests @@ -133,7 +135,6 @@ jobs: command: yarn run cypress run wait-on: 'http://localhost:5601' browser: chrome - video: false env: CYPRESS_CACHE_FOLDER: ${{ matrix.cypress_cache_folder }}