Allow deleting closed indices with shadow replicas
Previously we skip deleting the index store for indices on a shared filesystem, because we don't want to delete the data when the shard is relocating around the cluster. This adds a flag to the `deleteIndexStore` method signifying that the index is closed and that we should allow deleting the contents even if it is on a shared filesystem. Includes a unit test for the IndicesService.canDeleteIndexContents and integration tests ensure a closed shadow replica index deletes files correctly. Resolves #13297
This commit is contained in:
parent
ac951e86d1
commit
701dc340a8
|
@ -437,7 +437,7 @@ public class IndicesService extends AbstractLifecycleComponent<IndicesService> i
|
||||||
final Settings indexSettings = indexService.getIndexSettings();
|
final Settings indexSettings = indexService.getIndexSettings();
|
||||||
indicesLifecycle.afterIndexDeleted(indexService.index(), indexSettings);
|
indicesLifecycle.afterIndexDeleted(indexService.index(), indexSettings);
|
||||||
// now we are done - try to wipe data on disk if possible
|
// now we are done - try to wipe data on disk if possible
|
||||||
deleteIndexStore(reason, indexService.index(), indexSettings);
|
deleteIndexStore(reason, indexService.index(), indexSettings, false);
|
||||||
}
|
}
|
||||||
} catch (IOException ex) {
|
} catch (IOException ex) {
|
||||||
throw new ElasticsearchException("failed to remove index " + index, ex);
|
throw new ElasticsearchException("failed to remove index " + index, ex);
|
||||||
|
@ -490,7 +490,7 @@ public class IndicesService extends AbstractLifecycleComponent<IndicesService> i
|
||||||
final IndexMetaData index = clusterState.metaData().index(indexName);
|
final IndexMetaData index = clusterState.metaData().index(indexName);
|
||||||
throw new IllegalStateException("Can't delete closed index store for [" + indexName + "] - it's still part of the cluster state [" + index.getIndexUUID() + "] [" + metaData.getIndexUUID() + "]");
|
throw new IllegalStateException("Can't delete closed index store for [" + indexName + "] - it's still part of the cluster state [" + index.getIndexUUID() + "] [" + metaData.getIndexUUID() + "]");
|
||||||
}
|
}
|
||||||
deleteIndexStore(reason, metaData, clusterState);
|
deleteIndexStore(reason, metaData, clusterState, true);
|
||||||
} catch (IOException e) {
|
} catch (IOException e) {
|
||||||
logger.warn("[{}] failed to delete closed index", e, metaData.index());
|
logger.warn("[{}] failed to delete closed index", e, metaData.index());
|
||||||
}
|
}
|
||||||
|
@ -501,7 +501,7 @@ public class IndicesService extends AbstractLifecycleComponent<IndicesService> i
|
||||||
* Deletes the index store trying to acquire all shards locks for this index.
|
* Deletes the index store trying to acquire all shards locks for this index.
|
||||||
* This method will delete the metadata for the index even if the actual shards can't be locked.
|
* This method will delete the metadata for the index even if the actual shards can't be locked.
|
||||||
*/
|
*/
|
||||||
public void deleteIndexStore(String reason, IndexMetaData metaData, ClusterState clusterState) throws IOException {
|
public void deleteIndexStore(String reason, IndexMetaData metaData, ClusterState clusterState, boolean closed) throws IOException {
|
||||||
if (nodeEnv.hasNodeFile()) {
|
if (nodeEnv.hasNodeFile()) {
|
||||||
synchronized (this) {
|
synchronized (this) {
|
||||||
String indexName = metaData.index();
|
String indexName = metaData.index();
|
||||||
|
@ -518,18 +518,18 @@ public class IndicesService extends AbstractLifecycleComponent<IndicesService> i
|
||||||
}
|
}
|
||||||
Index index = new Index(metaData.index());
|
Index index = new Index(metaData.index());
|
||||||
final Settings indexSettings = buildIndexSettings(metaData);
|
final Settings indexSettings = buildIndexSettings(metaData);
|
||||||
deleteIndexStore(reason, index, indexSettings);
|
deleteIndexStore(reason, index, indexSettings, closed);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
private void deleteIndexStore(String reason, Index index, Settings indexSettings) throws IOException {
|
private void deleteIndexStore(String reason, Index index, Settings indexSettings, boolean closed) throws IOException {
|
||||||
boolean success = false;
|
boolean success = false;
|
||||||
try {
|
try {
|
||||||
// we are trying to delete the index store here - not a big deal if the lock can't be obtained
|
// we are trying to delete the index store here - not a big deal if the lock can't be obtained
|
||||||
// the store metadata gets wiped anyway even without the lock this is just best effort since
|
// the store metadata gets wiped anyway even without the lock this is just best effort since
|
||||||
// every shards deletes its content under the shard lock it owns.
|
// every shards deletes its content under the shard lock it owns.
|
||||||
logger.debug("{} deleting index store reason [{}]", index, reason);
|
logger.debug("{} deleting index store reason [{}]", index, reason);
|
||||||
if (canDeleteIndexContents(index, indexSettings)) {
|
if (canDeleteIndexContents(index, indexSettings, closed)) {
|
||||||
nodeEnv.deleteIndexDirectorySafe(index, 0, indexSettings);
|
nodeEnv.deleteIndexDirectorySafe(index, 0, indexSettings);
|
||||||
}
|
}
|
||||||
success = true;
|
success = true;
|
||||||
|
@ -583,11 +583,11 @@ public class IndicesService extends AbstractLifecycleComponent<IndicesService> i
|
||||||
logger.debug("{} deleted shard reason [{}]", shardId, reason);
|
logger.debug("{} deleted shard reason [{}]", shardId, reason);
|
||||||
|
|
||||||
if (clusterState.nodes().localNode().isMasterNode() == false && // master nodes keep the index meta data, even if having no shards..
|
if (clusterState.nodes().localNode().isMasterNode() == false && // master nodes keep the index meta data, even if having no shards..
|
||||||
canDeleteIndexContents(shardId.index(), indexSettings)) {
|
canDeleteIndexContents(shardId.index(), indexSettings, false)) {
|
||||||
if (nodeEnv.findAllShardIds(shardId.index()).isEmpty()) {
|
if (nodeEnv.findAllShardIds(shardId.index()).isEmpty()) {
|
||||||
try {
|
try {
|
||||||
// note that deleteIndexStore have more safety checks and may throw an exception if index was concurrently created.
|
// note that deleteIndexStore have more safety checks and may throw an exception if index was concurrently created.
|
||||||
deleteIndexStore("no longer used", metaData, clusterState);
|
deleteIndexStore("no longer used", metaData, clusterState, false);
|
||||||
} catch (Exception e) {
|
} catch (Exception e) {
|
||||||
// wrap the exception to indicate we already deleted the shard
|
// wrap the exception to indicate we already deleted the shard
|
||||||
throw new ElasticsearchException("failed to delete unused index after deleting its last shard (" + shardId + ")", e);
|
throw new ElasticsearchException("failed to delete unused index after deleting its last shard (" + shardId + ")", e);
|
||||||
|
@ -606,9 +606,11 @@ public class IndicesService extends AbstractLifecycleComponent<IndicesService> i
|
||||||
* @param indexSettings {@code Settings} for the given index
|
* @param indexSettings {@code Settings} for the given index
|
||||||
* @return true if the index can be deleted on this node
|
* @return true if the index can be deleted on this node
|
||||||
*/
|
*/
|
||||||
public boolean canDeleteIndexContents(Index index, Settings indexSettings) {
|
public boolean canDeleteIndexContents(Index index, Settings indexSettings, boolean closed) {
|
||||||
final IndexServiceInjectorPair indexServiceInjectorPair = this.indices.get(index.name());
|
final IndexServiceInjectorPair indexServiceInjectorPair = this.indices.get(index.name());
|
||||||
if (IndexMetaData.isOnSharedFilesystem(indexSettings) == false) {
|
// Closed indices may be deleted, even if they are on a shared
|
||||||
|
// filesystem. Since it is closed we aren't deleting it for relocation
|
||||||
|
if (IndexMetaData.isOnSharedFilesystem(indexSettings) == false || closed) {
|
||||||
if (indexServiceInjectorPair == null && nodeEnv.hasNodeFile()) {
|
if (indexServiceInjectorPair == null && nodeEnv.hasNodeFile()) {
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
|
@ -44,6 +44,7 @@ import org.elasticsearch.search.sort.SortOrder;
|
||||||
import org.elasticsearch.snapshots.SnapshotState;
|
import org.elasticsearch.snapshots.SnapshotState;
|
||||||
import org.elasticsearch.test.ESIntegTestCase;
|
import org.elasticsearch.test.ESIntegTestCase;
|
||||||
import org.elasticsearch.test.InternalTestCluster;
|
import org.elasticsearch.test.InternalTestCluster;
|
||||||
|
import org.elasticsearch.test.junit.annotations.TestLogging;
|
||||||
import org.elasticsearch.test.transport.MockTransportService;
|
import org.elasticsearch.test.transport.MockTransportService;
|
||||||
import org.elasticsearch.transport.TransportException;
|
import org.elasticsearch.transport.TransportException;
|
||||||
import org.elasticsearch.transport.TransportRequest;
|
import org.elasticsearch.transport.TransportRequest;
|
||||||
|
@ -759,4 +760,52 @@ public class IndexWithShadowReplicasIT extends ESIntegTestCase {
|
||||||
assertShardCountOn(newFooNode, 5);
|
assertShardCountOn(newFooNode, 5);
|
||||||
assertNoShardsOn(barNodes.get());
|
assertNoShardsOn(barNodes.get());
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public void testDeletingClosedIndexRemovesFiles() throws Exception {
|
||||||
|
Path dataPath = createTempDir();
|
||||||
|
Path dataPath2 = createTempDir();
|
||||||
|
Settings nodeSettings = nodeSettings(dataPath.getParent());
|
||||||
|
|
||||||
|
internalCluster().startNodesAsync(2, nodeSettings).get();
|
||||||
|
String IDX = "test";
|
||||||
|
String IDX2 = "test2";
|
||||||
|
|
||||||
|
Settings idxSettings = Settings.builder()
|
||||||
|
.put(IndexMetaData.SETTING_NUMBER_OF_SHARDS, 5)
|
||||||
|
.put(IndexMetaData.SETTING_NUMBER_OF_REPLICAS, 1)
|
||||||
|
.put(IndexMetaData.SETTING_DATA_PATH, dataPath.toAbsolutePath().toString())
|
||||||
|
.put(IndexMetaData.SETTING_SHADOW_REPLICAS, true)
|
||||||
|
.put(IndexMetaData.SETTING_SHARED_FILESYSTEM, true)
|
||||||
|
.build();
|
||||||
|
Settings idx2Settings = Settings.builder()
|
||||||
|
.put(IndexMetaData.SETTING_NUMBER_OF_SHARDS, 5)
|
||||||
|
.put(IndexMetaData.SETTING_NUMBER_OF_REPLICAS, 1)
|
||||||
|
.put(IndexMetaData.SETTING_DATA_PATH, dataPath2.toAbsolutePath().toString())
|
||||||
|
.put(IndexMetaData.SETTING_SHADOW_REPLICAS, true)
|
||||||
|
.put(IndexMetaData.SETTING_SHARED_FILESYSTEM, true)
|
||||||
|
.build();
|
||||||
|
|
||||||
|
prepareCreate(IDX).setSettings(idxSettings).addMapping("doc", "foo", "type=string").get();
|
||||||
|
prepareCreate(IDX2).setSettings(idx2Settings).addMapping("doc", "foo", "type=string").get();
|
||||||
|
ensureGreen(IDX, IDX2);
|
||||||
|
|
||||||
|
int docCount = randomIntBetween(10, 100);
|
||||||
|
List<IndexRequestBuilder> builders = new ArrayList<>();
|
||||||
|
for (int i = 0; i < docCount; i++) {
|
||||||
|
builders.add(client().prepareIndex(IDX, "doc", i + "").setSource("foo", "bar"));
|
||||||
|
builders.add(client().prepareIndex(IDX2, "doc", i + "").setSource("foo", "bar"));
|
||||||
|
}
|
||||||
|
indexRandom(true, true, true, builders);
|
||||||
|
flushAndRefresh(IDX, IDX2);
|
||||||
|
|
||||||
|
logger.info("--> closing index {}", IDX);
|
||||||
|
client().admin().indices().prepareClose(IDX).get();
|
||||||
|
|
||||||
|
logger.info("--> deleting non-closed index");
|
||||||
|
client().admin().indices().prepareDelete(IDX2).get();
|
||||||
|
assertPathHasBeenCleared(dataPath2);
|
||||||
|
logger.info("--> deleting closed index");
|
||||||
|
client().admin().indices().prepareDelete(IDX).get();
|
||||||
|
assertPathHasBeenCleared(dataPath);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -23,9 +23,11 @@ import org.elasticsearch.Version;
|
||||||
import org.elasticsearch.cluster.ClusterService;
|
import org.elasticsearch.cluster.ClusterService;
|
||||||
import org.elasticsearch.cluster.metadata.IndexMetaData;
|
import org.elasticsearch.cluster.metadata.IndexMetaData;
|
||||||
import org.elasticsearch.cluster.metadata.MetaData;
|
import org.elasticsearch.cluster.metadata.MetaData;
|
||||||
|
import org.elasticsearch.common.settings.Settings;
|
||||||
import org.elasticsearch.common.unit.TimeValue;
|
import org.elasticsearch.common.unit.TimeValue;
|
||||||
import org.elasticsearch.env.NodeEnvironment;
|
import org.elasticsearch.env.NodeEnvironment;
|
||||||
import org.elasticsearch.gateway.GatewayMetaState;
|
import org.elasticsearch.gateway.GatewayMetaState;
|
||||||
|
import org.elasticsearch.index.Index;
|
||||||
import org.elasticsearch.index.IndexService;
|
import org.elasticsearch.index.IndexService;
|
||||||
import org.elasticsearch.index.shard.ShardId;
|
import org.elasticsearch.index.shard.ShardId;
|
||||||
import org.elasticsearch.index.shard.ShardPath;
|
import org.elasticsearch.index.shard.ShardPath;
|
||||||
|
@ -51,6 +53,19 @@ public class IndicesServiceTests extends ESSingleNodeTestCase {
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public void testCanDeleteIndexContent() {
|
||||||
|
IndicesService indicesService = getIndicesService();
|
||||||
|
|
||||||
|
Settings idxSettings = settings(Version.CURRENT)
|
||||||
|
.put(IndexMetaData.SETTING_SHADOW_REPLICAS, true)
|
||||||
|
.put(IndexMetaData.SETTING_DATA_PATH, "/foo/bar")
|
||||||
|
.put(IndexMetaData.SETTING_NUMBER_OF_SHARDS, randomIntBetween(1, 4))
|
||||||
|
.put(IndexMetaData.SETTING_NUMBER_OF_REPLICAS, randomIntBetween(0, 3))
|
||||||
|
.build();
|
||||||
|
assertFalse("shard on shared filesystem", indicesService.canDeleteIndexContents(new Index("test"), idxSettings, false));
|
||||||
|
assertTrue("shard on shared filesystem and closed", indicesService.canDeleteIndexContents(new Index("test"), idxSettings, true));
|
||||||
|
}
|
||||||
|
|
||||||
public void testCanDeleteShardContent() {
|
public void testCanDeleteShardContent() {
|
||||||
IndicesService indicesService = getIndicesService();
|
IndicesService indicesService = getIndicesService();
|
||||||
IndexMetaData meta = IndexMetaData.builder("test").settings(settings(Version.CURRENT)).numberOfShards(1).numberOfReplicas(
|
IndexMetaData meta = IndexMetaData.builder("test").settings(settings(Version.CURRENT)).numberOfShards(1).numberOfReplicas(
|
||||||
|
@ -71,7 +86,7 @@ public class IndicesServiceTests extends ESSingleNodeTestCase {
|
||||||
assertTrue(test.hasShard(0));
|
assertTrue(test.hasShard(0));
|
||||||
|
|
||||||
try {
|
try {
|
||||||
indicesService.deleteIndexStore("boom", firstMetaData, clusterService.state());
|
indicesService.deleteIndexStore("boom", firstMetaData, clusterService.state(), false);
|
||||||
fail();
|
fail();
|
||||||
} catch (IllegalStateException ex) {
|
} catch (IllegalStateException ex) {
|
||||||
// all good
|
// all good
|
||||||
|
@ -98,7 +113,7 @@ public class IndicesServiceTests extends ESSingleNodeTestCase {
|
||||||
assertTrue(path.exists());
|
assertTrue(path.exists());
|
||||||
|
|
||||||
try {
|
try {
|
||||||
indicesService.deleteIndexStore("boom", secondMetaData, clusterService.state());
|
indicesService.deleteIndexStore("boom", secondMetaData, clusterService.state(), false);
|
||||||
fail();
|
fail();
|
||||||
} catch (IllegalStateException ex) {
|
} catch (IllegalStateException ex) {
|
||||||
// all good
|
// all good
|
||||||
|
@ -108,7 +123,7 @@ public class IndicesServiceTests extends ESSingleNodeTestCase {
|
||||||
|
|
||||||
// now delete the old one and make sure we resolve against the name
|
// now delete the old one and make sure we resolve against the name
|
||||||
try {
|
try {
|
||||||
indicesService.deleteIndexStore("boom", firstMetaData, clusterService.state());
|
indicesService.deleteIndexStore("boom", firstMetaData, clusterService.state(), false);
|
||||||
fail();
|
fail();
|
||||||
} catch (IllegalStateException ex) {
|
} catch (IllegalStateException ex) {
|
||||||
// all good
|
// all good
|
||||||
|
|
Loading…
Reference in New Issue