From 858c614a444bd195e42f33fdc2a22b12b675819f Mon Sep 17 00:00:00 2001 From: Daniel Mitterdorfer Date: Mon, 12 Feb 2024 10:26:17 +0100 Subject: [PATCH] [Profiling] Allow wildcards for custom indices (#105372) Several profiling APIs allow to specify custom event indices with the `indices` parameter. So far this parameter has allowed only one index and also disallowed wildcards. With this commit we extend this parameter to support multiple indices and each index can use wildcards. * Allow to distinguish whether indices are user-provided As we set indices now internally in all cases due to wildcard resolution, we need a means to distinguish whether `indices` have been provided originally via the API or whether Elasticsearch has set them internally after wildcard resolution. --- .../profiling/GetStackTracesActionIT.java | 6 +- .../profiling/GetStackTracesRequest.java | 89 ++++++++++++++----- .../TransportGetStackTracesAction.java | 6 +- .../profiling/GetStackTracesRequestTests.java | 82 +++++++++++++++-- .../rest-api-spec/test/profiling/10_basic.yml | 2 +- 5 files changed, 149 insertions(+), 36 deletions(-) diff --git a/x-pack/plugin/profiling/src/internalClusterTest/java/org/elasticsearch/xpack/profiling/GetStackTracesActionIT.java b/x-pack/plugin/profiling/src/internalClusterTest/java/org/elasticsearch/xpack/profiling/GetStackTracesActionIT.java index 0ed2105fdde49..d42bc59e7dbb5 100644 --- a/x-pack/plugin/profiling/src/internalClusterTest/java/org/elasticsearch/xpack/profiling/GetStackTracesActionIT.java +++ b/x-pack/plugin/profiling/src/internalClusterTest/java/org/elasticsearch/xpack/profiling/GetStackTracesActionIT.java @@ -55,7 +55,7 @@ public void testGetStackTracesFromAPMWithMatchNoDownsampling() throws Exception 1.0d, 1.0d, query, - "apm-test-*", + new String[] { "apm-test-*" }, "transaction.profiler_stack_trace_ids", null, null, @@ -99,7 +99,7 @@ public void testGetStackTracesFromAPMWithMatchAndDownsampling() throws Exception 1.0d, 1.0d, query, - "apm-test-*", + new String[] { "apm-test-*" }, "transaction.profiler_stack_trace_ids", null, null, @@ -165,7 +165,7 @@ public void testGetStackTracesFromAPMNoMatch() throws Exception { 1.0d, 1.0d, query, - "apm-test-*", + new String[] { "apm-test-*" }, "transaction.profiler_stack_trace_ids", null, null, diff --git a/x-pack/plugin/profiling/src/main/java/org/elasticsearch/xpack/profiling/GetStackTracesRequest.java b/x-pack/plugin/profiling/src/main/java/org/elasticsearch/xpack/profiling/GetStackTracesRequest.java index 785ed525287da..3dfe48744cb97 100644 --- a/x-pack/plugin/profiling/src/main/java/org/elasticsearch/xpack/profiling/GetStackTracesRequest.java +++ b/x-pack/plugin/profiling/src/main/java/org/elasticsearch/xpack/profiling/GetStackTracesRequest.java @@ -21,7 +21,10 @@ import org.elasticsearch.xcontent.XContentParser; import java.io.IOException; +import java.util.ArrayList; +import java.util.Arrays; import java.util.HashSet; +import java.util.List; import java.util.Map; import java.util.Objects; import java.util.Set; @@ -32,7 +35,7 @@ /** * A request to get profiling details */ -public class GetStackTracesRequest extends ActionRequest implements IndicesRequest { +public class GetStackTracesRequest extends ActionRequest implements IndicesRequest.Replaceable { public static final ParseField QUERY_FIELD = new ParseField("query"); public static final ParseField SAMPLE_SIZE_FIELD = new ParseField("sample_size"); public static final ParseField INDICES_FIELD = new ParseField("indices"); @@ -49,7 +52,8 @@ public class GetStackTracesRequest extends ActionRequest implements IndicesReque private QueryBuilder query; private int sampleSize; - private String indices; + private String[] indices; + private boolean userProvidedIndices; private String stackTraceIdsField; private Double requestedDuration; private Double awsCostFactor; @@ -75,7 +79,7 @@ public GetStackTracesRequest( Double awsCostFactor, Double azureCostFactor, QueryBuilder query, - String indices, + String[] indices, String stackTraceIdsField, Double customCO2PerKWH, Double customDatacenterPUE, @@ -89,6 +93,7 @@ public GetStackTracesRequest( this.azureCostFactor = azureCostFactor; this.query = query; this.indices = indices; + this.userProvidedIndices = indices != null && indices.length > 0; this.stackTraceIdsField = stackTraceIdsField; this.customCO2PerKWH = customCO2PerKWH; this.customDatacenterPUE = customDatacenterPUE; @@ -142,10 +147,14 @@ public QueryBuilder getQuery() { return query; } - public String getIndices() { + public String[] getIndices() { return indices; } + public boolean isUserProvidedIndices() { + return userProvidedIndices; + } + public String getStackTraceIdsField() { return stackTraceIdsField; } @@ -164,8 +173,7 @@ public void parseXContent(XContentParser parser) throws IOException { if (token != XContentParser.Token.START_OBJECT && (token = parser.nextToken()) != XContentParser.Token.START_OBJECT) { throw new ParsingException( parser.getTokenLocation(), - "Expected [" + XContentParser.Token.START_OBJECT + "] but found [" + token + "]", - parser.getTokenLocation() + "Expected [" + XContentParser.Token.START_OBJECT + "] but found [" + token + "]." ); } @@ -175,8 +183,6 @@ public void parseXContent(XContentParser parser) throws IOException { } else if (token.isValue()) { if (SAMPLE_SIZE_FIELD.match(currentFieldName, parser.getDeprecationHandler())) { this.sampleSize = parser.intValue(); - } else if (INDICES_FIELD.match(currentFieldName, parser.getDeprecationHandler())) { - this.indices = parser.text(); } else if (STACKTRACE_IDS_FIELD.match(currentFieldName, parser.getDeprecationHandler())) { this.stackTraceIdsField = parser.text(); } else if (REQUESTED_DURATION_FIELD.match(currentFieldName, parser.getDeprecationHandler())) { @@ -196,22 +202,21 @@ public void parseXContent(XContentParser parser) throws IOException { } else if (CUSTOM_COST_PER_CORE_HOUR.match(currentFieldName, parser.getDeprecationHandler())) { this.customCostPerCoreHour = parser.doubleValue(); } else { - throw new ParsingException( - parser.getTokenLocation(), - "Unknown key for a " + token + " in [" + currentFieldName + "].", - parser.getTokenLocation() - ); + throw new ParsingException(parser.getTokenLocation(), "Unknown key for a " + token + " in [" + currentFieldName + "]."); } } else if (token == XContentParser.Token.START_OBJECT) { if (QUERY_FIELD.match(currentFieldName, parser.getDeprecationHandler())) { this.query = parseTopLevelQuery(parser); } + } else if (token == XContentParser.Token.START_ARRAY) { + if (INDICES_FIELD.match(currentFieldName, parser.getDeprecationHandler())) { + this.indices = parseIndices(parser); + this.userProvidedIndices = true; + } else { + throw new ParsingException(parser.getTokenLocation(), "Unexpected token " + token + " in [" + currentFieldName + "]."); + } } else { - throw new ParsingException( - parser.getTokenLocation(), - "Unknown key for a " + token + " in [" + currentFieldName + "].", - parser.getTokenLocation() - ); + throw new ParsingException(parser.getTokenLocation(), "Unknown key for a " + token + " in [" + currentFieldName + "]."); } } @@ -221,10 +226,32 @@ public void parseXContent(XContentParser parser) throws IOException { } } + private String[] parseIndices(XContentParser parser) throws IOException { + XContentParser.Token token; + List indices = new ArrayList<>(); + while ((token = parser.nextToken()) != XContentParser.Token.END_ARRAY) { + if (token == XContentParser.Token.VALUE_STRING) { + indices.add(parser.text()); + } else { + throw new ParsingException( + parser.getTokenLocation(), + "Expected [" + + XContentParser.Token.VALUE_STRING + + "] but found [" + + token + + "] in [" + + INDICES_FIELD.getPreferredName() + + "]." + ); + } + } + return indices.toArray(new String[0]); + } + @Override public ActionRequestValidationException validate() { ActionRequestValidationException validationException = null; - if (indices != null) { + if (userProvidedIndices) { if (stackTraceIdsField == null || stackTraceIdsField.isEmpty()) { validationException = addValidationError( "[" + STACKTRACE_IDS_FIELD.getPreferredName() + "] is mandatory", @@ -306,7 +333,7 @@ public boolean equals(Object o) { GetStackTracesRequest that = (GetStackTracesRequest) o; return Objects.equals(query, that.query) && Objects.equals(sampleSize, that.sampleSize) - && Objects.equals(indices, that.indices) + && Arrays.equals(indices, that.indices) && Objects.equals(stackTraceIdsField, that.stackTraceIdsField); } @@ -318,7 +345,7 @@ public int hashCode() { // Resampler to produce a consistent downsampling results, relying on the default hashCode implementation of `query` will // produce consistent results per node but not across the cluster. To avoid this, we produce the hashCode based on the // string representation instead, which will produce consistent results for the entire cluster and across node restarts. - return Objects.hash(Objects.toString(query, "null"), sampleSize, indices, stackTraceIdsField); + return Objects.hash(Objects.toString(query, "null"), sampleSize, Arrays.hashCode(indices), stackTraceIdsField); } @Override @@ -327,10 +354,10 @@ public String[] indices() { indices.add("profiling-stacktraces"); indices.add("profiling-stackframes"); indices.add("profiling-executables"); - if (this.indices == null) { - indices.addAll(EventsIndex.indexNames()); + if (userProvidedIndices) { + indices.addAll(List.of(this.indices)); } else { - indices.add(this.indices); + indices.addAll(EventsIndex.indexNames()); } return indices.toArray(new String[0]); } @@ -344,4 +371,18 @@ public IndicesOptions indicesOptions() { public boolean includeDataStreams() { return true; } + + @Override + public IndicesRequest indices(String... indices) { + validateIndices(indices); + this.indices = indices; + return null; + } + + private static void validateIndices(String... indices) { + Objects.requireNonNull(indices, "indices must not be null"); + for (String index : indices) { + Objects.requireNonNull(index, "index must not be null"); + } + } } diff --git a/x-pack/plugin/profiling/src/main/java/org/elasticsearch/xpack/profiling/TransportGetStackTracesAction.java b/x-pack/plugin/profiling/src/main/java/org/elasticsearch/xpack/profiling/TransportGetStackTracesAction.java index d2cee7825eb21..d49f42ce69737 100644 --- a/x-pack/plugin/profiling/src/main/java/org/elasticsearch/xpack/profiling/TransportGetStackTracesAction.java +++ b/x-pack/plugin/profiling/src/main/java/org/elasticsearch/xpack/profiling/TransportGetStackTracesAction.java @@ -148,10 +148,10 @@ protected void doExecute(Task submitTask, GetStackTracesRequest request, ActionL licenseChecker.requireSupportedLicense(); GetStackTracesResponseBuilder responseBuilder = new GetStackTracesResponseBuilder(request); Client client = new ParentTaskAssigningClient(this.nodeClient, transportService.getLocalNode(), submitTask); - if (request.getIndices() == null) { - searchProfilingEvents(submitTask, client, request, submitListener, responseBuilder); - } else { + if (request.isUserProvidedIndices()) { searchGenericEvents(submitTask, client, request, submitListener, responseBuilder); + } else { + searchProfilingEvents(submitTask, client, request, submitListener, responseBuilder); } } diff --git a/x-pack/plugin/profiling/src/test/java/org/elasticsearch/xpack/profiling/GetStackTracesRequestTests.java b/x-pack/plugin/profiling/src/test/java/org/elasticsearch/xpack/profiling/GetStackTracesRequestTests.java index bdd2d9868363c..a6fd6f39d88a2 100644 --- a/x-pack/plugin/profiling/src/test/java/org/elasticsearch/xpack/profiling/GetStackTracesRequestTests.java +++ b/x-pack/plugin/profiling/src/test/java/org/elasticsearch/xpack/profiling/GetStackTracesRequestTests.java @@ -66,7 +66,7 @@ public void testParseValidXContentWithCustomIndex() throws IOException { //tag::noformat .startObject() .field("sample_size", 2000) - .field("indices", "my-traces") + .field("indices", new String[] {"my-traces"}) .field("stacktrace_ids_field", "stacktraces") .startObject("query") .startObject("range") @@ -83,7 +83,7 @@ public void testParseValidXContentWithCustomIndex() throws IOException { request.parseXContent(content); assertEquals(2000, request.getSampleSize()); - assertEquals("my-traces", request.getIndices()); + assertArrayEquals(new String[] { "my-traces" }, request.getIndices()); assertEquals("stacktraces", request.getStackTraceIdsField()); // a basic check suffices here assertEquals("@timestamp", ((RangeQueryBuilder) request.getQuery()).fieldName()); @@ -169,6 +169,78 @@ public void testParseXContentUnrecognizedField() throws IOException { } } + public void testParseXContentCustomIndexNoArray() throws IOException { + try (XContentParser content = createParser(XContentFactory.jsonBuilder() + //tag::noformat + .startObject() + .field("sample_size", 2000) + .field("indices", "my-traces") + .field("stacktrace_ids_field", "stacktraces") + .startObject("query") + .startObject("range") + .startObject("@timestamp") + .field("gte", "2022-10-05") + .endObject() + .endObject() + .endObject() + .endObject() + //end::noformat + )) { + + GetStackTracesRequest request = new GetStackTracesRequest(); + ParsingException ex = expectThrows(ParsingException.class, () -> request.parseXContent(content)); + assertEquals("Unknown key for a VALUE_STRING in [indices].", ex.getMessage()); + } + } + + public void testParseXContentCustomIndexNullValues() throws IOException { + try (XContentParser content = createParser(XContentFactory.jsonBuilder() + //tag::noformat + .startObject() + .field("sample_size", 2000) + .field("indices", new String[] {"my-traces", null}) + .field("stacktrace_ids_field", "stacktraces") + .startObject("query") + .startObject("range") + .startObject("@timestamp") + .field("gte", "2022-10-05") + .endObject() + .endObject() + .endObject() + .endObject() + //end::noformat + )) { + + GetStackTracesRequest request = new GetStackTracesRequest(); + ParsingException ex = expectThrows(ParsingException.class, () -> request.parseXContent(content)); + assertEquals("Expected [VALUE_STRING] but found [VALUE_NULL] in [indices].", ex.getMessage()); + } + } + + public void testParseXContentCustomIndexInvalidTypes() throws IOException { + try (XContentParser content = createParser(XContentFactory.jsonBuilder() + //tag::noformat + .startObject() + .field("sample_size", 2000) + .field("indices", new int[] {1, 2, 3}) + .field("stacktrace_ids_field", "stacktraces") + .startObject("query") + .startObject("range") + .startObject("@timestamp") + .field("gte", "2022-10-05") + .endObject() + .endObject() + .endObject() + .endObject() + //end::noformat + )) { + + GetStackTracesRequest request = new GetStackTracesRequest(); + ParsingException ex = expectThrows(ParsingException.class, () -> request.parseXContent(content)); + assertEquals("Expected [VALUE_STRING] but found [VALUE_NUMBER] in [indices].", ex.getMessage()); + } + } + public void testValidateWrongSampleSize() { GetStackTracesRequest request = new GetStackTracesRequest( randomIntBetween(Integer.MIN_VALUE, 0), @@ -196,7 +268,7 @@ public void testValidateSampleSizeIsValidWithCustomIndices() { 1.0d, 1.0d, null, - randomAlphaOfLength(7), + new String[] { randomAlphaOfLength(7) }, randomAlphaOfLength(3), null, null, @@ -234,7 +306,7 @@ public void testValidateIndicesWithoutStacktraces() { 1.0d, 1.0d, null, - randomAlphaOfLength(5), + new String[] { randomAlphaOfLength(5) }, randomFrom("", null), null, null, @@ -255,7 +327,7 @@ public void testConsidersCustomIndicesInRelatedIndices() { 1.0d, 1.0d, null, - customIndex, + new String[] { customIndex }, randomAlphaOfLength(3), null, null, diff --git a/x-pack/plugin/src/yamlRestTest/resources/rest-api-spec/test/profiling/10_basic.yml b/x-pack/plugin/src/yamlRestTest/resources/rest-api-spec/test/profiling/10_basic.yml index 684d554f08e58..367655ba89388 100644 --- a/x-pack/plugin/src/yamlRestTest/resources/rest-api-spec/test/profiling/10_basic.yml +++ b/x-pack/plugin/src/yamlRestTest/resources/rest-api-spec/test/profiling/10_basic.yml @@ -197,7 +197,7 @@ teardown: body: > { "sample_size": 20000, - "indices": "test-events", + "indices": ["test-event*"], "stacktrace_ids_field": "events", "requested_duration": 86400, "query": {