From 8540820fe13f51e8ce622a1d0bf571835c02bd6d Mon Sep 17 00:00:00 2001 From: Michael McCandless Date: Tue, 6 May 2008 18:41:10 +0000 Subject: [PATCH] LUCENE-1274: add preparCommit() to IW to do phase 1 of 2-phase commit git-svn-id: https://svn.apache.org/repos/asf/lucene/java/trunk@653878 13f79535-47bb-0310-9956-ffa450edef68 --- CHANGES.txt | 6 + .../index/ConcurrentMergeScheduler.java | 3 + .../apache/lucene/index/DocumentsWriter.java | 11 +- .../org/apache/lucene/index/FieldsWriter.java | 29 ++- .../org/apache/lucene/index/IndexWriter.java | 217 +++++++++++++----- .../org/apache/lucene/index/SegmentInfos.java | 153 +++++++----- .../lucene/store/ChecksumIndexOutput.java | 24 ++ .../apache/lucene/index/TestAtomicUpdate.java | 11 +- .../apache/lucene/index/TestIndexWriter.java | 122 +++++++++- .../lucene/index/TestStressIndexing.java | 1 + .../apache/lucene/index/TestTransactions.java | 217 ++++++++++++++++++ .../apache/lucene/store/MockRAMDirectory.java | 2 +- 12 files changed, 662 insertions(+), 134 deletions(-) create mode 100644 src/test/org/apache/lucene/index/TestTransactions.java diff --git a/CHANGES.txt b/CHANGES.txt index 28064156499..0d59408e569 100644 --- a/CHANGES.txt +++ b/CHANGES.txt @@ -97,6 +97,12 @@ Bug fixes 8. LUCENE-1267: Added numDocs() and maxDoc() to IndexWriter; deprecated docCount(). (Mike McCandless) + + 9. LUCENE-1274: Added new prepareCommit() method to IndexWriter, + which does phase 1 of a 2-phase commit (commit() does phase 2). + This is needed when you want to update an index as part of a + transaction involving external resources (eg a database). Also + deprecated abort(), renaming it to rollback(). (Mike McCandless) New features diff --git a/src/java/org/apache/lucene/index/ConcurrentMergeScheduler.java b/src/java/org/apache/lucene/index/ConcurrentMergeScheduler.java index 9ea7903b120..5ef05a96cfd 100644 --- a/src/java/org/apache/lucene/index/ConcurrentMergeScheduler.java +++ b/src/java/org/apache/lucene/index/ConcurrentMergeScheduler.java @@ -138,6 +138,9 @@ public class ConcurrentMergeScheduler extends MergeScheduler { public void merge(IndexWriter writer) throws CorruptIndexException, IOException { + // TODO: enable this once we are on JRE 1.5 + // assert !Thread.holdsLock(writer); + this.writer = writer; initMergeThreadPriority(); diff --git a/src/java/org/apache/lucene/index/DocumentsWriter.java b/src/java/org/apache/lucene/index/DocumentsWriter.java index 2232f48f9e8..681b3897edb 100644 --- a/src/java/org/apache/lucene/index/DocumentsWriter.java +++ b/src/java/org/apache/lucene/index/DocumentsWriter.java @@ -348,11 +348,12 @@ final class DocumentsWriter { abortCount++; } - /** Called if we hit an exception when adding docs, - * flushing, etc. This resets our state, discarding any - * docs added since last flush. If ae is non-null, it - * contains the root cause exception (which we re-throw - * after we are done aborting). */ + /** Called if we hit an exception at a bad time (when + * updating the index files) and must discard all + * currently buffered docs. This resets our state, + * discarding any docs added since last flush. If ae is + * non-null, it contains the root cause exception (which + * we re-throw after we are done aborting). */ synchronized void abort(AbortException ae) throws IOException { // Anywhere that throws an AbortException must first diff --git a/src/java/org/apache/lucene/index/FieldsWriter.java b/src/java/org/apache/lucene/index/FieldsWriter.java index 5dfb5b6f665..e6dff0fc1b1 100644 --- a/src/java/org/apache/lucene/index/FieldsWriter.java +++ b/src/java/org/apache/lucene/index/FieldsWriter.java @@ -64,8 +64,16 @@ final class FieldsWriter success = true; } finally { if (!success) { - close(); - d.deleteFile(fieldsName); + try { + close(); + } catch (Throwable t) { + // Suppress so we keep throwing the original exception + } + try { + d.deleteFile(fieldsName); + } catch (Throwable t) { + // Suppress so we keep throwing the original exception + } } } @@ -77,9 +85,20 @@ final class FieldsWriter success = true; } finally { if (!success) { - close(); - d.deleteFile(fieldsName); - d.deleteFile(indexName); + try { + close(); + } catch (IOException ioe) { + } + try { + d.deleteFile(fieldsName); + } catch (Throwable t) { + // Suppress so we keep throwing the original exception + } + try { + d.deleteFile(indexName); + } catch (Throwable t) { + // Suppress so we keep throwing the original exception + } } } diff --git a/src/java/org/apache/lucene/index/IndexWriter.java b/src/java/org/apache/lucene/index/IndexWriter.java index 3d273edab68..c62984a3d97 100644 --- a/src/java/org/apache/lucene/index/IndexWriter.java +++ b/src/java/org/apache/lucene/index/IndexWriter.java @@ -306,6 +306,9 @@ public class IndexWriter { private SegmentInfos rollbackSegmentInfos; // segmentInfos we will fallback to if the commit fails private HashMap rollbackSegments; + volatile SegmentInfos pendingCommit; // set when a commit is pending (after prepareCommit() & before commit()) + volatile long pendingCommitChangeCount; + private SegmentInfos localRollbackSegmentInfos; // segmentInfos we will fallback to if the commit fails private boolean localAutoCommit; // saved autoCommit during local transaction private int localFlushedDocCount; // saved docWriter.getFlushedDocCount during local transaction @@ -364,12 +367,13 @@ public class IndexWriter { infoStream.println("IW " + messageID + " [" + Thread.currentThread().getName() + "]: " + message); } - private synchronized void setMessageID() { + private synchronized void setMessageID(PrintStream infoStream) { if (infoStream != null && messageID == -1) { synchronized(MESSAGE_ID_LOCK) { messageID = MESSAGE_ID++; } } + this.infoStream = infoStream; } /** @@ -1082,9 +1086,8 @@ public class IndexWriter { this.closeDir = closeDir; directory = d; analyzer = a; - this.infoStream = defaultInfoStream; + setMessageID(defaultInfoStream); this.maxFieldLength = maxFieldLength; - setMessageID(); if (create) { // Clear the write lock in case it's leftover: @@ -1496,8 +1499,7 @@ public class IndexWriter { */ public void setInfoStream(PrintStream infoStream) { ensureOpen(); - this.infoStream = infoStream; - setMessageID(); + setMessageID(infoStream); docWriter.setInfoStream(infoStream); deleter.setInfoStream(infoStream); if (infoStream != null) @@ -1672,7 +1674,7 @@ public class IndexWriter { if (infoStream != null) message("now call final commit()"); - commit(true, 0); + commit(0); if (infoStream != null) message("at close: " + segString()); @@ -2571,7 +2573,7 @@ public class IndexWriter { if (autoCommit) { boolean success = false; try { - commit(true, 0); + commit(0); success = true; } finally { if (!success) { @@ -2587,6 +2589,13 @@ public class IndexWriter { localRollbackSegmentInfos = null; } + /** + * @deprecated Please use {@link #rollback} instead. + */ + public void abort() throws IOException { + rollback(); + } + /** * Close the IndexWriter without committing * any of the changes that have occurred since it was @@ -2594,18 +2603,27 @@ public class IndexWriter { * created, after which the state of the index will be the * same as it was when this writer was first opened. This * can only be called when this IndexWriter was opened - * with autoCommit=false. + * with autoCommit=false. This also clears a + * previous call to {@link #prepareCommit}. * @throws IllegalStateException if this is called when * the writer was opened with autoCommit=true. * @throws IOException if there is a low-level IO error */ - public void abort() throws IOException { + public void rollback() throws IOException { ensureOpen(); if (autoCommit) throw new IllegalStateException("abort() can only be called when IndexWriter was opened with autoCommit=false"); boolean doClose; synchronized(this) { + + if (pendingCommit != null) { + pendingCommit.rollbackCommit(directory); + deleter.decRef(pendingCommit); + pendingCommit = null; + notifyAll(); + } + // Ensure that only one thread actually gets to do the closing: if (!closing) { doClose = true; @@ -3113,10 +3131,54 @@ public class IndexWriter { flush(true, false, true); } + /**

Expert: prepare for commit. This does the first + * phase of 2-phase commit. You can only call this when + * autoCommit is false. This method does all steps + * necessary to commit changes since this writer was + * opened: flushes pending added and deleted docs, syncs + * the index files, writes most of next segments_N file. + * After calling this you must call either {@link + * #commit()} to finish the commit, or {@link + * #rollback()} to revert the commit and undo all changes + * done since the writer was opened.

+ * + * You can also just call {@link #commit()} directly + * without prepareCommit first in which case that method + * will internally call prepareCommit. + */ + public final void prepareCommit() throws CorruptIndexException, IOException { + prepareCommit(false); + } + + private final void prepareCommit(boolean internal) throws CorruptIndexException, IOException { + + if (hitOOM) + throw new IllegalStateException("this writer hit an OutOfMemoryError; cannot commit"); + + if (autoCommit && !internal) + throw new IllegalStateException("this method can only be used when autoCommit is false"); + + if (!autoCommit && pendingCommit != null) + throw new IllegalStateException("prepareCommit was already called with no corresponding call to commit"); + + message("prepareCommit: flush"); + + flush(true, true, true); + + startCommit(0); + } + + private void commit(long sizeInBytes) throws IOException { + startCommit(sizeInBytes); + finishCommit(); + } + /** - *

Commits all pending updates (added & deleted documents) - * to the index, and syncs all referenced index files, - * such that a reader will see the changes. Note that + *

Commits all pending updates (added & deleted + * documents) to the index, and syncs all referenced index + * files, such that a reader will see the changes and the + * index updates will survive an OS or machine crash or + * power loss (though, see the note below). Note that * this does not wait for any running background merges to * finish. This may be a costly operation, so you should * test the cost in your application and do it only when @@ -3135,12 +3197,38 @@ public class IndexWriter { * consistency on such devices.

*/ public final void commit() throws CorruptIndexException, IOException { - commit(true); + + message("commit: start"); + + if (autoCommit || pendingCommit == null) { + message("commit: now prepare"); + prepareCommit(true); + } else + message("commit: already prepared"); + + finishCommit(); } - private final void commit(boolean triggerMerges) throws CorruptIndexException, IOException { - flush(triggerMerges, true, true); - commit(true, 0); + private synchronized final void finishCommit() throws CorruptIndexException, IOException { + + if (pendingCommit != null) { + try { + message("commit: pendingCommit != null"); + pendingCommit.finishCommit(directory); + lastCommitChangeCount = pendingCommitChangeCount; + segmentInfos.updateGeneration(pendingCommit); + setRollbackSegmentInfos(); + deleter.checkpoint(pendingCommit, true); + } finally { + deleter.decRef(pendingCommit); + pendingCommit = null; + notifyAll(); + } + + } else + message("commit: pendingCommit == null; skip"); + + message("commit: done"); } /** @@ -3176,8 +3264,7 @@ public class IndexWriter { // when flushing a segment; otherwise deletes may become // visible before their corresponding added document // from an updateDocument call - if (autoCommit) - flushDeletes = true; + flushDeletes |= autoCommit; // Returns true if docWriter is currently aborting, in // which case we skip flushing this segment @@ -3935,7 +4022,7 @@ public class IndexWriter { synchronized(this) { size = merge.info.sizeInBytes(); } - commit(false, size); + commit(size); } success = false; @@ -3988,7 +4075,7 @@ public class IndexWriter { synchronized(this) { size = merge.info.sizeInBytes(); } - commit(false, size); + commit(size); } return mergedDocCount; @@ -4151,13 +4238,13 @@ public class IndexWriter { } /** Walk through all files referenced by the current - * segmentInfos, minus flushes, and ask the Directory to - * sync each file, if it wasn't already. If that - * succeeds, then we write a new segments_N file & sync - * that. */ - private void commit(boolean skipWait, long sizeInBytes) throws IOException { + * segmentInfos and ask the Directory to sync each file, + * if it wasn't already. If that succeeds, then we + * prepare a new segments_N file but do not fully commit + * it. */ + private void startCommit(long sizeInBytes) throws IOException { - assert testPoint("startCommit"); + assert testPoint("startStartCommit"); if (hitOOM) return; @@ -4165,9 +4252,9 @@ public class IndexWriter { try { if (infoStream != null) - message("start commit() skipWait=" + skipWait + " sizeInBytes=" + sizeInBytes); + message("startCommit(): start sizeInBytes=" + sizeInBytes); - if (!skipWait) + if (sizeInBytes > 0) syncPause(sizeInBytes); SegmentInfos toSync = null; @@ -4179,7 +4266,7 @@ public class IndexWriter { if (changeCount == lastCommitChangeCount) { if (infoStream != null) - message(" skip commit(): no changes pending"); + message(" skip startCommit(): no changes pending"); return; } @@ -4189,15 +4276,17 @@ public class IndexWriter { // threads can be doing this at once, if say a large // merge and a small merge finish at the same time: + if (infoStream != null) + message("startCommit index=" + segString(segmentInfos) + " changeCount=" + changeCount); + toSync = (SegmentInfos) segmentInfos.clone(); deleter.incRef(toSync, false); myChangeCount = changeCount; } - if (infoStream != null) - message("commit index=" + segString(toSync)); + assert testPoint("midStartCommit"); - assert testPoint("midCommit"); + boolean setPending = false; try { @@ -4237,54 +4326,72 @@ public class IndexWriter { break; } - assert testPoint("midCommit2"); - + assert testPoint("midStartCommit2"); + synchronized(this) { // If someone saved a newer version of segments file // since I first started syncing my version, I can // safely skip saving myself since I've been // superseded: - if (myChangeCount > lastCommitChangeCount) { - + if (myChangeCount > lastCommitChangeCount && (pendingCommit == null || myChangeCount > pendingCommitChangeCount)) { + + // Wait now for any current pending commit to complete: + while(pendingCommit != null) { + message("wait for existing pendingCommit to finish..."); + try { + wait(); + } catch (InterruptedException ie) { + Thread.currentThread().interrupt(); + } + } + if (segmentInfos.getGeneration() > toSync.getGeneration()) toSync.updateGeneration(segmentInfos); boolean success = false; try { - toSync.commit(directory); + + // Exception here means nothing is prepared + // (this method unwinds everything it did on + // an exception) + try { + toSync.prepareCommit(directory); + } finally { + // Have our master segmentInfos record the + // generations we just prepared. We do this + // on error or success so we don't + // double-write a segments_N file. + segmentInfos.updateGeneration(toSync); + } + + assert pendingCommit == null; + setPending = true; + pendingCommit = toSync; + pendingCommitChangeCount = myChangeCount; success = true; } finally { - // Have our master segmentInfos record the - // generations we just sync'd - segmentInfos.updateGeneration(toSync); if (!success) message("hit exception committing segments file"); } - - message("commit complete"); - - lastCommitChangeCount = myChangeCount; - - deleter.checkpoint(toSync, true); - setRollbackSegmentInfos(); } else message("sync superseded by newer infos"); } message("done all syncs"); - assert testPoint("midCommitSuccess"); + assert testPoint("midStartCommitSuccess"); } finally { synchronized(this) { - deleter.decRef(toSync); + if (!setPending) + deleter.decRef(toSync); } } } catch (OutOfMemoryError oom) { hitOOM = true; throw oom; } - assert testPoint("finishCommit"); + assert testPoint("finishStartCommit"); } /** @@ -4377,11 +4484,11 @@ public class IndexWriter { // Used only by assert for testing. Current points: // startDoFlush // startCommitMerge - // startCommit - // midCommit - // midCommit2 - // midCommitSuccess - // finishCommit + // startStartCommit + // midStartCommit + // midStartCommit2 + // midStartCommitSuccess + // finishStartCommit // startCommitMergeDeletes // startMergeInit // startApplyDeletes diff --git a/src/java/org/apache/lucene/index/SegmentInfos.java b/src/java/org/apache/lucene/index/SegmentInfos.java index 334a3848bbc..90e79aa0ef5 100644 --- a/src/java/org/apache/lucene/index/SegmentInfos.java +++ b/src/java/org/apache/lucene/index/SegmentInfos.java @@ -274,6 +274,10 @@ final class SegmentInfos extends Vector { }.run(); } + // Only non-null after prepareCommit has been called and + // before finishCommit is called + ChecksumIndexOutput pendingOutput; + private final void write(Directory directory) throws IOException { String segmentFileName = getNextSegmentFileName(); @@ -298,53 +302,27 @@ final class SegmentInfos extends Vector { for (int i = 0; i < size(); i++) { info(i).write(output); } - final long checksum = output.getChecksum(); - output.writeLong(checksum); + output.prepareCommit(); success = true; + pendingOutput = output; } finally { - boolean success2 = false; - try { - if (!success) { - // We hit an exception above; try to close the file - // but suppress any exception: - try { - output.close(); - success2 = true; - } catch (Throwable t) { - // Suppress so we keep throwing the original exception - } - } else { + if (!success) { + // We hit an exception above; try to close the file + // but suppress any exception: + try { output.close(); - success2 = true; + } catch (Throwable t) { + // Suppress so we keep throwing the original exception } - } finally { - if (!success || !success2) { - try { - // Try not to leave a truncated segments_N file in - // the index: - directory.deleteFile(segmentFileName); - } catch (Throwable t) { - // Suppress so we keep throwing the original exception - } + try { + // Try not to leave a truncated segments_N file in + // the index: + directory.deleteFile(segmentFileName); + } catch (Throwable t) { + // Suppress so we keep throwing the original exception } } } - - try { - IndexOutput genOutput = directory.createOutput(IndexFileNames.SEGMENTS_GEN); - try { - genOutput.writeInt(FORMAT_LOCKLESS); - genOutput.writeLong(generation); - genOutput.writeLong(generation); - } finally { - genOutput.close(); - } - } catch (IOException e) { - // It's OK if we fail to write this file since it's - // used only as one of the retry fallbacks. - } - - lastGeneration = generation; } /** @@ -355,7 +333,7 @@ final class SegmentInfos extends Vector { public Object clone() { SegmentInfos sis = (SegmentInfos) super.clone(); for(int i=0;i generation; lastGeneration = other.lastGeneration; generation = other.generation; version = other.version; } - /** Writes & syncs to the Directory dir, taking care to - * remove the segments file on exception */ - public final void commit(Directory dir) throws IOException { - boolean success = false; - try { - write(dir); - success = true; - } finally { - if (!success) { - // Must carefully compute fileName from "generation" - // since lastGeneration isn't incremented: + public final void rollbackCommit(Directory dir) throws IOException { + if (pendingOutput != null) { + try { + pendingOutput.close(); + } catch (Throwable t) { + // Suppress so we keep throwing the original exception + // in our caller + } + + // Must carefully compute fileName from "generation" + // since lastGeneration isn't incremented: + try { final String segmentFileName = IndexFileNames.fileNameFromGeneration(IndexFileNames.SEGMENTS, "", generation); - try { - dir.deleteFile(segmentFileName); - } catch (Throwable t) { - // Suppress so we keep throwing the original exception - } + dir.deleteFile(segmentFileName); + } catch (Throwable t) { + // Suppress so we keep throwing the original exception + // in our caller } + pendingOutput = null; + } + } + + /** Call this to start a commit. This writes the new + * segments file, but writes an invalid checksum at the + * end, so that it is not visible to readers. Once this + * is called you must call {@link #finishCommit} to complete + * the commit or {@link #rollbackCommit} to abort it. */ + public final void prepareCommit(Directory dir) throws IOException { + if (pendingOutput != null) + throw new IllegalStateException("prepareCommit was already called"); + write(dir); + } + + public final void finishCommit(Directory dir) throws IOException { + if (pendingOutput == null) + throw new IllegalStateException("prepareCommit was not called"); + boolean success = false; + try { + pendingOutput.finishCommit(); + pendingOutput.close(); + pendingOutput = null; + success = true; + } finally { + if (!success) + rollbackCommit(dir); } // NOTE: if we crash here, we have left a segments_N // file in the directory in a possibly corrupt state (if // some bytes made it to stable storage and others - // didn't). But, the segments_N file now includes - // checksum at the end, which should catch this case. - // So when a reader tries to read it, it will throw a + // didn't). But, the segments_N file includes checksum + // at the end, which should catch this case. So when a + // reader tries to read it, it will throw a // CorruptIndexException, which should cause the retry // logic in SegmentInfos to kick in and load the last // good (previous) segments_N-1 file. - final String fileName = getCurrentSegmentFileName(); + final String fileName = IndexFileNames.fileNameFromGeneration(IndexFileNames.SEGMENTS, + "", + generation); success = false; try { dir.sync(fileName); @@ -791,5 +797,28 @@ final class SegmentInfos extends Vector { } } } + + lastGeneration = generation; + + try { + IndexOutput genOutput = dir.createOutput(IndexFileNames.SEGMENTS_GEN); + try { + genOutput.writeInt(FORMAT_LOCKLESS); + genOutput.writeLong(generation); + genOutput.writeLong(generation); + } finally { + genOutput.close(); + } + } catch (Throwable t) { + // It's OK if we fail to write this file since it's + // used only as one of the retry fallbacks. + } + } + + /** Writes & syncs to the Directory dir, taking care to + * remove the segments file on exception */ + public final void commit(Directory dir) throws IOException { + prepareCommit(dir); + finishCommit(dir); } } diff --git a/src/java/org/apache/lucene/store/ChecksumIndexOutput.java b/src/java/org/apache/lucene/store/ChecksumIndexOutput.java index 9b2562b8699..ad2895b8a2b 100644 --- a/src/java/org/apache/lucene/store/ChecksumIndexOutput.java +++ b/src/java/org/apache/lucene/store/ChecksumIndexOutput.java @@ -62,6 +62,30 @@ public class ChecksumIndexOutput extends IndexOutput { throw new RuntimeException("not allowed"); } + /** + * Starts but does not complete the commit of this file (= + * writing of the final checksum at the end). After this + * is called must call {@link #finishCommit} and the + * {@link #close} to complete the commit. + */ + public void prepareCommit() throws IOException { + final long checksum = getChecksum(); + // Intentionally write a mismatched checksum. This is + // because we want to 1) test, as best we can, that we + // are able to write a long to the file, but 2) not + // actually "commit" the file yet. This (prepare + // commit) is phase 1 of a two-phase commit. + final long pos = main.getFilePointer(); + main.writeLong(checksum-1); + main.flush(); + main.seek(pos); + } + + /** See {@link #prepareCommit} */ + public void finishCommit() throws IOException { + main.writeLong(getChecksum()); + } + public long length() throws IOException { return main.length(); } diff --git a/src/test/org/apache/lucene/index/TestAtomicUpdate.java b/src/test/org/apache/lucene/index/TestAtomicUpdate.java index 329759fffff..d5b94621ecc 100644 --- a/src/test/org/apache/lucene/index/TestAtomicUpdate.java +++ b/src/test/org/apache/lucene/index/TestAtomicUpdate.java @@ -68,6 +68,7 @@ public class TestAtomicUpdate extends LuceneTestCase { count++; } } catch (Throwable e) { + System.out.println(Thread.currentThread().getName() + ": exc"); e.printStackTrace(System.out); failed = true; } @@ -111,11 +112,7 @@ public class TestAtomicUpdate extends LuceneTestCase { public void doWork() throws Throwable { IndexReader r = IndexReader.open(directory); - try { - assertEquals(100, r.numDocs()); - } catch (Throwable t) { - throw t; - } + assertEquals(100, r.numDocs()); r.close(); } } @@ -141,6 +138,10 @@ public class TestAtomicUpdate extends LuceneTestCase { } writer.commit(); + IndexReader r = IndexReader.open(directory); + assertEquals(100, r.numDocs()); + r.close(); + IndexerThread indexerThread = new IndexerThread(writer, threads); threads[0] = indexerThread; indexerThread.start(); diff --git a/src/test/org/apache/lucene/index/TestIndexWriter.java b/src/test/org/apache/lucene/index/TestIndexWriter.java index 229f63723f1..242fd7e1507 100644 --- a/src/test/org/apache/lucene/index/TestIndexWriter.java +++ b/src/test/org/apache/lucene/index/TestIndexWriter.java @@ -3302,7 +3302,7 @@ public class TestIndexWriter extends LuceneTestCase boolean isCommit = false; boolean isDelete = false; for (int i = 0; i < trace.length; i++) { - if ("org.apache.lucene.index.SegmentInfos".equals(trace[i].getClassName()) && "commit".equals(trace[i].getMethodName())) + if ("org.apache.lucene.index.SegmentInfos".equals(trace[i].getClassName()) && "prepareCommit".equals(trace[i].getMethodName())) isCommit = true; if ("org.apache.lucene.store.MockRAMDirectory".equals(trace[i].getClassName()) && "deleteFile".equals(trace[i].getMethodName())) isDelete = true; @@ -3603,4 +3603,124 @@ public class TestIndexWriter extends LuceneTestCase s.close(); dir.close(); } + + // LUCENE-1274: test writer.prepareCommit() + public void testPrepareCommit() throws IOException { + Directory dir = new MockRAMDirectory(); + + IndexWriter writer = new IndexWriter(dir, false, new WhitespaceAnalyzer(), IndexWriter.MaxFieldLength.LIMITED); + writer.setMaxBufferedDocs(2); + writer.setMergeFactor(5); + + for (int i = 0; i < 23; i++) + addDoc(writer); + + IndexReader reader = IndexReader.open(dir); + assertEquals(0, reader.numDocs()); + + writer.prepareCommit(); + + IndexReader reader2 = IndexReader.open(dir); + assertEquals(0, reader2.numDocs()); + + writer.commit(); + + IndexReader reader3 = reader.reopen(); + assertEquals(0, reader.numDocs()); + assertEquals(0, reader2.numDocs()); + assertEquals(23, reader3.numDocs()); + reader.close(); + reader2.close(); + + for (int i = 0; i < 17; i++) + addDoc(writer); + + assertEquals(23, reader3.numDocs()); + reader3.close(); + reader = IndexReader.open(dir); + assertEquals(23, reader.numDocs()); + reader.close(); + + writer.prepareCommit(); + + reader = IndexReader.open(dir); + assertEquals(23, reader.numDocs()); + reader.close(); + + writer.commit(); + reader = IndexReader.open(dir); + assertEquals(40, reader.numDocs()); + reader.close(); + writer.close(); + dir.close(); + } + + // LUCENE-1274: test writer.prepareCommit() + public void testPrepareCommitRollback() throws IOException { + MockRAMDirectory dir = new MockRAMDirectory(); + dir.setPreventDoubleWrite(false); + + IndexWriter writer = new IndexWriter(dir, false, new WhitespaceAnalyzer(), IndexWriter.MaxFieldLength.LIMITED); + + writer.setMaxBufferedDocs(2); + writer.setMergeFactor(5); + + for (int i = 0; i < 23; i++) + addDoc(writer); + + IndexReader reader = IndexReader.open(dir); + assertEquals(0, reader.numDocs()); + + writer.prepareCommit(); + + IndexReader reader2 = IndexReader.open(dir); + assertEquals(0, reader2.numDocs()); + + writer.rollback(); + + IndexReader reader3 = reader.reopen(); + assertEquals(0, reader.numDocs()); + assertEquals(0, reader2.numDocs()); + assertEquals(0, reader3.numDocs()); + reader.close(); + reader2.close(); + + writer = new IndexWriter(dir, false, new WhitespaceAnalyzer(), IndexWriter.MaxFieldLength.LIMITED); + for (int i = 0; i < 17; i++) + addDoc(writer); + + assertEquals(0, reader3.numDocs()); + reader3.close(); + reader = IndexReader.open(dir); + assertEquals(0, reader.numDocs()); + reader.close(); + + writer.prepareCommit(); + + reader = IndexReader.open(dir); + assertEquals(0, reader.numDocs()); + reader.close(); + + writer.commit(); + reader = IndexReader.open(dir); + assertEquals(17, reader.numDocs()); + reader.close(); + writer.close(); + dir.close(); + } + + // LUCENE-1274 + public void testPrepareCommitNoChanges() throws IOException { + MockRAMDirectory dir = new MockRAMDirectory(); + + IndexWriter writer = new IndexWriter(dir, false, new WhitespaceAnalyzer(), IndexWriter.MaxFieldLength.LIMITED); + writer.prepareCommit(); + writer.commit(); + writer.close(); + + IndexReader reader = IndexReader.open(dir); + assertEquals(0, reader.numDocs()); + reader.close(); + dir.close(); + } } diff --git a/src/test/org/apache/lucene/index/TestStressIndexing.java b/src/test/org/apache/lucene/index/TestStressIndexing.java index 80f7f761598..a18e0b42266 100644 --- a/src/test/org/apache/lucene/index/TestStressIndexing.java +++ b/src/test/org/apache/lucene/index/TestStressIndexing.java @@ -53,6 +53,7 @@ public class TestStressIndexing extends LuceneTestCase { count++; } } catch (Throwable e) { + System.out.println(Thread.currentThread() + ": exc"); e.printStackTrace(System.out); failed = true; } diff --git a/src/test/org/apache/lucene/index/TestTransactions.java b/src/test/org/apache/lucene/index/TestTransactions.java new file mode 100644 index 00000000000..b939bc4780e --- /dev/null +++ b/src/test/org/apache/lucene/index/TestTransactions.java @@ -0,0 +1,217 @@ +package org.apache.lucene.index; + +/** + * Licensed to the Apache Software Foundation (ASF) under one or more + * contributor license agreements. See the NOTICE file distributed with + * this work for additional information regarding copyright ownership. + * The ASF licenses this file to You under the Apache License, Version 2.0 + * (the "License"); you may not use this file except in compliance with + * the License. You may obtain a copy of the License at + * + * http://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ + +import java.io.IOException; +import java.util.Random; +import org.apache.lucene.store.*; +import org.apache.lucene.util.*; +import org.apache.lucene.analysis.*; +import org.apache.lucene.document.*; + +public class TestTransactions extends LuceneTestCase +{ + private static final Random RANDOM = new Random(); + private static volatile boolean doFail; + + private class RandomFailure extends MockRAMDirectory.Failure { + public void eval(MockRAMDirectory dir) throws IOException { + if (TestTransactions.doFail && RANDOM.nextInt() % 10 <= 3) + throw new IOException("now failing randomly but on purpose"); + } + } + + private static abstract class TimedThread extends Thread { + boolean failed; + private static int RUN_TIME_SEC = 6; + private TimedThread[] allThreads; + + abstract public void doWork() throws Throwable; + + TimedThread(TimedThread[] threads) { + this.allThreads = threads; + } + + public void run() { + final long stopTime = System.currentTimeMillis() + 1000*RUN_TIME_SEC; + + try { + while(System.currentTimeMillis() < stopTime && !anyErrors()) + doWork(); + } catch (Throwable e) { + System.out.println(Thread.currentThread() + ": exc"); + e.printStackTrace(System.out); + failed = true; + } + } + + private boolean anyErrors() { + for(int i=0;i