mirror of https://github.com/apache/lucene.git
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:
parent
c9eb9e10e4
commit
06ae3eb2ec
|
@ -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
|
||||||
|
|
|
@ -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);
|
||||||
|
|
|
@ -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).
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -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);
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -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;
|
||||||
|
|
|
@ -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);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -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);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -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();
|
||||||
|
|
Loading…
Reference in New Issue