LUCENE-8212: Never swallow Exceptions in IndexWriter and DocumentsWriter

IndexWriter as well as DocumentsWriter caught Throwable and ignored
it. This is mainly a relict from pre Java 7 were exceptions didn't have
the needed API to suppress exceptions. This change handles exceptions
correctly where the original exception is rethrown and all other exceptions
are added as suppressed.
This commit is contained in:
Simon Willnauer 2018-03-15 15:00:35 +01:00
parent 6eac67fe56
commit 916ed60eea
3 changed files with 127 additions and 115 deletions

View File

@ -19,10 +19,13 @@ package org.apache.lucene.index;
import java.io.Closeable; import java.io.Closeable;
import java.io.IOException; import java.io.IOException;
import java.util.ArrayList;
import java.util.Collection; import java.util.Collection;
import java.util.List;
import java.util.Locale; import java.util.Locale;
import java.util.Queue; import java.util.Queue;
import java.util.concurrent.ConcurrentLinkedQueue; import java.util.concurrent.ConcurrentLinkedQueue;
import java.util.concurrent.atomic.AtomicBoolean;
import java.util.concurrent.atomic.AtomicInteger; import java.util.concurrent.atomic.AtomicInteger;
import org.apache.lucene.analysis.Analyzer; import org.apache.lucene.analysis.Analyzer;
@ -261,16 +264,30 @@ final class DocumentsWriter implements Closeable, Accountable {
return false; // we didn't flush anything here return false; // we didn't flush anything here
} }
/** Returns how many documents were aborted. */ /** Locks all currently active DWPT and aborts them.
synchronized long lockAndAbortAll(IndexWriter indexWriter) throws IOException { * The returned Closeable should be closed once the locks for the aborted
* DWPTs can be released. */
synchronized Closeable lockAndAbortAll(IndexWriter indexWriter) throws IOException {
assert indexWriter.holdsFullFlushLock(); assert indexWriter.holdsFullFlushLock();
if (infoStream.isEnabled("DW")) { if (infoStream.isEnabled("DW")) {
infoStream.message("DW", "lockAndAbortAll"); infoStream.message("DW", "lockAndAbortAll");
} }
// Make sure we move all pending tickets into the flush queue: // Make sure we move all pending tickets into the flush queue:
ticketQueue.forcePurge(indexWriter); ticketQueue.forcePurge(indexWriter);
long abortedDocCount = 0; List<ThreadState> threadStates = new ArrayList<>();
boolean success = false; AtomicBoolean released = new AtomicBoolean(false);
final Closeable release = () -> {
assert indexWriter.holdsFullFlushLock();
if (released.compareAndSet(false, true)) { // only once
if (infoStream.isEnabled("DW")) {
infoStream.message("DW", "unlockAllAbortedThread");
}
perThreadPool.clearAbort();
for (ThreadState state : threadStates) {
state.unlock();
}
}
};
try { try {
deleteQueue.clear(); deleteQueue.clear();
final int limit = perThreadPool.getMaxThreadStates(); final int limit = perThreadPool.getMaxThreadStates();
@ -278,25 +295,31 @@ final class DocumentsWriter implements Closeable, Accountable {
for (int i = 0; i < limit; i++) { for (int i = 0; i < limit; i++) {
final ThreadState perThread = perThreadPool.getThreadState(i); final ThreadState perThread = perThreadPool.getThreadState(i);
perThread.lock(); perThread.lock();
abortedDocCount += abortThreadState(perThread); threadStates.add(perThread);
abortThreadState(perThread);
} }
deleteQueue.clear(); deleteQueue.clear();
// jump over any possible in flight ops: // jump over any possible in flight ops:
deleteQueue.skipSequenceNumbers(perThreadPool.getActiveThreadStateCount()+1); deleteQueue.skipSequenceNumbers(perThreadPool.getActiveThreadStateCount() + 1);
flushControl.abortPendingFlushes(); flushControl.abortPendingFlushes();
flushControl.waitForFlush(); flushControl.waitForFlush();
success = true;
return abortedDocCount;
} finally {
if (infoStream.isEnabled("DW")) { if (infoStream.isEnabled("DW")) {
infoStream.message("DW", "finished lockAndAbortAll success=" + success); infoStream.message("DW", "finished lockAndAbortAll success=true");
} }
if (success == false) { return release;
} catch (Throwable t) {
if (infoStream.isEnabled("DW")) {
infoStream.message("DW", "finished lockAndAbortAll success=false");
}
try {
// if something happens here we unlock all states again // if something happens here we unlock all states again
unlockAllAfterAbortAll(indexWriter); release.close();
} catch (Throwable t1) {
t.addSuppressed(t1);
} }
throw t;
} }
} }
@ -318,28 +341,6 @@ final class DocumentsWriter implements Closeable, Accountable {
return 0; return 0;
} }
} }
synchronized void unlockAllAfterAbortAll(IndexWriter indexWriter) {
assert indexWriter.holdsFullFlushLock();
if (infoStream.isEnabled("DW")) {
infoStream.message("DW", "unlockAll");
}
final int limit = perThreadPool.getMaxThreadStates();
perThreadPool.clearAbort();
for (int i = 0; i < limit; i++) {
try {
final ThreadState perThread = perThreadPool.getThreadState(i);
if (perThread.isHeldByCurrentThread()) {
perThread.unlock();
}
} catch (Throwable e) {
if (infoStream.isEnabled("DW")) {
infoStream.message("DW", "unlockAll: could not unlock state: " + i + " msg:" + e.getMessage());
}
// ignore & keep on unlocking
}
}
}
/** returns the maximum sequence number for all previously completed operations */ /** returns the maximum sequence number for all previously completed operations */
public long getMaxCompletedSequenceNumber() { public long getMaxCompletedSequenceNumber() {

View File

@ -746,10 +746,9 @@ public class IndexWriter implements Closeable, TwoPhaseCommit, Accountable {
checkpointNoSIS(); checkpointNoSIS();
} }
} catch (Throwable t) { } catch (Throwable t) {
priorE = IOUtils.useOrSuppress(priorE, t);
if (doSave) { if (doSave) {
throw IOUtils.rethrowAlways(t); throw t;
} else if (priorE == null) {
priorE = t;
} }
} }
@ -766,10 +765,9 @@ public class IndexWriter implements Closeable, TwoPhaseCommit, Accountable {
try { try {
rld.dropReaders(); rld.dropReaders();
} catch (Throwable t) { } catch (Throwable t) {
priorE = IOUtils.useOrSuppress(priorE, t);
if (doSave) { if (doSave) {
throw IOUtils.rethrowAlways(t); throw t;
} else if (priorE == null) {
priorE = t;
} }
} }
} }
@ -2447,8 +2445,6 @@ public class IndexWriter implements Closeable, TwoPhaseCommit, Accountable {
} }
private void rollbackInternalNoCommit() throws IOException { private void rollbackInternalNoCommit() throws IOException {
boolean success = false;
if (infoStream.isEnabled("IW")) { if (infoStream.isEnabled("IW")) {
infoStream.message("IW", "rollback"); infoStream.message("IW", "rollback");
} }
@ -2467,7 +2463,7 @@ public class IndexWriter implements Closeable, TwoPhaseCommit, Accountable {
docWriter.abort(this); // don't sync on IW here docWriter.abort(this); // don't sync on IW here
docWriter.flushControl.waitForFlush(); // wait for all concurrently running flushes docWriter.flushControl.waitForFlush(); // wait for all concurrently running flushes
purge(true); // empty the flush ticket queue otherwise we might not have cleaned up all resources purge(true); // empty the flush ticket queue otherwise we might not have cleaned up all resources
synchronized(this) { synchronized (this) {
if (pendingCommit != null) { if (pendingCommit != null) {
pendingCommit.rollbackCommit(directory); pendingCommit.rollbackCommit(directory);
@ -2488,8 +2484,8 @@ public class IndexWriter implements Closeable, TwoPhaseCommit, Accountable {
int rollbackMaxDoc = segmentInfos.totalMaxDoc(); int rollbackMaxDoc = segmentInfos.totalMaxDoc();
// now we need to adjust this back to the rolled back SI but don't set it to the absolute value // now we need to adjust this back to the rolled back SI but don't set it to the absolute value
// otherwise we might hide internal bugsf // otherwise we might hide internal bugsf
adjustPendingNumDocs(-(totalMaxDoc-rollbackMaxDoc)); adjustPendingNumDocs(-(totalMaxDoc - rollbackMaxDoc));
if (infoStream.isEnabled("IW") ) { if (infoStream.isEnabled("IW")) {
infoStream.message("IW", "rollback: infos=" + segString(segmentInfos)); infoStream.message("IW", "rollback: infos=" + segString(segmentInfos));
} }
@ -2510,45 +2506,53 @@ public class IndexWriter implements Closeable, TwoPhaseCommit, Accountable {
// after we leave this sync block and before we enter the sync block in the finally clause below that sets closed: // after we leave this sync block and before we enter the sync block in the finally clause below that sets closed:
closed = true; closed = true;
IOUtils.close(writeLock); // release write lock IOUtils.close(writeLock); // release write lock
writeLock = null; writeLock = null;
closed = true;
closing = false;
// So any "concurrently closing" threads wake up and see that the close has now completed:
notifyAll();
} }
} catch (Throwable throwable) {
success = true; try {
} catch (VirtualMachineError tragedy) {
tragicEvent(tragedy, "rollbackInternal");
throw tragedy;
} finally {
if (success == false) {
// Must not hold IW's lock while closing // Must not hold IW's lock while closing
// mergeScheduler: this can lead to deadlock, // mergeScheduler: this can lead to deadlock,
// e.g. TestIW.testThreadInterruptDeadlock // e.g. TestIW.testThreadInterruptDeadlock
IOUtils.closeWhileHandlingException(mergeScheduler); IOUtils.closeWhileHandlingException(mergeScheduler);
} synchronized (this) {
synchronized(this) {
if (success == false) {
// we tried to be nice about it: do the minimum // we tried to be nice about it: do the minimum
// don't leak a segments_N file if there is a pending commit // don't leak a segments_N file if there is a pending commit
if (pendingCommit != null) { if (pendingCommit != null) {
try { try {
pendingCommit.rollbackCommit(directory); pendingCommit.rollbackCommit(directory);
deleter.decRef(pendingCommit); deleter.decRef(pendingCommit);
} catch (Throwable t) { } catch (Throwable t) {
throwable.addSuppressed(t);
} }
pendingCommit = null; pendingCommit = null;
} }
// close all the closeables we can (but important is readerPool and writeLock to prevent leaks) // close all the closeables we can (but important is readerPool and writeLock to prevent leaks)
IOUtils.closeWhileHandlingException(readerPool, deleter, writeLock); IOUtils.closeWhileHandlingException(readerPool, deleter, writeLock);
writeLock = null; writeLock = null;
} closed = true;
closed = true; closing = false;
closing = false;
// So any "concurrently closing" threads wake up and see that the close has now completed: // So any "concurrently closing" threads wake up and see that the close has now completed:
notifyAll(); notifyAll();
}
} catch (Throwable t) {
throwable.addSuppressed(t);
} finally {
if (throwable instanceof VirtualMachineError) {
try {
tragicEvent(throwable, "rollbackInternal");
} catch (Throwable t1){
throwable.addSuppressed(t1);
}
}
} }
throw throwable;
} }
} }
@ -2599,42 +2603,42 @@ public class IndexWriter implements Closeable, TwoPhaseCommit, Accountable {
* sure it's just like a fresh index. * sure it's just like a fresh index.
*/ */
try { try {
synchronized (fullFlushLock) { synchronized (fullFlushLock) {
docWriter.lockAndAbortAll(this); try (Closeable release = docWriter.lockAndAbortAll(this)) {
processEvents(false); processEvents(false);
synchronized (this) { synchronized (this) {
try { try {
// Abort any running merges // Abort any running merges
abortMerges(); abortMerges();
// Let merges run again // Let merges run again
stopMerges = false; stopMerges = false;
adjustPendingNumDocs(-segmentInfos.totalMaxDoc()); adjustPendingNumDocs(-segmentInfos.totalMaxDoc());
// Remove all segments // Remove all segments
segmentInfos.clear(); segmentInfos.clear();
// Ask deleter to locate unreferenced files & remove them: // Ask deleter to locate unreferenced files & remove them:
deleter.checkpoint(segmentInfos, false); deleter.checkpoint(segmentInfos, false);
/* don't refresh the deleter here since there might /* don't refresh the deleter here since there might
* be concurrent indexing requests coming in opening * be concurrent indexing requests coming in opening
* files on the directory after we called DW#abort() * files on the directory after we called DW#abort()
* if we do so these indexing requests might hit FNF exceptions. * if we do so these indexing requests might hit FNF exceptions.
* We will remove the files incrementally as we go... * We will remove the files incrementally as we go...
*/ */
// Don't bother saving any changes in our segmentInfos // Don't bother saving any changes in our segmentInfos
readerPool.dropAll(false); readerPool.dropAll(false);
// Mark that the index has changed // Mark that the index has changed
changeCount.incrementAndGet(); changeCount.incrementAndGet();
segmentInfos.changed(); segmentInfos.changed();
globalFieldNumberMap.clear(); globalFieldNumberMap.clear();
success = true; success = true;
long seqNo = docWriter.deleteQueue.getNextSequenceNumber(); long seqNo = docWriter.deleteQueue.getNextSequenceNumber();
docWriter.setLastSeqNo(seqNo); docWriter.setLastSeqNo(seqNo);
return seqNo; return seqNo;
} finally { } finally {
docWriter.unlockAllAfterAbortAll(this); if (success == false) {
if (!success) { if (infoStream.isEnabled("IW")) {
if (infoStream.isEnabled("IW")) { infoStream.message("IW", "hit exception during deleteAll");
infoStream.message("IW", "hit exception during deleteAll"); }
} }
} }
} }
@ -4065,26 +4069,22 @@ public class IndexWriter implements Closeable, TwoPhaseCommit, Accountable {
deleteNewFiles(merge.info.files()); deleteNewFiles(merge.info.files());
} }
boolean success = false;
try { try {
// Must close before checkpoint, otherwise IFD won't be // Must close before checkpoint, otherwise IFD won't be
// able to delete the held-open files from the merge // able to delete the held-open files from the merge
// readers: // readers:
closeMergeReaders(merge, false); closeMergeReaders(merge, false);
success = true; checkpoint();
} finally { } catch (Throwable t) {
// Must note the change to segmentInfos so any commits // Must note the change to segmentInfos so any commits
// in-flight don't lose it (IFD will incRef/protect the // in-flight don't lose it (IFD will incRef/protect the
// new files we created): // new files we created):
if (success) { try {
checkpoint(); checkpoint();
} else { } catch (Throwable t1) {
try { t.addSuppressed(t1);
checkpoint();
} catch (Throwable t) {
// Ignore so we keep throwing original exception.
}
} }
throw t;
} }
if (infoStream.isEnabled("IW")) { if (infoStream.isEnabled("IW")) {
@ -4120,7 +4120,7 @@ public class IndexWriter implements Closeable, TwoPhaseCommit, Accountable {
// in which case we must throw it so, for example, the // in which case we must throw it so, for example, the
// rollbackTransaction code in addIndexes* is // rollbackTransaction code in addIndexes* is
// executed. // executed.
if (merge.isExternal) { if (merge.isExternal) { // TODO can we simplify this and just throw all the time? this would simplify this a lot
throw (MergePolicy.MergeAbortedException) t; throw (MergePolicy.MergeAbortedException) t;
} }
} else { } else {
@ -4427,9 +4427,7 @@ public class IndexWriter implements Closeable, TwoPhaseCommit, Accountable {
readerPool.drop(rld.info); readerPool.drop(rld.info);
} }
} catch (Throwable t) { } catch (Throwable t) {
if (th == null) { th = IOUtils.useOrSuppress(th, t);
th = t;
}
} }
merge.readers.set(i, null); merge.readers.set(i, null);
} }
@ -4438,9 +4436,7 @@ public class IndexWriter implements Closeable, TwoPhaseCommit, Accountable {
try { try {
merge.mergeFinished(); merge.mergeFinished();
} catch (Throwable t) { } catch (Throwable t) {
if (th == null) { th = IOUtils.useOrSuppress(th, t);
th = t;
}
} }
// If any error occurred, throw it. // If any error occurred, throw it.

View File

@ -122,6 +122,8 @@ public final class IOUtils {
if (object != null) { if (object != null) {
object.close(); object.close();
} }
} catch (VirtualMachineError e) {
throw e;
} catch (Throwable t) { } catch (Throwable t) {
} }
} }
@ -623,4 +625,17 @@ public final class IOUtils {
return desc; return desc;
} }
} }
/**
* Returns the second throwable if the first is null otherwise adds the second as suppressed to the first
* and returns it.
*/
public static <T extends Throwable> T useOrSuppress(T first, T second) {
if (first == null) {
return second;
} else {
first.addSuppressed(second);
}
return first;
}
} }