From fa4d10baf9cd3b450e8008c54c5a5f9e43a1f8e3 Mon Sep 17 00:00:00 2001 From: Michael McCandless Date: Sun, 9 Mar 2008 08:19:00 +0000 Subject: [PATCH] LUCENE-1210: fix deadlock case on hitting exception in mergeInit git-svn-id: https://svn.apache.org/repos/asf/lucene/java/trunk@635190 13f79535-47bb-0310-9956-ffa450edef68 --- .../org/apache/lucene/index/IndexWriter.java | 15 ++++++- .../apache/lucene/index/TestIndexWriter.java | 44 ++++++++++++++++++- 2 files changed, 57 insertions(+), 2 deletions(-) diff --git a/src/java/org/apache/lucene/index/IndexWriter.java b/src/java/org/apache/lucene/index/IndexWriter.java index 075f7fa80e5..cadf92d3e2d 100644 --- a/src/java/org/apache/lucene/index/IndexWriter.java +++ b/src/java/org/apache/lucene/index/IndexWriter.java @@ -3588,8 +3588,21 @@ public class IndexWriter { } /** Does initial setup for a merge, which is fast but holds - * the synchronized lock on IndexWriter instance. */ + * the synchronized lock on IndexWriter instance. */ final synchronized void mergeInit(MergePolicy.OneMerge merge) throws IOException { + boolean success = false; + try { + _mergeInit(merge); + success = true; + } finally { + if (!success) { + mergeFinish(merge); + runningMerges.remove(merge); + } + } + } + + final synchronized private void _mergeInit(MergePolicy.OneMerge merge) throws IOException { assert testPoint("startMergeInit"); diff --git a/src/test/org/apache/lucene/index/TestIndexWriter.java b/src/test/org/apache/lucene/index/TestIndexWriter.java index 917ce03fcbc..12a558a4808 100644 --- a/src/test/org/apache/lucene/index/TestIndexWriter.java +++ b/src/test/org/apache/lucene/index/TestIndexWriter.java @@ -3181,7 +3181,7 @@ public class TestIndexWriter extends LuceneTestCase Field.Index.TOKENIZED)); try { w.addDocument(crashDoc, analyzer); - fail("did not hit exxpected exception"); + fail("did not hit expected exception"); } catch (IOException ioe) { // expected } @@ -3189,4 +3189,46 @@ public class TestIndexWriter extends LuceneTestCase w.close(); dir.close(); } + + public class MockIndexWriter2 extends IndexWriter { + + public MockIndexWriter2(Directory dir, boolean autoCommit, Analyzer a, boolean create, MaxFieldLength mfl) throws IOException { + super(dir, autoCommit, a, create, mfl); + } + + boolean doFail; + boolean failed; + + boolean testPoint(String name) { + if (doFail && name.equals("startMergeInit")) { + failed = true; + throw new RuntimeException("intentionally failing"); + } + return true; + } + } + + // LUCENE-1210 + public void testExceptionOnMergeInit() throws IOException { + MockRAMDirectory dir = new MockRAMDirectory(); + MockIndexWriter2 w = new MockIndexWriter2(dir, false, new WhitespaceAnalyzer(), true, IndexWriter.MaxFieldLength.UNLIMITED); + w.setMaxBufferedDocs(2); + w.setMergeFactor(2); + w.doFail = true; + w.setMergeScheduler(new ConcurrentMergeScheduler()); + Document doc = new Document(); + doc.add(new Field("field", "a field", Field.Store.YES, + Field.Index.TOKENIZED)); + for(int i=0;i<10;i++) + try { + w.addDocument(doc); + } catch (RuntimeException re) { + break; + } + + ((ConcurrentMergeScheduler) w.getMergeScheduler()).sync(); + assertTrue(w.failed); + w.close(); + dir.close(); + } }