From 723e652fbb41ab6cc981ef4c58b3b47e11e3aea9 Mon Sep 17 00:00:00 2001 From: Martijn van Groningen Date: Tue, 30 Nov 2021 09:31:26 +0100 Subject: [PATCH] Fix data stream alias validation. (#81040) In case of restoring a snapshot, it is possible to overwrite an existing data stream with a data stream alias from a snapshot. This change fixes this by improving the generic duplicate name validation. On top of this the lack of data stream alias validation in Metadata.Builder#build() method resulted in cases where data stream aliases could be added for existing index aliases, data streams or indices with the same name. Closes #80972 --- .../cluster/metadata/Metadata.java | 19 +- .../cluster/metadata/MetadataTests.java | 75 +++++++ .../datastreams/DataStreamIT.java | 184 +++++++++++++++++- .../datastreams/DataStreamsSnapshotsIT.java | 31 +++ 4 files changed, 305 insertions(+), 4 deletions(-) diff --git a/server/src/main/java/org/elasticsearch/cluster/metadata/Metadata.java b/server/src/main/java/org/elasticsearch/cluster/metadata/Metadata.java index 59bf5f98319a9..24c083e463857 100644 --- a/server/src/main/java/org/elasticsearch/cluster/metadata/Metadata.java +++ b/server/src/main/java/org/elasticsearch/cluster/metadata/Metadata.java @@ -1602,17 +1602,23 @@ public Metadata build(boolean builtIndicesLookupEagerly) { indexMetadata.getAliases().keysIt().forEachRemaining(allAliases::add); } + final ArrayList duplicates = new ArrayList<>(); final Set allDataStreams = new HashSet<>(); DataStreamMetadata dataStreamMetadata = (DataStreamMetadata) this.customs.get(DataStreamMetadata.TYPE); if (dataStreamMetadata != null) { for (DataStream dataStream : dataStreamMetadata.dataStreams().values()) { allDataStreams.add(dataStream.getName()); } + // Adding data stream aliases: + for (String dataStreamAlias : dataStreamMetadata.getDataStreamAliases().keySet()) { + if (allAliases.add(dataStreamAlias) == false) { + duplicates.add("data stream alias and indices alias have the same name (" + dataStreamAlias + ")"); + } + } } final Set aliasDuplicatesWithIndices = new HashSet<>(allAliases); aliasDuplicatesWithIndices.retainAll(allIndices); - ArrayList duplicates = new ArrayList<>(); if (aliasDuplicatesWithIndices.isEmpty() == false) { // iterate again and constructs a helpful message for (ObjectCursor cursor : indices.values()) { @@ -1628,12 +1634,19 @@ public Metadata build(boolean builtIndicesLookupEagerly) { aliasDuplicatesWithDataStreams.retainAll(allDataStreams); if (aliasDuplicatesWithDataStreams.isEmpty() == false) { // iterate again and constructs a helpful message - for (ObjectCursor cursor : indices.values()) { - for (String alias : aliasDuplicatesWithDataStreams) { + for (String alias : aliasDuplicatesWithDataStreams) { + // reported var avoids adding a message twice if an index alias has the same name as a data stream. + boolean reported = false; + for (ObjectCursor cursor : indices.values()) { if (cursor.value.getAliases().containsKey(alias)) { duplicates.add(alias + " (alias of " + cursor.value.getIndex() + ") conflicts with data stream"); + reported = true; } } + // This is for adding an error message for when a data steam alias has the same name as a data stream. + if (reported == false && dataStreamMetadata != null && dataStreamMetadata.dataStreams().containsKey(alias)) { + duplicates.add("data stream alias and data stream have the same name (" + alias + ")"); + } } } diff --git a/server/src/test/java/org/elasticsearch/cluster/metadata/MetadataTests.java b/server/src/test/java/org/elasticsearch/cluster/metadata/MetadataTests.java index 7061d9556b82c..fde75d6f608c1 100644 --- a/server/src/test/java/org/elasticsearch/cluster/metadata/MetadataTests.java +++ b/server/src/test/java/org/elasticsearch/cluster/metadata/MetadataTests.java @@ -1215,6 +1215,71 @@ public void testBuildIndicesLookupForDataStreamAliases() { assertThat(value.getAliases(), nullValue()); } + public void testDataStreamAliasValidation() { + Metadata.Builder b = Metadata.builder(); + addDataStream("my-alias", b); + b.put("my-alias", "my-alias", null, null); + var e = expectThrows(IllegalStateException.class, b::build); + assertThat(e.getMessage(), containsString("data stream alias and data stream have the same name (my-alias)")); + + b = Metadata.builder(); + addDataStream("d1", b); + addDataStream("my-alias", b); + b.put("my-alias", "d1", null, null); + e = expectThrows(IllegalStateException.class, b::build); + assertThat(e.getMessage(), containsString("data stream alias and data stream have the same name (my-alias)")); + + b = Metadata.builder(); + b.put( + IndexMetadata.builder("index1") + .settings( + Settings.builder() + .put(IndexMetadata.SETTING_VERSION_CREATED, Version.CURRENT) + .put(IndexMetadata.SETTING_NUMBER_OF_SHARDS, 1) + .put(IndexMetadata.SETTING_NUMBER_OF_REPLICAS, 0) + ) + .putAlias(new AliasMetadata.Builder("my-alias")) + ); + + addDataStream("d1", b); + b.put("my-alias", "d1", null, null); + e = expectThrows(IllegalStateException.class, b::build); + assertThat(e.getMessage(), containsString("data stream alias and indices alias have the same name (my-alias)")); + } + + public void testDataStreamAliasValidationRestoreScenario() { + Metadata.Builder b = Metadata.builder(); + b.dataStreams( + Map.of("my-alias", createDataStream("my-alias")), + Map.of("my-alias", new DataStreamAlias("my-alias", List.of("my-alias"), null, null)) + ); + var e = expectThrows(IllegalStateException.class, b::build); + assertThat(e.getMessage(), containsString("data stream alias and data stream have the same name (my-alias)")); + + b = Metadata.builder(); + b.dataStreams( + Map.of("d1", createDataStream("d1"), "my-alias", createDataStream("my-alias")), + Map.of("my-alias", new DataStreamAlias("my-alias", List.of("d1"), null, null)) + ); + e = expectThrows(IllegalStateException.class, b::build); + assertThat(e.getMessage(), containsString("data stream alias and data stream have the same name (my-alias)")); + + b = Metadata.builder(); + b.put( + IndexMetadata.builder("index1") + .settings( + Settings.builder() + .put(IndexMetadata.SETTING_VERSION_CREATED, Version.CURRENT) + .put(IndexMetadata.SETTING_NUMBER_OF_SHARDS, 1) + .put(IndexMetadata.SETTING_NUMBER_OF_REPLICAS, 0) + ) + .putAlias(new AliasMetadata.Builder("my-alias")) + ); + b.dataStreams(Map.of("d1", createDataStream("d1")), Map.of("my-alias", new DataStreamAlias("my-alias", List.of("d1"), null, null))); + e = expectThrows(IllegalStateException.class, b::build); + assertThat(e.getMessage(), containsString("data stream alias and indices alias have the same name (my-alias)")); + } + private void addDataStream(String name, Metadata.Builder b) { int numBackingIndices = randomIntBetween(1, 4); List indices = new ArrayList<>(numBackingIndices); @@ -1226,6 +1291,16 @@ private void addDataStream(String name, Metadata.Builder b) { b.put(new DataStream(name, createTimestampField("@timestamp"), indices)); } + private DataStream createDataStream(String name) { + int numBackingIndices = randomIntBetween(1, 4); + List indices = new ArrayList<>(numBackingIndices); + for (int j = 1; j <= numBackingIndices; j++) { + IndexMetadata idx = createBackingIndex(name, j).build(); + indices.add(idx.getIndex()); + } + return new DataStream(name, createTimestampField("@timestamp"), indices); + } + public void testIndicesLookupRecordsDataStreamForBackingIndices() { final int numIndices = randomIntBetween(2, 5); final int numBackingIndices = randomIntBetween(2, 5); diff --git a/x-pack/plugin/data-streams/src/internalClusterTest/java/org/elasticsearch/datastreams/DataStreamIT.java b/x-pack/plugin/data-streams/src/internalClusterTest/java/org/elasticsearch/datastreams/DataStreamIT.java index be073aa840d4d..a820b4d382d28 100644 --- a/x-pack/plugin/data-streams/src/internalClusterTest/java/org/elasticsearch/datastreams/DataStreamIT.java +++ b/x-pack/plugin/data-streams/src/internalClusterTest/java/org/elasticsearch/datastreams/DataStreamIT.java @@ -14,10 +14,12 @@ import org.elasticsearch.action.ActionRequestBuilder; import org.elasticsearch.action.DocWriteRequest; import org.elasticsearch.action.admin.cluster.state.ClusterStateRequest; +import org.elasticsearch.action.admin.indices.alias.Alias; 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.alias.get.GetAliasesResponse; +import org.elasticsearch.action.admin.indices.create.CreateIndexRequest; import org.elasticsearch.action.admin.indices.get.GetIndexRequest; import org.elasticsearch.action.admin.indices.get.GetIndexResponse; import org.elasticsearch.action.admin.indices.mapping.get.GetMappingsResponse; @@ -58,6 +60,7 @@ import org.elasticsearch.index.mapper.MapperParsingException; import org.elasticsearch.index.query.TermQueryBuilder; import org.elasticsearch.indices.InvalidAliasNameException; +import org.elasticsearch.indices.InvalidIndexNameException; import org.elasticsearch.plugins.Plugin; import org.elasticsearch.rest.RestStatus; import org.elasticsearch.search.SearchHit; @@ -1531,6 +1534,174 @@ public void testSegmentsSortedOnTimestampDesc() throws Exception { assertTrue(timestamp2 > timestamp3); } + public void testCreateDataStreamWithSameNameAsIndexAlias() throws Exception { + CreateIndexRequest createIndexRequest = new CreateIndexRequest("my-index").alias(new Alias("my-alias")); + assertAcked(client().admin().indices().create(createIndexRequest).actionGet()); + + // Important detail: create template with data stream template after the index has been created + DataStreamIT.putComposableIndexTemplate("my-template", List.of("my-*")); + + var request = new CreateDataStreamAction.Request("my-alias"); + var e = expectThrows(IllegalStateException.class, () -> client().execute(CreateDataStreamAction.INSTANCE, request).actionGet()); + assertThat(e.getMessage(), containsString("[my-alias (alias of [")); + assertThat(e.getMessage(), containsString("]) conflicts with data stream")); + } + + public void testCreateDataStreamWithSameNameAsIndex() throws Exception { + CreateIndexRequest createIndexRequest = new CreateIndexRequest("my-index").alias(new Alias("my-alias")); + assertAcked(client().admin().indices().create(createIndexRequest).actionGet()); + + // Important detail: create template with data stream template after the index has been created + DataStreamIT.putComposableIndexTemplate("my-template", List.of("my-*")); + + var request = new CreateDataStreamAction.Request("my-index"); + var e = expectThrows(IllegalStateException.class, () -> client().execute(CreateDataStreamAction.INSTANCE, request).actionGet()); + assertThat(e.getMessage(), containsString("data stream [my-index] conflicts with index")); + } + + public void testCreateDataStreamWithSameNameAsDataStreamAlias() throws Exception { + { + DataStreamIT.putComposableIndexTemplate("my-template", List.of("my-*")); + var request = new CreateDataStreamAction.Request("my-ds"); + assertAcked(client().execute(CreateDataStreamAction.INSTANCE, request).actionGet()); + var aliasesAddRequest = new IndicesAliasesRequest(); + aliasesAddRequest.addAliasAction(new AliasActions(AliasActions.Type.ADD).index("my-ds").aliases("my-alias")); + assertAcked(client().admin().indices().aliases(aliasesAddRequest).actionGet()); + + var request2 = new CreateDataStreamAction.Request("my-alias"); + var e = expectThrows( + IllegalStateException.class, + () -> client().execute(CreateDataStreamAction.INSTANCE, request2).actionGet() + ); + assertThat(e.getMessage(), containsString("data stream alias and data stream have the same name (my-alias)")); + } + { + assertAcked(client().execute(DeleteDataStreamAction.INSTANCE, new DeleteDataStreamAction.Request("*")).actionGet()); + DataStreamIT.putComposableIndexTemplate( + "my-template", + null, + List.of("my-*"), + null, + null, + Map.of("my-alias", AliasMetadata.builder("my-alias").build()) + ); + var request = new CreateDataStreamAction.Request("my-ds"); + assertAcked(client().execute(CreateDataStreamAction.INSTANCE, request).actionGet()); + + var request2 = new CreateDataStreamAction.Request("my-alias"); + var e = expectThrows( + IllegalStateException.class, + () -> client().execute(CreateDataStreamAction.INSTANCE, request2).actionGet() + ); + assertThat(e.getMessage(), containsString("data stream alias and data stream have the same name (my-alias)")); + } + } + + public void testCreateDataStreamAliasWithSameNameAsIndexAlias() throws Exception { + { + DataStreamIT.putComposableIndexTemplate("my-template", List.of("logs-*")); + CreateIndexRequest createIndexRequest = new CreateIndexRequest("es-logs").alias(new Alias("logs")); + assertAcked(client().admin().indices().create(createIndexRequest).actionGet()); + + var request = new CreateDataStreamAction.Request("logs-es"); + assertAcked(client().execute(CreateDataStreamAction.INSTANCE, request).actionGet()); + IndicesAliasesRequest aliasesAddRequest = new IndicesAliasesRequest(); + aliasesAddRequest.addAliasAction(new AliasActions(AliasActions.Type.ADD).index("logs-es").aliases("logs")); + var e = expectThrows(IllegalStateException.class, () -> client().admin().indices().aliases(aliasesAddRequest).actionGet()); + assertThat(e.getMessage(), containsString("data stream alias and indices alias have the same name (logs)")); + } + { + assertAcked(client().execute(DeleteDataStreamAction.INSTANCE, new DeleteDataStreamAction.Request("*")).actionGet()); + DataStreamIT.putComposableIndexTemplate( + "my-template", + null, + List.of("logs-*"), + null, + null, + Map.of("logs", AliasMetadata.builder("logs").build()) + ); + + var request = new CreateDataStreamAction.Request("logs-es"); + var e = expectThrows(IllegalStateException.class, () -> client().execute(CreateDataStreamAction.INSTANCE, request).actionGet()); + assertThat(e.getMessage(), containsString("data stream alias and indices alias have the same name (logs)")); + } + } + + public void testCreateDataStreamAliasWithSameNameAsIndex() throws Exception { + DataStreamIT.putComposableIndexTemplate("my-template", List.of("logs-*")); + + CreateIndexRequest createIndexRequest = new CreateIndexRequest("logs"); + assertAcked(client().admin().indices().create(createIndexRequest).actionGet()); + + { + var request = new CreateDataStreamAction.Request("logs-es"); + assertAcked(client().execute(CreateDataStreamAction.INSTANCE, request).actionGet()); + IndicesAliasesRequest aliasesAddRequest = new IndicesAliasesRequest(); + aliasesAddRequest.addAliasAction(new AliasActions(AliasActions.Type.ADD).index("logs-es").aliases("logs")); + var e = expectThrows(InvalidAliasNameException.class, () -> client().admin().indices().aliases(aliasesAddRequest).actionGet()); + assertThat( + e.getMessage(), + equalTo("Invalid alias name [logs]: an index or data stream exists with the same name as the alias") + ); + } + { + assertAcked(client().execute(DeleteDataStreamAction.INSTANCE, new DeleteDataStreamAction.Request("*")).actionGet()); + var e = expectThrows( + IllegalArgumentException.class, + () -> DataStreamIT.putComposableIndexTemplate( + "my-template", + null, + List.of("logs-*"), + null, + null, + Map.of("logs", AliasMetadata.builder("logs").build()) + ) + ); + assertThat( + e.getCause().getMessage(), + equalTo("Invalid alias name [logs]: an index or data stream exists with the same name as the alias") + ); + } + } + + public void testCreateIndexWithSameNameAsDataStreamAlias() throws Exception { + DataStreamIT.putComposableIndexTemplate("my-template", List.of("logs-*")); + + var request = new CreateDataStreamAction.Request("logs-es"); + assertAcked(client().execute(CreateDataStreamAction.INSTANCE, request).actionGet()); + IndicesAliasesRequest aliasesAddRequest = new IndicesAliasesRequest(); + aliasesAddRequest.addAliasAction(new AliasActions(AliasActions.Type.ADD).index("logs-es").aliases("logs")); + assertAcked(client().admin().indices().aliases(aliasesAddRequest).actionGet()); + + CreateIndexRequest createIndexRequest = new CreateIndexRequest("logs"); + var e = expectThrows(InvalidIndexNameException.class, () -> client().admin().indices().create(createIndexRequest).actionGet()); + assertThat(e.getMessage(), equalTo("Invalid index name [logs], already exists as alias")); + } + + public void testCreateIndexAliasWithSameNameAsDataStreamAlias() throws Exception { + DataStreamIT.putComposableIndexTemplate("my-template", List.of("logs-*")); + + var request = new CreateDataStreamAction.Request("logs-es"); + assertAcked(client().execute(CreateDataStreamAction.INSTANCE, request).actionGet()); + IndicesAliasesRequest aliasesAddRequest = new IndicesAliasesRequest(); + aliasesAddRequest.addAliasAction(new AliasActions(AliasActions.Type.ADD).index("logs-es").aliases("logs")); + assertAcked(client().admin().indices().aliases(aliasesAddRequest).actionGet()); + + { + CreateIndexRequest createIndexRequest = new CreateIndexRequest("my-index").alias(new Alias("logs")); + var e = expectThrows(IllegalStateException.class, () -> client().admin().indices().create(createIndexRequest).actionGet()); + assertThat(e.getMessage(), containsString("data stream alias and indices alias have the same name (logs)")); + } + { + CreateIndexRequest createIndexRequest = new CreateIndexRequest("my-index"); + assertAcked(client().admin().indices().create(createIndexRequest).actionGet()); + IndicesAliasesRequest addAliasRequest = new IndicesAliasesRequest(); + addAliasRequest.addAliasAction(new AliasActions(AliasActions.Type.ADD).index("my-index").aliases("logs")); + var e = expectThrows(IllegalStateException.class, () -> client().admin().indices().aliases(addAliasRequest).actionGet()); + assertThat(e.getMessage(), containsString("data stream alias and indices alias have the same name (logs)")); + } + } + private static void verifyResolvability(String dataStream, ActionRequestBuilder requestBuilder, boolean fail) { verifyResolvability(dataStream, requestBuilder, fail, 0); } @@ -1723,12 +1894,23 @@ static void putComposableIndexTemplate( List patterns, @Nullable Settings settings, @Nullable Map metadata + ) throws IOException { + putComposableIndexTemplate(id, mappings, patterns, settings, metadata, null); + } + + static void putComposableIndexTemplate( + String id, + @Nullable String mappings, + List patterns, + @Nullable Settings settings, + @Nullable Map metadata, + @Nullable Map aliases ) throws IOException { PutComposableIndexTemplateAction.Request request = new PutComposableIndexTemplateAction.Request(id); request.indexTemplate( new ComposableIndexTemplate( patterns, - new Template(settings, mappings == null ? null : new CompressedXContent(mappings), null), + new Template(settings, mappings == null ? null : new CompressedXContent(mappings), aliases), null, null, null, diff --git a/x-pack/plugin/data-streams/src/internalClusterTest/java/org/elasticsearch/datastreams/DataStreamsSnapshotsIT.java b/x-pack/plugin/data-streams/src/internalClusterTest/java/org/elasticsearch/datastreams/DataStreamsSnapshotsIT.java index 683ef7310a016..38f69173b8686 100644 --- a/x-pack/plugin/data-streams/src/internalClusterTest/java/org/elasticsearch/datastreams/DataStreamsSnapshotsIT.java +++ b/x-pack/plugin/data-streams/src/internalClusterTest/java/org/elasticsearch/datastreams/DataStreamsSnapshotsIT.java @@ -14,11 +14,13 @@ import org.elasticsearch.action.admin.cluster.snapshots.create.CreateSnapshotResponse; import org.elasticsearch.action.admin.cluster.snapshots.restore.RestoreSnapshotRequest; import org.elasticsearch.action.admin.cluster.snapshots.restore.RestoreSnapshotResponse; +import org.elasticsearch.action.admin.indices.alias.Alias; 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.alias.get.GetAliasesResponse; import org.elasticsearch.action.admin.indices.close.CloseIndexRequest; +import org.elasticsearch.action.admin.indices.create.CreateIndexRequest; import org.elasticsearch.action.admin.indices.rollover.RolloverRequest; import org.elasticsearch.action.admin.indices.rollover.RolloverResponse; import org.elasticsearch.action.index.IndexResponse; @@ -1075,4 +1077,33 @@ public void testRestoreSnapshotFully() throws Exception { assertThat(client.execute(GetDataStreamAction.INSTANCE, getRequest).get().getDataStreams(), hasSize(2)); assertNotNull(client.admin().indices().prepareGetIndex().setIndices(indexName).get()); } + + public void testRestoreDataStreamAliasWithConflictingDataStream() throws Exception { + var snapshotName = "test-snapshot"; + createFullSnapshot(REPO, snapshotName); + client.execute(DeleteDataStreamAction.INSTANCE, new DeleteDataStreamAction.Request("*")).actionGet(); + DataStreamIT.putComposableIndexTemplate("my-template", List.of("my-*")); + var request = new CreateDataStreamAction.Request("my-alias"); + assertAcked(client.execute(CreateDataStreamAction.INSTANCE, request).actionGet()); + + var e = expectThrows( + IllegalStateException.class, + () -> client.admin().cluster().prepareRestoreSnapshot(REPO, snapshotName).setWaitForCompletion(true).get() + ); + assertThat(e.getMessage(), containsString("data stream alias and data stream have the same name (my-alias)")); + } + + public void testRestoreDataStreamAliasWithConflictingIndicesAlias() throws Exception { + var snapshotName = "test-snapshot"; + createFullSnapshot(REPO, snapshotName); + client.execute(DeleteDataStreamAction.INSTANCE, new DeleteDataStreamAction.Request("*")).actionGet(); + CreateIndexRequest createIndexRequest = new CreateIndexRequest("my-index").alias(new Alias("my-alias")); + assertAcked(client.admin().indices().create(createIndexRequest).actionGet()); + + var e = expectThrows( + IllegalStateException.class, + () -> client.admin().cluster().prepareRestoreSnapshot(REPO, snapshotName).setWaitForCompletion(true).get() + ); + assertThat(e.getMessage(), containsString("data stream alias and indices alias have the same name (my-alias)")); + } }