From 271da22a44bced0a2119318eb47defa6a47fa624 Mon Sep 17 00:00:00 2001 From: javanna Date: Thu, 1 Mar 2018 12:11:44 +0100 Subject: [PATCH 1/6] REST high-level client: add clear cache API Relates to #27205 --- .../elasticsearch/client/IndicesClient.java | 25 ++++ .../org/elasticsearch/client/Request.java | 12 ++ .../elasticsearch/client/IndicesClientIT.java | 33 +++++- .../elasticsearch/client/RequestTests.java | 35 ++++++ .../IndicesClientDocumentationIT.java | 87 ++++++++++++++ .../high-level/indices/clear_cache.asciidoc | 109 ++++++++++++++++++ .../high-level/supported-apis.asciidoc | 2 + .../clear/ClearIndicesCacheResponse.java | 31 ++--- .../indices/RestClearIndicesCacheAction.java | 4 +- .../clear/ClearIndicesCacheResponseTests.java | 40 +++++++ 10 files changed, 359 insertions(+), 19 deletions(-) create mode 100644 docs/java-rest/high-level/indices/clear_cache.asciidoc create mode 100644 server/src/test/java/org/elasticsearch/action/admin/indices/cache/clear/ClearIndicesCacheResponseTests.java diff --git a/client/rest-high-level/src/main/java/org/elasticsearch/client/IndicesClient.java b/client/rest-high-level/src/main/java/org/elasticsearch/client/IndicesClient.java index b39b7801e27e5..5560a8df51cf5 100644 --- a/client/rest-high-level/src/main/java/org/elasticsearch/client/IndicesClient.java +++ b/client/rest-high-level/src/main/java/org/elasticsearch/client/IndicesClient.java @@ -24,6 +24,8 @@ import org.elasticsearch.action.admin.indices.alias.IndicesAliasesRequest; import org.elasticsearch.action.admin.indices.alias.IndicesAliasesResponse; import org.elasticsearch.action.admin.indices.alias.get.GetAliasesRequest; +import org.elasticsearch.action.admin.indices.cache.clear.ClearIndicesCacheRequest; +import org.elasticsearch.action.admin.indices.cache.clear.ClearIndicesCacheResponse; import org.elasticsearch.action.admin.indices.close.CloseIndexRequest; import org.elasticsearch.action.admin.indices.close.CloseIndexResponse; import org.elasticsearch.action.admin.indices.create.CreateIndexRequest; @@ -259,6 +261,29 @@ public void flushAsync(FlushRequest flushRequest, ActionListener listener, emptySet(), headers); } + /** + * Clears the cache of one or more indices using the Clear Cache API + *

+ * See + * Clear Cache API on elastic.co + */ + public ClearIndicesCacheResponse clearCache(ClearIndicesCacheRequest clearIndicesCacheRequest, Header... headers) throws IOException { + return restHighLevelClient.performRequestAndParseEntity(clearIndicesCacheRequest, Request::clearCache, + ClearIndicesCacheResponse::fromXContent, emptySet(), headers); + } + + /** + * Asynchronously clears the cache of one or more indices using the Clear Cache API + *

+ * See + * Clear Cache API on elastic.co + */ + public void clearCacheAsync(ClearIndicesCacheRequest clearIndicesCacheRequest, ActionListener listener, + Header... headers) { + restHighLevelClient.performRequestAsyncAndParseEntity(clearIndicesCacheRequest, Request::clearCache, + ClearIndicesCacheResponse::fromXContent, listener, emptySet(), headers); + } + /** * Checks if the index (indices) exists or not. *

diff --git a/client/rest-high-level/src/main/java/org/elasticsearch/client/Request.java b/client/rest-high-level/src/main/java/org/elasticsearch/client/Request.java index 53bd6b9ecd77d..488811c265506 100755 --- a/client/rest-high-level/src/main/java/org/elasticsearch/client/Request.java +++ b/client/rest-high-level/src/main/java/org/elasticsearch/client/Request.java @@ -32,6 +32,7 @@ import org.elasticsearch.action.admin.cluster.settings.ClusterUpdateSettingsRequest; import org.elasticsearch.action.admin.indices.alias.IndicesAliasesRequest; import org.elasticsearch.action.admin.indices.alias.get.GetAliasesRequest; +import org.elasticsearch.action.admin.indices.cache.clear.ClearIndicesCacheRequest; import org.elasticsearch.action.admin.indices.close.CloseIndexRequest; import org.elasticsearch.action.admin.indices.create.CreateIndexRequest; import org.elasticsearch.action.admin.indices.delete.DeleteIndexRequest; @@ -234,6 +235,17 @@ static Request flush(FlushRequest flushRequest) { return new Request(HttpPost.METHOD_NAME, endpoint, parameters.getParams(), null); } + static Request clearCache(ClearIndicesCacheRequest clearIndicesCacheRequest) { + String endpoint = endpoint(clearIndicesCacheRequest.indices(), "_cache", "clear"); + Params parameters = Params.builder(); + parameters.withIndicesOptions(clearIndicesCacheRequest.indicesOptions()); + parameters.putParam("query", Boolean.toString(clearIndicesCacheRequest.queryCache())); + parameters.putParam("fielddata", Boolean.toString(clearIndicesCacheRequest.fieldDataCache())); + parameters.putParam("request", Boolean.toString(clearIndicesCacheRequest.requestCache())); + parameters.putParam("fields", String.join(",", clearIndicesCacheRequest.fields())); + return new Request(HttpPost.METHOD_NAME, endpoint, parameters.getParams(), null); + } + static Request info() { return new Request(HttpGet.METHOD_NAME, "/", Collections.emptyMap(), null); } diff --git a/client/rest-high-level/src/test/java/org/elasticsearch/client/IndicesClientIT.java b/client/rest-high-level/src/test/java/org/elasticsearch/client/IndicesClientIT.java index 7f777531440d9..8a2ba44791149 100755 --- a/client/rest-high-level/src/test/java/org/elasticsearch/client/IndicesClientIT.java +++ b/client/rest-high-level/src/test/java/org/elasticsearch/client/IndicesClientIT.java @@ -28,6 +28,8 @@ import org.elasticsearch.action.admin.indices.alias.IndicesAliasesRequest.AliasActions; import org.elasticsearch.action.admin.indices.alias.IndicesAliasesResponse; import org.elasticsearch.action.admin.indices.alias.get.GetAliasesRequest; +import org.elasticsearch.action.admin.indices.cache.clear.ClearIndicesCacheRequest; +import org.elasticsearch.action.admin.indices.cache.clear.ClearIndicesCacheResponse; import org.elasticsearch.action.admin.indices.close.CloseIndexRequest; import org.elasticsearch.action.admin.indices.close.CloseIndexResponse; import org.elasticsearch.action.admin.indices.create.CreateIndexRequest; @@ -431,9 +433,36 @@ public void testFlush() throws IOException { { String nonExistentIndex = "non_existent_index"; assertFalse(indexExists(nonExistentIndex)); - FlushRequest refreshRequest = new FlushRequest(nonExistentIndex); + FlushRequest flushRequest = new FlushRequest(nonExistentIndex); ElasticsearchException exception = expectThrows(ElasticsearchException.class, - () -> execute(refreshRequest, highLevelClient().indices()::flush, highLevelClient().indices()::flushAsync)); + () -> execute(flushRequest, highLevelClient().indices()::flush, highLevelClient().indices()::flushAsync)); + assertEquals(RestStatus.NOT_FOUND, exception.status()); + } + } + + public void testClearCache() throws IOException { + { + String index = "index"; + Settings settings = Settings.builder() + .put("number_of_shards", 1) + .put("number_of_replicas", 0) + .build(); + createIndex(index, settings); + ClearIndicesCacheRequest clearCacheRequest = new ClearIndicesCacheRequest(index); + ClearIndicesCacheResponse clearCacheResponse = + execute(clearCacheRequest, highLevelClient().indices()::clearCache, highLevelClient().indices()::clearCacheAsync); + assertThat(clearCacheResponse.getTotalShards(), equalTo(1)); + assertThat(clearCacheResponse.getSuccessfulShards(), equalTo(1)); + assertThat(clearCacheResponse.getFailedShards(), equalTo(0)); + assertThat(clearCacheResponse.getShardFailures(), equalTo(BroadcastResponse.EMPTY)); + } + { + String nonExistentIndex = "non_existent_index"; + assertFalse(indexExists(nonExistentIndex)); + ClearIndicesCacheRequest clearCacheRequest = new ClearIndicesCacheRequest(nonExistentIndex); + ElasticsearchException exception = expectThrows(ElasticsearchException.class, + () -> execute(clearCacheRequest, highLevelClient().indices()::clearCache, + highLevelClient().indices()::clearCacheAsync)); assertEquals(RestStatus.NOT_FOUND, exception.status()); } } diff --git a/client/rest-high-level/src/test/java/org/elasticsearch/client/RequestTests.java b/client/rest-high-level/src/test/java/org/elasticsearch/client/RequestTests.java index 5a7965ad446cb..19e9fcc359a39 100755 --- a/client/rest-high-level/src/test/java/org/elasticsearch/client/RequestTests.java +++ b/client/rest-high-level/src/test/java/org/elasticsearch/client/RequestTests.java @@ -34,6 +34,7 @@ import org.elasticsearch.action.admin.indices.alias.IndicesAliasesRequest; import org.elasticsearch.action.admin.indices.alias.IndicesAliasesRequest.AliasActions; import org.elasticsearch.action.admin.indices.alias.get.GetAliasesRequest; +import org.elasticsearch.action.admin.indices.cache.clear.ClearIndicesCacheRequest; import org.elasticsearch.action.admin.indices.close.CloseIndexRequest; import org.elasticsearch.action.admin.indices.create.CreateIndexRequest; import org.elasticsearch.action.admin.indices.delete.DeleteIndexRequest; @@ -580,6 +581,40 @@ public void testFlush() { assertThat(request.getMethod(), equalTo(HttpPost.METHOD_NAME)); } + public void testClearCache() { + String[] indices = randomIndicesNames(0, 5); + ClearIndicesCacheRequest clearIndicesCacheRequest = new ClearIndicesCacheRequest(indices); + Map expectedParams = new HashMap<>(); + setRandomIndicesOptions(clearIndicesCacheRequest::indicesOptions, clearIndicesCacheRequest::indicesOptions, expectedParams); + if (randomBoolean()) { + clearIndicesCacheRequest.queryCache(randomBoolean()); + } + expectedParams.put("query", Boolean.toString(clearIndicesCacheRequest.queryCache())); + if (randomBoolean()) { + clearIndicesCacheRequest.fieldDataCache(randomBoolean()); + } + expectedParams.put("fielddata", Boolean.toString(clearIndicesCacheRequest.fieldDataCache())); + if (randomBoolean()) { + clearIndicesCacheRequest.requestCache(randomBoolean()); + } + expectedParams.put("request", Boolean.toString(clearIndicesCacheRequest.requestCache())); + if (randomBoolean()) { + clearIndicesCacheRequest.fields(randomIndicesNames(1, 5)); + expectedParams.put("fields", String.join(",", clearIndicesCacheRequest.fields())); + } + + Request request = Request.clearCache(clearIndicesCacheRequest); + StringJoiner endpoint = new StringJoiner("/", "/", ""); + if (indices.length > 0) { + endpoint.add(String.join(",", indices)); + } + endpoint.add("_cache/clear"); + assertThat(request.getEndpoint(), equalTo(endpoint.toString())); + assertThat(request.getParameters(), equalTo(expectedParams)); + assertThat(request.getEntity(), nullValue()); + assertThat(request.getMethod(), equalTo(HttpPost.METHOD_NAME)); + } + public void testUpdate() throws IOException { XContentType xContentType = randomFrom(XContentType.values()); diff --git a/client/rest-high-level/src/test/java/org/elasticsearch/client/documentation/IndicesClientDocumentationIT.java b/client/rest-high-level/src/test/java/org/elasticsearch/client/documentation/IndicesClientDocumentationIT.java index 4bbc00fb41111..fb9e56d222022 100644 --- a/client/rest-high-level/src/test/java/org/elasticsearch/client/documentation/IndicesClientDocumentationIT.java +++ b/client/rest-high-level/src/test/java/org/elasticsearch/client/documentation/IndicesClientDocumentationIT.java @@ -27,6 +27,8 @@ import org.elasticsearch.action.admin.indices.alias.IndicesAliasesRequest.AliasActions; import org.elasticsearch.action.admin.indices.alias.IndicesAliasesResponse; import org.elasticsearch.action.admin.indices.alias.get.GetAliasesRequest; +import org.elasticsearch.action.admin.indices.cache.clear.ClearIndicesCacheRequest; +import org.elasticsearch.action.admin.indices.cache.clear.ClearIndicesCacheResponse; import org.elasticsearch.action.admin.indices.close.CloseIndexRequest; import org.elasticsearch.action.admin.indices.close.CloseIndexResponse; import org.elasticsearch.action.admin.indices.create.CreateIndexRequest; @@ -769,6 +771,91 @@ public void onFailure(Exception e) { } } + public void testClearCache() throws Exception { + RestHighLevelClient client = highLevelClient(); + + { + createIndex("index1", Settings.EMPTY); + } + + { + // tag::clear-cache-request + ClearIndicesCacheRequest request = new ClearIndicesCacheRequest("index1"); // <1> + ClearIndicesCacheRequest requestMultiple = new ClearIndicesCacheRequest("index1", "index2"); // <2> + ClearIndicesCacheRequest requestAll = new ClearIndicesCacheRequest(); // <3> + // end::clear-cache-request + + // tag::clear-cache-request-indicesOptions + request.indicesOptions(IndicesOptions.lenientExpandOpen()); // <1> + // end::clear-cache-request-indicesOptions + + // tag::clear-cache-request-query + request.queryCache(true); // <1> + // end::clear-cache-request-query + + // tag::clear-cache-request-request + request.requestCache(true); // <1> + // end::clear-cache-request-request + + // tag::clear-cache-request-fielddata + request.fieldDataCache(true); // <1> + // end::clear-cache-request-fielddata + + // tag::clear-cache-request-fields + request.fields("field1", "field2", "field3"); // <1> + // end::clear-cache-request-fields + + // tag::clear-cache-execute + ClearIndicesCacheResponse clearCacheResponse = client.indices().clearCache(request); + // end::clear-cache-execute + + // tag::clear-cache-response + int totalShards = clearCacheResponse.getTotalShards(); // <1> + int successfulShards = clearCacheResponse.getSuccessfulShards(); // <2> + int failedShards = clearCacheResponse.getFailedShards(); // <3> + DefaultShardOperationFailedException[] failures = clearCacheResponse.getShardFailures(); // <4> + // end::clear-cache-response + + // tag::clear-cache-execute-listener + ActionListener listener = new ActionListener() { + @Override + public void onResponse(ClearIndicesCacheResponse clearCacheResponse) { + // <1> + } + + @Override + public void onFailure(Exception e) { + // <2> + } + }; + // end::clear-cache-execute-listener + + // Replace the empty listener by a blocking listener in test + final CountDownLatch latch = new CountDownLatch(1); + listener = new LatchedActionListener<>(listener, latch); + + // tag::clear-cache-execute-async + client.indices().clearCacheAsync(request, listener); // <1> + // end::clear-cache-execute-async + + assertTrue(latch.await(30L, TimeUnit.SECONDS)); + } + + { + // tag::clear-cache-notfound + try { + ClearIndicesCacheRequest request = new ClearIndicesCacheRequest("does_not_exist"); + client.indices().clearCache(request); + } catch (ElasticsearchException exception) { + if (exception.status() == RestStatus.NOT_FOUND) { + // <1> + } + } + // end::clear-cache-notfound + } + } + + public void testCloseIndex() throws Exception { RestHighLevelClient client = highLevelClient(); diff --git a/docs/java-rest/high-level/indices/clear_cache.asciidoc b/docs/java-rest/high-level/indices/clear_cache.asciidoc new file mode 100644 index 0000000000000..9e73283abd815 --- /dev/null +++ b/docs/java-rest/high-level/indices/clear_cache.asciidoc @@ -0,0 +1,109 @@ +[[java-rest-high-clear-cache]] +=== Clear Cache API + +[[java-rest-high-clear-cache-request]] +==== Clear Cache Request + +A `ClearIndicesCacheRquest` can be applied to one or more indices, or even on +`_all` the indices: + +["source","java",subs="attributes,callouts,macros"] +-------------------------------------------------- +include-tagged::{doc-tests}/IndicesClientDocumentationIT.java[clear-cache-request] +-------------------------------------------------- +<1> Clears the cache of one index +<2> Clears the cache of multiple indices +<3> Clears the cache of all the indices + +==== Optional arguments + +["source","java",subs="attributes,callouts,macros"] +-------------------------------------------------- +include-tagged::{doc-tests}/IndicesClientDocumentationIT.java[clear-cache-request-indicesOptions] +-------------------------------------------------- +<1> Setting `IndicesOptions` controls how unavailable indices are resolved and +how wildcard expressions are expanded + +["source","java",subs="attributes,callouts,macros"] +-------------------------------------------------- +include-tagged::{doc-tests}/IndicesClientDocumentationIT.java[clear-cache-request-query] +-------------------------------------------------- +<1> Set the `query` flag to `true` + +["source","java",subs="attributes,callouts,macros"] +-------------------------------------------------- +include-tagged::{doc-tests}/IndicesClientDocumentationIT.java[clear-cache-request-fielddata] +-------------------------------------------------- +<1> Set the `fielddata` flag to `true` + +["source","java",subs="attributes,callouts,macros"] +-------------------------------------------------- +include-tagged::{doc-tests}/IndicesClientDocumentationIT.java[clear-cache-request-request] +-------------------------------------------------- +<1> Set the `request` flag to `true` + +["source","java",subs="attributes,callouts,macros"] +-------------------------------------------------- +include-tagged::{doc-tests}/IndicesClientDocumentationIT.java[clear-cache-request-fields] +-------------------------------------------------- +<1> Set the `fields` parameter + +[[java-rest-high-clear-cache-sync]] +==== Synchronous Execution + +["source","java",subs="attributes,callouts,macros"] +-------------------------------------------------- +include-tagged::{doc-tests}/IndicesClientDocumentationIT.java[clear-cache-execute] +-------------------------------------------------- + +[[java-rest-high-clear-cache-async]] +==== Asynchronous Execution + +The asynchronous execution of a clear cache request requires both the `ClearIndicesCacheRequest` +instance and an `ActionListener` instance to be passed to the asynchronous +method: + +["source","java",subs="attributes,callouts,macros"] +-------------------------------------------------- +include-tagged::{doc-tests}/IndicesClientDocumentationIT.java[clear-cache-execute-async] +-------------------------------------------------- +<1> The `ClearIndicesCacheRequest` to execute and the `ActionListener` to use when +the execution completes + +The asynchronous method does not block and returns immediately. Once it is +completed the `ActionListener` is called back using the `onResponse` method +if the execution successfully completed or using the `onFailure` method if +it failed. + +A typical listener for `ClearIndicesCacheResponse` looks like: + +["source","java",subs="attributes,callouts,macros"] +-------------------------------------------------- +include-tagged::{doc-tests}/IndicesClientDocumentationIT.java[clear-cache-execute-listener] +-------------------------------------------------- +<1> Called when the execution is successfully completed. The response is +provided as an argument +<2> Called in case of failure. The raised exception is provided as an argument + +[[java-rest-high-clear-cache-response]] +==== Clear Cache Response + +The returned `ClearIndicesCacheResponse` allows to retrieve information about the +executed operation as follows: + +["source","java",subs="attributes,callouts,macros"] +-------------------------------------------------- +include-tagged::{doc-tests}/IndicesClientDocumentationIT.java[clear-cache-response] +-------------------------------------------------- +<1> Total number of shards hit by the clear cache request +<2> Number of shards where the clear cache has succeeded +<3> Number of shards where the clear cache has failed +<4> A list of failures if the operation failed on one or more shards + +By default, if the indices were not found, an `ElasticsearchException` will be thrown: + +["source","java",subs="attributes,callouts,macros"] +-------------------------------------------------- +include-tagged::{doc-tests}/IndicesClientDocumentationIT.java[clear-cache-notfound] +-------------------------------------------------- +<1> Do something if the indices to be cleared were not found \ No newline at end of file diff --git a/docs/java-rest/high-level/supported-apis.asciidoc b/docs/java-rest/high-level/supported-apis.asciidoc index 9fb8bd8c66700..ed1aa395b6772 100644 --- a/docs/java-rest/high-level/supported-apis.asciidoc +++ b/docs/java-rest/high-level/supported-apis.asciidoc @@ -54,6 +54,7 @@ Index Management:: * <> * <> * <> +* <> * <> Mapping Management:: @@ -72,6 +73,7 @@ include::indices/shrink_index.asciidoc[] include::indices/split_index.asciidoc[] include::indices/refresh.asciidoc[] include::indices/flush.asciidoc[] +include::indices/clear_cache.asciidoc[] include::indices/rollover.asciidoc[] include::indices/put_mapping.asciidoc[] include::indices/update_aliases.asciidoc[] diff --git a/server/src/main/java/org/elasticsearch/action/admin/indices/cache/clear/ClearIndicesCacheResponse.java b/server/src/main/java/org/elasticsearch/action/admin/indices/cache/clear/ClearIndicesCacheResponse.java index d0f4b3cc20beb..47b516137d795 100644 --- a/server/src/main/java/org/elasticsearch/action/admin/indices/cache/clear/ClearIndicesCacheResponse.java +++ b/server/src/main/java/org/elasticsearch/action/admin/indices/cache/clear/ClearIndicesCacheResponse.java @@ -21,19 +21,28 @@ import org.elasticsearch.action.support.DefaultShardOperationFailedException; import org.elasticsearch.action.support.broadcast.BroadcastResponse; -import org.elasticsearch.common.io.stream.StreamInput; -import org.elasticsearch.common.io.stream.StreamOutput; +import org.elasticsearch.common.xcontent.ConstructingObjectParser; +import org.elasticsearch.common.xcontent.XContentParser; -import java.io.IOException; +import java.util.Arrays; import java.util.List; /** - * The response of a refresh action. - * - * + * The response of a clear cache action. */ public class ClearIndicesCacheResponse extends BroadcastResponse { + private static final ConstructingObjectParser PARSER = new ConstructingObjectParser<>("clear_cache", + true, arg -> { + BroadcastResponse response = (BroadcastResponse) arg[0]; + return new ClearIndicesCacheResponse(response.getTotalShards(), response.getSuccessfulShards(), response.getFailedShards(), + Arrays.asList(response.getShardFailures())); + }); + + static { + declareBroadcastFields(PARSER); + } + ClearIndicesCacheResponse() { } @@ -43,13 +52,7 @@ public class ClearIndicesCacheResponse extends BroadcastResponse { super(totalShards, successfulShards, failedShards, shardFailures); } - @Override - public void readFrom(StreamInput in) throws IOException { - super.readFrom(in); - } - - @Override - public void writeTo(StreamOutput out) throws IOException { - super.writeTo(out); + public static ClearIndicesCacheResponse fromXContent(XContentParser parser) { + return PARSER.apply(parser, null); } } \ No newline at end of file diff --git a/server/src/main/java/org/elasticsearch/rest/action/admin/indices/RestClearIndicesCacheAction.java b/server/src/main/java/org/elasticsearch/rest/action/admin/indices/RestClearIndicesCacheAction.java index b96ada4cdd974..a44c2d5a779ba 100644 --- a/server/src/main/java/org/elasticsearch/rest/action/admin/indices/RestClearIndicesCacheAction.java +++ b/server/src/main/java/org/elasticsearch/rest/action/admin/indices/RestClearIndicesCacheAction.java @@ -41,7 +41,6 @@ import static org.elasticsearch.rest.RestRequest.Method.GET; import static org.elasticsearch.rest.RestRequest.Method.POST; import static org.elasticsearch.rest.RestStatus.OK; -import static org.elasticsearch.rest.action.RestActions.buildBroadcastShardsHeader; public class RestClearIndicesCacheAction extends BaseRestHandler { public RestClearIndicesCacheAction(Settings settings, RestController controller) { @@ -69,7 +68,7 @@ public RestChannelConsumer prepareRequest(final RestRequest request, final NodeC @Override public RestResponse buildResponse(ClearIndicesCacheResponse response, XContentBuilder builder) throws Exception { builder.startObject(); - buildBroadcastShardsHeader(builder, request, response); + response.toXContent(builder, request); builder.endObject(); return new BytesRestResponse(OK, builder); } @@ -104,5 +103,4 @@ public static class Fields { public static final ParseField FIELD_DATA = new ParseField("field_data", "fielddata"); public static final ParseField FIELDS = new ParseField("fields"); } - } diff --git a/server/src/test/java/org/elasticsearch/action/admin/indices/cache/clear/ClearIndicesCacheResponseTests.java b/server/src/test/java/org/elasticsearch/action/admin/indices/cache/clear/ClearIndicesCacheResponseTests.java new file mode 100644 index 0000000000000..4b58d5fb70246 --- /dev/null +++ b/server/src/test/java/org/elasticsearch/action/admin/indices/cache/clear/ClearIndicesCacheResponseTests.java @@ -0,0 +1,40 @@ +/* + * Licensed to Elasticsearch under one or more contributor + * license agreements. See the NOTICE file distributed with + * this work for additional information regarding copyright + * ownership. Elasticsearch licenses this file to you under + * the Apache License, Version 2.0 (the "License"); you may + * not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * http://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, + * software distributed under the License is distributed on an + * "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY + * KIND, either express or implied. See the License for the + * specific language governing permissions and limitations + * under the License. + */ + +package org.elasticsearch.action.admin.indices.cache.clear; + +import org.elasticsearch.action.support.DefaultShardOperationFailedException; +import org.elasticsearch.action.support.broadcast.AbstractBroadcastResponseTestCase; +import org.elasticsearch.common.xcontent.XContentParser; + +import java.util.List; + +public class ClearIndicesCacheResponseTests extends AbstractBroadcastResponseTestCase { + + @Override + protected ClearIndicesCacheResponse createTestInstance(int totalShards, int successfulShards, int failedShards, + List failures) { + return new ClearIndicesCacheResponse(totalShards, successfulShards, failedShards, failures); + } + + @Override + protected ClearIndicesCacheResponse doParseInstance(XContentParser parser) { + return ClearIndicesCacheResponse.fromXContent(parser); + } +} From cddc2e0650e3768ef883fdc1f0db3f57993a83a3 Mon Sep 17 00:00:00 2001 From: javanna Date: Tue, 6 Mar 2018 14:28:38 +0100 Subject: [PATCH 2/6] address some of the review comments --- .../src/main/java/org/elasticsearch/client/IndicesClient.java | 4 ++-- .../main/resources/rest-api-spec/api/indices.clear_cache.json | 4 ---- 2 files changed, 2 insertions(+), 6 deletions(-) diff --git a/client/rest-high-level/src/main/java/org/elasticsearch/client/IndicesClient.java b/client/rest-high-level/src/main/java/org/elasticsearch/client/IndicesClient.java index 5560a8df51cf5..0b366aa99e188 100644 --- a/client/rest-high-level/src/main/java/org/elasticsearch/client/IndicesClient.java +++ b/client/rest-high-level/src/main/java/org/elasticsearch/client/IndicesClient.java @@ -265,7 +265,7 @@ public void flushAsync(FlushRequest flushRequest, ActionListener * Clears the cache of one or more indices using the Clear Cache API *

* See - * Clear Cache API on elastic.co + * Clear Cache API on elastic.co */ public ClearIndicesCacheResponse clearCache(ClearIndicesCacheRequest clearIndicesCacheRequest, Header... headers) throws IOException { return restHighLevelClient.performRequestAndParseEntity(clearIndicesCacheRequest, Request::clearCache, @@ -276,7 +276,7 @@ public ClearIndicesCacheResponse clearCache(ClearIndicesCacheRequest clearIndice * Asynchronously clears the cache of one or more indices using the Clear Cache API *

* See - * Clear Cache API on elastic.co + * Clear Cache API on elastic.co */ public void clearCacheAsync(ClearIndicesCacheRequest clearIndicesCacheRequest, ActionListener listener, Header... headers) { diff --git a/rest-api-spec/src/main/resources/rest-api-spec/api/indices.clear_cache.json b/rest-api-spec/src/main/resources/rest-api-spec/api/indices.clear_cache.json index 1f24199fad468..c81b514f24b86 100644 --- a/rest-api-spec/src/main/resources/rest-api-spec/api/indices.clear_cache.json +++ b/rest-api-spec/src/main/resources/rest-api-spec/api/indices.clear_cache.json @@ -46,10 +46,6 @@ "type" : "list", "description" : "A comma-separated list of index name to limit the operation" }, - "recycler": { - "type" : "boolean", - "description" : "Clear the recycler cache" - }, "request_cache": { "type" : "boolean", "description" : "Clear request cache" From 8acef57e6f38427481e495deca0df66ead67d440 Mon Sep 17 00:00:00 2001 From: javanna Date: Mon, 19 Mar 2018 17:56:17 +0100 Subject: [PATCH 3/6] prevent NPEs if indices are null, add tests and skip encoding of known endpoints --- .../org/elasticsearch/client/Request.java | 136 +++++++----- .../elasticsearch/client/RequestTests.java | 209 ++++++++++++++---- .../admin/indices/get/GetIndexRequest.java | 9 +- 3 files changed, 260 insertions(+), 94 deletions(-) diff --git a/client/rest-high-level/src/main/java/org/elasticsearch/client/Request.java b/client/rest-high-level/src/main/java/org/elasticsearch/client/Request.java index deeb9ca537ff5..36f65a3a33438 100755 --- a/client/rest-high-level/src/main/java/org/elasticsearch/client/Request.java +++ b/client/rest-high-level/src/main/java/org/elasticsearch/client/Request.java @@ -171,7 +171,8 @@ static Request openIndex(OpenIndexRequest openIndexRequest) { } static Request closeIndex(CloseIndexRequest closeIndexRequest) { - String endpoint = endpoint(closeIndexRequest.indices(), "_close"); + String[] indices = closeIndexRequest.indices() == null ? Strings.EMPTY_ARRAY : closeIndexRequest.indices(); + String endpoint = endpoint(indices, "_close"); Params parameters = Params.builder(); @@ -220,14 +221,16 @@ static Request putMapping(PutMappingRequest putMappingRequest) throws IOExceptio } static Request refresh(RefreshRequest refreshRequest) { - String endpoint = endpoint(refreshRequest.indices(), "_refresh"); + String[] indices = refreshRequest.indices() == null ? Strings.EMPTY_ARRAY : refreshRequest.indices(); + String endpoint = endpoint(indices, "_refresh"); Params parameters = Params.builder(); parameters.withIndicesOptions(refreshRequest.indicesOptions()); return new Request(HttpPost.METHOD_NAME, endpoint, parameters.getParams(), null); } static Request flush(FlushRequest flushRequest) { - String endpoint = endpoint(flushRequest.indices(), "_flush"); + String[] indices = flushRequest.indices() == null ? Strings.EMPTY_ARRAY : flushRequest.indices(); + String endpoint = endpoint(indices, "_flush"); Params parameters = Params.builder(); parameters.withIndicesOptions(flushRequest.indicesOptions()); parameters.putParam("wait_if_ongoing", Boolean.toString(flushRequest.waitIfOngoing())); @@ -236,7 +239,8 @@ static Request flush(FlushRequest flushRequest) { } static Request clearCache(ClearIndicesCacheRequest clearIndicesCacheRequest) { - String endpoint = endpoint(clearIndicesCacheRequest.indices(), "_cache", "clear"); + String[] indices = clearIndicesCacheRequest.indices() == null ? Strings.EMPTY_ARRAY :clearIndicesCacheRequest.indices(); + String endpoint = endpoint(indices, "_cache/clear"); Params parameters = Params.builder(); parameters.withIndicesOptions(clearIndicesCacheRequest.indicesOptions()); parameters.putParam("query", Boolean.toString(clearIndicesCacheRequest.queryCache())); @@ -521,10 +525,12 @@ static Request existsAlias(GetAliasesRequest getAliasesRequest) { Params params = Params.builder(); params.withIndicesOptions(getAliasesRequest.indicesOptions()); params.withLocal(getAliasesRequest.local()); - if (getAliasesRequest.indices().length == 0 && getAliasesRequest.aliases().length == 0) { + if ((getAliasesRequest.indices() == null || getAliasesRequest.indices().length == 0) && getAliasesRequest.aliases().length == 0) { throw new IllegalArgumentException("existsAlias requires at least an alias or an index"); } - String endpoint = endpoint(getAliasesRequest.indices(), "_alias", getAliasesRequest.aliases()); + String[] indices = getAliasesRequest.indices() == null ? Strings.EMPTY_ARRAY : getAliasesRequest.indices(); + String[] aliases = getAliasesRequest.aliases() == null ? Strings.EMPTY_ARRAY : getAliasesRequest.aliases(); + String endpoint = endpoint(indices, "_alias", aliases); return new Request(HttpHead.METHOD_NAME, endpoint, params.getParams(), null); } @@ -555,8 +561,9 @@ private static Request resize(ResizeRequest resizeRequest) throws IOException { params.withTimeout(resizeRequest.timeout()); params.withMasterTimeout(resizeRequest.masterNodeTimeout()); params.withWaitForActiveShards(resizeRequest.getTargetIndexRequest().waitForActiveShards()); - String endpoint = buildEndpoint(resizeRequest.getSourceIndex(), "_" + resizeRequest.getResizeType().name().toLowerCase(Locale.ROOT), - resizeRequest.getTargetIndexRequest().index()); + String endpoint = new EndpointBuilder().addPathPart(resizeRequest.getSourceIndex()) + .addPathPartAsIs("_" + resizeRequest.getResizeType().name().toLowerCase(Locale.ROOT)) + .addPathPart(resizeRequest.getTargetIndexRequest().index()).build(); HttpEntity entity = createEntity(resizeRequest, REQUEST_BODY_CONTENT_TYPE); return new Request(HttpPut.METHOD_NAME, endpoint, params.getParams(), entity); } @@ -566,10 +573,8 @@ static Request clusterPutSettings(ClusterUpdateSettingsRequest clusterUpdateSett parameters.withFlatSettings(clusterUpdateSettingsRequest.flatSettings()); parameters.withTimeout(clusterUpdateSettingsRequest.timeout()); parameters.withMasterTimeout(clusterUpdateSettingsRequest.masterNodeTimeout()); - - String endpoint = buildEndpoint("_cluster", "settings"); HttpEntity entity = createEntity(clusterUpdateSettingsRequest, REQUEST_BODY_CONTENT_TYPE); - return new Request(HttpPut.METHOD_NAME, endpoint, parameters.getParams(), entity); + return new Request(HttpPut.METHOD_NAME, "/_cluster/settings", parameters.getParams(), entity); } static Request rollover(RolloverRequest rolloverRequest) throws IOException { @@ -580,64 +585,56 @@ static Request rollover(RolloverRequest rolloverRequest) throws IOException { if (rolloverRequest.isDryRun()) { params.putParam("dry_run", Boolean.TRUE.toString()); } - String endpoint = buildEndpoint(rolloverRequest.getAlias(), "_rollover", rolloverRequest.getNewIndexName()); + String endpoint = new EndpointBuilder().addPathPart(rolloverRequest.getAlias()).addPathPartAsIs("_rollover") + .addPathPart(rolloverRequest.getNewIndexName()).build(); HttpEntity entity = createEntity(rolloverRequest, REQUEST_BODY_CONTENT_TYPE); return new Request(HttpPost.METHOD_NAME, endpoint, params.getParams(), entity); } + static Request indicesExist(GetIndexRequest request) { + String endpoint = endpoint(request.indices(), ""); + Params params = Params.builder(); + params.withLocal(request.local()); + params.withHuman(request.humanReadable()); + params.withIndicesOptions(request.indicesOptions()); + params.withFlatSettings(request.flatSettings()); + params.withIncludeDefaults(request.includeDefaults()); + return new Request(HttpHead.METHOD_NAME, endpoint, params.getParams(), null); + } + private static HttpEntity createEntity(ToXContent toXContent, XContentType xContentType) throws IOException { BytesRef source = XContentHelper.toXContent(toXContent, xContentType, false).toBytesRef(); return new ByteArrayEntity(source.bytes, source.offset, source.length, createContentType(xContentType)); } static String endpoint(String index, String type, String id) { - return buildEndpoint(index, type, id); + return new EndpointBuilder().addPathPart(index, type, id).build(); } static String endpoint(String index, String type, String id, String endpoint) { - return buildEndpoint(index, type, id, endpoint); + return new EndpointBuilder().addPathPart(index, type, id).addPathPartAsIs(endpoint).build(); } static String endpoint(String[] indices) { - return buildEndpoint(String.join(",", indices)); + return new EndpointBuilder().addCommaSeparatedPathParts(indices).build(); } static String endpoint(String[] indices, String endpoint) { - return buildEndpoint(String.join(",", indices), endpoint); + return new EndpointBuilder().addCommaSeparatedPathParts(indices).addPathPartAsIs(endpoint).build(); } static String endpoint(String[] indices, String[] types, String endpoint) { - return buildEndpoint(String.join(",", indices), String.join(",", types), endpoint); + return new EndpointBuilder().addCommaSeparatedPathParts(indices).addCommaSeparatedPathParts(types) + .addPathPartAsIs(endpoint).build(); } static String endpoint(String[] indices, String endpoint, String[] suffixes) { - return buildEndpoint(String.join(",", indices), endpoint, String.join(",", suffixes)); + return new EndpointBuilder().addCommaSeparatedPathParts(indices).addPathPartAsIs(endpoint) + .addCommaSeparatedPathParts(suffixes).build(); } static String endpoint(String[] indices, String endpoint, String type) { - return endpoint(String.join(",", indices), endpoint, type); - } - - /** - * Utility method to build request's endpoint given its parts as strings - */ - static String buildEndpoint(String... parts) { - StringJoiner joiner = new StringJoiner("/", "/", ""); - for (String part : parts) { - if (Strings.hasLength(part)) { - try { - //encode each part (e.g. index, type and id) separately before merging them into the path - //we prepend "/" to the path part to make this pate absolute, otherwise there can be issues with - //paths that start with `-` or contain `:` - URI uri = new URI(null, null, null, -1, "/" + part, null, null); - //manually encode any slash that each part may contain - joiner.add(uri.getRawPath().substring(1).replaceAll("/", "%2F")); - } catch (URISyntaxException e) { - throw new IllegalArgumentException("Path part [" + part + "] couldn't be encoded", e); - } - } - } - return joiner.toString(); + return new EndpointBuilder().addCommaSeparatedPathParts(indices).addPathPartAsIs(endpoint).addPathPart(type).build(); } /** @@ -651,17 +648,6 @@ public static ContentType createContentType(final XContentType xContentType) { return ContentType.create(xContentType.mediaTypeWithoutParameters(), (Charset) null); } - static Request indicesExist(GetIndexRequest request) { - String endpoint = endpoint(request.indices(), Strings.EMPTY_ARRAY, ""); - Params params = Params.builder(); - params.withLocal(request.local()); - params.withHuman(request.humanReadable()); - params.withIndicesOptions(request.indicesOptions()); - params.withFlatSettings(request.flatSettings()); - params.withIncludeDefaults(request.includeDefaults()); - return new Request(HttpHead.METHOD_NAME, endpoint, params.getParams(), null); - } - /** * Utility class to build request's parameters map and centralize all parameter names. */ @@ -867,4 +853,50 @@ static XContentType enforceSameContentType(IndexRequest indexRequest, @Nullable } return xContentType; } + + /** + * Utility class to build request's endpoint given its parts as strings + */ + static class EndpointBuilder { + + private final StringJoiner joiner = new StringJoiner("/", "/", ""); + + EndpointBuilder addPathPart(String... parts) { + for (String part : parts) { + if (Strings.hasLength(part)) { + joiner.add(encodePart(part)); + } + } + return this; + } + + EndpointBuilder addCommaSeparatedPathParts(String[] parts) { + addPathPart(String.join(",", parts)); + return this; + } + + EndpointBuilder addPathPartAsIs(String part) { + if (Strings.hasLength(part)) { + joiner.add(part); + } + return this; + } + + String build() { + return joiner.toString(); + } + + private static String encodePart(String pathPart) { + try { + //encode each part (e.g. index, type and id) separately before merging them into the path + //we prepend "/" to the path part to make this pate absolute, otherwise there can be issues with + //paths that start with `-` or contain `:` + URI uri = new URI(null, null, null, -1, "/" + pathPart, null, null); + //manually encode any slash that each part may contain + return uri.getRawPath().substring(1).replaceAll("/", "%2F"); + } catch (URISyntaxException e) { + throw new IllegalArgumentException("Path part [" + pathPart + "] couldn't be encoded", e); + } + } + } } diff --git a/client/rest-high-level/src/test/java/org/elasticsearch/client/RequestTests.java b/client/rest-high-level/src/test/java/org/elasticsearch/client/RequestTests.java index 19e9fcc359a39..ac05cfae66849 100755 --- a/client/rest-high-level/src/test/java/org/elasticsearch/client/RequestTests.java +++ b/client/rest-high-level/src/test/java/org/elasticsearch/client/RequestTests.java @@ -29,6 +29,7 @@ import org.apache.http.entity.ContentType; import org.apache.http.entity.StringEntity; import org.apache.http.util.EntityUtils; +import org.elasticsearch.action.ActionRequestValidationException; import org.elasticsearch.action.DocWriteRequest; import org.elasticsearch.action.admin.cluster.settings.ClusterUpdateSettingsRequest; import org.elasticsearch.action.admin.indices.alias.IndicesAliasesRequest; @@ -290,6 +291,12 @@ public void testIndicesExist() { assertNull(request.getEntity()); } + public void testIndicesExistEmptyIndices() { + String[] indices = randomBoolean() ? null : Strings.EMPTY_ARRAY; + ActionRequestValidationException validationException = new GetIndexRequest().indices(indices).validate(); + assertNotNull(validationException); + } + private static void getAndExistsTest(Function requestConverter, String method) { String index = randomAlphaOfLengthBetween(3, 10); String type = randomAlphaOfLengthBetween(3, 10); @@ -363,6 +370,11 @@ public void testCreateIndex() throws IOException { assertToXContentBody(createIndexRequest, request.getEntity()); } + public void testCreateIndexNullIndex() { + ActionRequestValidationException validationException = new CreateIndexRequest(null).validate(); + assertNotNull(validationException); + } + public void testUpdateAliases() throws IOException { IndicesAliasesRequest indicesAliasesRequest = new IndicesAliasesRequest(); AliasActions aliasAction = randomAliasAction(); @@ -424,6 +436,12 @@ public void testDeleteIndex() { assertNull(request.getEntity()); } + public void testDeleteIndexEmptyIndices() { + String[] indices = randomBoolean() ? null : Strings.EMPTY_ARRAY; + ActionRequestValidationException validationException = new DeleteIndexRequest(indices).validate(); + assertNotNull(validationException); + } + public void testOpenIndex() { String[] indices = randomIndicesNames(1, 5); OpenIndexRequest openIndexRequest = new OpenIndexRequest(indices); @@ -443,6 +461,12 @@ public void testOpenIndex() { assertThat(request.getEntity(), nullValue()); } + public void testOpenIndexEmptyIndices() { + String[] indices = randomBoolean() ? null : Strings.EMPTY_ARRAY; + ActionRequestValidationException validationException = new OpenIndexRequest(indices).validate(); + assertNotNull(validationException); + } + public void testCloseIndex() { String[] indices = randomIndicesNames(1, 5); CloseIndexRequest closeIndexRequest = new CloseIndexRequest(indices); @@ -460,6 +484,12 @@ public void testCloseIndex() { assertThat(request.getEntity(), nullValue()); } + public void testCloseIndexEmptyIndices() { + String[] indices = randomBoolean() ? null : Strings.EMPTY_ARRAY; + ActionRequestValidationException validationException = new CloseIndexRequest(indices).validate(); + assertNotNull(validationException); + } + public void testIndex() throws IOException { String index = randomAlphaOfLengthBetween(3, 10); String type = randomAlphaOfLengthBetween(3, 10); @@ -539,13 +569,19 @@ public void testIndex() throws IOException { } public void testRefresh() { - String[] indices = randomIndicesNames(0, 5); - RefreshRequest refreshRequest = new RefreshRequest(indices); + String[] indices = randomBoolean() ? null : randomIndicesNames(0, 5); + RefreshRequest refreshRequest; + if (randomBoolean()) { + refreshRequest = new RefreshRequest(indices); + } else { + refreshRequest = new RefreshRequest(); + refreshRequest.indices(indices); + } Map expectedParams = new HashMap<>(); setRandomIndicesOptions(refreshRequest::indicesOptions, refreshRequest::indicesOptions, expectedParams); Request request = Request.refresh(refreshRequest); StringJoiner endpoint = new StringJoiner("/", "/", ""); - if (indices.length > 0) { + if (indices != null && indices.length > 0) { endpoint.add(String.join(",", indices)); } endpoint.add("_refresh"); @@ -556,8 +592,14 @@ public void testRefresh() { } public void testFlush() { - String[] indices = randomIndicesNames(0, 5); - FlushRequest flushRequest = new FlushRequest(indices); + String[] indices = randomBoolean() ? null : randomIndicesNames(0, 5); + FlushRequest flushRequest; + if (randomBoolean()) { + flushRequest = new FlushRequest(indices); + } else { + flushRequest = new FlushRequest(); + flushRequest.indices(indices); + } Map expectedParams = new HashMap<>(); setRandomIndicesOptions(flushRequest::indicesOptions, flushRequest::indicesOptions, expectedParams); if (randomBoolean()) { @@ -571,7 +613,7 @@ public void testFlush() { Request request = Request.flush(flushRequest); StringJoiner endpoint = new StringJoiner("/", "/", ""); - if (indices.length > 0) { + if (indices != null && indices.length > 0) { endpoint.add(String.join(",", indices)); } endpoint.add("_flush"); @@ -582,8 +624,14 @@ public void testFlush() { } public void testClearCache() { - String[] indices = randomIndicesNames(0, 5); - ClearIndicesCacheRequest clearIndicesCacheRequest = new ClearIndicesCacheRequest(indices); + String[] indices = randomBoolean() ? null : randomIndicesNames(0, 5); + ClearIndicesCacheRequest clearIndicesCacheRequest; + if (randomBoolean()) { + clearIndicesCacheRequest = new ClearIndicesCacheRequest(indices); + } else { + clearIndicesCacheRequest = new ClearIndicesCacheRequest(); + clearIndicesCacheRequest.indices(indices); + } Map expectedParams = new HashMap<>(); setRandomIndicesOptions(clearIndicesCacheRequest::indicesOptions, clearIndicesCacheRequest::indicesOptions, expectedParams); if (randomBoolean()) { @@ -605,7 +653,7 @@ public void testClearCache() { Request request = Request.clearCache(clearIndicesCacheRequest); StringJoiner endpoint = new StringJoiner("/", "/", ""); - if (indices.length > 0) { + if (indices != null && indices.length > 0) { endpoint.add(String.join(",", indices)); } endpoint.add("_cache/clear"); @@ -1027,6 +1075,12 @@ public void testSearch() throws Exception { assertToXContentBody(searchSourceBuilder, request.getEntity()); } + public void testSearchNullIndicesAndTypes() { + expectThrows(NullPointerException.class, () -> new SearchRequest((String[]) null)); + expectThrows(NullPointerException.class, () -> new SearchRequest().indices((String[]) null)); + expectThrows(NullPointerException.class, () -> new SearchRequest().types((String[]) null)); + } + public void testMultiSearch() throws IOException { int numberOfSearchRequests = randomIntBetween(0, 32); MultiSearchRequest multiSearchRequest = new MultiSearchRequest(); @@ -1107,10 +1161,16 @@ public void testClearScroll() throws IOException { public void testExistsAlias() { GetAliasesRequest getAliasesRequest = new GetAliasesRequest(); - String[] indices = randomIndicesNames(0, 5); + String[] indices = randomBoolean() ? null : randomIndicesNames(0, 5); getAliasesRequest.indices(indices); //the HEAD endpoint requires at least an alias or an index - String[] aliases = randomIndicesNames(indices.length == 0 ? 1 : 0, 5); + boolean hasIndices = indices != null && indices.length > 0; + String[] aliases; + if (hasIndices) { + aliases = randomBoolean() ? null : randomIndicesNames(0, 5); + } else { + aliases = randomIndicesNames(1, 5); + } getAliasesRequest.aliases(aliases); Map expectedParams = new HashMap<>(); setRandomLocal(getAliasesRequest, expectedParams); @@ -1118,14 +1178,12 @@ public void testExistsAlias() { Request request = Request.existsAlias(getAliasesRequest); StringJoiner expectedEndpoint = new StringJoiner("/", "/", ""); - String index = String.join(",", indices); - if (Strings.hasLength(index)) { - expectedEndpoint.add(index); + if (indices != null && indices.length > 0) { + expectedEndpoint.add(String.join(",", indices)); } expectedEndpoint.add("_alias"); - String alias = String.join(",", aliases); - if (Strings.hasLength(alias)) { - expectedEndpoint.add(alias); + if (aliases != null && aliases.length > 0) { + expectedEndpoint.add(String.join(",", aliases)); } assertEquals(HttpHead.METHOD_NAME, request.getMethod()); assertEquals(expectedEndpoint.toString(), request.getEndpoint()); @@ -1296,30 +1354,99 @@ public void testParamsNoDuplicates() { assertEquals("1", requestParams.values().iterator().next()); } - public void testBuildEndpoint() { - assertEquals("/", Request.buildEndpoint()); - assertEquals("/", Request.buildEndpoint(Strings.EMPTY_ARRAY)); - assertEquals("/", Request.buildEndpoint("")); - assertEquals("/a/b", Request.buildEndpoint("a", "b")); - assertEquals("/a/b/_create", Request.buildEndpoint("a", "b", "_create")); - assertEquals("/a/b/c/_create", Request.buildEndpoint("a", "b", "c", "_create")); - assertEquals("/a/_create", Request.buildEndpoint("a", null, null, "_create")); - } - - public void testBuildEndPointEncodeParts() { - assertEquals("/-%23index1,index%232/type/id", Request.buildEndpoint("-#index1,index#2", "type", "id")); - assertEquals("/index/type%232/id", Request.buildEndpoint("index", "type#2", "id")); - assertEquals("/index/type/this%2Fis%2Fthe%2Fid", Request.buildEndpoint("index", "type", "this/is/the/id")); - assertEquals("/index/type/this%7Cis%7Cthe%7Cid", Request.buildEndpoint("index", "type", "this|is|the|id")); - assertEquals("/index/type/id%231", Request.buildEndpoint("index", "type", "id#1")); - assertEquals("/%3Clogstash-%7Bnow%2FM%7D%3E/_search", Request.buildEndpoint("", "_search")); - assertEquals("/中文", Request.buildEndpoint("中文")); - assertEquals("/foo%20bar", Request.buildEndpoint("foo bar")); - assertEquals("/foo+bar", Request.buildEndpoint("foo+bar")); - assertEquals("/foo%2Fbar", Request.buildEndpoint("foo/bar")); - assertEquals("/foo%5Ebar", Request.buildEndpoint("foo^bar")); - assertEquals("/cluster1:index1,index2/_search", Request.buildEndpoint("cluster1:index1,index2", "_search")); - assertEquals("/*", Request.buildEndpoint("*")); + public void testEndpointBuilder() { + { + Request.EndpointBuilder endpointBuilder = new Request.EndpointBuilder(); + assertEquals("/", endpointBuilder.build()); + } + { + Request.EndpointBuilder endpointBuilder = new Request.EndpointBuilder().addPathPart(Strings.EMPTY_ARRAY); + assertEquals("/", endpointBuilder.build()); + } + { + Request.EndpointBuilder endpointBuilder = new Request.EndpointBuilder().addPathPart(""); + assertEquals("/", endpointBuilder.build()); + } + { + Request.EndpointBuilder endpointBuilder = new Request.EndpointBuilder().addPathPart("a", "b"); + assertEquals("/a/b", endpointBuilder.build()); + } + { + Request.EndpointBuilder endpointBuilder = new Request.EndpointBuilder().addPathPart("a").addPathPart("b") + .addPathPartAsIs("_create"); + assertEquals("/a/b/_create", endpointBuilder.build()); + } + + { + Request.EndpointBuilder endpointBuilder = new Request.EndpointBuilder().addPathPart("a", "b", "c") + .addPathPartAsIs("_create"); + assertEquals("/a/b/c/_create", endpointBuilder.build()); + } + { + Request.EndpointBuilder endpointBuilder = new Request.EndpointBuilder().addPathPart("a").addPathPartAsIs("_create"); + assertEquals("/a/_create", endpointBuilder.build()); + } + } + + public void testEndpointBuilderEncodeParts() { + { + Request.EndpointBuilder endpointBuilder = new Request.EndpointBuilder().addPathPart("-#index1,index#2", "type", "id"); + assertEquals("/-%23index1,index%232/type/id", endpointBuilder.build()); + } + { + Request.EndpointBuilder endpointBuilder = new Request.EndpointBuilder().addPathPart("index", "type#2", "id"); + assertEquals("/index/type%232/id", endpointBuilder.build()); + } + { + Request.EndpointBuilder endpointBuilder = new Request.EndpointBuilder().addPathPart("index", "type", "this/is/the/id"); + assertEquals("/index/type/this%2Fis%2Fthe%2Fid", endpointBuilder.build()); + } + { + Request.EndpointBuilder endpointBuilder = new Request.EndpointBuilder().addPathPart("index", "type", "this|is|the|id"); + assertEquals("/index/type/this%7Cis%7Cthe%7Cid", endpointBuilder.build()); + } + { + Request.EndpointBuilder endpointBuilder = new Request.EndpointBuilder().addPathPart("index", "type", "id#1"); + assertEquals("/index/type/id%231", endpointBuilder.build()); + } + { + Request.EndpointBuilder endpointBuilder = new Request.EndpointBuilder().addPathPart("", "_search"); + assertEquals("/%3Clogstash-%7Bnow%2FM%7D%3E/_search", endpointBuilder.build()); + } + { + Request.EndpointBuilder endpointBuilder = new Request.EndpointBuilder().addPathPart("中文"); + assertEquals("/中文", endpointBuilder.build()); + } + { + Request.EndpointBuilder endpointBuilder = new Request.EndpointBuilder().addPathPart("foo bar"); + assertEquals("/foo%20bar", endpointBuilder.build()); + } + { + Request.EndpointBuilder endpointBuilder = new Request.EndpointBuilder().addPathPart("foo+bar"); + assertEquals("/foo+bar", endpointBuilder.build()); + } + { + Request.EndpointBuilder endpointBuilder = new Request.EndpointBuilder().addPathPart("foo+bar"); + assertEquals("/foo+bar", endpointBuilder.build()); + } + { + Request.EndpointBuilder endpointBuilder = new Request.EndpointBuilder().addPathPart("foo/bar"); + assertEquals("/foo%2Fbar", endpointBuilder.build()); + } + { + Request.EndpointBuilder endpointBuilder = new Request.EndpointBuilder().addPathPart("foo^bar"); + assertEquals("/foo%5Ebar", endpointBuilder.build()); + } + { + Request.EndpointBuilder endpointBuilder = new Request.EndpointBuilder().addPathPart("cluster1:index1,index2") + .addPathPartAsIs("_search"); + assertEquals("/cluster1:index1,index2/_search", endpointBuilder.build()); + } + { + Request.EndpointBuilder endpointBuilder = new Request.EndpointBuilder() + .addCommaSeparatedPathParts(new String[]{"index1", "index2"}).addPathPartAsIs("cache/clear"); + assertEquals("/index1,index2/cache/clear", endpointBuilder.build()); + } } public void testEndpoint() { diff --git a/server/src/main/java/org/elasticsearch/action/admin/indices/get/GetIndexRequest.java b/server/src/main/java/org/elasticsearch/action/admin/indices/get/GetIndexRequest.java index 2a70aa836454e..fc260b6febcb4 100644 --- a/server/src/main/java/org/elasticsearch/action/admin/indices/get/GetIndexRequest.java +++ b/server/src/main/java/org/elasticsearch/action/admin/indices/get/GetIndexRequest.java @@ -24,9 +24,12 @@ import org.elasticsearch.common.io.stream.StreamInput; import org.elasticsearch.common.io.stream.StreamOutput; import org.elasticsearch.common.util.ArrayUtils; +import org.elasticsearch.common.util.CollectionUtils; import java.io.IOException; +import static org.elasticsearch.action.ValidateActions.addValidationError; + /** * A request to delete an index. Best created with {@link org.elasticsearch.client.Requests#deleteIndexRequest(String)}. */ @@ -106,7 +109,11 @@ public Feature[] features() { @Override public ActionRequestValidationException validate() { - return null; + ActionRequestValidationException validationException = null; + if (CollectionUtils.isEmpty(indices())) { + validationException = addValidationError("index / indices is missing", validationException); + } + return validationException; } public GetIndexRequest humanReadable(boolean humanReadable) { From 1611f65b0d911157fbb26e554bfb1f61d314b1cd Mon Sep 17 00:00:00 2001 From: javanna Date: Mon, 19 Mar 2018 18:02:07 +0100 Subject: [PATCH 4/6] remove unnecessary null check --- .../src/main/java/org/elasticsearch/client/Request.java | 6 +----- 1 file changed, 1 insertion(+), 5 deletions(-) diff --git a/client/rest-high-level/src/main/java/org/elasticsearch/client/Request.java b/client/rest-high-level/src/main/java/org/elasticsearch/client/Request.java index 6d9dba8ca3041..2aa985c8f94f5 100755 --- a/client/rest-high-level/src/main/java/org/elasticsearch/client/Request.java +++ b/client/rest-high-level/src/main/java/org/elasticsearch/client/Request.java @@ -170,15 +170,11 @@ static Request openIndex(OpenIndexRequest openIndexRequest) { } static Request closeIndex(CloseIndexRequest closeIndexRequest) { - String[] indices = closeIndexRequest.indices() == null ? Strings.EMPTY_ARRAY : closeIndexRequest.indices(); - String endpoint = endpoint(indices, "_close"); - + String endpoint = endpoint(closeIndexRequest.indices(), "_close"); Params parameters = Params.builder(); - parameters.withTimeout(closeIndexRequest.timeout()); parameters.withMasterTimeout(closeIndexRequest.masterNodeTimeout()); parameters.withIndicesOptions(closeIndexRequest.indicesOptions()); - return new Request(HttpPost.METHOD_NAME, endpoint, parameters.getParams(), null); } From 0ee3c95b9174e7c468012bca64d0959cf33c7e89 Mon Sep 17 00:00:00 2001 From: javanna Date: Mon, 19 Mar 2018 18:06:13 +0100 Subject: [PATCH 5/6] iter --- .../java/org/elasticsearch/client/Request.java | 3 ++- .../org/elasticsearch/client/RequestTests.java | 14 +++++++++++--- 2 files changed, 13 insertions(+), 4 deletions(-) diff --git a/client/rest-high-level/src/main/java/org/elasticsearch/client/Request.java b/client/rest-high-level/src/main/java/org/elasticsearch/client/Request.java index 2aa985c8f94f5..abbf4cc244b4a 100755 --- a/client/rest-high-level/src/main/java/org/elasticsearch/client/Request.java +++ b/client/rest-high-level/src/main/java/org/elasticsearch/client/Request.java @@ -520,7 +520,8 @@ static Request existsAlias(GetAliasesRequest getAliasesRequest) { Params params = Params.builder(); params.withIndicesOptions(getAliasesRequest.indicesOptions()); params.withLocal(getAliasesRequest.local()); - if ((getAliasesRequest.indices() == null || getAliasesRequest.indices().length == 0) && getAliasesRequest.aliases().length == 0) { + if ((getAliasesRequest.indices() == null || getAliasesRequest.indices().length == 0) && + (getAliasesRequest.aliases() == null || getAliasesRequest.aliases().length == 0)) { throw new IllegalArgumentException("existsAlias requires at least an alias or an index"); } String[] indices = getAliasesRequest.indices() == null ? Strings.EMPTY_ARRAY : getAliasesRequest.indices(); diff --git a/client/rest-high-level/src/test/java/org/elasticsearch/client/RequestTests.java b/client/rest-high-level/src/test/java/org/elasticsearch/client/RequestTests.java index bc2361dfb0e37..d58d0621a8e2c 100755 --- a/client/rest-high-level/src/test/java/org/elasticsearch/client/RequestTests.java +++ b/client/rest-high-level/src/test/java/org/elasticsearch/client/RequestTests.java @@ -1191,9 +1191,17 @@ public void testExistsAlias() { } public void testExistsAliasNoAliasNoIndex() { - GetAliasesRequest getAliasesRequest = new GetAliasesRequest(); - IllegalArgumentException iae = expectThrows(IllegalArgumentException.class, () -> Request.existsAlias(getAliasesRequest)); - assertEquals("existsAlias requires at least an alias or an index", iae.getMessage()); + { + GetAliasesRequest getAliasesRequest = new GetAliasesRequest(); + IllegalArgumentException iae = expectThrows(IllegalArgumentException.class, () -> Request.existsAlias(getAliasesRequest)); + assertEquals("existsAlias requires at least an alias or an index", iae.getMessage()); + } + { + GetAliasesRequest getAliasesRequest = new GetAliasesRequest((String[])null); + getAliasesRequest.indices((String[])null); + IllegalArgumentException iae = expectThrows(IllegalArgumentException.class, () -> Request.existsAlias(getAliasesRequest)); + assertEquals("existsAlias requires at least an alias or an index", iae.getMessage()); + } } public void testRankEval() throws Exception { From ee5d389d753a28005fb047828bf0047d30a74b71 Mon Sep 17 00:00:00 2001 From: javanna Date: Mon, 19 Mar 2018 18:31:16 +0100 Subject: [PATCH 6/6] fix get index issue --- .../src/main/java/org/elasticsearch/client/Request.java | 4 ++++ .../test/java/org/elasticsearch/client/RequestTests.java | 5 ++--- .../action/admin/indices/get/GetIndexRequest.java | 9 +-------- 3 files changed, 7 insertions(+), 11 deletions(-) diff --git a/client/rest-high-level/src/main/java/org/elasticsearch/client/Request.java b/client/rest-high-level/src/main/java/org/elasticsearch/client/Request.java index abbf4cc244b4a..66b34da777b6a 100755 --- a/client/rest-high-level/src/main/java/org/elasticsearch/client/Request.java +++ b/client/rest-high-level/src/main/java/org/elasticsearch/client/Request.java @@ -586,6 +586,10 @@ static Request rollover(RolloverRequest rolloverRequest) throws IOException { } static Request indicesExist(GetIndexRequest request) { + //this can be called with no indices as argument by transport client, not via REST though + if (request.indices() == null || request.indices().length == 0) { + throw new IllegalArgumentException("indices are mandatory"); + } String endpoint = endpoint(request.indices(), ""); Params params = Params.builder(); params.withLocal(request.local()); diff --git a/client/rest-high-level/src/test/java/org/elasticsearch/client/RequestTests.java b/client/rest-high-level/src/test/java/org/elasticsearch/client/RequestTests.java index d58d0621a8e2c..f79135c44f5ec 100755 --- a/client/rest-high-level/src/test/java/org/elasticsearch/client/RequestTests.java +++ b/client/rest-high-level/src/test/java/org/elasticsearch/client/RequestTests.java @@ -291,9 +291,8 @@ public void testIndicesExist() { } public void testIndicesExistEmptyIndices() { - String[] indices = randomBoolean() ? null : Strings.EMPTY_ARRAY; - ActionRequestValidationException validationException = new GetIndexRequest().indices(indices).validate(); - assertNotNull(validationException); + expectThrows(IllegalArgumentException.class, () -> Request.indicesExist(new GetIndexRequest())); + expectThrows(IllegalArgumentException.class, () -> Request.indicesExist(new GetIndexRequest().indices((String[])null))); } private static void getAndExistsTest(Function requestConverter, String method) { diff --git a/server/src/main/java/org/elasticsearch/action/admin/indices/get/GetIndexRequest.java b/server/src/main/java/org/elasticsearch/action/admin/indices/get/GetIndexRequest.java index fc260b6febcb4..2a70aa836454e 100644 --- a/server/src/main/java/org/elasticsearch/action/admin/indices/get/GetIndexRequest.java +++ b/server/src/main/java/org/elasticsearch/action/admin/indices/get/GetIndexRequest.java @@ -24,12 +24,9 @@ import org.elasticsearch.common.io.stream.StreamInput; import org.elasticsearch.common.io.stream.StreamOutput; import org.elasticsearch.common.util.ArrayUtils; -import org.elasticsearch.common.util.CollectionUtils; import java.io.IOException; -import static org.elasticsearch.action.ValidateActions.addValidationError; - /** * A request to delete an index. Best created with {@link org.elasticsearch.client.Requests#deleteIndexRequest(String)}. */ @@ -109,11 +106,7 @@ public Feature[] features() { @Override public ActionRequestValidationException validate() { - ActionRequestValidationException validationException = null; - if (CollectionUtils.isEmpty(indices())) { - validationException = addValidationError("index / indices is missing", validationException); - } - return validationException; + return null; } public GetIndexRequest humanReadable(boolean humanReadable) {