SOLR-9642: Refactor the snapshot cleanup mechanism to rely on Lucene

The current snapshot cleanup mechanism is based on reference counting
the index files shared between multiple segments. Since this mechanism
completely skips the Lucene APIs, it is not portable (e.g. it doesn't
work on 4.10.x version).

This patch provides an alternate implementation which relies exclusively
on Lucene IndexWriter (+ IndexDeletionPolicy) for cleanup.

mend
This commit is contained in:
Hrishikesh Gadre 2016-08-10 16:59:31 -07:00 committed by yonik
parent c9eb9e10e4
commit 06ae3eb2ec
8 changed files with 185 additions and 254 deletions

View File

@ -246,6 +246,8 @@ Other Changes
* SOLR-9625: Add HelloWorldSolrCloudTestCase class (Christine Poerschke, Alan Woodward, Alexandre Rafalovitch) * SOLR-9625: Add HelloWorldSolrCloudTestCase class (Christine Poerschke, Alan Woodward, Alexandre Rafalovitch)
* SOLR-9642: Refactor the core level snapshot cleanup mechanism to rely on Lucene (Hrishikesh Gadre via yonik)
================== 6.2.1 ================== ================== 6.2.1 ==================
Bug Fixes Bug Fixes

View File

@ -39,6 +39,7 @@ import java.util.LinkedHashMap;
import java.util.LinkedList; import java.util.LinkedList;
import java.util.List; import java.util.List;
import java.util.Map; import java.util.Map;
import java.util.Optional;
import java.util.Properties; import java.util.Properties;
import java.util.Set; import java.util.Set;
import java.util.concurrent.Callable; import java.util.concurrent.Callable;
@ -81,7 +82,9 @@ import org.apache.solr.common.util.ObjectReleaseTracker;
import org.apache.solr.common.util.SimpleOrderedMap; import org.apache.solr.common.util.SimpleOrderedMap;
import org.apache.solr.common.util.Utils; import org.apache.solr.common.util.Utils;
import org.apache.solr.core.DirectoryFactory.DirContext; import org.apache.solr.core.DirectoryFactory.DirContext;
import org.apache.solr.core.snapshots.SolrSnapshotManager;
import org.apache.solr.core.snapshots.SolrSnapshotMetaDataManager; import org.apache.solr.core.snapshots.SolrSnapshotMetaDataManager;
import org.apache.solr.core.snapshots.SolrSnapshotMetaDataManager.SnapshotMetaData;
import org.apache.solr.handler.IndexFetcher; import org.apache.solr.handler.IndexFetcher;
import org.apache.solr.handler.ReplicationHandler; import org.apache.solr.handler.ReplicationHandler;
import org.apache.solr.handler.RequestHandlerBase; import org.apache.solr.handler.RequestHandlerBase;
@ -194,6 +197,7 @@ public final class SolrCore implements SolrInfoMBean, Closeable {
private final List<Runnable> confListeners = new CopyOnWriteArrayList<>(); private final List<Runnable> confListeners = new CopyOnWriteArrayList<>();
private final ReentrantLock ruleExpiryLock; private final ReentrantLock ruleExpiryLock;
private final ReentrantLock snapshotDelLock; // A lock instance to guard against concurrent deletions.
public Date getStartTimeStamp() { return startTime; } public Date getStartTimeStamp() { return startTime; }
@ -431,6 +435,83 @@ public final class SolrCore implements SolrInfoMBean, Closeable {
} }
} }
/**
* This method deletes the snapshot with the specified name. If the directory
* storing the snapshot is not the same as the *current* core index directory,
* then delete the files corresponding to this snapshot. Otherwise we leave the
* index files related to snapshot as is (assuming the underlying Solr IndexDeletionPolicy
* will clean them up appropriately).
*
* @param commitName The name of the snapshot to be deleted.
* @throws IOException in case of I/O error.
*/
public void deleteNamedSnapshot(String commitName) throws IOException {
// Note this lock is required to prevent multiple snapshot deletions from
// opening multiple IndexWriter instances simultaneously.
this.snapshotDelLock.lock();
try {
Optional<SnapshotMetaData> metadata = snapshotMgr.release(commitName);
if (metadata.isPresent()) {
long gen = metadata.get().getGenerationNumber();
String indexDirPath = metadata.get().getIndexDirPath();
if (!indexDirPath.equals(getIndexDir())) {
Directory d = getDirectoryFactory().get(indexDirPath, DirContext.DEFAULT, "none");
try {
Collection<SnapshotMetaData> snapshots = snapshotMgr.listSnapshotsInIndexDir(indexDirPath);
log.info("Following snapshots exist in the index directory {} : {}", indexDirPath, snapshots);
if (snapshots.isEmpty()) {// No snapshots remain in this directory. Can be cleaned up!
log.info("Removing index directory {} since all named snapshots are deleted.", indexDirPath);
getDirectoryFactory().remove(d);
} else {
SolrSnapshotManager.deleteSnapshotIndexFiles(this, d, gen);
}
} finally {
getDirectoryFactory().release(d);
}
}
}
} finally {
snapshotDelLock.unlock();
}
}
/**
* This method deletes the index files not associated with any named snapshot only
* if the specified indexDirPath is not the *current* index directory.
*
* @param indexDirPath The path of the directory
* @throws IOException In case of I/O error.
*/
public void deleteNonSnapshotIndexFiles(String indexDirPath) throws IOException {
// Skip if the specified indexDirPath is the *current* index directory.
if (getIndexDir().equals(indexDirPath)) {
return;
}
// Note this lock is required to prevent multiple snapshot deletions from
// opening multiple IndexWriter instances simultaneously.
this.snapshotDelLock.lock();
Directory dir = getDirectoryFactory().get(indexDirPath, DirContext.DEFAULT, "none");
try {
Collection<SnapshotMetaData> snapshots = snapshotMgr.listSnapshotsInIndexDir(indexDirPath);
log.info("Following snapshots exist in the index directory {} : {}", indexDirPath, snapshots);
// Delete the old index directory only if no snapshot exists in that directory.
if (snapshots.isEmpty()) {
log.info("Removing index directory {} since all named snapshots are deleted.", indexDirPath);
getDirectoryFactory().remove(dir);
} else {
SolrSnapshotManager.deleteNonSnapshotIndexFiles(this, dir, snapshots);
}
} finally {
snapshotDelLock.unlock();
if (dir != null) {
getDirectoryFactory().release(dir);
}
}
}
private void initListeners() { private void initListeners() {
final Class<SolrEventListener> clazz = SolrEventListener.class; final Class<SolrEventListener> clazz = SolrEventListener.class;
final String label = "Event Listener"; final String label = "Event Listener";
@ -845,6 +926,8 @@ public final class SolrCore implements SolrInfoMBean, Closeable {
bufferUpdatesIfConstructing(coreDescriptor); bufferUpdatesIfConstructing(coreDescriptor);
this.ruleExpiryLock = new ReentrantLock(); this.ruleExpiryLock = new ReentrantLock();
this.snapshotDelLock = new ReentrantLock();
registerConfListener(); registerConfListener();
assert ObjectReleaseTracker.track(this); assert ObjectReleaseTracker.track(this);

View File

@ -19,18 +19,18 @@ package org.apache.solr.core.snapshots;
import java.io.IOException; import java.io.IOException;
import java.lang.invoke.MethodHandles; import java.lang.invoke.MethodHandles;
import java.util.Collection; import java.util.Collection;
import java.util.HashMap; import java.util.HashSet;
import java.util.List; import java.util.List;
import java.util.Map;
import java.util.Set; import java.util.Set;
import java.util.function.Function;
import java.util.stream.Collectors;
import com.google.common.annotations.VisibleForTesting;
import org.apache.lucene.index.DirectoryReader;
import org.apache.lucene.index.IndexCommit; import org.apache.lucene.index.IndexCommit;
import org.apache.lucene.index.IndexDeletionPolicy;
import org.apache.lucene.index.IndexWriterConfig;
import org.apache.lucene.index.IndexWriterConfig.OpenMode;
import org.apache.lucene.index.NoMergePolicy;
import org.apache.lucene.store.Directory; import org.apache.lucene.store.Directory;
import org.apache.solr.core.SolrCore;
import org.apache.solr.core.snapshots.SolrSnapshotMetaDataManager.SnapshotMetaData; import org.apache.solr.core.snapshots.SolrSnapshotMetaDataManager.SnapshotMetaData;
import org.apache.solr.update.SolrIndexWriter;
import org.slf4j.Logger; import org.slf4j.Logger;
import org.slf4j.LoggerFactory; import org.slf4j.LoggerFactory;
@ -43,92 +43,78 @@ public class SolrSnapshotManager {
/** /**
* This method deletes index files of the {@linkplain IndexCommit} for the specified generation number. * This method deletes index files of the {@linkplain IndexCommit} for the specified generation number.
* *
* @param core The Solr core
* @param dir The index directory storing the snapshot. * @param dir The index directory storing the snapshot.
* @param gen The generation number for the {@linkplain IndexCommit} * @param gen The generation number of the {@linkplain IndexCommit} to be deleted.
* @throws IOException in case of I/O errors. * @throws IOException in case of I/O errors.
*/ */
public static void deleteIndexFiles ( Directory dir, Collection<SnapshotMetaData> snapshots, long gen ) throws IOException { public static void deleteSnapshotIndexFiles(SolrCore core, Directory dir, final long gen) throws IOException {
List<IndexCommit> commits = DirectoryReader.listCommits(dir); deleteSnapshotIndexFiles(core, dir, new IndexDeletionPolicy() {
Map<String, Integer> refCounts = buildRefCounts(snapshots, commits); @Override
for (IndexCommit ic : commits) { public void onInit(List<? extends IndexCommit> commits) throws IOException {
if (ic.getGeneration() == gen) { for (IndexCommit ic : commits) {
deleteIndexFiles(dir,refCounts, ic); if (gen == ic.getGeneration()) {
break; log.info("Deleting non-snapshotted index commit with generation {}", ic.getGeneration());
} ic.delete();
} }
}
/**
* This method deletes all files not corresponding to a configured snapshot in the specified index directory.
*
* @param dir The index directory to search for.
* @throws IOException in case of I/O errors.
*/
public static void deleteNonSnapshotIndexFiles (Directory dir, Collection<SnapshotMetaData> snapshots) throws IOException {
List<IndexCommit> commits = DirectoryReader.listCommits(dir);
Map<String, Integer> refCounts = buildRefCounts(snapshots, commits);
Set<Long> snapshotGenNumbers = snapshots.stream()
.map(SnapshotMetaData::getGenerationNumber)
.collect(Collectors.toSet());
for (IndexCommit ic : commits) {
if (!snapshotGenNumbers.contains(ic.getGeneration())) {
deleteIndexFiles(dir,refCounts, ic);
}
}
}
/**
* This method computes reference count for the index files by taking into consideration
* (a) configured snapshots and (b) files sharing between two or more {@linkplain IndexCommit} instances.
*
* @param snapshots A collection of user configured snapshots
* @param commits A list of {@linkplain IndexCommit} instances
* @return A map containing reference count for each index file referred in one of the {@linkplain IndexCommit} instances.
* @throws IOException in case of I/O error.
*/
@VisibleForTesting
static Map<String, Integer> buildRefCounts (Collection<SnapshotMetaData> snapshots, List<IndexCommit> commits) throws IOException {
Map<String, Integer> result = new HashMap<>();
Map<Long, IndexCommit> commitsByGen = commits.stream().collect(
Collectors.toMap(IndexCommit::getGeneration, Function.identity()));
for(SnapshotMetaData md : snapshots) {
IndexCommit ic = commitsByGen.get(md.getGenerationNumber());
if (ic != null) {
Collection<String> fileNames = ic.getFileNames();
for(String fileName : fileNames) {
int refCount = result.getOrDefault(fileName, 0);
result.put(fileName, refCount+1);
} }
} }
}
return result; @Override
public void onCommit(List<? extends IndexCommit> commits)
throws IOException {}
});
} }
/** /**
* This method deletes the index files associated with specified <code>indexCommit</code> provided they * This method deletes index files not associated with the specified <code>snapshots</code>.
* are not referred by some other {@linkplain IndexCommit}.
* *
* @param dir The index directory containing the {@linkplain IndexCommit} to be deleted. * @param core The Solr core
* @param refCounts A map containing reference counts for each file associated with every {@linkplain IndexCommit} * @param dir The index directory storing the snapshot.
* in the specified directory. * @param snapshots The snapshots to be preserved.
* @param indexCommit The {@linkplain IndexCommit} whose files need to be deleted.
* @throws IOException in case of I/O errors. * @throws IOException in case of I/O errors.
*/ */
private static void deleteIndexFiles ( Directory dir, Map<String, Integer> refCounts, IndexCommit indexCommit ) throws IOException { public static void deleteNonSnapshotIndexFiles(SolrCore core, Directory dir, Collection<SnapshotMetaData> snapshots) throws IOException {
log.info("Deleting index files for index commit with generation {} in directory {}", indexCommit.getGeneration(), dir); final Set<Long> genNumbers = new HashSet<>();
for (String fileName : indexCommit.getFileNames()) { for (SnapshotMetaData m : snapshots) {
try { genNumbers.add(m.getGenerationNumber());
// Ensure that a file being deleted is not referred by some other commit. }
int ref = refCounts.getOrDefault(fileName, 0);
log.debug("Reference count for file {} is {}", fileName, ref); deleteSnapshotIndexFiles(core, dir, new IndexDeletionPolicy() {
if (ref == 0) { @Override
dir.deleteFile(fileName); public void onInit(List<? extends IndexCommit> commits) throws IOException {
for (IndexCommit ic : commits) {
if (!genNumbers.contains(ic.getGeneration())) {
log.info("Deleting non-snapshotted index commit with generation {}", ic.getGeneration());
ic.delete();
}
} }
} catch (IOException e) {
log.warn("Unable to delete file {} in directory {} due to exception {}", fileName, dir, e.getMessage());
} }
@Override
public void onCommit(List<? extends IndexCommit> commits)
throws IOException {}
});
}
/**
* This method deletes index files of the {@linkplain IndexCommit} for the specified generation number.
*
* @param core The Solr core
* @param dir The index directory storing the snapshot.
* @throws IOException in case of I/O errors.
*/
private static void deleteSnapshotIndexFiles(SolrCore core, Directory dir, IndexDeletionPolicy delPolicy) throws IOException {
IndexWriterConfig conf = core.getSolrConfig().indexConfig.toIndexWriterConfig(core);
conf.setOpenMode(OpenMode.APPEND);
conf.setMergePolicy(NoMergePolicy.INSTANCE);//Don't want to merge any commits here!
conf.setIndexDeletionPolicy(delPolicy);
conf.setCodec(core.getCodec());
try (SolrIndexWriter iw = new SolrIndexWriter("SolrSnapshotCleaner", dir, conf)) {
// Do nothing. The only purpose of opening index writer is to invoke the Lucene IndexDeletionPolicy#onInit
// method so that we can cleanup the files associated with specified index commit.
// Note the index writer creates a new commit during the close() operation (which is harmless).
} }
} }
} }

View File

@ -81,9 +81,6 @@ import org.apache.solr.core.DirectoryFactory;
import org.apache.solr.core.DirectoryFactory.DirContext; import org.apache.solr.core.DirectoryFactory.DirContext;
import org.apache.solr.core.IndexDeletionPolicyWrapper; import org.apache.solr.core.IndexDeletionPolicyWrapper;
import org.apache.solr.core.SolrCore; import org.apache.solr.core.SolrCore;
import org.apache.solr.core.snapshots.SolrSnapshotManager;
import org.apache.solr.core.snapshots.SolrSnapshotMetaDataManager;
import org.apache.solr.core.snapshots.SolrSnapshotMetaDataManager.SnapshotMetaData;
import org.apache.solr.handler.ReplicationHandler.*; import org.apache.solr.handler.ReplicationHandler.*;
import org.apache.solr.request.LocalSolrQueryRequest; import org.apache.solr.request.LocalSolrQueryRequest;
import org.apache.solr.request.SolrQueryRequest; import org.apache.solr.request.SolrQueryRequest;
@ -464,17 +461,8 @@ public class IndexFetcher {
// may be closed // may be closed
if (indexDir != null) { if (indexDir != null) {
solrCore.getDirectoryFactory().doneWithDirectory(indexDir); solrCore.getDirectoryFactory().doneWithDirectory(indexDir);
// Cleanup all index files not associated with any *named* snapshot.
SolrSnapshotMetaDataManager snapshotsMgr = solrCore.getSnapshotMetaDataManager(); solrCore.deleteNonSnapshotIndexFiles(indexDirPath);
Collection<SnapshotMetaData> snapshots = snapshotsMgr.listSnapshotsInIndexDir(indexDirPath);
// Delete the old index directory only if no snapshot exists in that directory.
if(snapshots.isEmpty()) {
LOG.info("removing old index directory " + indexDir);
solrCore.getDirectoryFactory().remove(indexDir);
} else {
SolrSnapshotManager.deleteNonSnapshotIndexFiles(indexDir, snapshots);
}
} }
} }

View File

@ -19,7 +19,6 @@ package org.apache.solr.handler;
import java.lang.invoke.MethodHandles; import java.lang.invoke.MethodHandles;
import java.net.URI; import java.net.URI;
import java.text.SimpleDateFormat; import java.text.SimpleDateFormat;
import java.util.Collection;
import java.util.Date; import java.util.Date;
import java.util.Locale; import java.util.Locale;
import java.util.concurrent.Callable; import java.util.concurrent.Callable;
@ -33,9 +32,6 @@ import org.apache.solr.common.SolrException;
import org.apache.solr.core.DirectoryFactory; import org.apache.solr.core.DirectoryFactory;
import org.apache.solr.core.SolrCore; import org.apache.solr.core.SolrCore;
import org.apache.solr.core.backup.repository.BackupRepository; import org.apache.solr.core.backup.repository.BackupRepository;
import org.apache.solr.core.snapshots.SolrSnapshotManager;
import org.apache.solr.core.snapshots.SolrSnapshotMetaDataManager;
import org.apache.solr.core.snapshots.SolrSnapshotMetaDataManager.SnapshotMetaData;
import org.slf4j.Logger; import org.slf4j.Logger;
import org.slf4j.LoggerFactory; import org.slf4j.LoggerFactory;
@ -135,16 +131,8 @@ public class RestoreCore implements Callable<Boolean> {
} }
if (success) { if (success) {
core.getDirectoryFactory().doneWithDirectory(indexDir); core.getDirectoryFactory().doneWithDirectory(indexDir);
// Cleanup all index files not associated with any *named* snapshot.
SolrSnapshotMetaDataManager snapshotsMgr = core.getSnapshotMetaDataManager(); core.deleteNonSnapshotIndexFiles(indexDirPath);
Collection<SnapshotMetaData> snapshots = snapshotsMgr.listSnapshotsInIndexDir(indexDirPath);
// Delete the old index directory only if no snapshot exists in that directory.
if (snapshots.isEmpty()) {
core.getDirectoryFactory().remove(indexDir);
} else {
SolrSnapshotManager.deleteNonSnapshotIndexFiles(indexDir, snapshots);
}
} }
return true; return true;

View File

@ -17,17 +17,11 @@
package org.apache.solr.handler.admin; package org.apache.solr.handler.admin;
import java.util.Optional;
import org.apache.lucene.store.Directory;
import org.apache.solr.common.SolrException; import org.apache.solr.common.SolrException;
import org.apache.solr.common.params.CoreAdminParams; import org.apache.solr.common.params.CoreAdminParams;
import org.apache.solr.common.params.SolrParams; import org.apache.solr.common.params.SolrParams;
import org.apache.solr.core.CoreContainer; import org.apache.solr.core.CoreContainer;
import org.apache.solr.core.DirectoryFactory;
import org.apache.solr.core.SolrCore; import org.apache.solr.core.SolrCore;
import org.apache.solr.core.snapshots.SolrSnapshotManager;
import org.apache.solr.core.snapshots.SolrSnapshotMetaDataManager;
class DeleteSnapshotOp implements CoreAdminHandler.CoreAdminOp { class DeleteSnapshotOp implements CoreAdminHandler.CoreAdminOp {
@ -39,30 +33,15 @@ class DeleteSnapshotOp implements CoreAdminHandler.CoreAdminOp {
String commitName = params.required().get(CoreAdminParams.COMMIT_NAME); String commitName = params.required().get(CoreAdminParams.COMMIT_NAME);
String cname = params.required().get(CoreAdminParams.CORE); String cname = params.required().get(CoreAdminParams.CORE);
try (SolrCore core = cc.getCore(cname)) { SolrCore core = cc.getCore(cname);
if (core == null) { if (core == null) {
throw new SolrException(SolrException.ErrorCode.BAD_REQUEST, "Unable to locate core " + cname); throw new SolrException(SolrException.ErrorCode.BAD_REQUEST, "Unable to locate core " + cname);
} }
SolrSnapshotMetaDataManager mgr = core.getSnapshotMetaDataManager(); try {
Optional<SolrSnapshotMetaDataManager.SnapshotMetaData> metadata = mgr.release(commitName); core.deleteNamedSnapshot(commitName);
if (metadata.isPresent()) { } finally {
long gen = metadata.get().getGenerationNumber(); core.close();
String indexDirPath = metadata.get().getIndexDirPath();
// If the directory storing the snapshot is not the same as the *current* core
// index directory, then delete the files corresponding to this snapshot.
// Otherwise we leave the index files related to snapshot as is (assuming the
// underlying Solr IndexDeletionPolicy will clean them up appropriately).
if (!indexDirPath.equals(core.getIndexDir())) {
Directory d = core.getDirectoryFactory().get(indexDirPath, DirectoryFactory.DirContext.DEFAULT, DirectoryFactory.LOCK_TYPE_NONE);
try {
SolrSnapshotManager.deleteIndexFiles(d, mgr.listSnapshotsInIndexDir(indexDirPath), gen);
} finally {
core.getDirectoryFactory().release(d);
}
}
}
} }
} }
} }

View File

@ -77,6 +77,15 @@ public class SolrIndexWriter extends IndexWriter {
} }
} }
public SolrIndexWriter(String name, Directory d, IndexWriterConfig conf) throws IOException {
super(d, conf);
this.name = name;
this.infoStream = conf.getInfoStream();
this.directory = d;
numOpens.incrementAndGet();
log.debug("Opened Writer " + name);
}
private SolrIndexWriter(SolrCore core, String name, String path, Directory directory, boolean create, IndexSchema schema, SolrIndexConfig config, IndexDeletionPolicy delPolicy, Codec codec) throws IOException { private SolrIndexWriter(SolrCore core, String name, String path, Directory directory, boolean create, IndexSchema schema, SolrIndexConfig config, IndexDeletionPolicy delPolicy, Codec codec) throws IOException {
super(directory, super(directory,
config.toIndexWriterConfig(core). config.toIndexWriterConfig(core).
@ -182,7 +191,10 @@ public class SolrIndexWriter extends IndexWriter {
IOUtils.closeQuietly(infoStream); IOUtils.closeQuietly(infoStream);
} }
numCloses.incrementAndGet(); numCloses.incrementAndGet();
directoryFactory.release(directory);
if (directoryFactory != null) {
directoryFactory.release(directory);
}
} }
} }

View File

@ -17,8 +17,6 @@
package org.apache.solr.core.snapshots; package org.apache.solr.core.snapshots;
import java.lang.invoke.MethodHandles; import java.lang.invoke.MethodHandles;
import java.nio.file.Files;
import java.nio.file.Path;
import java.nio.file.Paths; import java.nio.file.Paths;
import java.util.ArrayList; import java.util.ArrayList;
import java.util.Collection; import java.util.Collection;
@ -124,7 +122,7 @@ public class TestSolrCoreSnapshots extends SolrCloudTestCase {
// and the other containing document deletions. // and the other containing document deletions.
{ {
List<IndexCommit> commits = listCommits(metaData.getIndexDirPath()); List<IndexCommit> commits = listCommits(metaData.getIndexDirPath());
assertTrue(2 <= commits.size()); assertTrue(commits.size() >= 2);
} }
// Backup the earlier created snapshot. // Backup the earlier created snapshot.
@ -146,9 +144,11 @@ public class TestSolrCoreSnapshots extends SolrCloudTestCase {
} }
// Verify that the old index directory (before restore) contains only those index commits referred by snapshots. // Verify that the old index directory (before restore) contains only those index commits referred by snapshots.
// The IndexWriter (used to cleanup index files) creates an additional commit during closing. Hence we expect 2 commits (instead
// of 1).
{ {
List<IndexCommit> commits = listCommits(metaData.getIndexDirPath()); List<IndexCommit> commits = listCommits(metaData.getIndexDirPath());
assertEquals(1, commits.size()); assertEquals(2, commits.size());
assertEquals(metaData.getGenerationNumber(), commits.get(0).getGeneration()); assertEquals(metaData.getGenerationNumber(), commits.get(0).getGeneration());
} }
@ -161,122 +161,15 @@ public class TestSolrCoreSnapshots extends SolrCloudTestCase {
// Delete second snapshot // Delete second snapshot
deleteSnapshot(adminClient, coreName, duplicateCommit.getName()); deleteSnapshot(adminClient, coreName, duplicateCommit.getName());
// Verify that corresponding index files have been deleted. // Verify that corresponding index files have been deleted. Ideally this directory should
assertTrue(listCommits(duplicateCommit.getIndexDirPath()).isEmpty()); // be removed immediately. But the current DirectoryFactory impl waits until the
// closing the core (or the directoryFactory) for actual removal. Since the IndexWriter
// (used to cleanup index files) creates an additional commit during closing, we expect a single
// commit (instead of 0).
assertEquals(1, listCommits(duplicateCommit.getIndexDirPath()).size());
} }
} }
@Test
public void testHandlingSharedIndexFiles() throws Exception {
CloudSolrClient solrClient = cluster.getSolrClient();
String collectionName = "SolrCoreSnapshots_IndexFileSharing";
CollectionAdminRequest.Create create = CollectionAdminRequest.createCollection(collectionName, "conf1", 1, 1);
create.process(solrClient);
int nDocs = BackupRestoreUtils.indexDocs(cluster.getSolrClient(), collectionName, docsSeed);
DocCollection collectionState = solrClient.getZkStateReader().getClusterState().getCollection(collectionName);
assertEquals(1, collectionState.getActiveSlices().size());
Slice shard = collectionState.getActiveSlices().iterator().next();
assertEquals(1, shard.getReplicas().size());
Replica replica = shard.getReplicas().iterator().next();
String replicaBaseUrl = replica.getStr(BASE_URL_PROP);
String coreName = replica.getStr(ZkStateReader.CORE_NAME_PROP);
String backupName = TestUtil.randomSimpleString(random(), 1, 5);
String location = createTempDir().toFile().getAbsolutePath();
try (
SolrClient adminClient = getHttpSolrClient(cluster.getJettySolrRunners().get(0).getBaseUrl().toString());
SolrClient masterClient = getHttpSolrClient(replica.getCoreUrl())) {
int numTests = TestUtil.nextInt(random(), 2, 5);
List<SnapshotMetaData> snapshots = new ArrayList<>(numTests);
// Create multiple commits and create a snapshot per commit.
// This should result in Lucene reusing some of the segments for later index commits.
for (int attempt=0; attempt<numTests; attempt++) {
if (nDocs > 0) {
//Delete a few docs
int numDeletes = TestUtil.nextInt(random(), 1, nDocs);
for(int i=0; i<numDeletes; i++) {
masterClient.deleteByQuery("id:" + i);
}
}
// Add a few more
int moreAdds = TestUtil.nextInt(random(), 1, 100);
for (int i = 0; i < moreAdds; i++) {
SolrInputDocument doc = new SolrInputDocument();
doc.addField("id", i + nDocs);
doc.addField("name", "name = " + (i + nDocs));
masterClient.add(doc);
}
masterClient.commit();
// Create a snapshot
snapshots.add(createSnapshot(adminClient, coreName, "snapshot_" + attempt));
}
// Backup the earlier created snapshot.
{
Map<String,String> params = new HashMap<>();
params.put("name", backupName);
params.put("commitName", snapshots.get(0).getName());
params.put("location", location);
BackupRestoreUtils.runCoreAdminCommand(replicaBaseUrl, coreName, CoreAdminAction.BACKUPCORE.toString(), params);
}
// Restore the backup. The purpose of the restore operation is to change the *current* index directory.
// This is required since we delegate the file deletion to underlying IndexDeletionPolicy in case of
// *current* index directory. Hence for the purpose of this test, we want to ensure that the created
// snapshots are NOT in the *current* index directory.
{
Map<String,String> params = new HashMap<>();
params.put("name", "snapshot." + backupName);
params.put("location", location);
BackupRestoreUtils.runCoreAdminCommand(replicaBaseUrl, coreName, CoreAdminAction.RESTORECORE.toString(), params);
}
{
SnapshotMetaData snapshotMetaData = snapshots.get(0);
List<IndexCommit> commits = listCommits(snapshotMetaData.getIndexDirPath());
// Check if number of index commits are > 0 to ensure index file sharing.
assertTrue(commits.size() > 0);
Map<String,Integer> refCounts = SolrSnapshotManager.buildRefCounts(snapshots, commits);
Optional<IndexCommit> ic = commits.stream()
.filter(entry -> entry.getGeneration() == snapshotMetaData.getGenerationNumber())
.findFirst();
assertTrue(ic.isPresent());
Collection<String> nonSharedFiles = new ArrayList<>();
Collection<String> sharedFiles = new ArrayList<>();
for (String fileName : ic.get().getFileNames()) {
if (refCounts.getOrDefault(fileName, 0) > 1) {
sharedFiles.add(fileName);
} else {
nonSharedFiles.add(fileName);
}
}
// Delete snapshot
deleteSnapshot(adminClient, coreName, snapshotMetaData.getName());
// Verify that the shared files are not deleted.
for (String fileName : sharedFiles) {
Path path = Paths.get(snapshotMetaData.getIndexDirPath(), fileName);
assertTrue(path + " should exist.", Files.exists(path));
}
// Verify that the non-shared files are deleted.
for (String fileName : nonSharedFiles) {
Path path = Paths.get(snapshotMetaData.getIndexDirPath(), fileName);
assertFalse(path + " should not exist.", Files.exists(path));
}
}
}
}
@Test @Test
public void testIndexOptimization() throws Exception { public void testIndexOptimization() throws Exception {
CloudSolrClient solrClient = cluster.getSolrClient(); CloudSolrClient solrClient = cluster.getSolrClient();