From 9c0e846db3d7975eca8441914239be87bcb7bd67 Mon Sep 17 00:00:00 2001 From: Lee Hinman Date: Tue, 17 Mar 2020 13:23:28 -0600 Subject: [PATCH] [7.x] Add REST API for ComponentTemplate CRUD (#53558) (#53681) * Add REST API for ComponentTemplate CRUD This adds the Put/Get/DeleteComponentTemplate APIs that allow inserting, retrieving, and removing ComponentTemplateMetadata into the cluster state metadata. These APIs are currently only available behind a feature flag system property - `es.itv2_feature_flag_registered`. Relates to #53101 Co-authored-by: Elastic Machine Co-authored-by: Elastic Machine --- .../client/RestHighLevelClientTests.java | 5 +- .../archives/integ-test-zip/build.gradle | 7 + .../cluster.delete_component_template.json | 35 ++++ .../api/cluster.get_component_template.json | 41 ++++ .../api/cluster.put_component_template.json | 45 +++++ .../cluster.component_template/10_basic.yml | 47 +++++ .../elasticsearch/action/ActionModule.java | 38 ++++ .../delete/DeleteComponentTemplateAction.java | 90 +++++++++ ...ransportDeleteComponentTemplateAction.java | 78 ++++++++ .../get/GetComponentTemplateAction.java | 171 ++++++++++++++++ .../TransportGetComponentTemplateAction.java | 93 +++++++++ .../put/PutComponentTemplateAction.java | 151 ++++++++++++++ .../TransportPutComponentTemplateAction.java | 87 ++++++++ .../cluster/metadata/ComponentTemplate.java | 149 +------------- .../MetaDataIndexTemplateService.java | 100 ++++++++++ .../cluster/metadata/Template.java | 187 ++++++++++++++++++ .../RestDeleteComponentTemplateAction.java | 54 +++++ .../RestGetComponentTemplateAction.java | 83 ++++++++ .../RestPutComponentTemplateAction.java | 61 ++++++ .../GetComponentTemplateResponseTests.java | 54 +++++ .../metadata/ComponentTemplateTests.java | 14 +- .../MetaDataIndexTemplateServiceTests.java | 24 ++- .../metadata/ToAndFromJsonMetaDataTests.java | 4 +- 23 files changed, 1454 insertions(+), 164 deletions(-) create mode 100644 rest-api-spec/src/main/resources/rest-api-spec/api/cluster.delete_component_template.json create mode 100644 rest-api-spec/src/main/resources/rest-api-spec/api/cluster.get_component_template.json create mode 100644 rest-api-spec/src/main/resources/rest-api-spec/api/cluster.put_component_template.json create mode 100644 rest-api-spec/src/main/resources/rest-api-spec/test/cluster.component_template/10_basic.yml create mode 100644 server/src/main/java/org/elasticsearch/action/admin/indices/template/delete/DeleteComponentTemplateAction.java create mode 100644 server/src/main/java/org/elasticsearch/action/admin/indices/template/delete/TransportDeleteComponentTemplateAction.java create mode 100644 server/src/main/java/org/elasticsearch/action/admin/indices/template/get/GetComponentTemplateAction.java create mode 100644 server/src/main/java/org/elasticsearch/action/admin/indices/template/get/TransportGetComponentTemplateAction.java create mode 100644 server/src/main/java/org/elasticsearch/action/admin/indices/template/put/PutComponentTemplateAction.java create mode 100644 server/src/main/java/org/elasticsearch/action/admin/indices/template/put/TransportPutComponentTemplateAction.java create mode 100644 server/src/main/java/org/elasticsearch/cluster/metadata/Template.java create mode 100644 server/src/main/java/org/elasticsearch/rest/action/admin/indices/RestDeleteComponentTemplateAction.java create mode 100644 server/src/main/java/org/elasticsearch/rest/action/admin/indices/RestGetComponentTemplateAction.java create mode 100644 server/src/main/java/org/elasticsearch/rest/action/admin/indices/RestPutComponentTemplateAction.java create mode 100644 server/src/test/java/org/elasticsearch/action/admin/indices/template/get/GetComponentTemplateResponseTests.java rename server/src/test/java/org/elasticsearch/{action/admin/indices/template/put => cluster/metadata}/MetaDataIndexTemplateServiceTests.java (93%) diff --git a/client/rest-high-level/src/test/java/org/elasticsearch/client/RestHighLevelClientTests.java b/client/rest-high-level/src/test/java/org/elasticsearch/client/RestHighLevelClientTests.java index b37c1404c82c5..f781856bfe6f6 100644 --- a/client/rest-high-level/src/test/java/org/elasticsearch/client/RestHighLevelClientTests.java +++ b/client/rest-high-level/src/test/java/org/elasticsearch/client/RestHighLevelClientTests.java @@ -790,7 +790,10 @@ public void testApiNamingConventions() throws Exception { "indices.get_upgrade", "indices.put_alias", "render_search_template", - "scripts_painless_execute" + "scripts_painless_execute", + "cluster.put_component_template", + "cluster.get_component_template", + "cluster.delete_component_template" }; //These API are not required for high-level client feature completeness String[] notRequiredApi = new String[] { diff --git a/distribution/archives/integ-test-zip/build.gradle b/distribution/archives/integ-test-zip/build.gradle index ed8f893d7f840..5fb5b36553b91 100644 --- a/distribution/archives/integ-test-zip/build.gradle +++ b/distribution/archives/integ-test-zip/build.gradle @@ -1,3 +1,4 @@ +import org.elasticsearch.gradle.info.BuildParams /* * Licensed to Elasticsearch under one or more contributor * license agreements. See the NOTICE file distributed with @@ -32,3 +33,9 @@ integTest.runner { systemProperty 'tests.logfile', '--external--' } } + +testClusters.integTest { + if (BuildParams.isSnapshotBuild() == false) { + systemProperty 'es.itv2_feature_flag_registered', 'true' + } +} diff --git a/rest-api-spec/src/main/resources/rest-api-spec/api/cluster.delete_component_template.json b/rest-api-spec/src/main/resources/rest-api-spec/api/cluster.delete_component_template.json new file mode 100644 index 0000000000000..6ddfe6c7ead5f --- /dev/null +++ b/rest-api-spec/src/main/resources/rest-api-spec/api/cluster.delete_component_template.json @@ -0,0 +1,35 @@ +{ + "cluster.delete_component_template":{ + "documentation":{ + "url":"https://www.elastic.co/guide/en/elasticsearch/reference/master/indices-component-templates.html", + "description":"Deletes a component template" + }, + "stability":"stable", + "url":{ + "paths":[ + { + "path":"/_component_template/{name}", + "methods":[ + "DELETE" + ], + "parts":{ + "name":{ + "type":"string", + "description":"The name of the template" + } + } + } + ] + }, + "params":{ + "timeout":{ + "type":"time", + "description":"Explicit operation timeout" + }, + "master_timeout":{ + "type":"time", + "description":"Specify timeout for connection to master" + } + } + } +} diff --git a/rest-api-spec/src/main/resources/rest-api-spec/api/cluster.get_component_template.json b/rest-api-spec/src/main/resources/rest-api-spec/api/cluster.get_component_template.json new file mode 100644 index 0000000000000..27bd093e620f0 --- /dev/null +++ b/rest-api-spec/src/main/resources/rest-api-spec/api/cluster.get_component_template.json @@ -0,0 +1,41 @@ +{ + "cluster.get_component_template":{ + "documentation":{ + "url":"https://www.elastic.co/guide/en/elasticsearch/reference/master/indices-component-templates.html", + "description":"Returns one or more component templates" + }, + "stability":"stable", + "url":{ + "paths":[ + { + "path":"/_component_template", + "methods":[ + "GET" + ] + }, + { + "path":"/_component_template/{name}", + "methods":[ + "GET" + ], + "parts":{ + "name":{ + "type":"list", + "description":"The comma separated names of the component templates" + } + } + } + ] + }, + "params":{ + "master_timeout":{ + "type":"time", + "description":"Explicit operation timeout for connection to master node" + }, + "local":{ + "type":"boolean", + "description":"Return local information, do not retrieve the state from master node (default: false)" + } + } + } +} diff --git a/rest-api-spec/src/main/resources/rest-api-spec/api/cluster.put_component_template.json b/rest-api-spec/src/main/resources/rest-api-spec/api/cluster.put_component_template.json new file mode 100644 index 0000000000000..3bdac2357d023 --- /dev/null +++ b/rest-api-spec/src/main/resources/rest-api-spec/api/cluster.put_component_template.json @@ -0,0 +1,45 @@ +{ + "cluster.put_component_template":{ + "documentation":{ + "url":"https://www.elastic.co/guide/en/elasticsearch/reference/master/indices-component-templates.html", + "description":"Creates or updates a component template" + }, + "stability":"stable", + "url":{ + "paths":[ + { + "path":"/_component_template/{name}", + "methods":[ + "PUT", + "POST" + ], + "parts":{ + "name":{ + "type":"string", + "description":"The name of the template" + } + } + } + ] + }, + "params":{ + "create":{ + "type":"boolean", + "description":"Whether the index template should only be added if new or can also replace an existing one", + "default":false + }, + "timeout":{ + "type":"time", + "description":"Explicit operation timeout" + }, + "master_timeout":{ + "type":"time", + "description":"Specify timeout for connection to master" + } + }, + "body":{ + "description":"The template definition", + "required":true + } + } +} diff --git a/rest-api-spec/src/main/resources/rest-api-spec/test/cluster.component_template/10_basic.yml b/rest-api-spec/src/main/resources/rest-api-spec/test/cluster.component_template/10_basic.yml new file mode 100644 index 0000000000000..1ce79e3ca18c7 --- /dev/null +++ b/rest-api-spec/src/main/resources/rest-api-spec/test/cluster.component_template/10_basic.yml @@ -0,0 +1,47 @@ +--- +"Basic CRUD": + - skip: + version: " - 7.99.99" + reason: not backported yet + + - do: + cluster.put_component_template: + name: test + body: + template: + settings: + number_of_shards: 1 + number_of_replicas: 0 + mappings: + properties: + field: + type: keyword + aliases: + aliasname: {} + version: 2 + _meta: + foo: bar + baz: + eggplant: true + + - do: + cluster.get_component_template: + name: test + + - match: {component_templates.0.name: test} + - match: {component_templates.0.component_template.version: 2} + - match: {component_templates.0.component_template._meta: {foo: bar, baz: {eggplant: true}}} + - match: {component_templates.0.component_template.template.settings: {index: {number_of_shards: '1', number_of_replicas: '0'}}} + - match: {component_templates.0.component_template.template.mappings: {properties: {field: {type: keyword}}}} + - match: {component_templates.0.component_template.template.aliases: {aliasname: {}}} + + - do: + cluster.delete_component_template: + name: test + + - do: + catch: missing + cluster.get_component_template: + name: test + + - is_false: test diff --git a/server/src/main/java/org/elasticsearch/action/ActionModule.java b/server/src/main/java/org/elasticsearch/action/ActionModule.java index f0c223f75bdf4..45a9e1c3b5da5 100644 --- a/server/src/main/java/org/elasticsearch/action/ActionModule.java +++ b/server/src/main/java/org/elasticsearch/action/ActionModule.java @@ -21,6 +21,7 @@ import org.apache.logging.log4j.LogManager; import org.apache.logging.log4j.Logger; +import org.elasticsearch.Build; import org.elasticsearch.action.admin.cluster.allocation.ClusterAllocationExplainAction; import org.elasticsearch.action.admin.cluster.allocation.TransportClusterAllocationExplainAction; import org.elasticsearch.action.admin.cluster.configuration.AddVotingConfigExclusionsAction; @@ -147,11 +148,17 @@ import org.elasticsearch.action.admin.indices.shrink.TransportResizeAction; import org.elasticsearch.action.admin.indices.stats.IndicesStatsAction; import org.elasticsearch.action.admin.indices.stats.TransportIndicesStatsAction; +import org.elasticsearch.action.admin.indices.template.delete.DeleteComponentTemplateAction; import org.elasticsearch.action.admin.indices.template.delete.DeleteIndexTemplateAction; +import org.elasticsearch.action.admin.indices.template.delete.TransportDeleteComponentTemplateAction; import org.elasticsearch.action.admin.indices.template.delete.TransportDeleteIndexTemplateAction; +import org.elasticsearch.action.admin.indices.template.get.GetComponentTemplateAction; import org.elasticsearch.action.admin.indices.template.get.GetIndexTemplatesAction; +import org.elasticsearch.action.admin.indices.template.get.TransportGetComponentTemplateAction; import org.elasticsearch.action.admin.indices.template.get.TransportGetIndexTemplatesAction; +import org.elasticsearch.action.admin.indices.template.put.PutComponentTemplateAction; import org.elasticsearch.action.admin.indices.template.put.PutIndexTemplateAction; +import org.elasticsearch.action.admin.indices.template.put.TransportPutComponentTemplateAction; import org.elasticsearch.action.admin.indices.template.put.TransportPutIndexTemplateAction; import org.elasticsearch.action.admin.indices.upgrade.get.TransportUpgradeStatusAction; import org.elasticsearch.action.admin.indices.upgrade.get.UpgradeStatusAction; @@ -270,11 +277,13 @@ import org.elasticsearch.rest.action.admin.indices.RestClearIndicesCacheAction; import org.elasticsearch.rest.action.admin.indices.RestCloseIndexAction; import org.elasticsearch.rest.action.admin.indices.RestCreateIndexAction; +import org.elasticsearch.rest.action.admin.indices.RestDeleteComponentTemplateAction; import org.elasticsearch.rest.action.admin.indices.RestDeleteIndexAction; import org.elasticsearch.rest.action.admin.indices.RestDeleteIndexTemplateAction; import org.elasticsearch.rest.action.admin.indices.RestFlushAction; import org.elasticsearch.rest.action.admin.indices.RestForceMergeAction; import org.elasticsearch.rest.action.admin.indices.RestGetAliasesAction; +import org.elasticsearch.rest.action.admin.indices.RestGetComponentTemplateAction; import org.elasticsearch.rest.action.admin.indices.RestGetFieldMappingAction; import org.elasticsearch.rest.action.admin.indices.RestGetIndexTemplateAction; import org.elasticsearch.rest.action.admin.indices.RestGetIndicesAction; @@ -287,6 +296,7 @@ import org.elasticsearch.rest.action.admin.indices.RestIndicesShardStoresAction; import org.elasticsearch.rest.action.admin.indices.RestIndicesStatsAction; import org.elasticsearch.rest.action.admin.indices.RestOpenIndexAction; +import org.elasticsearch.rest.action.admin.indices.RestPutComponentTemplateAction; import org.elasticsearch.rest.action.admin.indices.RestPutIndexTemplateAction; import org.elasticsearch.rest.action.admin.indices.RestPutMappingAction; import org.elasticsearch.rest.action.admin.indices.RestRecoveryAction; @@ -363,6 +373,24 @@ public class ActionModule extends AbstractModule { private static final Logger logger = LogManager.getLogger(ActionModule.class); private final boolean transportClient; + + private static final boolean ITV2_FEATURE_FLAG_REGISTERED; + + static { + final String property = System.getProperty("es.itv2_feature_flag_registered"); + if (Build.CURRENT.isSnapshot() && property != null) { + throw new IllegalArgumentException("es.itv2_feature_flag_registered is only supported in non-snapshot builds"); + } + if (Build.CURRENT.isSnapshot() || "true".equals(property)) { + ITV2_FEATURE_FLAG_REGISTERED = true; + } else if ("false".equals(property) || property == null) { + ITV2_FEATURE_FLAG_REGISTERED = false; + } else { + throw new IllegalArgumentException("expected es.itv2_feature_flag_registered to be unset, true, or false but was [" + + property + "]"); + } + } + private final Settings settings; private final IndexNameExpressionResolver indexNameExpressionResolver; private final IndexScopedSettings indexScopedSettings; @@ -497,6 +525,11 @@ public void reg actions.register(PutIndexTemplateAction.INSTANCE, TransportPutIndexTemplateAction.class); actions.register(GetIndexTemplatesAction.INSTANCE, TransportGetIndexTemplatesAction.class); actions.register(DeleteIndexTemplateAction.INSTANCE, TransportDeleteIndexTemplateAction.class); + if (ITV2_FEATURE_FLAG_REGISTERED) { + actions.register(PutComponentTemplateAction.INSTANCE, TransportPutComponentTemplateAction.class); + actions.register(GetComponentTemplateAction.INSTANCE, TransportGetComponentTemplateAction.class); + actions.register(DeleteComponentTemplateAction.INSTANCE, TransportDeleteComponentTemplateAction.class); + } actions.register(ValidateQueryAction.INSTANCE, TransportValidateQueryAction.class); actions.register(RefreshAction.INSTANCE, TransportRefreshAction.class); actions.register(FlushAction.INSTANCE, TransportFlushAction.class); @@ -624,6 +657,11 @@ public void initRestHandlers(Supplier nodesInCluster) { registerHandler.accept(new RestGetIndexTemplateAction()); registerHandler.accept(new RestPutIndexTemplateAction()); registerHandler.accept(new RestDeleteIndexTemplateAction()); + if (ITV2_FEATURE_FLAG_REGISTERED) { + registerHandler.accept(new RestPutComponentTemplateAction()); + registerHandler.accept(new RestGetComponentTemplateAction()); + registerHandler.accept(new RestDeleteComponentTemplateAction()); + } registerHandler.accept(new RestPutMappingAction()); registerHandler.accept(new RestGetMappingAction()); diff --git a/server/src/main/java/org/elasticsearch/action/admin/indices/template/delete/DeleteComponentTemplateAction.java b/server/src/main/java/org/elasticsearch/action/admin/indices/template/delete/DeleteComponentTemplateAction.java new file mode 100644 index 0000000000000..a0dc94a92f14b --- /dev/null +++ b/server/src/main/java/org/elasticsearch/action/admin/indices/template/delete/DeleteComponentTemplateAction.java @@ -0,0 +1,90 @@ +/* + * 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.template.delete; + +import org.elasticsearch.action.ActionRequestValidationException; +import org.elasticsearch.action.ActionType; +import org.elasticsearch.action.support.master.AcknowledgedResponse; +import org.elasticsearch.action.support.master.MasterNodeRequest; +import org.elasticsearch.common.io.stream.StreamInput; +import org.elasticsearch.common.io.stream.StreamOutput; + +import java.io.IOException; + +import static org.elasticsearch.action.ValidateActions.addValidationError; + +public class DeleteComponentTemplateAction extends ActionType { + + public static final DeleteComponentTemplateAction INSTANCE = new DeleteComponentTemplateAction(); + public static final String NAME = "cluster:admin/component_template/delete"; + + private DeleteComponentTemplateAction() { + super(NAME, AcknowledgedResponse::new); + } + + public static class Request extends MasterNodeRequest { + + private String name; + + public Request(StreamInput in) throws IOException { + super(in); + name = in.readString(); + } + + public Request() { } + + /** + * Constructs a new delete index request for the specified name. + */ + public Request(String name) { + this.name = name; + } + + /** + * Set the index template name to delete. + */ + public Request name(String name) { + this.name = name; + return this; + } + + @Override + public ActionRequestValidationException validate() { + ActionRequestValidationException validationException = null; + if (name == null) { + validationException = addValidationError("name is missing", validationException); + } + return validationException; + } + + /** + * The index template name to delete. + */ + public String name() { + return name; + } + + @Override + public void writeTo(StreamOutput out) throws IOException { + super.writeTo(out); + out.writeString(name); + } + } +} diff --git a/server/src/main/java/org/elasticsearch/action/admin/indices/template/delete/TransportDeleteComponentTemplateAction.java b/server/src/main/java/org/elasticsearch/action/admin/indices/template/delete/TransportDeleteComponentTemplateAction.java new file mode 100644 index 0000000000000..df6f53576cb32 --- /dev/null +++ b/server/src/main/java/org/elasticsearch/action/admin/indices/template/delete/TransportDeleteComponentTemplateAction.java @@ -0,0 +1,78 @@ +/* + * 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.template.delete; + +import org.apache.logging.log4j.LogManager; +import org.apache.logging.log4j.Logger; +import org.elasticsearch.action.ActionListener; +import org.elasticsearch.action.support.ActionFilters; +import org.elasticsearch.action.support.master.AcknowledgedResponse; +import org.elasticsearch.action.support.master.TransportMasterNodeAction; +import org.elasticsearch.cluster.ClusterState; +import org.elasticsearch.cluster.block.ClusterBlockException; +import org.elasticsearch.cluster.block.ClusterBlockLevel; +import org.elasticsearch.cluster.metadata.IndexNameExpressionResolver; +import org.elasticsearch.cluster.metadata.MetaDataIndexTemplateService; +import org.elasticsearch.cluster.service.ClusterService; +import org.elasticsearch.common.inject.Inject; +import org.elasticsearch.common.io.stream.StreamInput; +import org.elasticsearch.threadpool.ThreadPool; +import org.elasticsearch.transport.TransportService; + +import java.io.IOException; + +public class TransportDeleteComponentTemplateAction + extends TransportMasterNodeAction { + + private static final Logger logger = LogManager.getLogger(TransportDeleteComponentTemplateAction.class); + + private final MetaDataIndexTemplateService indexTemplateService; + + @Inject + public TransportDeleteComponentTemplateAction(TransportService transportService, ClusterService clusterService, + ThreadPool threadPool, MetaDataIndexTemplateService indexTemplateService, + ActionFilters actionFilters, IndexNameExpressionResolver indexNameExpressionResolver) { + super(DeleteComponentTemplateAction.NAME, transportService, clusterService, threadPool, actionFilters, + DeleteComponentTemplateAction.Request::new, indexNameExpressionResolver); + this.indexTemplateService = indexTemplateService; + } + + @Override + protected String executor() { + // we go async right away + return ThreadPool.Names.SAME; + } + + @Override + protected AcknowledgedResponse read(StreamInput in) throws IOException { + return new AcknowledgedResponse(in); + } + + @Override + protected ClusterBlockException checkBlock(DeleteComponentTemplateAction.Request request, ClusterState state) { + return state.blocks().globalBlockedException(ClusterBlockLevel.METADATA_WRITE); + } + + @Override + protected void masterOperation(final DeleteComponentTemplateAction.Request request, final ClusterState state, + final ActionListener listener) { + indexTemplateService.removeComponentTemplate(request.name(), request.masterNodeTimeout(), listener); + } +} diff --git a/server/src/main/java/org/elasticsearch/action/admin/indices/template/get/GetComponentTemplateAction.java b/server/src/main/java/org/elasticsearch/action/admin/indices/template/get/GetComponentTemplateAction.java new file mode 100644 index 0000000000000..34242c1d6aa9d --- /dev/null +++ b/server/src/main/java/org/elasticsearch/action/admin/indices/template/get/GetComponentTemplateAction.java @@ -0,0 +1,171 @@ +/* + * 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.template.get; + +import org.elasticsearch.action.ActionRequestValidationException; +import org.elasticsearch.action.ActionResponse; +import org.elasticsearch.action.ActionType; +import org.elasticsearch.action.support.master.MasterNodeReadRequest; +import org.elasticsearch.cluster.metadata.ComponentTemplate; +import org.elasticsearch.common.ParseField; +import org.elasticsearch.common.Strings; +import org.elasticsearch.common.io.stream.StreamInput; +import org.elasticsearch.common.io.stream.StreamOutput; +import org.elasticsearch.common.xcontent.ToXContentObject; +import org.elasticsearch.common.xcontent.XContentBuilder; + +import java.io.IOException; +import java.util.HashMap; +import java.util.Map; +import java.util.Objects; + +import static org.elasticsearch.action.ValidateActions.addValidationError; + +/** + * Action to retrieve one or more component templates + */ +public class GetComponentTemplateAction extends ActionType { + + public static final GetComponentTemplateAction INSTANCE = new GetComponentTemplateAction(); + public static final String NAME = "cluster:admin/component_template/get"; + + private GetComponentTemplateAction() { + super(NAME, GetComponentTemplateAction.Response::new); + } + + /** + * Request that to retrieve one or more component templates + */ + public static class Request extends MasterNodeReadRequest { + + private String[] names; + + public Request() { } + + public Request(String... names) { + this.names = names; + } + + public Request(StreamInput in) throws IOException { + super(in); + names = in.readStringArray(); + } + + @Override + public void writeTo(StreamOutput out) throws IOException { + super.writeTo(out); + out.writeStringArray(names); + } + + @Override + public ActionRequestValidationException validate() { + ActionRequestValidationException validationException = null; + if (names == null) { + validationException = addValidationError("names is null or empty", validationException); + } else { + for (String name : names) { + if (name == null || Strings.hasText(name) == false) { + validationException = addValidationError("name is missing", validationException); + } + } + } + return validationException; + } + + /** + * Sets the names of the component templates. + */ + public Request names(String... names) { + this.names = names; + return this; + } + + /** + * The names of the component templates. + */ + public String[] names() { + return this.names; + } + } + + public static class Response extends ActionResponse implements ToXContentObject { + public static final ParseField NAME = new ParseField("name"); + public static final ParseField COMPONENT_TEMPLATES = new ParseField("component_templates"); + public static final ParseField COMPONENT_TEMPLATE = new ParseField("component_template"); + + private final Map componentTemplates; + + public Response(StreamInput in) throws IOException { + super(in); + int size = in.readVInt(); + componentTemplates = new HashMap<>(); + for (int i = 0 ; i < size ; i++) { + componentTemplates.put(in.readString(), new ComponentTemplate(in)); + } + } + + public Response(Map componentTemplates) { + this.componentTemplates = componentTemplates; + } + + public Map getComponentTemplates() { + return componentTemplates; + } + + @Override + public void writeTo(StreamOutput out) throws IOException { + out.writeVInt(componentTemplates.size()); + for (Map.Entry componentTemplate : componentTemplates.entrySet()) { + out.writeString(componentTemplate.getKey()); + componentTemplate.getValue().writeTo(out); + } + } + + @Override + public boolean equals(Object o) { + if (this == o) return true; + if (o == null || getClass() != o.getClass()) return false; + Response that = (Response) o; + return Objects.equals(componentTemplates, that.componentTemplates); + } + + @Override + public int hashCode() { + return Objects.hash(componentTemplates); + } + + @Override + public XContentBuilder toXContent(XContentBuilder builder, Params params) throws IOException { + builder.startObject(); + builder.startArray(COMPONENT_TEMPLATES.getPreferredName()); + for (Map.Entry componentTemplate : this.componentTemplates.entrySet()) { + builder.startObject(); + builder.field(NAME.getPreferredName(), componentTemplate.getKey()); + builder.field(COMPONENT_TEMPLATE.getPreferredName(), componentTemplate.getValue()); + builder.endObject(); + } + builder.endArray(); + builder.endObject(); + return builder; + } + + } + +} diff --git a/server/src/main/java/org/elasticsearch/action/admin/indices/template/get/TransportGetComponentTemplateAction.java b/server/src/main/java/org/elasticsearch/action/admin/indices/template/get/TransportGetComponentTemplateAction.java new file mode 100644 index 0000000000000..250f943a5912f --- /dev/null +++ b/server/src/main/java/org/elasticsearch/action/admin/indices/template/get/TransportGetComponentTemplateAction.java @@ -0,0 +1,93 @@ +/* + * 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.template.get; + +import org.elasticsearch.action.ActionListener; +import org.elasticsearch.action.support.ActionFilters; +import org.elasticsearch.action.support.master.TransportMasterNodeReadAction; +import org.elasticsearch.cluster.ClusterState; +import org.elasticsearch.cluster.block.ClusterBlockException; +import org.elasticsearch.cluster.block.ClusterBlockLevel; +import org.elasticsearch.cluster.metadata.ComponentTemplate; +import org.elasticsearch.cluster.metadata.IndexNameExpressionResolver; +import org.elasticsearch.cluster.service.ClusterService; +import org.elasticsearch.common.inject.Inject; +import org.elasticsearch.common.io.stream.StreamInput; +import org.elasticsearch.common.regex.Regex; +import org.elasticsearch.threadpool.ThreadPool; +import org.elasticsearch.transport.TransportService; + +import java.io.IOException; +import java.util.HashMap; +import java.util.Map; + +public class TransportGetComponentTemplateAction extends + TransportMasterNodeReadAction { + + @Inject + public TransportGetComponentTemplateAction(TransportService transportService, ClusterService clusterService, + ThreadPool threadPool, ActionFilters actionFilters, + IndexNameExpressionResolver indexNameExpressionResolver) { + super(GetComponentTemplateAction.NAME, transportService, clusterService, threadPool, actionFilters, + GetComponentTemplateAction.Request::new, indexNameExpressionResolver); + } + + @Override + protected String executor() { + return ThreadPool.Names.SAME; + } + + @Override + protected GetComponentTemplateAction.Response read(StreamInput in) throws IOException { + return new GetComponentTemplateAction.Response(in); + } + + @Override + protected ClusterBlockException checkBlock(GetComponentTemplateAction.Request request, ClusterState state) { + return state.blocks().globalBlockedException(ClusterBlockLevel.METADATA_READ); + } + + @Override + protected void masterOperation(GetComponentTemplateAction.Request request, ClusterState state, + ActionListener listener) { + Map allTemplates = state.metaData().componentTemplates(); + + // If we did not ask for a specific name, then we return all templates + if (request.names().length == 0) { + listener.onResponse(new GetComponentTemplateAction.Response(allTemplates)); + return; + } + + final Map results = new HashMap<>(); + for (String name : request.names()) { + if (Regex.isSimpleMatchPattern(name)) { + for (Map.Entry entry : allTemplates.entrySet()) { + if (Regex.simpleMatch(name, entry.getKey())) { + results.put(entry.getKey(), entry.getValue()); + } + } + } else if (allTemplates.containsKey(name)) { + results.put(name, allTemplates.get(name)); + } + } + + listener.onResponse(new GetComponentTemplateAction.Response(results)); + } +} diff --git a/server/src/main/java/org/elasticsearch/action/admin/indices/template/put/PutComponentTemplateAction.java b/server/src/main/java/org/elasticsearch/action/admin/indices/template/put/PutComponentTemplateAction.java new file mode 100644 index 0000000000000..448be77793e5e --- /dev/null +++ b/server/src/main/java/org/elasticsearch/action/admin/indices/template/put/PutComponentTemplateAction.java @@ -0,0 +1,151 @@ +/* + * 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.template.put; + +import org.elasticsearch.action.ActionRequestValidationException; +import org.elasticsearch.action.ActionType; +import org.elasticsearch.action.support.master.AcknowledgedResponse; +import org.elasticsearch.action.support.master.MasterNodeRequest; +import org.elasticsearch.cluster.metadata.ComponentTemplate; +import org.elasticsearch.common.Nullable; +import org.elasticsearch.common.Strings; +import org.elasticsearch.common.io.stream.StreamInput; +import org.elasticsearch.common.io.stream.StreamOutput; + +import java.io.IOException; + +import static org.elasticsearch.action.ValidateActions.addValidationError; + +/** + * An action for putting a single component template into the cluster state + */ +public class PutComponentTemplateAction extends ActionType { + + public static final PutComponentTemplateAction INSTANCE = new PutComponentTemplateAction(); + public static final String NAME = "cluster:admin/component_template/put"; + + private PutComponentTemplateAction() { + super(NAME, AcknowledgedResponse::new); + } + + /** + * A request for putting a single component template into the cluster state + */ + public static class Request extends MasterNodeRequest { + private final String name; + @Nullable + private String cause; + private boolean create; + private ComponentTemplate componentTemplate; + + public Request(StreamInput in) throws IOException { + super(in); + this.name = in.readString(); + this.cause = in.readOptionalString(); + this.create = in.readBoolean(); + this.componentTemplate = new ComponentTemplate(in); + } + + /** + * Constructs a new put component template request with the provided name. + */ + public Request(String name) { + this.name = name; + } + + @Override + public void writeTo(StreamOutput out) throws IOException { + super.writeTo(out); + out.writeString(name); + out.writeOptionalString(cause); + out.writeBoolean(create); + this.componentTemplate.writeTo(out); + } + + @Override + public ActionRequestValidationException validate() { + ActionRequestValidationException validationException = null; + if (name == null || Strings.hasText(name) == false) { + validationException = addValidationError("name is missing", validationException); + } + if (componentTemplate == null) { + validationException = addValidationError("a component template is required", validationException); + } + return validationException; + } + + /** + * The name of the index template. + */ + public String name() { + return this.name; + } + + /** + * Set to {@code true} to force only creation, not an update of an index template. If it already + * exists, it will fail with an {@link IllegalArgumentException}. + */ + public Request create(boolean create) { + this.create = create; + return this; + } + + public boolean create() { + return create; + } + + /** + * The cause for this index template creation. + */ + public Request cause(@Nullable String cause) { + this.cause = cause; + return this; + } + + @Nullable + public String cause() { + return this.cause; + } + + /** + * The component template that will be inserted into the cluster state + */ + public Request componentTemplate(ComponentTemplate template) { + this.componentTemplate = template; + return this; + } + + public ComponentTemplate componentTemplate() { + return this.componentTemplate; + } + + @Override + public String toString() { + StringBuilder sb = new StringBuilder("PutComponentRequest["); + sb.append("name=").append(name); + sb.append(", cause=").append(cause); + sb.append(", create=").append(create); + sb.append(", component_template=").append(componentTemplate); + sb.append("]"); + return sb.toString(); + } + } + +} diff --git a/server/src/main/java/org/elasticsearch/action/admin/indices/template/put/TransportPutComponentTemplateAction.java b/server/src/main/java/org/elasticsearch/action/admin/indices/template/put/TransportPutComponentTemplateAction.java new file mode 100644 index 0000000000000..6e71afaaa5d8d --- /dev/null +++ b/server/src/main/java/org/elasticsearch/action/admin/indices/template/put/TransportPutComponentTemplateAction.java @@ -0,0 +1,87 @@ +/* + * 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.template.put; + +import org.elasticsearch.action.ActionListener; +import org.elasticsearch.action.support.ActionFilters; +import org.elasticsearch.action.support.master.AcknowledgedResponse; +import org.elasticsearch.action.support.master.TransportMasterNodeAction; +import org.elasticsearch.cluster.ClusterState; +import org.elasticsearch.cluster.block.ClusterBlockException; +import org.elasticsearch.cluster.block.ClusterBlockLevel; +import org.elasticsearch.cluster.metadata.ComponentTemplate; +import org.elasticsearch.cluster.metadata.IndexMetaData; +import org.elasticsearch.cluster.metadata.IndexNameExpressionResolver; +import org.elasticsearch.cluster.metadata.MetaDataIndexTemplateService; +import org.elasticsearch.cluster.metadata.Template; +import org.elasticsearch.cluster.service.ClusterService; +import org.elasticsearch.common.inject.Inject; +import org.elasticsearch.common.io.stream.StreamInput; +import org.elasticsearch.common.settings.Settings; +import org.elasticsearch.threadpool.ThreadPool; +import org.elasticsearch.transport.TransportService; + +import java.io.IOException; + +public class TransportPutComponentTemplateAction + extends TransportMasterNodeAction { + + private final MetaDataIndexTemplateService indexTemplateService; + + @Inject + public TransportPutComponentTemplateAction(TransportService transportService, ClusterService clusterService, + ThreadPool threadPool, MetaDataIndexTemplateService indexTemplateService, + ActionFilters actionFilters, IndexNameExpressionResolver indexNameExpressionResolver) { + super(PutComponentTemplateAction.NAME, transportService, clusterService, threadPool, actionFilters, + PutComponentTemplateAction.Request::new, indexNameExpressionResolver); + this.indexTemplateService = indexTemplateService; + } + + @Override + protected String executor() { + // we go async right away + return ThreadPool.Names.SAME; + } + + @Override + protected AcknowledgedResponse read(StreamInput in) throws IOException { + return new AcknowledgedResponse(in); + } + + @Override + protected ClusterBlockException checkBlock(PutComponentTemplateAction.Request request, ClusterState state) { + return state.blocks().globalBlockedException(ClusterBlockLevel.METADATA_WRITE); + } + + @Override + protected void masterOperation(final PutComponentTemplateAction.Request request, final ClusterState state, + final ActionListener listener) { + ComponentTemplate componentTemplate = request.componentTemplate(); + Template template = componentTemplate.template(); + // Normalize the index settings if necessary + if (template.settings() != null) { + Settings.Builder settings = Settings.builder().put(template.settings()).normalizePrefix(IndexMetaData.INDEX_SETTING_PREFIX); + template = new Template(settings.build(), template.mappings(), template.aliases()); + componentTemplate = new ComponentTemplate(template, componentTemplate.version(), componentTemplate.metadata()); + } + indexTemplateService.putComponentTemplate(request.cause(), request.create(), request.name(), request.masterNodeTimeout(), + componentTemplate, listener); + } +} diff --git a/server/src/main/java/org/elasticsearch/cluster/metadata/ComponentTemplate.java b/server/src/main/java/org/elasticsearch/cluster/metadata/ComponentTemplate.java index 2b160b1b2fb11..0cc8c8f8a940d 100644 --- a/server/src/main/java/org/elasticsearch/cluster/metadata/ComponentTemplate.java +++ b/server/src/main/java/org/elasticsearch/cluster/metadata/ComponentTemplate.java @@ -24,26 +24,19 @@ import org.elasticsearch.common.Nullable; import org.elasticsearch.common.ParseField; import org.elasticsearch.common.Strings; -import org.elasticsearch.common.bytes.BytesArray; -import org.elasticsearch.common.compress.CompressedXContent; import org.elasticsearch.common.io.stream.StreamInput; import org.elasticsearch.common.io.stream.StreamOutput; -import org.elasticsearch.common.settings.Settings; import org.elasticsearch.common.xcontent.ConstructingObjectParser; import org.elasticsearch.common.xcontent.ToXContentObject; import org.elasticsearch.common.xcontent.XContentBuilder; -import org.elasticsearch.common.xcontent.XContentFactory; -import org.elasticsearch.common.xcontent.XContentHelper; import org.elasticsearch.common.xcontent.XContentParser; -import org.elasticsearch.common.xcontent.XContentType; import java.io.IOException; -import java.util.HashMap; import java.util.Map; import java.util.Objects; /** - * A component template is a re-usable template as well as metadata about the template. Each + * A component template is a re-usable {@link Template} as well as metadata about the template. Each * component template is expected to be valid on its own. For example, if a component template * contains a field "foo", it's expected to contain all the necessary settings/mappings/etc for the * "foo" field. These component templates make up the individual pieces composing an index template. @@ -157,144 +150,4 @@ public XContentBuilder toXContent(XContentBuilder builder, Params params) throws builder.endObject(); return builder; } - - static class Template extends AbstractDiffable