Skip to content

Commit

Permalink
[ML] Adds anomaly detectors forecast API test (#125214)
Browse files Browse the repository at this point in the history
* [ML] Adds anomaly detectors forecast API test

* [ML] Add check that forecast results exist
  • Loading branch information
peteharverson authored Feb 10, 2022
1 parent 0cdc636 commit 2c7d96b
Show file tree
Hide file tree
Showing 3 changed files with 143 additions and 0 deletions.
Original file line number Diff line number Diff line change
@@ -0,0 +1,101 @@
/*
* Copyright Elasticsearch B.V. and/or licensed to Elasticsearch B.V. under one
* or more contributor license agreements. Licensed under the Elastic License
* 2.0; you may not use this file except in compliance with the Elastic License
* 2.0.
*/

import { FtrProviderContext } from '../../../ftr_provider_context';
import { USER } from '../../../../functional/services/ml/security_common';
import { JOB_STATE, DATAFEED_STATE } from '../../../../../plugins/ml/common/constants/states';
import { COMMON_REQUEST_HEADERS } from '../../../../functional/services/ml/common_api';

export default ({ getService }: FtrProviderContext) => {
const esArchiver = getService('esArchiver');
const supertest = getService('supertestWithoutAuth');
const ml = getService('ml');
const spacesService = getService('spaces');

const forecastJobId = 'fq_single_forecast';
const forecastJobDatafeedId = `datafeed-${forecastJobId}`;
const idSpace1 = 'space1';
const idSpace2 = 'space2';

async function runForecast(
jobId: string,
space: string,
duration: string | number,
user: USER,
expectedStatusCode: number
) {
const { body } = await supertest
.post(`${space ? `/s/${space}` : ''}/api/ml/anomaly_detectors/${jobId}/_forecast`)
.auth(user, ml.securityCommon.getPasswordForUser(user))
.set(COMMON_REQUEST_HEADERS)
.send({ duration })
.expect(expectedStatusCode);

return body;
}

describe('POST anomaly_detectors _forecast with spaces', function () {
before(async () => {
await esArchiver.loadIfNeeded('x-pack/test/functional/es_archives/ml/farequote');
await ml.testResources.setKibanaTimeZoneToUTC();

await spacesService.create({ id: idSpace1, name: 'space_one', disabledFeatures: [] });
await spacesService.create({ id: idSpace2, name: 'space_two', disabledFeatures: [] });

const jobConfig = ml.commonConfig.getADFqSingleMetricJobConfig(forecastJobId);
const datafeedConfig = ml.commonConfig.getADFqDatafeedConfig(forecastJobId);

await ml.api.createAnomalyDetectionJob(jobConfig, idSpace1);
await ml.api.createDatafeed(datafeedConfig, idSpace1);
});

after(async () => {
await ml.api.closeAnomalyDetectionJob(forecastJobId);
await spacesService.delete(idSpace1);
await spacesService.delete(idSpace2);
await ml.api.cleanMlIndices();
await ml.testResources.cleanMLSavedObjects();
});

it('should not forecast for a job that has not been opened', async () => {
await runForecast(forecastJobId, idSpace1, '1d', USER.ML_POWERUSER, 409);
});

it('should not forecast for a job that has not been run', async () => {
await ml.api.openAnomalyDetectionJob(forecastJobId);
await runForecast(forecastJobId, idSpace1, '1d', USER.ML_POWERUSER, 400);
});

it('should not forecast for a job with an invalid job ID', async () => {
await runForecast(`${forecastJobId}_invalid`, idSpace1, '1d', USER.ML_POWERUSER, 404);
});

it('should run forecast for open job with valid job ID', async () => {
await ml.api.startDatafeed(forecastJobDatafeedId, { start: '0', end: `${Date.now()}` });
await ml.api.waitForDatafeedState(forecastJobDatafeedId, DATAFEED_STATE.STOPPED);
await ml.api.waitForJobState(forecastJobId, JOB_STATE.CLOSED);
await ml.api.openAnomalyDetectionJob(forecastJobId);
await runForecast(forecastJobId, idSpace1, '1d', USER.ML_POWERUSER, 200);
await ml.testExecution.logTestStep(
`forecast results should exist for job '${forecastJobId}'`
);
await ml.api.assertForecastResultsExist(forecastJobId);
});

it('should not run forecast for open job with invalid duration', async () => {
await runForecast(forecastJobId, idSpace1, 3600000, USER.ML_POWERUSER, 400);
});

it('should not run forecast for open job with valid job ID as ML Viewer', async () => {
await runForecast(forecastJobId, idSpace1, '1d', USER.ML_VIEWER, 403);
});

it('should not run forecast for open job with valid job ID in wrong space', async () => {
await runForecast(forecastJobId, idSpace2, '1d', USER.ML_POWERUSER, 404);
});
});
};
Original file line number Diff line number Diff line change
Expand Up @@ -17,5 +17,6 @@ export default function ({ loadTestFile }: FtrProviderContext) {
loadTestFile(require.resolve('./close_with_spaces'));
loadTestFile(require.resolve('./delete_with_spaces'));
loadTestFile(require.resolve('./create_with_spaces'));
loadTestFile(require.resolve('./forecast_with_spaces'));
});
}
41 changes: 41 additions & 0 deletions x-pack/test/functional/services/ml/api.ts
Original file line number Diff line number Diff line change
Expand Up @@ -126,6 +126,47 @@ export function MachineLearningAPIProvider({ getService }: FtrProviderContext) {
);
},

async hasForecastResults(jobId: string): Promise<boolean> {
const body = await es.search({
index: '.ml-anomalies-*',
body: {
size: 1,
query: {
bool: {
must: [
{
match: {
job_id: jobId,
},
},
{
match: {
result_type: 'model_forecast',
},
},
],
},
},
},
});

return body.hits.hits.length > 0;
},

async assertForecastResultsExist(jobId: string) {
await retry.waitForWithTimeout(
`forecast results for job ${jobId} to exist`,
30 * 1000,
async () => {
if ((await this.hasForecastResults(jobId)) === true) {
return true;
} else {
throw new Error(`expected forecast results for job '${jobId}' to exist`);
}
}
);
},

async createIndex(
indices: string,
mappings?: Record<string, estypes.MappingTypeMapping> | estypes.MappingTypeMapping
Expand Down

0 comments on commit 2c7d96b

Please sign in to comment.