Skip to content
This repository has been archived by the owner on Aug 2, 2022. It is now read-only.

Commit

Permalink
add IT cases for filtering out non-server exceptions for HC detector (#…
Browse files Browse the repository at this point in the history
  • Loading branch information
ylwu-amzn authored Dec 29, 2020
1 parent 7df38a1 commit bb79a0f
Show file tree
Hide file tree
Showing 3 changed files with 125 additions and 18 deletions.
Original file line number Diff line number Diff line change
Expand Up @@ -629,15 +629,14 @@ void handleExecuteException(Exception ex, ActionListener<AnomalyResultResponse>
}

private boolean invalidQuery(SearchPhaseExecutionException ex) {
boolean invalidQuery = true;
// If all shards return bad request and failure cause is IllegalArgumentException, we
// consider the feature query is invalid and will not count the error in failure stats.
for (ShardSearchFailure failure : ex.shardFailures()) {
if (RestStatus.BAD_REQUEST != failure.status() || !(failure.getCause() instanceof IllegalArgumentException)) {
invalidQuery = false;
return false;
}
}
return invalidQuery;
return true;
}

class RCFActionListener implements ActionListener<RCFResultResponse> {
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -403,6 +403,10 @@ public static QueryBuilder randomQuery() throws IOException {
String query = "{\"bool\":{\"must\":{\"term\":{\"user\":\"kimchy\"}},\"filter\":{\"term\":{\"tag\":"
+ "\"tech\"}},\"must_not\":{\"range\":{\"age\":{\"gte\":10,\"lte\":20}}},\"should\":[{\"term\":"
+ "{\"tag\":\"wow\"}},{\"term\":{\"tag\":\"elasticsearch\"}}],\"minimum_should_match\":1,\"boost\":1}}";
return randomQuery(query);
}

public static QueryBuilder randomQuery(String query) throws IOException {
XContentParser parser = TestHelpers.parser(query);
return parseInnerQueryBuilder(parser);
}
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -15,22 +15,26 @@

package com.amazon.opendistroforelasticsearch.ad.transport;

import static com.amazon.opendistroforelasticsearch.ad.TestHelpers.randomQuery;

import java.io.IOException;
import java.time.Instant;
import java.time.temporal.ChronoUnit;
import java.util.List;
import java.util.Map;

import org.elasticsearch.action.get.GetResponse;
import org.elasticsearch.common.io.stream.NotSerializableExceptionWrapper;
import org.elasticsearch.search.aggregations.AggregationBuilder;
import org.elasticsearch.test.rest.ESRestTestCase;
import org.junit.Before;

import com.amazon.opendistroforelasticsearch.ad.ADIntegTestCase;
import com.amazon.opendistroforelasticsearch.ad.TestHelpers;
import com.amazon.opendistroforelasticsearch.ad.common.exception.AnomalyDetectionException;
import com.amazon.opendistroforelasticsearch.ad.model.AnomalyDetector;
import com.amazon.opendistroforelasticsearch.ad.model.AnomalyDetectorType;
import com.amazon.opendistroforelasticsearch.ad.model.Feature;
import com.amazon.opendistroforelasticsearch.ad.model.IntervalTimeConfiguration;
import com.google.common.collect.ImmutableList;
import com.google.common.collect.ImmutableMap;

Expand All @@ -39,6 +43,8 @@ public class AnomalyResultTransportActionTests extends ADIntegTestCase {
private Instant testDataTimeStamp;
private long start;
private long end;
private String timeField = "timestamp";
private String categoryField = "type";

@Override
@Before
Expand All @@ -52,8 +58,12 @@ public void setUp() throws Exception {
}

private void ingestTestData() throws IOException {
String mappings = "{\"properties\":{\"timestamp\":{\"type\":\"date\",\"format\":\"strict_date_time||epoch_millis\"},"
+ "\"value\":{\"type\":\"double\"}, \"type\":{\"type\":\"keyword\"},"
String mappings = "{\"properties\":{\""
+ timeField
+ "\":{\"type\":\"date\",\"format\":\"strict_date_time||epoch_millis\"},"
+ "\"value\":{\"type\":\"double\"}, \""
+ categoryField
+ "\":{\"type\":\"keyword\"},"
+ "\"is_error\":{\"type\":\"boolean\"}, \"message\":{\"type\":\"text\"}}}";
createIndex(testIndex, mappings);
double value = randomDouble();
Expand All @@ -63,11 +73,11 @@ private void ingestTestData() throws IOException {
String id = indexDoc(
testIndex,
ImmutableMap
.of("timestamp", testDataTimeStamp.toEpochMilli(), "value", value, "type", type, "is_error", isError, "message", message)
.of(timeField, testDataTimeStamp.toEpochMilli(), "value", value, "type", type, "is_error", isError, "message", message)
);
GetResponse doc = getDoc(testIndex, id);
Map<String, Object> sourceAsMap = doc.getSourceAsMap();
assertEquals(testDataTimeStamp.toEpochMilli(), sourceAsMap.get("timestamp"));
assertEquals(testDataTimeStamp.toEpochMilli(), sourceAsMap.get(timeField));
assertEquals(value, sourceAsMap.get("value"));
assertEquals(type, sourceAsMap.get("type"));
assertEquals(isError, sourceAsMap.get("is_error"));
Expand Down Expand Up @@ -130,23 +140,117 @@ public void testFeatureWithCardinalityOfTextField() throws IOException {
assertErrorMessage(adId, "Text fields are not optimised for operations");
}

public void testFeatureQueryWithTermsAggregationForHCDetector() throws IOException {
String adId = createDetectorWithFeatureAgg("{\"test\":{\"terms\":{\"field\":\"type\"}}}", true);
assertErrorMessage(adId, "Failed to parse aggregation");
}

public void testFeatureWithSumOfTextFieldForHCDetector() throws IOException {
String adId = createDetectorWithFeatureAgg("{\"test\":{\"sum\":{\"field\":\"message\"}}}", true);
assertErrorMessage(adId, "Text fields are not optimised for operations");
}

public void testFeatureWithSumOfTypeFieldForHCDetector() throws IOException {
String adId = createDetectorWithFeatureAgg("{\"test\":{\"sum\":{\"field\":\"type\"}}}", true);
assertErrorMessage(adId, "Field [type] of type [keyword] is not supported for aggregation [sum]");
}

public void testFeatureWithMaxOfTextFieldForHCDetector() throws IOException {
String adId = createDetectorWithFeatureAgg("{\"test\":{\"max\":{\"field\":\"message\"}}}", true);
assertErrorMessage(adId, "Text fields are not optimised for operations");
}

public void testFeatureWithMaxOfTypeFieldForHCDetector() throws IOException {
String adId = createDetectorWithFeatureAgg("{\"test\":{\"max\":{\"field\":\"type\"}}}", true);
assertErrorMessage(adId, "Field [type] of type [keyword] is not supported for aggregation [max]");
}

public void testFeatureWithMinOfTextFieldForHCDetector() throws IOException {
String adId = createDetectorWithFeatureAgg("{\"test\":{\"min\":{\"field\":\"message\"}}}", true);
assertErrorMessage(adId, "Text fields are not optimised for operations");
}

public void testFeatureWithMinOfTypeFieldForHCDetector() throws IOException {
String adId = createDetectorWithFeatureAgg("{\"test\":{\"min\":{\"field\":\"type\"}}}", true);
assertErrorMessage(adId, "Field [type] of type [keyword] is not supported for aggregation [min]");
}

public void testFeatureWithAvgOfTextFieldForHCDetector() throws IOException {
String adId = createDetectorWithFeatureAgg("{\"test\":{\"avg\":{\"field\":\"message\"}}}", true);
assertErrorMessage(adId, "Text fields are not optimised for operations");
}

public void testFeatureWithAvgOfTypeFieldForHCDetector() throws IOException {
String adId = createDetectorWithFeatureAgg("{\"test\":{\"avg\":{\"field\":\"type\"}}}", true);
assertErrorMessage(adId, "Field [type] of type [keyword] is not supported for aggregation [avg]");
}

public void testFeatureWithCountOfTextFieldForHCDetector() throws IOException {
String adId = createDetectorWithFeatureAgg("{\"test\":{\"value_count\":{\"field\":\"message\"}}}", true);
assertErrorMessage(adId, "Text fields are not optimised for operations");
}

public void testFeatureWithCardinalityOfTextFieldForHCDetector() throws IOException {
String adId = createDetectorWithFeatureAgg("{\"test\":{\"cardinality\":{\"field\":\"message\"}}}", true);
assertErrorMessage(adId, "Text fields are not optimised for operations");
}

private String createDetectorWithFeatureAgg(String aggQuery) throws IOException {
return createDetectorWithFeatureAgg(aggQuery, false);
}

private String createDetectorWithFeatureAgg(String aggQuery, boolean hcDetector) throws IOException {
AggregationBuilder aggregationBuilder = TestHelpers.parseAggregation(aggQuery);
Feature feature = new Feature(randomAlphaOfLength(5), randomAlphaOfLength(10), true, aggregationBuilder);
AnomalyDetector detector = TestHelpers
.randomAnomalyDetector(
ImmutableList.of(testIndex),
ImmutableList.of(feature),
ImmutableMap.of(),
Instant.now(),
AnomalyDetectorType.REALTIME_SINGLE_ENTITY.name(),
null,
false
);
AnomalyDetector detector = hcDetector
? randomHCDetector(ImmutableList.of(testIndex), ImmutableList.of(feature))
: randomDetector(ImmutableList.of(testIndex), ImmutableList.of(feature));
String adId = createDetectors(detector);
return adId;
}

private AnomalyDetector randomDetector(List<String> indices, List<Feature> features) throws IOException {
return new AnomalyDetector(
randomAlphaOfLength(10),
randomLong(),
randomAlphaOfLength(20),
randomAlphaOfLength(30),
timeField,
indices,
features,
randomQuery("{\"bool\":{\"filter\":[{\"exists\":{\"field\":\"value\"}}]}}"),
new IntervalTimeConfiguration(ESRestTestCase.randomLongBetween(1, 5), ChronoUnit.MINUTES),
new IntervalTimeConfiguration(ESRestTestCase.randomLongBetween(1, 5), ChronoUnit.MINUTES),
8,
null,
randomInt(),
Instant.now(),
null,
null
);
}

private AnomalyDetector randomHCDetector(List<String> indices, List<Feature> features) throws IOException {
return new AnomalyDetector(
randomAlphaOfLength(10),
randomLong(),
randomAlphaOfLength(20),
randomAlphaOfLength(30),
timeField,
indices,
features,
randomQuery("{\"bool\":{\"filter\":[{\"exists\":{\"field\":\"value\"}}]}}"),
new IntervalTimeConfiguration(ESRestTestCase.randomLongBetween(1, 5), ChronoUnit.MINUTES),
new IntervalTimeConfiguration(ESRestTestCase.randomLongBetween(1, 5), ChronoUnit.MINUTES),
8,
null,
randomInt(),
Instant.now(),
ImmutableList.of(categoryField),
null
);
}

private void assertErrorMessage(String adId, String errorMessage) {
AnomalyResultRequest resultRequest = new AnomalyResultRequest(adId, start, end);
RuntimeException e = expectThrowsAnyOf(
Expand Down

0 comments on commit bb79a0f

Please sign in to comment.